Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(257)

Side by Side Diff: net/base/net_util_win.cc

Issue 566243005: Cast: Allow extension to control wifi options on windows (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: comments addressed Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « net/base/net_util_win.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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
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
OLDNEW
« no previous file with comments | « net/base/net_util_win.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698