зеркало из https://github.com/mozilla/pjs.git
Bug 482613 - Support for WiFi Scanning on Windows Mobile r=dougt, sr=jst
This commit is contained in:
Родитель
2602acc8f6
Коммит
510f46a8ce
|
@ -7886,6 +7886,8 @@ if test "$NECKO_WIFI"; then
|
||||||
NECKO_WIFI=1
|
NECKO_WIFI=1
|
||||||
elif test "$OS_ARCH" = "WINNT"; then
|
elif test "$OS_ARCH" = "WINNT"; then
|
||||||
NECKO_WIFI=1
|
NECKO_WIFI=1
|
||||||
|
elif test "$OS_ARCH" = "WINCE"; then
|
||||||
|
NECKO_WIFI=1
|
||||||
else
|
else
|
||||||
AC_MSG_WARN([Necko WiFi scanning not supported on your platform.])
|
AC_MSG_WARN([Necko WiFi scanning not supported on your platform.])
|
||||||
fi
|
fi
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
*
|
*
|
||||||
* Contributor(s):
|
* Contributor(s):
|
||||||
* Doug Turner <dougt@meer.net> (Original Author)
|
* Doug Turner <dougt@meer.net> (Original Author)
|
||||||
|
* Nino D'Aversa <ninodaversa@gmail.com>
|
||||||
*
|
*
|
||||||
* Alternatively, the contents of this file may be used under the terms of
|
* 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
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||||||
|
@ -54,6 +55,13 @@
|
||||||
#include "nsComponentManagerUtils.h"
|
#include "nsComponentManagerUtils.h"
|
||||||
#include "nsIMutableArray.h"
|
#include "nsIMutableArray.h"
|
||||||
|
|
||||||
|
#ifdef WINCE
|
||||||
|
#include <Iphlpapi.h> // For GetAdaptersInfo()
|
||||||
|
#include <nuiouser.h> // For NDISUIO stuff
|
||||||
|
// GetAdaptersInfo
|
||||||
|
typedef DWORD (*GETADAPTERSINFO)(PIP_ADAPTER_INFO pAdapterInfo, PULONG pOutBufLen);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
// Taken from ndis.h
|
// Taken from ndis.h
|
||||||
#define NDIS_STATUS_INVALID_LENGTH ((NDIS_STATUS)0xC0010014L)
|
#define NDIS_STATUS_INVALID_LENGTH ((NDIS_STATUS)0xC0010014L)
|
||||||
|
@ -75,7 +83,7 @@ PRBool ResizeBuffer(int requested_size, BYTE **buffer)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
BYTE *new_buffer = reinterpret_cast<BYTE*>(realloc(*buffer, requested_size));
|
BYTE *new_buffer = (BYTE*)realloc(*buffer, requested_size);
|
||||||
if (new_buffer == NULL) {
|
if (new_buffer == NULL) {
|
||||||
free(*buffer);
|
free(*buffer);
|
||||||
*buffer = NULL;
|
*buffer = NULL;
|
||||||
|
@ -87,7 +95,92 @@ PRBool ResizeBuffer(int requested_size, BYTE **buffer)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef WINCE
|
||||||
|
int PerformQuery(HANDLE &ndis_handle,
|
||||||
|
TCHAR *device_name,
|
||||||
|
BYTE *buffer,
|
||||||
|
DWORD buffer_size,
|
||||||
|
BYTE *&data,
|
||||||
|
DWORD *bytes_out) {
|
||||||
|
// Form the query parameters.
|
||||||
|
NDISUIO_QUERY_OID *query = (NDISUIO_QUERY_OID*)(buffer);
|
||||||
|
query->ptcDeviceName = device_name;
|
||||||
|
query->Oid = OID_802_11_BSSID_LIST;
|
||||||
|
|
||||||
|
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
|
||||||
|
if (pGetAdaptersInfo(NULL, &buffer_size) != ERROR_BUFFER_OVERFLOW)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pGetAdaptersInfo(adapter_info, &buffer_size) != ERROR_SUCCESS){
|
||||||
|
free (adapter_info);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Walk through the list of adapters.
|
||||||
|
while (adapter_info) {
|
||||||
|
// AdapterName e.g. TNETW12511
|
||||||
|
nsString adapterName;
|
||||||
|
// AdapterName is in ASCII
|
||||||
|
adapterName.AppendWithConversion(adapter_info->AdapterName);
|
||||||
|
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;
|
||||||
|
|
||||||
|
HINSTANCE hIpDLL = LoadLibraryW(L"Iphlpapi.dll");
|
||||||
|
if (!hIpDLL)
|
||||||
|
return NS_ERROR_NOT_AVAILABLE;
|
||||||
|
|
||||||
|
pGetAdaptersInfo = (GETADAPTERSINFO) GetProcAddress(hIpDLL, "GetAdaptersInfo");
|
||||||
|
|
||||||
|
if (!pGetAdaptersInfo)
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
int PerformQuery(HANDLE adapter_handle,
|
int PerformQuery(HANDLE adapter_handle,
|
||||||
BYTE *buffer,
|
BYTE *buffer,
|
||||||
DWORD buffer_size,
|
DWORD buffer_size,
|
||||||
|
@ -166,7 +259,7 @@ bool DefineDosDeviceIfNotExists(unsigned short* device_name) {
|
||||||
target_path.Equals(target);
|
target_path.Equals(target);
|
||||||
}
|
}
|
||||||
|
|
||||||
void GetNetworkInterfacesFromRegistry(nsStringArray& interfaces)
|
void GetNetworkInterfaces(nsStringArray& interfaces)
|
||||||
{
|
{
|
||||||
HKEY network_cards_key = NULL;
|
HKEY network_cards_key = NULL;
|
||||||
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
if (RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
||||||
|
@ -213,7 +306,7 @@ void GetNetworkInterfacesFromRegistry(nsStringArray& interfaces)
|
||||||
L"ServiceName",
|
L"ServiceName",
|
||||||
NULL,
|
NULL,
|
||||||
&type,
|
&type,
|
||||||
reinterpret_cast<LPBYTE>(service_name),
|
(LPBYTE)service_name,
|
||||||
&service_name_size) == ERROR_SUCCESS) {
|
&service_name_size) == ERROR_SUCCESS) {
|
||||||
|
|
||||||
interfaces.AppendString(nsString(service_name)); // is this allowed?
|
interfaces.AppendString(nsString(service_name)); // is this allowed?
|
||||||
|
@ -238,11 +331,20 @@ PRBool IsRunningOnVista() {
|
||||||
|
|
||||||
return (6 == os_major_version) && (VER_PLATFORM_WIN32_NT == platform_id);
|
return (6 == os_major_version) && (VER_PLATFORM_WIN32_NT == platform_id);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
nsresult
|
nsresult
|
||||||
nsWifiMonitor::DoScan()
|
nsWifiMonitor::DoScan()
|
||||||
{
|
{
|
||||||
|
#ifndef WINCE
|
||||||
if (!IsRunningOnVista()) {
|
if (!IsRunningOnVista()) {
|
||||||
|
#else
|
||||||
|
HANDLE ndis_handle;
|
||||||
|
GETADAPTERSINFO pGetAdaptersInfo;
|
||||||
|
nsresult rc = SetupWince(ndis_handle, pGetAdaptersInfo);
|
||||||
|
if (rc != NS_OK)
|
||||||
|
return rc;
|
||||||
|
#endif
|
||||||
|
|
||||||
nsCOMArray<nsWifiAccessPoint> lastAccessPoints;
|
nsCOMArray<nsWifiAccessPoint> lastAccessPoints;
|
||||||
nsCOMArray<nsWifiAccessPoint> accessPoints;
|
nsCOMArray<nsWifiAccessPoint> accessPoints;
|
||||||
|
@ -251,12 +353,17 @@ nsWifiMonitor::DoScan()
|
||||||
accessPoints.Clear();
|
accessPoints.Clear();
|
||||||
|
|
||||||
nsStringArray interfaces;
|
nsStringArray interfaces;
|
||||||
GetNetworkInterfacesFromRegistry(interfaces);
|
#ifdef WINCE
|
||||||
|
GetNetworkInterfaces(pGetAdaptersInfo, interfaces);
|
||||||
|
#else
|
||||||
|
GetNetworkInterfaces(interfaces);
|
||||||
|
#endif
|
||||||
|
|
||||||
for (int i = 0; i < interfaces.Count(); i++) {
|
for (int i = 0; i < interfaces.Count(); i++) {
|
||||||
nsString *s = interfaces.StringAt(i);
|
nsString *s = interfaces.StringAt(i);
|
||||||
unsigned short *service_name = (PRUnichar*) s->get();
|
unsigned short *service_name = (PRUnichar*) s->get();
|
||||||
|
|
||||||
|
#ifndef WINCE
|
||||||
if (!DefineDosDeviceIfNotExists(service_name))
|
if (!DefineDosDeviceIfNotExists(service_name))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -266,10 +373,13 @@ nsWifiMonitor::DoScan()
|
||||||
HANDLE adapter_handle = GetFileHandle(service_name);
|
HANDLE adapter_handle = GetFileHandle(service_name);
|
||||||
if (adapter_handle == INVALID_HANDLE_VALUE)
|
if (adapter_handle == INVALID_HANDLE_VALUE)
|
||||||
continue;
|
continue;
|
||||||
|
#else
|
||||||
|
BYTE *data; // will store address of NDIS_802_11_BSSID_LIST data
|
||||||
|
#endif
|
||||||
|
|
||||||
// Get the data.
|
// Get the data.
|
||||||
|
|
||||||
BYTE *buffer = reinterpret_cast<BYTE*>(malloc(oid_buffer_size_));
|
BYTE *buffer = (BYTE*)malloc(oid_buffer_size_);
|
||||||
if (buffer == NULL)
|
if (buffer == NULL)
|
||||||
return NS_ERROR_OUT_OF_MEMORY;
|
return NS_ERROR_OUT_OF_MEMORY;
|
||||||
|
|
||||||
|
@ -278,7 +388,11 @@ nsWifiMonitor::DoScan()
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
bytes_out = 0;
|
bytes_out = 0;
|
||||||
|
#ifdef WINCE
|
||||||
|
result = PerformQuery(ndis_handle, service_name, buffer, oid_buffer_size_, data, &bytes_out);
|
||||||
|
#else
|
||||||
result = PerformQuery(adapter_handle, buffer, oid_buffer_size_, &bytes_out);
|
result = PerformQuery(adapter_handle, buffer, oid_buffer_size_, &bytes_out);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (result == ERROR_GEN_FAILURE || // Returned by some Intel cards.
|
if (result == ERROR_GEN_FAILURE || // Returned by some Intel cards.
|
||||||
result == ERROR_INSUFFICIENT_BUFFER ||
|
result == ERROR_INSUFFICIENT_BUFFER ||
|
||||||
|
@ -306,15 +420,19 @@ nsWifiMonitor::DoScan()
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result == ERROR_SUCCESS) {
|
if (result == ERROR_SUCCESS) {
|
||||||
NDIS_802_11_BSSID_LIST* bssid_list = reinterpret_cast<NDIS_802_11_BSSID_LIST*>(buffer);
|
#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
|
||||||
|
|
||||||
// Walk through the BSS IDs.
|
// Walk through the BSS IDs.
|
||||||
int found = 0;
|
int found = 0;
|
||||||
const uint8 *iterator = reinterpret_cast<const uint8*>(&bssid_list->Bssid[0]);
|
const uint8 *iterator = (const uint8*)&bssid_list->Bssid[0];
|
||||||
const uint8 *end_of_buffer = reinterpret_cast<const uint8*>(&bssid_list) + oid_buffer_size_;
|
const uint8 *end_of_buffer = (const uint8*)buffer + oid_buffer_size_;
|
||||||
for (int i = 0; i < static_cast<int>(bssid_list->NumberOfItems); ++i) {
|
for (int i = 0; i < static_cast<int>(bssid_list->NumberOfItems); ++i) {
|
||||||
|
|
||||||
const NDIS_WLAN_BSSID *bss_id = reinterpret_cast<const NDIS_WLAN_BSSID*>(iterator);
|
const NDIS_WLAN_BSSID *bss_id = (const NDIS_WLAN_BSSID*)iterator;
|
||||||
|
|
||||||
// Check that the length of this BSS ID is reasonable.
|
// Check that the length of this BSS ID is reasonable.
|
||||||
if (bss_id->Length < sizeof(NDIS_WLAN_BSSID) ||
|
if (bss_id->Length < sizeof(NDIS_WLAN_BSSID) ||
|
||||||
|
@ -339,10 +457,14 @@ nsWifiMonitor::DoScan()
|
||||||
free(buffer);
|
free(buffer);
|
||||||
|
|
||||||
// Clean up.
|
// Clean up.
|
||||||
|
#ifndef WINCE
|
||||||
CloseHandle(adapter_handle);
|
CloseHandle(adapter_handle);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef WINCE
|
||||||
UndefineDosDevice(service_name);
|
UndefineDosDevice(service_name);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -407,6 +529,10 @@ nsWifiMonitor::DoScan()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef WINCE
|
||||||
|
//Clean up
|
||||||
|
CloseHandle(ndis_handle);
|
||||||
|
#else
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
|
@ -558,5 +684,6 @@ nsWifiMonitor::DoScan()
|
||||||
}
|
}
|
||||||
while (mKeepGoing == PR_TRUE);
|
while (mKeepGoing == PR_TRUE);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
return NS_OK;
|
return NS_OK;
|
||||||
}
|
}
|
||||||
|
|
Загрузка…
Ссылка в новой задаче