tenfourfox/netwerk/wifi/osx_corewlan.mm

126 lines
3.7 KiB
Plaintext

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#import <Cocoa/Cocoa.h>
#if(0)
#import <CoreWLAN/CoreWLAN.h>
#endif
#include <dlfcn.h>
#include <unistd.h>
#include <objc/objc.h>
#include <objc/objc-runtime.h>
#include "nsObjCExceptions.h"
#include "nsAutoPtr.h"
#include "nsCOMArray.h"
#include "nsWifiMonitor.h"
#include "nsWifiAccessPoint.h"
// Undo bug 848435. We still sort of support running under Rosetta on
// Snow Leopard, so we still need to support CoreWLAN, sort of. Also
// see bug 514626. This is actually called from nsWifiScannerMac now
// to make our patch a little more atomic.
bool WiFiUsingSnowLeopard() {
static int32_t gOSXVersion = 0x0;
if (gOSXVersion == 0x0) {
Gestalt(gestaltSystemVersion, (SInt32*)&gOSXVersion);
}
// Force this to act like a C++ boolean, not a Obj-C++ BOOL.
return (gOSXVersion >= 0x00001060) ? true : false;
}
nsresult
GetAccessPointsFromWLAN(nsCOMArray<nsWifiAccessPoint> &accessPoints)
{
NS_OBJC_BEGIN_TRY_ABORT_BLOCK_RETURN;
accessPoints.Clear();
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
@try {
NSBundle * bundle = [[[NSBundle alloc] initWithPath:@"/System/Library/Frameworks/CoreWLAN.framework"] autorelease];
if (!bundle) {
[pool release];
return NS_ERROR_NOT_AVAILABLE;
}
Class CWI_class = [bundle classNamed:@"CWInterface"];
if (!CWI_class) {
[pool release];
return NS_ERROR_NOT_AVAILABLE;
}
id scanResult = [[CWI_class interface] scanForNetworksWithParameters:nil error:nil];
if (!scanResult) {
[pool release];
return NS_ERROR_NOT_AVAILABLE;
}
NSArray* scan = [NSMutableArray arrayWithArray:scanResult];
NSEnumerator *enumerator = [scan objectEnumerator];
while (id anObject = [enumerator nextObject]) {
nsWifiAccessPoint* ap = new nsWifiAccessPoint();
if (!ap) {
[pool release];
return NS_ERROR_OUT_OF_MEMORY;
}
// [CWInterface bssidData] is deprecated on OS X 10.7 and up. Which is
// is a pain, so we'll use it for as long as it's available.
unsigned char macData[6] = {0};
if ([anObject respondsToSelector:@selector(bssidData)]) {
NSData* data = [anObject bssidData];
if (data) {
memcpy(macData, [data bytes], 6);
}
} else {
// [CWInterface bssid] returns a string formatted "00:00:00:00:00:00".
NSString* macString = [anObject bssid];
if (macString && ([macString length] == 17)) {
#define NSUInteger PRUint32
for (NSUInteger i = 0; i < 6; ++i) {
NSString* part = [macString substringWithRange:NSMakeRange(i * 3, 2)];
NSScanner* scanner = [NSScanner scannerWithString:part];
unsigned int data = 0;
if (![scanner scanHexInt:&data]) {
data = 0;
}
macData[i] = (unsigned char) data;
}
}
}
// [CWInterface rssiValue] is available on OS X 10.7 and up (and
// [CWInterface rssi] is deprecated).
int signal = 0;
if ([anObject respondsToSelector:@selector(rssiValue)]) {
#define NSInteger PRInt32
signal = (int) ((NSInteger) [anObject rssiValue]);
} else {
signal = [[anObject rssi] intValue];
}
ap->setMac(macData);
ap->setSignal(signal);
ap->setSSID([[anObject ssid] UTF8String], 32);
accessPoints.AppendObject(ap);
}
}
@catch(NSException*) {
[pool release];
return NS_ERROR_NOT_AVAILABLE;
}
[pool release];
return NS_OK;
NS_OBJC_END_TRY_ABORT_BLOCK_RETURN(NS_ERROR_NOT_AVAILABLE);
}