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 | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |