2009-03-21 04:49:51 -07:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is Geolocation.
|
|
|
|
*
|
2009-12-21 13:50:30 -08:00
|
|
|
* The Initial Developer of the Original Code is Mozilla Foundation
|
2009-03-21 04:49:51 -07:00
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2008
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* This is a derivative of work done by Google under a BSD style License.
|
|
|
|
* See: http://gears.googlecode.com/svn/trunk/gears/geolocation/
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Doug Turner <dougt@meer.net> (Original Author)
|
2009-04-21 07:30:25 -07:00
|
|
|
* Nino D'Aversa <ninodaversa@gmail.com>
|
2009-03-21 04:49:51 -07:00
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
|
|
|
|
#include "windows.h"
|
|
|
|
#include "wlanapi.h"
|
|
|
|
|
|
|
|
#include <ntddndis.h>
|
|
|
|
#include "winioctl.h"
|
|
|
|
#include "stdlib.h"
|
|
|
|
|
|
|
|
#include "nsAutoPtr.h"
|
|
|
|
#include "nsWifiMonitor.h"
|
|
|
|
#include "nsWifiAccessPoint.h"
|
|
|
|
|
|
|
|
#include "nsIProxyObjectManager.h"
|
|
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
#include "nsComponentManagerUtils.h"
|
|
|
|
#include "nsIMutableArray.h"
|
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
#include <Iphlpapi.h> // For GetAdaptersInfo()
|
|
|
|
#include <nuiouser.h> // For NDISUIO stuff
|
|
|
|
// GetAdaptersInfo
|
|
|
|
typedef DWORD (*GETADAPTERSINFO)(PIP_ADAPTER_INFO pAdapterInfo, PULONG pOutBufLen);
|
|
|
|
#endif
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
// Taken from ndis.h
|
|
|
|
#define NDIS_STATUS_INVALID_LENGTH ((NDIS_STATUS)0xC0010014L)
|
|
|
|
#define NDIS_STATUS_BUFFER_TOO_SHORT ((NDIS_STATUS)0xC0010016L)
|
|
|
|
|
|
|
|
static const int kStringLength = 512;
|
|
|
|
|
|
|
|
// The limits on the size of the buffer used for the OID query.
|
|
|
|
static const int kInitialBufferSize = 2 << 12; // Good for about 50 APs.
|
|
|
|
static const int kMaximumBufferSize = 2 << 20; // 2MB
|
|
|
|
|
|
|
|
static int oid_buffer_size_ = kStringLength;
|
|
|
|
|
|
|
|
PRBool ResizeBuffer(int requested_size, BYTE **buffer)
|
|
|
|
{
|
|
|
|
if (requested_size > kMaximumBufferSize) {
|
|
|
|
free(*buffer);
|
|
|
|
*buffer = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
BYTE *new_buffer = (BYTE*)realloc(*buffer, requested_size);
|
2009-03-21 04:49:51 -07:00
|
|
|
if (new_buffer == NULL) {
|
|
|
|
free(*buffer);
|
|
|
|
*buffer = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*buffer = new_buffer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
int PerformQuery(HANDLE &ndis_handle,
|
2009-08-05 11:40:42 -07:00
|
|
|
const TCHAR *device_name,
|
2009-04-21 07:30:25 -07:00
|
|
|
BYTE *buffer,
|
|
|
|
DWORD buffer_size,
|
|
|
|
BYTE *&data,
|
|
|
|
DWORD *bytes_out) {
|
2009-09-08 13:40:20 -07:00
|
|
|
|
|
|
|
NS_ASSERTION(buffer, "Buffer is null. OOM?");
|
|
|
|
|
|
|
|
if (!buffer)
|
|
|
|
return ERROR_INSUFFICIENT_BUFFER;
|
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
// Form the query parameters.
|
|
|
|
NDISUIO_QUERY_OID *query = (NDISUIO_QUERY_OID*)(buffer);
|
2009-08-05 11:40:42 -07:00
|
|
|
query->ptcDeviceName = const_cast<PTCHAR>(device_name);
|
2009-04-21 07:30:25 -07:00
|
|
|
query->Oid = OID_802_11_BSSID_LIST;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
if (!DeviceIoControl(ndis_handle,
|
|
|
|
IOCTL_NDISUIO_QUERY_OID_VALUE,
|
|
|
|
query,
|
|
|
|
sizeof(NDISUIO_QUERY_OID),
|
|
|
|
query,
|
|
|
|
buffer_size,
|
|
|
|
bytes_out,
|
|
|
|
NULL)) {
|
|
|
|
return GetLastError();
|
|
|
|
}
|
|
|
|
// The start of the NDIS_802_11_BSSID_LIST is at Data[0]
|
|
|
|
data = &query->Data[0];
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void GetNetworkInterfaces(GETADAPTERSINFO pGetAdaptersInfo, nsStringArray& interfaces){
|
|
|
|
// Get the list of adapters. First determine the buffer size.
|
|
|
|
ULONG buffer_size = 0;
|
|
|
|
// since buffer_size is zero before this, we should get ERROR_BUFFER_OVERFLOW
|
|
|
|
// after this error the value of buffer_size will reflect the size needed
|
2010-03-01 21:56:06 -08:00
|
|
|
if (pGetAdaptersInfo(NULL, &buffer_size) != ERROR_BUFFER_OVERFLOW)
|
2009-04-21 07:30:25 -07:00
|
|
|
return;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
// Allocate adapter_info with correct size.
|
|
|
|
IP_ADAPTER_INFO *adapter_info = (IP_ADAPTER_INFO*)malloc(buffer_size);
|
|
|
|
if (adapter_info == NULL){
|
|
|
|
free (adapter_info);
|
|
|
|
return;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
if (pGetAdaptersInfo(adapter_info, &buffer_size) != ERROR_SUCCESS){
|
|
|
|
free (adapter_info);
|
|
|
|
return;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
// Walk through the list of adapters.
|
2010-03-01 21:56:06 -08:00
|
|
|
while (adapter_info) {
|
2009-04-21 07:30:25 -07:00
|
|
|
// AdapterName e.g. TNETW12511
|
|
|
|
nsString adapterName;
|
|
|
|
// AdapterName is in ASCII
|
2010-03-01 21:56:06 -08:00
|
|
|
adapterName.AppendWithConversion(adapter_info->AdapterName);
|
2009-04-21 07:30:25 -07:00
|
|
|
interfaces.AppendString(adapterName);
|
|
|
|
adapter_info = adapter_info->Next;
|
|
|
|
}
|
|
|
|
free (adapter_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult SetupWince(HANDLE& ndis_handle, GETADAPTERSINFO& pGetAdaptersInfo){
|
|
|
|
// Get the Network Driver Interface Specification (NDIS) handle for wireless
|
|
|
|
// devices. NDIS defines a standard API for Network Interface Cards (NICs).
|
|
|
|
// NDIS User Mode I/O (NDISUIO) is an NDIS protocol driver which offers
|
|
|
|
// support for wireless devices.
|
|
|
|
|
|
|
|
ndis_handle = CreateFile(NDISUIO_DEVICE_NAME, GENERIC_READ,
|
|
|
|
FILE_SHARE_READ, NULL, OPEN_EXISTING,
|
|
|
|
FILE_ATTRIBUTE_READONLY, NULL);
|
|
|
|
if (INVALID_HANDLE_VALUE == ndis_handle)
|
|
|
|
return NS_ERROR_FAILURE;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
HINSTANCE hIpDLL = LoadLibraryW(L"Iphlpapi.dll");
|
|
|
|
if (!hIpDLL)
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
pGetAdaptersInfo = (GETADAPTERSINFO) GetProcAddress(hIpDLL, "GetAdaptersInfo");
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
if (!pGetAdaptersInfo)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2009-03-21 04:49:51 -07:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
#else
|
2009-03-21 04:49:51 -07:00
|
|
|
int PerformQuery(HANDLE adapter_handle,
|
|
|
|
BYTE *buffer,
|
|
|
|
DWORD buffer_size,
|
|
|
|
DWORD *bytes_out) {
|
|
|
|
|
|
|
|
DWORD oid = OID_802_11_BSSID_LIST;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (!DeviceIoControl(adapter_handle,
|
|
|
|
IOCTL_NDIS_QUERY_GLOBAL_STATS,
|
|
|
|
&oid,
|
|
|
|
sizeof(oid),
|
|
|
|
buffer,
|
|
|
|
buffer_size,
|
|
|
|
bytes_out,
|
|
|
|
NULL)) {
|
|
|
|
return GetLastError();
|
|
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2009-08-04 19:39:06 -07:00
|
|
|
HANDLE GetFileHandle(const PRUnichar* device_name) {
|
2009-03-21 04:49:51 -07:00
|
|
|
// We access a device with DOS path \Device\<device_name> at
|
2010-03-01 21:56:06 -08:00
|
|
|
// \\.\<device_name>.
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
nsString formatted_device_name;
|
|
|
|
formatted_device_name.Assign(L"\\\\.\\");
|
|
|
|
formatted_device_name.Append(device_name);
|
|
|
|
|
|
|
|
return CreateFileW(formatted_device_name.get(),
|
|
|
|
GENERIC_READ,
|
|
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE, // share mode
|
|
|
|
0, // security attributes
|
|
|
|
OPEN_EXISTING,
|
|
|
|
0, // flags and attributes
|
|
|
|
INVALID_HANDLE_VALUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-04 19:39:06 -07:00
|
|
|
bool UndefineDosDevice(const PRUnichar* device_name) {
|
2009-03-21 04:49:51 -07:00
|
|
|
// We remove only the mapping we use, that is \Device\<device_name>.
|
|
|
|
nsString target_path;
|
|
|
|
target_path.Assign(L"\\Device\\");
|
|
|
|
target_path.Append(device_name);
|
|
|
|
|
|
|
|
return DefineDosDeviceW(DDD_RAW_TARGET_PATH | DDD_REMOVE_DEFINITION | DDD_EXACT_MATCH_ON_REMOVE,
|
|
|
|
device_name,
|
|
|
|
target_path.get()) == TRUE;
|
|
|
|
}
|
|
|
|
|
2009-08-04 19:39:06 -07:00
|
|
|
bool DefineDosDeviceIfNotExists(const PRUnichar* device_name, bool* dosDeviceDefined) {
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
// We create a DOS device name for the device at \Device\<device_name>.
|
|
|
|
nsString target_path;
|
|
|
|
target_path.Assign(L"\\Device\\");
|
|
|
|
target_path.Append(device_name);
|
|
|
|
|
2009-08-04 19:39:06 -07:00
|
|
|
WCHAR target[kStringLength];
|
2009-03-21 04:49:51 -07:00
|
|
|
|
2010-03-01 21:56:06 -08:00
|
|
|
if (QueryDosDeviceW(device_name, target, kStringLength) > 0 && target_path.Equals(target)) {
|
2009-03-21 04:49:51 -07:00
|
|
|
// Device already exists.
|
|
|
|
return true;
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
DWORD error = GetLastError();
|
|
|
|
if (error != ERROR_FILE_NOT_FOUND) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!DefineDosDeviceW(DDD_RAW_TARGET_PATH,
|
|
|
|
device_name,
|
|
|
|
target_path.get())) {
|
|
|
|
return false;
|
|
|
|
}
|
2009-05-18 15:58:23 -07:00
|
|
|
*dosDeviceDefined = true;
|
2010-03-01 21:56:06 -08:00
|
|
|
// Check that the device is really there.
|
2009-03-21 04:49:51 -07:00
|
|
|
return QueryDosDeviceW(device_name, target, kStringLength) > 0 &&
|
|
|
|
target_path.Equals(target);
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
2009-03-21 04:49:51 -07:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
void GetNetworkInterfaces(nsStringArray& interfaces)
|
2009-03-21 04:49:51 -07:00
|
|
|
{
|
|
|
|
HKEY network_cards_key = NULL;
|
|
|
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
L"Software\\Microsoft\\Windows NT\\CurrentVersion\\NetworkCards",
|
|
|
|
0,
|
|
|
|
KEY_READ,
|
|
|
|
&network_cards_key) != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
return;
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (int i = 0; ; ++i) {
|
2009-08-04 19:39:06 -07:00
|
|
|
WCHAR name[kStringLength];
|
2009-03-21 04:49:51 -07:00
|
|
|
DWORD name_size = kStringLength;
|
|
|
|
FILETIME time;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (RegEnumKeyExW(network_cards_key,
|
|
|
|
i,
|
|
|
|
name,
|
|
|
|
&name_size,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
2010-03-01 21:56:06 -08:00
|
|
|
&time) != ERROR_SUCCESS)
|
2009-03-21 04:49:51 -07:00
|
|
|
{
|
|
|
|
break;
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
HKEY hardware_key = NULL;
|
|
|
|
if (RegOpenKeyExW(network_cards_key,
|
|
|
|
name,
|
|
|
|
0,
|
|
|
|
KEY_READ,
|
|
|
|
&hardware_key) != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-08-04 19:39:06 -07:00
|
|
|
PRUnichar service_name[kStringLength];
|
2009-03-21 04:49:51 -07:00
|
|
|
DWORD service_name_size = kStringLength;
|
|
|
|
DWORD type = 0;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (RegQueryValueExW(hardware_key,
|
|
|
|
L"ServiceName",
|
|
|
|
NULL,
|
|
|
|
&type,
|
2009-04-21 07:30:25 -07:00
|
|
|
(LPBYTE)service_name,
|
2009-03-21 04:49:51 -07:00
|
|
|
&service_name_size) == ERROR_SUCCESS) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
interfaces.AppendString(nsString(service_name)); // is this allowed?
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
2009-03-21 04:49:51 -07:00
|
|
|
RegCloseKey(hardware_key);
|
|
|
|
}
|
|
|
|
RegCloseKey(network_cards_key);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
PRBool IsRunningOnVista() {
|
|
|
|
static DWORD os_major_version = 0;
|
|
|
|
static DWORD platform_id = 0;
|
|
|
|
|
|
|
|
if (0 == os_major_version) {
|
|
|
|
OSVERSIONINFO os_version = {0};
|
|
|
|
os_version.dwOSVersionInfoSize = sizeof(os_version);
|
|
|
|
GetVersionEx(&os_version);
|
|
|
|
os_major_version = os_version.dwMajorVersion;
|
|
|
|
platform_id = os_version.dwPlatformId;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (6 == os_major_version) && (VER_PLATFORM_WIN32_NT == platform_id);
|
|
|
|
}
|
2009-04-21 07:30:25 -07:00
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsWifiMonitor::DoScan()
|
|
|
|
{
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifndef WINCE
|
2009-03-21 04:49:51 -07:00
|
|
|
if (!IsRunningOnVista()) {
|
2009-04-21 07:30:25 -07:00
|
|
|
#else
|
|
|
|
HANDLE ndis_handle;
|
|
|
|
GETADAPTERSINFO pGetAdaptersInfo;
|
|
|
|
nsresult rc = SetupWince(ndis_handle, pGetAdaptersInfo);
|
|
|
|
if (rc != NS_OK)
|
|
|
|
return rc;
|
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
nsCOMArray<nsWifiAccessPoint> lastAccessPoints;
|
|
|
|
nsCOMArray<nsWifiAccessPoint> accessPoints;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
do {
|
|
|
|
accessPoints.Clear();
|
|
|
|
|
|
|
|
nsStringArray interfaces;
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
GetNetworkInterfaces(pGetAdaptersInfo, interfaces);
|
|
|
|
#else
|
|
|
|
GetNetworkInterfaces(interfaces);
|
|
|
|
#endif
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (int i = 0; i < interfaces.Count(); i++) {
|
|
|
|
nsString *s = interfaces.StringAt(i);
|
2009-08-04 19:39:06 -07:00
|
|
|
const PRUnichar *service_name = s->get();
|
2010-03-01 21:56:06 -08:00
|
|
|
|
|
|
|
#ifndef WINCE
|
2009-05-18 15:58:23 -07:00
|
|
|
bool dosDeviceDefined = false;
|
|
|
|
if (!DefineDosDeviceIfNotExists(service_name, &dosDeviceDefined))
|
2009-03-21 04:49:51 -07:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// Get the handle to the device. This will fail if the named device is not
|
|
|
|
// valid.
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
HANDLE adapter_handle = GetFileHandle(service_name);
|
|
|
|
if (adapter_handle == INVALID_HANDLE_VALUE)
|
|
|
|
continue;
|
2009-04-21 07:30:25 -07:00
|
|
|
#else
|
|
|
|
BYTE *data; // will store address of NDIS_802_11_BSSID_LIST data
|
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
// Get the data.
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
BYTE *buffer = (BYTE*)malloc(oid_buffer_size_);
|
2009-08-12 02:49:53 -07:00
|
|
|
if (buffer == NULL) {
|
|
|
|
#ifdef WINCE
|
|
|
|
CloseHandle(ndis_handle);
|
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
2009-08-12 02:49:53 -07:00
|
|
|
}
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
DWORD bytes_out;
|
|
|
|
int result;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
|
|
|
while (true) {
|
2009-09-08 13:40:20 -07:00
|
|
|
|
2009-09-08 14:37:55 -07:00
|
|
|
NS_ASSERTION(buffer && oid_buffer_size_ > 0, "buffer must not be null, and the size must be larger than 0");
|
2009-09-08 13:40:20 -07:00
|
|
|
|
2010-03-01 21:56:06 -08:00
|
|
|
bytes_out = 0;
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
result = PerformQuery(ndis_handle, service_name, buffer, oid_buffer_size_, data, &bytes_out);
|
|
|
|
#else
|
2009-03-21 04:49:51 -07:00
|
|
|
result = PerformQuery(adapter_handle, buffer, oid_buffer_size_, &bytes_out);
|
2009-04-21 07:30:25 -07:00
|
|
|
#endif
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (result == ERROR_GEN_FAILURE || // Returned by some Intel cards.
|
2009-09-25 11:11:07 -07:00
|
|
|
result == ERROR_INVALID_USER_BUFFER || // Returned on the Samsung Omnia II.
|
2009-03-21 04:49:51 -07:00
|
|
|
result == ERROR_INSUFFICIENT_BUFFER ||
|
|
|
|
result == ERROR_MORE_DATA ||
|
|
|
|
result == NDIS_STATUS_INVALID_LENGTH ||
|
|
|
|
result == NDIS_STATUS_BUFFER_TOO_SHORT) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
|
|
|
// The buffer we supplied is too small, so increase it. bytes_out should
|
|
|
|
// provide the required buffer size, but this is not always the case.
|
|
|
|
|
|
|
|
if (bytes_out > static_cast<DWORD>(oid_buffer_size_)) {
|
2009-03-21 04:49:51 -07:00
|
|
|
oid_buffer_size_ = bytes_out;
|
|
|
|
} else {
|
|
|
|
oid_buffer_size_ *= 2;
|
2010-03-01 21:56:06 -08:00
|
|
|
}
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (!ResizeBuffer(oid_buffer_size_, &buffer)) {
|
2010-03-01 21:56:06 -08:00
|
|
|
oid_buffer_size_ = kInitialBufferSize; // Reset for next time.
|
2009-03-21 04:49:51 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// The buffer is not too small.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (result == ERROR_SUCCESS) {
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
NDIS_802_11_BSSID_LIST* bssid_list = (NDIS_802_11_BSSID_LIST*)data;
|
|
|
|
#else
|
|
|
|
NDIS_802_11_BSSID_LIST* bssid_list = (NDIS_802_11_BSSID_LIST*)buffer;
|
|
|
|
#endif
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Walk through the BSS IDs.
|
2009-04-21 07:30:25 -07:00
|
|
|
const uint8 *iterator = (const uint8*)&bssid_list->Bssid[0];
|
|
|
|
const uint8 *end_of_buffer = (const uint8*)buffer + oid_buffer_size_;
|
2009-03-21 04:49:51 -07:00
|
|
|
for (int i = 0; i < static_cast<int>(bssid_list->NumberOfItems); ++i) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
const NDIS_WLAN_BSSID *bss_id = (const NDIS_WLAN_BSSID*)iterator;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Check that the length of this BSS ID is reasonable.
|
|
|
|
if (bss_id->Length < sizeof(NDIS_WLAN_BSSID) ||
|
|
|
|
iterator + bss_id->Length > end_of_buffer) {
|
|
|
|
break;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsWifiAccessPoint* ap = new nsWifiAccessPoint();
|
|
|
|
if (!ap)
|
|
|
|
continue;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
ap->setMac(bss_id->MacAddress);
|
|
|
|
ap->setSignal(bss_id->Rssi);
|
|
|
|
ap->setSSID((char*) bss_id->Ssid.Ssid, bss_id->Ssid.SsidLength);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
accessPoints.AppendObject(ap);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Move to the next BSS ID.
|
|
|
|
iterator += bss_id->Length;
|
|
|
|
}
|
|
|
|
|
2010-03-01 21:56:06 -08:00
|
|
|
free(buffer);
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
// Clean up.
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifndef WINCE
|
2009-03-21 04:49:51 -07:00
|
|
|
CloseHandle(adapter_handle);
|
2009-04-21 07:30:25 -07:00
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifndef WINCE
|
2009-05-18 15:58:23 -07:00
|
|
|
if (dosDeviceDefined)
|
|
|
|
UndefineDosDevice(service_name);
|
2009-04-21 07:30:25 -07:00
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
PRBool accessPointsChanged = !AccessPointsEqual(accessPoints, lastAccessPoints);
|
|
|
|
nsCOMArray<nsIWifiListener> currentListeners;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
{
|
|
|
|
nsAutoMonitor mon(mMonitor);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRUint32 i = 0; i < mListeners.Length(); i++) {
|
|
|
|
if (!mListeners[i].mHasSentData || accessPointsChanged) {
|
|
|
|
mListeners[i].mHasSentData = PR_TRUE;
|
|
|
|
currentListeners.AppendObject(mListeners[i].mListener);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
ReplaceArray(lastAccessPoints, accessPoints);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (currentListeners.Count() > 0) {
|
|
|
|
|
|
|
|
PRUint32 resultCount = lastAccessPoints.Count();
|
|
|
|
nsIWifiAccessPoint** result = static_cast<nsIWifiAccessPoint**> (nsMemory::Alloc(sizeof(nsIWifiAccessPoint*) * resultCount));
|
2009-08-12 02:49:53 -07:00
|
|
|
if (!result) {
|
|
|
|
#ifdef WINCE
|
|
|
|
CloseHandle(ndis_handle);
|
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
2009-08-12 02:49:53 -07:00
|
|
|
}
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRUint32 i = 0; i < resultCount; i++)
|
|
|
|
result[i] = lastAccessPoints[i];
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRInt32 i = 0; i < currentListeners.Count(); i++) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
LOG(("About to send data to the wifi listeners\n"));
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsCOMPtr<nsIWifiListener> proxy;
|
|
|
|
nsCOMPtr<nsIProxyObjectManager> proxyObjMgr = do_GetService("@mozilla.org/xpcomproxy;1");
|
|
|
|
proxyObjMgr->GetProxyForObject(NS_PROXY_TO_MAIN_THREAD,
|
|
|
|
NS_GET_IID(nsIWifiListener),
|
|
|
|
currentListeners[i],
|
|
|
|
NS_PROXY_SYNC | NS_PROXY_ALWAYS,
|
|
|
|
getter_AddRefs(proxy));
|
|
|
|
if (!proxy) {
|
|
|
|
LOG(("There is no proxy available. this should never happen\n"));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nsresult rv = proxy->OnChange(result, resultCount);
|
|
|
|
LOG( ("... sent %d\n", rv));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsMemory::Free(result);
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// wait for some reasonable amount of time. pref?
|
|
|
|
LOG(("waiting on monitor\n"));
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsAutoMonitor mon(mMonitor);
|
|
|
|
mon.Wait(PR_SecondsToInterval(60));
|
|
|
|
}
|
2010-02-07 07:52:43 -08:00
|
|
|
while (mKeepGoing);
|
2009-03-21 04:49:51 -07:00
|
|
|
|
|
|
|
|
|
|
|
|
2009-04-21 07:30:25 -07:00
|
|
|
#ifdef WINCE
|
|
|
|
//Clean up
|
|
|
|
CloseHandle(ndis_handle);
|
|
|
|
#else
|
2009-03-21 04:49:51 -07:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
|
|
|
|
HINSTANCE wlan_library = LoadLibrary("Wlanapi.dll");
|
|
|
|
if (!wlan_library)
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
WlanOpenHandleFunction WlanOpenHandle = (WlanOpenHandleFunction) GetProcAddress(wlan_library, "WlanOpenHandle");
|
|
|
|
WlanEnumInterfacesFunction WlanEnumInterfaces = (WlanEnumInterfacesFunction) GetProcAddress(wlan_library, "WlanEnumInterfaces");
|
|
|
|
WlanGetNetworkBssListFunction WlanGetNetworkBssList = (WlanGetNetworkBssListFunction) GetProcAddress(wlan_library, "WlanGetNetworkBssList");
|
|
|
|
WlanFreeMemoryFunction WlanFreeMemory = (WlanFreeMemoryFunction) GetProcAddress(wlan_library, "WlanFreeMemory");
|
|
|
|
WlanCloseHandleFunction WlanCloseHandle = (WlanCloseHandleFunction) GetProcAddress(wlan_library, "WlanCloseHandle");
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (!WlanOpenHandle ||
|
|
|
|
!WlanEnumInterfaces ||
|
|
|
|
!WlanGetNetworkBssList ||
|
|
|
|
!WlanFreeMemory ||
|
|
|
|
!WlanCloseHandle)
|
|
|
|
return NS_ERROR_FAILURE;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Regularly get the access point data.
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsCOMArray<nsWifiAccessPoint> lastAccessPoints;
|
|
|
|
nsCOMArray<nsWifiAccessPoint> accessPoints;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
do {
|
|
|
|
accessPoints.Clear();
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Get the handle to the WLAN API.
|
|
|
|
DWORD negotiated_version;
|
|
|
|
HANDLE wlan_handle = NULL;
|
|
|
|
// We could be executing on either Windows XP or Windows Vista, so use the
|
|
|
|
// lower version of the client WLAN API. It seems that the negotiated version
|
|
|
|
// is the Vista version irrespective of what we pass!
|
|
|
|
static const int kXpWlanClientVersion = 1;
|
|
|
|
if ((*WlanOpenHandle)(kXpWlanClientVersion,
|
|
|
|
NULL,
|
|
|
|
&negotiated_version,
|
|
|
|
&wlan_handle) != ERROR_SUCCESS) {
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// try again later.
|
|
|
|
if (!wlan_handle)
|
|
|
|
return NS_ERROR_FAILURE;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Get the list of interfaces. WlanEnumInterfaces allocates interface_list.
|
|
|
|
WLAN_INTERFACE_INFO_LIST *interface_list = NULL;
|
|
|
|
if ((*WlanEnumInterfaces)(wlan_handle, NULL, &interface_list) != ERROR_SUCCESS) {
|
|
|
|
// try again later
|
|
|
|
(*WlanCloseHandle)(wlan_handle, NULL);
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Go through the list of interfaces and get the data for each.
|
|
|
|
for (int i = 0; i < static_cast<int>(interface_list->dwNumberOfItems); ++i) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
WLAN_BSS_LIST *bss_list;
|
|
|
|
HRESULT rv = (*WlanGetNetworkBssList)(wlan_handle,
|
|
|
|
&interface_list->InterfaceInfo[i].InterfaceGuid,
|
|
|
|
NULL, // Use all SSIDs.
|
|
|
|
DOT11_BSS_TYPE_UNUSED,
|
|
|
|
false, // bSecurityEnabled - unused
|
|
|
|
NULL, // reserved
|
|
|
|
&bss_list);
|
|
|
|
if (rv != ERROR_SUCCESS) {
|
|
|
|
continue;
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (int j = 0; j < static_cast<int>(bss_list->dwNumberOfItems); ++j) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsWifiAccessPoint* ap = new nsWifiAccessPoint();
|
|
|
|
if (!ap)
|
|
|
|
continue;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
const WLAN_BSS_ENTRY bss_entry = bss_list->wlanBssEntries[j];
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
ap->setMac(bss_entry.dot11Bssid);
|
|
|
|
ap->setSignal(bss_entry.lRssi);
|
|
|
|
ap->setSSID((char*) bss_entry.dot11Ssid.ucSSID,
|
|
|
|
bss_entry.dot11Ssid.uSSIDLength);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
accessPoints.AppendObject(ap);
|
|
|
|
}
|
|
|
|
(*WlanFreeMemory)(bss_list);
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Free interface_list.
|
|
|
|
(*WlanFreeMemory)(interface_list);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// Close the handle.
|
|
|
|
(*WlanCloseHandle)(wlan_handle, NULL);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
PRBool accessPointsChanged = !AccessPointsEqual(accessPoints, lastAccessPoints);
|
|
|
|
nsCOMArray<nsIWifiListener> currentListeners;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
{
|
|
|
|
nsAutoMonitor mon(mMonitor);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRUint32 i = 0; i < mListeners.Length(); i++) {
|
|
|
|
if (!mListeners[i].mHasSentData || accessPointsChanged) {
|
|
|
|
mListeners[i].mHasSentData = PR_TRUE;
|
|
|
|
currentListeners.AppendObject(mListeners[i].mListener);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
ReplaceArray(lastAccessPoints, accessPoints);
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
if (currentListeners.Count() > 0) {
|
|
|
|
PRUint32 resultCount = lastAccessPoints.Count();
|
|
|
|
nsIWifiAccessPoint** result = static_cast<nsIWifiAccessPoint**> (nsMemory::Alloc(sizeof(nsIWifiAccessPoint*) * resultCount));
|
|
|
|
if (!result)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRUint32 i = 0; i < resultCount; i++)
|
|
|
|
result[i] = lastAccessPoints[i];
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
for (PRInt32 i = 0; i < currentListeners.Count(); i++) {
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
LOG(("About to send data to the wifi listeners\n"));
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsCOMPtr<nsIWifiListener> proxy;
|
|
|
|
nsCOMPtr<nsIProxyObjectManager> proxyObjMgr = do_GetService("@mozilla.org/xpcomproxy;1");
|
|
|
|
proxyObjMgr->GetProxyForObject(NS_PROXY_TO_MAIN_THREAD,
|
|
|
|
NS_GET_IID(nsIWifiListener),
|
|
|
|
currentListeners[i],
|
|
|
|
NS_PROXY_SYNC | NS_PROXY_ALWAYS,
|
|
|
|
getter_AddRefs(proxy));
|
|
|
|
if (!proxy) {
|
|
|
|
LOG(("There is no proxy available. this should never happen\n"));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nsresult rv = proxy->OnChange(result, resultCount);
|
|
|
|
LOG( ("... sent %d\n", rv));
|
|
|
|
}
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsMemory::Free(result);
|
|
|
|
}
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
// wait for some reasonable amount of time. pref?
|
|
|
|
LOG(("waiting on monitor\n"));
|
2010-03-01 21:56:06 -08:00
|
|
|
|
2009-03-21 04:49:51 -07:00
|
|
|
nsAutoMonitor mon(mMonitor);
|
|
|
|
mon.Wait(PR_SecondsToInterval(60));
|
|
|
|
}
|
2010-02-07 07:52:43 -08:00
|
|
|
while (mKeepGoing);
|
2009-03-21 04:49:51 -07:00
|
|
|
}
|
2009-04-21 07:30:25 -07:00
|
|
|
#endif
|
2009-03-21 04:49:51 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|