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

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: add missing NET_EXPORT 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
« net/base/net_util_win.h ('K') | « 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
54 base::LazyInstance<WlanApi>::Leaky lazy_wlanapi =
55 LAZY_INSTANCE_INITIALIZER;
56
57 WlanApi& WlanApi::GetInstance() {
58 return lazy_wlanapi.Get();
59 }
60
61 WlanApi::WlanApi() : initialized(false) {
62 // Use an absolute path to load the DLL to avoid DLL preloading attacks.
63 static const wchar_t* const kDLL = L"%WINDIR%\\system32\\wlanapi.dll";
64 wchar_t path[MAX_PATH] = {0};
65 ExpandEnvironmentStrings(kDLL, path, arraysize(path));
66 module = ::LoadLibraryEx(path, NULL, LOAD_WITH_ALTERED_SEARCH_PATH);
67 if (!module)
68 return;
69
70 open_handle_func = reinterpret_cast<WlanOpenHandleFunc>(
71 ::GetProcAddress(module, "WlanOpenHandle"));
72 enum_interfaces_func = reinterpret_cast<WlanEnumInterfacesFunc>(
73 ::GetProcAddress(module, "WlanEnumInterfaces"));
74 query_interface_func = reinterpret_cast<WlanQueryInterfaceFunc>(
75 ::GetProcAddress(module, "WlanQueryInterface"));
76 set_interface_func = reinterpret_cast<WlanSetInterfaceFunc>(
77 ::GetProcAddress(module, "WlanSetInterface"));
78 free_memory_func = reinterpret_cast<WlanFreeMemoryFunc>(
79 ::GetProcAddress(module, "WlanFreeMemory"));
80 close_handle_func = reinterpret_cast<WlanCloseHandleFunc>(
81 ::GetProcAddress(module, "WlanCloseHandle"));
82 initialized = open_handle_func && enum_interfaces_func &&
83 query_interface_func && set_interface_func &&
84 free_memory_func && close_handle_func;
85 }
86
107 bool GetNetworkList(NetworkInterfaceList* networks, int policy) { 87 bool GetNetworkList(NetworkInterfaceList* networks, int policy) {
108 // GetAdaptersAddresses() may require IO operations. 88 // GetAdaptersAddresses() may require IO operations.
109 base::ThreadRestrictions::AssertIOAllowed(); 89 base::ThreadRestrictions::AssertIOAllowed();
110 bool is_xp = base::win::GetVersion() < base::win::VERSION_VISTA; 90 bool is_xp = base::win::GetVersion() < base::win::VERSION_VISTA;
111 ULONG len = 0; 91 ULONG len = 0;
112 ULONG flags = is_xp ? GAA_FLAG_INCLUDE_PREFIX : 0; 92 ULONG flags = is_xp ? GAA_FLAG_INCLUDE_PREFIX : 0;
113 // First get number of networks. 93 // First get number of networks.
114 ULONG result = GetAdaptersAddresses(AF_UNSPEC, flags, NULL, NULL, &len); 94 ULONG result = GetAdaptersAddresses(AF_UNSPEC, flags, NULL, NULL, &len);
115 if (result != ERROR_BUFFER_OVERFLOW) { 95 if (result != ERROR_BUFFER_OVERFLOW) {
116 // There are 0 networks. 96 // There are 0 networks.
(...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after
167 for (IP_ADAPTER_PREFIX* prefix = adapter->FirstPrefix; prefix; 147 for (IP_ADAPTER_PREFIX* prefix = adapter->FirstPrefix; prefix;
168 prefix = prefix->Next) { 148 prefix = prefix->Next) {
169 int prefix_family = prefix->Address.lpSockaddr->sa_family; 149 int prefix_family = prefix->Address.lpSockaddr->sa_family;
170 IPEndPoint network_endpoint; 150 IPEndPoint network_endpoint;
171 if (prefix_family == family && 151 if (prefix_family == family &&
172 network_endpoint.FromSockAddr(prefix->Address.lpSockaddr, 152 network_endpoint.FromSockAddr(prefix->Address.lpSockaddr,
173 prefix->Address.iSockaddrLength) && 153 prefix->Address.iSockaddrLength) &&
174 IPNumberMatchesPrefix(endpoint.address(), 154 IPNumberMatchesPrefix(endpoint.address(),
175 network_endpoint.address(), 155 network_endpoint.address(),
176 prefix->PrefixLength)) { 156 prefix->PrefixLength)) {
177 net_prefix = std::max<size_t>(net_prefix, prefix->PrefixLength); 157 net_prefix = std::max<size_t>(net_prefix,
158 prefix->PrefixLength);
pauljensen 2014/09/23 14:03:55 unnecessary change?
hubbe 2014/09/23 17:37:50 Done.
178 } 159 }
179 } 160 }
180 } 161 }
181 uint32 index = 162 uint32 index =
182 (family == AF_INET) ? adapter->IfIndex : adapter->Ipv6IfIndex; 163 (family == AF_INET) ? adapter->IfIndex : adapter->Ipv6IfIndex;
183 // Pick one IPv6 address with least valid lifetime. 164 // Pick one IPv6 address with least valid lifetime.
184 // The reason we are checking |ValidLifeftime| as there is no other 165 // The reason we are checking |ValidLifeftime| as there is no other
185 // way identifying the interface type. Usually (and most likely) temp 166 // way identifying the interface type. Usually (and most likely) temp
186 // IPv6 will have a shorter ValidLifetime value then the permanent 167 // IPv6 will have a shorter ValidLifetime value then the permanent
187 // interface. 168 // interface.
(...skipping 21 matching lines...) Expand all
209 } 190 }
210 } 191 }
211 192
212 if (ipv6_address.get()) { 193 if (ipv6_address.get()) {
213 networks->push_back(*(ipv6_address.get())); 194 networks->push_back(*(ipv6_address.get()));
214 } 195 }
215 return true; 196 return true;
216 } 197 }
217 198
218 WifiPHYLayerProtocol GetWifiPHYLayerProtocol() { 199 WifiPHYLayerProtocol GetWifiPHYLayerProtocol() {
219 static base::LazyInstance<WlanApi>::Leaky lazy_wlanapi = 200 const WlanApi& wlanapi = 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) 201 if (!wlanapi.initialized)
249 return WIFI_PHY_LAYER_PROTOCOL_NONE; 202 return WIFI_PHY_LAYER_PROTOCOL_NONE;
250 203
251 WlanHandle client; 204 WlanHandle client;
252 DWORD cur_version = 0; 205 DWORD cur_version = 0;
253 const DWORD kMaxClientVersion = 2; 206 const DWORD kMaxClientVersion = 2;
254 DWORD result = wlanapi.OpenHandle(kMaxClientVersion, &cur_version, &client); 207 DWORD result = wlanapi.OpenHandle(kMaxClientVersion, &cur_version, &client);
255 if (result != ERROR_SUCCESS) 208 if (result != ERROR_SUCCESS)
256 return WIFI_PHY_LAYER_PROTOCOL_NONE; 209 return WIFI_PHY_LAYER_PROTOCOL_NONE;
257 210
(...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after
299 return WIFI_PHY_LAYER_PROTOCOL_B; 252 return WIFI_PHY_LAYER_PROTOCOL_B;
300 case dot11_phy_type_erp: 253 case dot11_phy_type_erp:
301 return WIFI_PHY_LAYER_PROTOCOL_G; 254 return WIFI_PHY_LAYER_PROTOCOL_G;
302 case dot11_phy_type_ht: 255 case dot11_phy_type_ht:
303 return WIFI_PHY_LAYER_PROTOCOL_N; 256 return WIFI_PHY_LAYER_PROTOCOL_N;
304 default: 257 default:
305 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN; 258 return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN;
306 } 259 }
307 } 260 }
308 261
262 // Note: There is no need to explicitly set the options back
263 // as the OS will automatically set them back when the WlanHandle
264 // is closed.
pauljensen 2014/09/23 14:03:55 I couldn't find any documentation for this. Have
hubbe 2014/09/23 17:37:49 It's very poorly documented, but the documentation
265 class WifiOptionSetter : public ScopedWifiOptions {
266 public:
267 WifiOptionSetter(int options) {
268 const WlanApi& wlanapi = WlanApi::GetInstance();
269 if (!wlanapi.initialized)
270 return;
271
272 DWORD cur_version = 0;
273 const DWORD kMaxClientVersion = 2;
274 DWORD result = wlanapi.OpenHandle(
275 kMaxClientVersion, &cur_version, &client_);
276 if (result != ERROR_SUCCESS)
277 return;
278
279 WLAN_INTERFACE_INFO_LIST* interface_list_ptr = NULL;
280 result = wlanapi.enum_interfaces_func(client_, NULL, &interface_list_ptr);
281 if (result != ERROR_SUCCESS)
282 return;
283 scoped_ptr<WLAN_INTERFACE_INFO_LIST, WlanApiDeleter> interface_list(
284 interface_list_ptr);
285
286 for (unsigned i = 0; i < interface_list->dwNumberOfItems; ++i) {
287 WLAN_INTERFACE_INFO* info = &interface_list->InterfaceInfo[i];
288 if (options & WIFI_OPTIONS_DISABLE_SCAN) {
289 BOOL data = false;
290 wlanapi.set_interface_func(client_,
291 &info->InterfaceGuid,
292 wlan_intf_opcode_background_scan_enabled,
293 sizeof(data),
294 &data,
295 NULL);
296 }
297 if (options & WIFI_OPTIONS_MEDIA_STREAMING_MODE) {
298 BOOL data = true;
299 wlanapi.set_interface_func(client_,
300 &info->InterfaceGuid,
301 wlan_intf_opcode_media_streaming_mode,
302 sizeof(data),
303 &data,
304 NULL);
305 }
306 }
307 }
308
309 private:
310 WlanHandle client_;
311 };
312
313 scoped_ptr<ScopedWifiOptions> SetWifiOptions(int options) {
314 return scoped_ptr<ScopedWifiOptions>(new WifiOptionSetter(options));
315 }
316
309 } // namespace net 317 } // namespace net
OLDNEW
« net/base/net_util_win.h ('K') | « net/base/net_util_win.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698