| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "net/base/net_util.h" | 5 #include "net/base/net_util.h" |
| 6 | 6 |
| 7 #include <iphlpapi.h> | 7 #include <iphlpapi.h> |
| 8 #include <wlanapi.h> | 8 #include <wlanapi.h> |
| 9 | 9 |
| 10 #include <algorithm> | 10 #include <algorithm> |
| 11 | 11 |
| 12 #include "base/files/file_path.h" | 12 #include "base/files/file_path.h" |
| 13 #include "base/lazy_instance.h" | 13 #include "base/lazy_instance.h" |
| 14 #include "base/memory/scoped_ptr.h" | 14 #include "base/memory/scoped_ptr.h" |
| 15 #include "base/strings/string_piece.h" | 15 #include "base/strings/string_piece.h" |
| 16 #include "base/strings/string_util.h" | 16 #include "base/strings/string_util.h" |
| 17 #include "base/strings/sys_string_conversions.h" | 17 #include "base/strings/sys_string_conversions.h" |
| 18 #include "base/strings/utf_string_conversions.h" | 18 #include "base/strings/utf_string_conversions.h" |
| 19 #include "base/threading/thread_restrictions.h" | 19 #include "base/threading/thread_restrictions.h" |
| 20 #include "base/win/scoped_handle.h" | 20 #include "base/win/scoped_handle.h" |
| 21 #include "base/win/windows_version.h" | 21 #include "base/win/windows_version.h" |
| 22 #include "net/base/escape.h" | 22 #include "net/base/escape.h" |
| 23 #include "net/base/ip_endpoint.h" | 23 #include "net/base/ip_endpoint.h" |
| 24 #include "net/base/net_errors.h" | 24 #include "net/base/net_errors.h" |
| 25 #include "net/base/net_util_win.h" |
| 25 #include "url/gurl.h" | 26 #include "url/gurl.h" |
| 26 | 27 |
| 27 namespace net { | 28 namespace net { |
| 28 | 29 |
| 29 namespace { | 30 namespace { |
| 30 | 31 |
| 31 struct WlanApi { | |
| 32 typedef DWORD (WINAPI *WlanOpenHandleFunc)( | |
| 33 DWORD, VOID*, DWORD*, HANDLE*); | |
| 34 typedef DWORD (WINAPI *WlanEnumInterfacesFunc)( | |
| 35 HANDLE, VOID*, WLAN_INTERFACE_INFO_LIST**); | |
| 36 typedef DWORD (WINAPI *WlanQueryInterfaceFunc)( | |
| 37 HANDLE, const GUID*, WLAN_INTF_OPCODE, VOID*, DWORD*, VOID**, | |
| 38 WLAN_OPCODE_VALUE_TYPE*); | |
| 39 typedef VOID (WINAPI *WlanFreeMemoryFunc)(VOID*); | |
| 40 typedef DWORD (WINAPI *WlanCloseHandleFunc)(HANDLE, VOID*); | |
| 41 | |
| 42 WlanApi() : initialized(false) { | |
| 43 // Use an absolute path to load the DLL to avoid DLL preloading attacks. | |
| 44 static const wchar_t* const kDLL = L"%WINDIR%\\system32\\wlanapi.dll"; | |
| 45 wchar_t path[MAX_PATH] = {0}; | |
| 46 ExpandEnvironmentStrings(kDLL, path, arraysize(path)); | |
| 47 module = ::LoadLibraryEx(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); | |
| 48 if (!module) | |
| 49 return; | |
| 50 | |
| 51 open_handle_func = reinterpret_cast<WlanOpenHandleFunc>( | |
| 52 ::GetProcAddress(module, "WlanOpenHandle")); | |
| 53 enum_interfaces_func = reinterpret_cast<WlanEnumInterfacesFunc>( | |
| 54 ::GetProcAddress(module, "WlanEnumInterfaces")); | |
| 55 query_interface_func = reinterpret_cast<WlanQueryInterfaceFunc>( | |
| 56 ::GetProcAddress(module, "WlanQueryInterface")); | |
| 57 free_memory_func = reinterpret_cast<WlanFreeMemoryFunc>( | |
| 58 ::GetProcAddress(module, "WlanFreeMemory")); | |
| 59 close_handle_func = reinterpret_cast<WlanCloseHandleFunc>( | |
| 60 ::GetProcAddress(module, "WlanCloseHandle")); | |
| 61 initialized = open_handle_func && enum_interfaces_func && | |
| 62 query_interface_func && free_memory_func && | |
| 63 close_handle_func; | |
| 64 } | |
| 65 | |
| 66 template <typename T> | |
| 67 DWORD OpenHandle(DWORD client_version, DWORD* cur_version, T* handle) const { | |
| 68 HANDLE temp_handle; | |
| 69 DWORD result = open_handle_func(client_version, NULL, cur_version, | |
| 70 &temp_handle); | |
| 71 if (result != ERROR_SUCCESS) | |
| 72 return result; | |
| 73 handle->Set(temp_handle); | |
| 74 return ERROR_SUCCESS; | |
| 75 } | |
| 76 | |
| 77 HMODULE module; | |
| 78 WlanOpenHandleFunc open_handle_func; | |
| 79 WlanEnumInterfacesFunc enum_interfaces_func; | |
| 80 WlanQueryInterfaceFunc query_interface_func; | |
| 81 WlanFreeMemoryFunc free_memory_func; | |
| 82 WlanCloseHandleFunc close_handle_func; | |
| 83 bool initialized; | |
| 84 }; | |
| 85 | |
| 86 // Converts Windows defined types to NetworkInterfaceType. | 32 // Converts Windows defined types to NetworkInterfaceType. |
| 87 NetworkChangeNotifier::ConnectionType GetNetworkInterfaceType(DWORD ifType) { | 33 NetworkChangeNotifier::ConnectionType GetNetworkInterfaceType(DWORD ifType) { |
| 88 // Bail out for pre-Vista versions of Windows which are documented to give | 34 // Bail out for pre-Vista versions of Windows which are documented to give |
| 89 // inaccurate results like returning Ethernet for WiFi. | 35 // inaccurate results like returning Ethernet for WiFi. |
| 90 // http://msdn.microsoft.com/en-us/library/windows/desktop/aa366058.aspx | 36 // http://msdn.microsoft.com/en-us/library/windows/desktop/aa366058.aspx |
| 91 if (base::win::GetVersion() < base::win::VERSION_VISTA) | 37 if (base::win::GetVersion() < base::win::VERSION_VISTA) |
| 92 return NetworkChangeNotifier::CONNECTION_UNKNOWN; | 38 return NetworkChangeNotifier::CONNECTION_UNKNOWN; |
| 93 | 39 |
| 94 NetworkChangeNotifier::ConnectionType type = | 40 NetworkChangeNotifier::ConnectionType type = |
| 95 NetworkChangeNotifier::CONNECTION_UNKNOWN; | 41 NetworkChangeNotifier::CONNECTION_UNKNOWN; |
| 96 if (ifType == IF_TYPE_ETHERNET_CSMACD) { | 42 if (ifType == IF_TYPE_ETHERNET_CSMACD) { |
| 97 type = NetworkChangeNotifier::CONNECTION_ETHERNET; | 43 type = NetworkChangeNotifier::CONNECTION_ETHERNET; |
| 98 } else if (ifType == IF_TYPE_IEEE80211) { | 44 } else if (ifType == IF_TYPE_IEEE80211) { |
| 99 type = NetworkChangeNotifier::CONNECTION_WIFI; | 45 type = NetworkChangeNotifier::CONNECTION_WIFI; |
| 100 } | 46 } |
| 101 // TODO(mallinath) - Cellular? | 47 // TODO(mallinath) - Cellular? |
| 102 return type; | 48 return type; |
| 103 } | 49 } |
| 104 | 50 |
| 105 } // namespace | 51 } // namespace |
| 106 | 52 |
| 53 namespace internal { |
| 54 |
| 55 base::LazyInstance<WlanApi>::Leaky lazy_wlanapi = |
| 56 LAZY_INSTANCE_INITIALIZER; |
| 57 |
| 58 WlanApi& WlanApi::GetInstance() { |
| 59 return lazy_wlanapi.Get(); |
| 60 } |
| 61 |
| 62 WlanApi::WlanApi() : initialized(false) { |
| 63 // Use an absolute path to load the DLL to avoid DLL preloading attacks. |
| 64 static const wchar_t* const kDLL = L"%WINDIR%\\system32\\wlanapi.dll"; |
| 65 wchar_t path[MAX_PATH] = {0}; |
| 66 ExpandEnvironmentStrings(kDLL, path, arraysize(path)); |
| 67 module = ::LoadLibraryEx(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH); |
| 68 if (!module) |
| 69 return; |
| 70 |
| 71 open_handle_func = reinterpret_cast<WlanOpenHandleFunc>( |
| 72 ::GetProcAddress(module, "WlanOpenHandle")); |
| 73 enum_interfaces_func = reinterpret_cast<WlanEnumInterfacesFunc>( |
| 74 ::GetProcAddress(module, "WlanEnumInterfaces")); |
| 75 query_interface_func = reinterpret_cast<WlanQueryInterfaceFunc>( |
| 76 ::GetProcAddress(module, "WlanQueryInterface")); |
| 77 set_interface_func = reinterpret_cast<WlanSetInterfaceFunc>( |
| 78 ::GetProcAddress(module, "WlanSetInterface")); |
| 79 free_memory_func = reinterpret_cast<WlanFreeMemoryFunc>( |
| 80 ::GetProcAddress(module, "WlanFreeMemory")); |
| 81 close_handle_func = reinterpret_cast<WlanCloseHandleFunc>( |
| 82 ::GetProcAddress(module, "WlanCloseHandle")); |
| 83 initialized = open_handle_func && enum_interfaces_func && |
| 84 query_interface_func && set_interface_func && |
| 85 free_memory_func && close_handle_func; |
| 86 } |
| 87 |
| 88 } // namespace internal |
| 89 |
| 107 bool GetNetworkList(NetworkInterfaceList* networks, int policy) { | 90 bool GetNetworkList(NetworkInterfaceList* networks, int policy) { |
| 108 // GetAdaptersAddresses() may require IO operations. | 91 // GetAdaptersAddresses() may require IO operations. |
| 109 base::ThreadRestrictions::AssertIOAllowed(); | 92 base::ThreadRestrictions::AssertIOAllowed(); |
| 110 bool is_xp = base::win::GetVersion() < base::win::VERSION_VISTA; | 93 bool is_xp = base::win::GetVersion() < base::win::VERSION_VISTA; |
| 111 ULONG len = 0; | 94 ULONG len = 0; |
| 112 ULONG flags = is_xp ? GAA_FLAG_INCLUDE_PREFIX : 0; | 95 ULONG flags = is_xp ? GAA_FLAG_INCLUDE_PREFIX : 0; |
| 113 // First get number of networks. | 96 // First get number of networks. |
| 114 ULONG result = GetAdaptersAddresses(AF_UNSPEC, flags, NULL, NULL, &len); | 97 ULONG result = GetAdaptersAddresses(AF_UNSPEC, flags, NULL, NULL, &len); |
| 115 if (result != ERROR_BUFFER_OVERFLOW) { | 98 if (result != ERROR_BUFFER_OVERFLOW) { |
| 116 // There are 0 networks. | 99 // There are 0 networks. |
| (...skipping 92 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 209 } | 192 } |
| 210 } | 193 } |
| 211 | 194 |
| 212 if (ipv6_address.get()) { | 195 if (ipv6_address.get()) { |
| 213 networks->push_back(*(ipv6_address.get())); | 196 networks->push_back(*(ipv6_address.get())); |
| 214 } | 197 } |
| 215 return true; | 198 return true; |
| 216 } | 199 } |
| 217 | 200 |
| 218 WifiPHYLayerProtocol GetWifiPHYLayerProtocol() { | 201 WifiPHYLayerProtocol GetWifiPHYLayerProtocol() { |
| 219 static base::LazyInstance<WlanApi>::Leaky lazy_wlanapi = | 202 const internal::WlanApi& wlanapi = internal::WlanApi::GetInstance(); |
| 220 LAZY_INSTANCE_INITIALIZER; | |
| 221 | |
| 222 struct WlanApiHandleTraits { | |
| 223 typedef HANDLE Handle; | |
| 224 | |
| 225 static bool CloseHandle(HANDLE handle) { | |
| 226 return lazy_wlanapi.Get().close_handle_func(handle, NULL) == | |
| 227 ERROR_SUCCESS; | |
| 228 } | |
| 229 static bool IsHandleValid(HANDLE handle) { | |
| 230 return base::win::HandleTraits::IsHandleValid(handle); | |
| 231 } | |
| 232 static HANDLE NullHandle() { | |
| 233 return base::win::HandleTraits::NullHandle(); | |
| 234 } | |
| 235 }; | |
| 236 | |
| 237 typedef base::win::GenericScopedHandle< | |
| 238 WlanApiHandleTraits, | |
| 239 base::win::DummyVerifierTraits> WlanHandle; | |
| 240 | |
| 241 struct WlanApiDeleter { | |
| 242 inline void operator()(void* ptr) const { | |
| 243 lazy_wlanapi.Get().free_memory_func(ptr); | |
| 244 } | |
| 245 }; | |
| 246 | |
| 247 const WlanApi& wlanapi = lazy_wlanapi.Get(); | |
| 248 if (!wlanapi.initialized) | 203 if (!wlanapi.initialized) |
| 249 return WIFI_PHY_LAYER_PROTOCOL_NONE; | 204 return WIFI_PHY_LAYER_PROTOCOL_NONE; |
| 250 | 205 |
| 251 WlanHandle client; | 206 internal::WlanHandle client; |
| 252 DWORD cur_version = 0; | 207 DWORD cur_version = 0; |
| 253 const DWORD kMaxClientVersion = 2; | 208 const DWORD kMaxClientVersion = 2; |
| 254 DWORD result = wlanapi.OpenHandle(kMaxClientVersion, &cur_version, &client); | 209 DWORD result = wlanapi.OpenHandle(kMaxClientVersion, &cur_version, &client); |
| 255 if (result != ERROR_SUCCESS) | 210 if (result != ERROR_SUCCESS) |
| 256 return WIFI_PHY_LAYER_PROTOCOL_NONE; | 211 return WIFI_PHY_LAYER_PROTOCOL_NONE; |
| 257 | 212 |
| 258 WLAN_INTERFACE_INFO_LIST* interface_list_ptr = NULL; | 213 WLAN_INTERFACE_INFO_LIST* interface_list_ptr = NULL; |
| 259 result = wlanapi.enum_interfaces_func(client, NULL, &interface_list_ptr); | 214 result = wlanapi.enum_interfaces_func(client, NULL, &interface_list_ptr); |
| 260 if (result != ERROR_SUCCESS) | 215 if (result != ERROR_SUCCESS) |
| 261 return WIFI_PHY_LAYER_PROTOCOL_NONE; | 216 return WIFI_PHY_LAYER_PROTOCOL_NONE; |
| 262 scoped_ptr<WLAN_INTERFACE_INFO_LIST, WlanApiDeleter> interface_list( | 217 scoped_ptr<WLAN_INTERFACE_INFO_LIST, internal::WlanApiDeleter> interface_list( |
| 263 interface_list_ptr); | 218 interface_list_ptr); |
| 264 | 219 |
| 265 // Assume at most one connected wifi interface. | 220 // Assume at most one connected wifi interface. |
| 266 WLAN_INTERFACE_INFO* info = NULL; | 221 WLAN_INTERFACE_INFO* info = NULL; |
| 267 for (unsigned i = 0; i < interface_list->dwNumberOfItems; ++i) { | 222 for (unsigned i = 0; i < interface_list->dwNumberOfItems; ++i) { |
| 268 if (interface_list->InterfaceInfo[i].isState == | 223 if (interface_list->InterfaceInfo[i].isState == |
| 269 wlan_interface_state_connected) { | 224 wlan_interface_state_connected) { |
| 270 info = &interface_list->InterfaceInfo[i]; | 225 info = &interface_list->InterfaceInfo[i]; |
| 271 break; | 226 break; |
| 272 } | 227 } |
| 273 } | 228 } |
| 274 | 229 |
| 275 if (info == NULL) | 230 if (info == NULL) |
| 276 return WIFI_PHY_LAYER_PROTOCOL_NONE; | 231 return WIFI_PHY_LAYER_PROTOCOL_NONE; |
| 277 | 232 |
| 278 WLAN_CONNECTION_ATTRIBUTES* conn_info_ptr; | 233 WLAN_CONNECTION_ATTRIBUTES* conn_info_ptr; |
| 279 DWORD conn_info_size = 0; | 234 DWORD conn_info_size = 0; |
| 280 WLAN_OPCODE_VALUE_TYPE op_code; | 235 WLAN_OPCODE_VALUE_TYPE op_code; |
| 281 result = wlanapi.query_interface_func( | 236 result = wlanapi.query_interface_func( |
| 282 client, &info->InterfaceGuid, wlan_intf_opcode_current_connection, NULL, | 237 client, &info->InterfaceGuid, wlan_intf_opcode_current_connection, NULL, |
| 283 &conn_info_size, reinterpret_cast<VOID**>(&conn_info_ptr), &op_code); | 238 &conn_info_size, reinterpret_cast<VOID**>(&conn_info_ptr), &op_code); |
| 284 if (result != ERROR_SUCCESS) | 239 if (result != ERROR_SUCCESS) |
| 285 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN; | 240 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN; |
| 286 scoped_ptr<WLAN_CONNECTION_ATTRIBUTES, WlanApiDeleter> conn_info( | 241 scoped_ptr<WLAN_CONNECTION_ATTRIBUTES, internal::WlanApiDeleter> conn_info( |
| 287 conn_info_ptr); | 242 conn_info_ptr); |
| 288 | 243 |
| 289 switch (conn_info->wlanAssociationAttributes.dot11PhyType) { | 244 switch (conn_info->wlanAssociationAttributes.dot11PhyType) { |
| 290 case dot11_phy_type_fhss: | 245 case dot11_phy_type_fhss: |
| 291 return WIFI_PHY_LAYER_PROTOCOL_ANCIENT; | 246 return WIFI_PHY_LAYER_PROTOCOL_ANCIENT; |
| 292 case dot11_phy_type_dsss: | 247 case dot11_phy_type_dsss: |
| 293 return WIFI_PHY_LAYER_PROTOCOL_B; | 248 return WIFI_PHY_LAYER_PROTOCOL_B; |
| 294 case dot11_phy_type_irbaseband: | 249 case dot11_phy_type_irbaseband: |
| 295 return WIFI_PHY_LAYER_PROTOCOL_ANCIENT; | 250 return WIFI_PHY_LAYER_PROTOCOL_ANCIENT; |
| 296 case dot11_phy_type_ofdm: | 251 case dot11_phy_type_ofdm: |
| 297 return WIFI_PHY_LAYER_PROTOCOL_A; | 252 return WIFI_PHY_LAYER_PROTOCOL_A; |
| 298 case dot11_phy_type_hrdsss: | 253 case dot11_phy_type_hrdsss: |
| 299 return WIFI_PHY_LAYER_PROTOCOL_B; | 254 return WIFI_PHY_LAYER_PROTOCOL_B; |
| 300 case dot11_phy_type_erp: | 255 case dot11_phy_type_erp: |
| 301 return WIFI_PHY_LAYER_PROTOCOL_G; | 256 return WIFI_PHY_LAYER_PROTOCOL_G; |
| 302 case dot11_phy_type_ht: | 257 case dot11_phy_type_ht: |
| 303 return WIFI_PHY_LAYER_PROTOCOL_N; | 258 return WIFI_PHY_LAYER_PROTOCOL_N; |
| 304 default: | 259 default: |
| 305 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN; | 260 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN; |
| 306 } | 261 } |
| 307 } | 262 } |
| 308 | 263 |
| 264 // Note: There is no need to explicitly set the options back |
| 265 // as the OS will automatically set them back when the WlanHandle |
| 266 // is closed. |
| 267 class WifiOptionSetter : public ScopedWifiOptions { |
| 268 public: |
| 269 WifiOptionSetter(int options) { |
| 270 const internal::WlanApi& wlanapi = internal::WlanApi::GetInstance(); |
| 271 if (!wlanapi.initialized) |
| 272 return; |
| 273 |
| 274 DWORD cur_version = 0; |
| 275 const DWORD kMaxClientVersion = 2; |
| 276 DWORD result = wlanapi.OpenHandle( |
| 277 kMaxClientVersion, &cur_version, &client_); |
| 278 if (result != ERROR_SUCCESS) |
| 279 return; |
| 280 |
| 281 WLAN_INTERFACE_INFO_LIST* interface_list_ptr = NULL; |
| 282 result = wlanapi.enum_interfaces_func(client_, NULL, &interface_list_ptr); |
| 283 if (result != ERROR_SUCCESS) |
| 284 return; |
| 285 scoped_ptr<WLAN_INTERFACE_INFO_LIST, internal::WlanApiDeleter> |
| 286 interface_list(interface_list_ptr); |
| 287 |
| 288 for (unsigned i = 0; i < interface_list->dwNumberOfItems; ++i) { |
| 289 WLAN_INTERFACE_INFO* info = &interface_list->InterfaceInfo[i]; |
| 290 if (options & WIFI_OPTIONS_DISABLE_SCAN) { |
| 291 BOOL data = false; |
| 292 wlanapi.set_interface_func(client_, |
| 293 &info->InterfaceGuid, |
| 294 wlan_intf_opcode_background_scan_enabled, |
| 295 sizeof(data), |
| 296 &data, |
| 297 NULL); |
| 298 } |
| 299 if (options & WIFI_OPTIONS_MEDIA_STREAMING_MODE) { |
| 300 BOOL data = true; |
| 301 wlanapi.set_interface_func(client_, |
| 302 &info->InterfaceGuid, |
| 303 wlan_intf_opcode_media_streaming_mode, |
| 304 sizeof(data), |
| 305 &data, |
| 306 NULL); |
| 307 } |
| 308 } |
| 309 } |
| 310 |
| 311 private: |
| 312 internal::WlanHandle client_; |
| 313 }; |
| 314 |
| 315 scoped_ptr<ScopedWifiOptions> SetWifiOptions(int options) { |
| 316 return scoped_ptr<ScopedWifiOptions>(new WifiOptionSetter(options)); |
| 317 } |
| 318 |
| 309 } // namespace net | 319 } // namespace net |
| OLD | NEW |