2 * Copyright (C) 2005-2013 Team XBMC
5 * This Program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2, or (at your option)
10 * This Program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with XBMC; see the file COPYING. If not, see
17 * <http://www.gnu.org/licenses/>.
24 #include "PlatformDefs.h"
25 #include "NetworkWin32.h"
26 #include "utils/log.h"
27 #include "threads/SingleLock.h"
28 #include "utils/CharsetConverter.h"
30 // undefine if you want to build without the wlan stuff
31 // might be needed for VS2003
32 #define HAS_WIN32_WLAN_API
34 #ifdef HAS_WIN32_WLAN_API
36 #pragma comment (lib,"Wlanapi.lib")
42 CNetworkInterfaceWin32::CNetworkInterfaceWin32(CNetworkWin32* network, IP_ADAPTER_INFO adapter)
47 m_adaptername = adapter.Description;
50 CNetworkInterfaceWin32::~CNetworkInterfaceWin32(void)
54 CStdString& CNetworkInterfaceWin32::GetName(void)
56 if (!g_charsetConverter.isValidUtf8(m_adaptername))
57 g_charsetConverter.unknownToUTF8(m_adaptername);
61 bool CNetworkInterfaceWin32::IsWireless()
63 return (m_adapter.Type == IF_TYPE_IEEE80211);
66 bool CNetworkInterfaceWin32::IsEnabled()
71 bool CNetworkInterfaceWin32::IsConnected()
73 CStdString strIP = m_adapter.IpAddressList.IpAddress.String;
74 return (strIP != "0.0.0.0");
77 CStdString CNetworkInterfaceWin32::GetMacAddress()
80 unsigned char* mAddr = m_adapter.Address;
81 result.Format("%02X:%02X:%02X:%02X:%02X:%02X", mAddr[0], mAddr[1], mAddr[2], mAddr[3], mAddr[4], mAddr[5]);
85 void CNetworkInterfaceWin32::GetMacAddressRaw(char rawMac[6])
87 memcpy(rawMac, m_adapter.Address, 6);
90 CStdString CNetworkInterfaceWin32::GetCurrentIPAddress(void)
92 return m_adapter.IpAddressList.IpAddress.String;
95 CStdString CNetworkInterfaceWin32::GetCurrentNetmask(void)
97 return m_adapter.IpAddressList.IpMask.String;
100 CStdString CNetworkInterfaceWin32::GetCurrentWirelessEssId(void)
102 CStdString result = "";
104 #ifdef HAS_WIN32_WLAN_API
107 HANDLE hClientHdl = NULL;
110 PWLAN_CONNECTION_ATTRIBUTES pAttributes;
113 if(WlanOpenHandle(1,NULL,&dwVersion, &hClientHdl) == ERROR_SUCCESS)
115 PWLAN_INTERFACE_INFO_LIST ppInterfaceList;
116 if(WlanEnumInterfaces(hClientHdl,NULL, &ppInterfaceList ) == ERROR_SUCCESS)
118 for(unsigned int i=0; i<ppInterfaceList->dwNumberOfItems;i++)
120 GUID guid = ppInterfaceList->InterfaceInfo[i].InterfaceGuid;
122 StringFromGUID2(guid, (LPOLESTR)&wcguid, 64);
123 CStdStringW strGuid = wcguid;
124 CStdStringW strAdaptername = m_adapter.AdapterName;
125 if( strGuid == strAdaptername)
127 if(WlanQueryInterface(hClientHdl,&ppInterfaceList->InterfaceInfo[i].InterfaceGuid,wlan_intf_opcode_current_connection, NULL, &dwSize, (PVOID*)&pAttributes, NULL ) == ERROR_SUCCESS)
129 result = (char*)pAttributes->wlanAssociationAttributes.dot11Ssid.ucSSID;
130 WlanFreeMemory((PVOID*)&pAttributes);
133 OutputDebugString("Can't query wlan interface\n");
137 WlanCloseHandle(&hClientHdl, NULL);
140 OutputDebugString("Can't open wlan handle\n");
146 CStdString CNetworkInterfaceWin32::GetCurrentDefaultGateway(void)
148 return m_adapter.GatewayList.IpAddress.String;
151 CNetworkWin32::CNetworkWin32(void)
153 queryInterfaceList();
156 CNetworkWin32::~CNetworkWin32(void)
158 CleanInterfaceList();
159 m_netrefreshTimer.Stop();
162 void CNetworkWin32::CleanInterfaceList()
164 vector<CNetworkInterface*>::iterator it = m_interfaces.begin();
165 while(it != m_interfaces.end())
167 CNetworkInterface* nInt = *it;
169 it = m_interfaces.erase(it);
173 std::vector<CNetworkInterface*>& CNetworkWin32::GetInterfaceList(void)
175 CSingleLock lock (m_critSection);
176 if(m_netrefreshTimer.GetElapsedSeconds() >= 5.0f)
177 queryInterfaceList();
182 void CNetworkWin32::queryInterfaceList()
184 CleanInterfaceList();
185 m_netrefreshTimer.StartZero();
187 PIP_ADAPTER_INFO adapterInfo;
188 PIP_ADAPTER_INFO adapter = NULL;
190 ULONG ulOutBufLen = sizeof (IP_ADAPTER_INFO);
192 adapterInfo = (IP_ADAPTER_INFO *) malloc(sizeof (IP_ADAPTER_INFO));
193 if (adapterInfo == NULL)
196 if (GetAdaptersInfo(adapterInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW)
199 adapterInfo = (IP_ADAPTER_INFO *) malloc(ulOutBufLen);
200 if (adapterInfo == NULL)
202 OutputDebugString("Error allocating memory needed to call GetAdaptersinfo\n");
207 if ((GetAdaptersInfo(adapterInfo, &ulOutBufLen)) == NO_ERROR)
209 adapter = adapterInfo;
212 m_interfaces.push_back(new CNetworkInterfaceWin32(this, *adapter));
213 adapter = adapter->Next;
219 std::vector<CStdString> CNetworkWin32::GetNameServers(void)
221 std::vector<CStdString> result;
223 FIXED_INFO *pFixedInfo;
225 IP_ADDR_STRING *pIPAddr;
227 pFixedInfo = (FIXED_INFO *) malloc(sizeof (FIXED_INFO));
228 if (pFixedInfo == NULL)
230 OutputDebugString("Error allocating memory needed to call GetNetworkParams\n");
233 ulOutBufLen = sizeof (FIXED_INFO);
234 if (GetNetworkParams(pFixedInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW)
237 pFixedInfo = (FIXED_INFO *) malloc(ulOutBufLen);
238 if (pFixedInfo == NULL)
240 OutputDebugString("Error allocating memory needed to call GetNetworkParams\n");
245 if (GetNetworkParams(pFixedInfo, &ulOutBufLen) == NO_ERROR)
247 result.push_back(pFixedInfo->DnsServerList.IpAddress.String);
248 pIPAddr = pFixedInfo->DnsServerList.Next;
251 result.push_back(pIPAddr->IpAddress.String);
252 pIPAddr = pIPAddr->Next;
261 void CNetworkWin32::SetNameServers(std::vector<CStdString> nameServers)
266 bool CNetworkWin32::PingHost(unsigned long host, unsigned int timeout_ms /* = 2000 */)
268 char SendData[] = "poke";
269 HANDLE hIcmpFile = IcmpCreateFile();
270 BYTE ReplyBuffer [sizeof(ICMP_ECHO_REPLY) + sizeof(SendData)];
272 SetLastError(ERROR_SUCCESS);
274 DWORD dwRetVal = IcmpSendEcho(hIcmpFile, host, SendData, sizeof(SendData),
275 NULL, ReplyBuffer, sizeof(ReplyBuffer), timeout_ms);
277 DWORD lastErr = GetLastError();
278 if (lastErr != ERROR_SUCCESS && lastErr != IP_REQ_TIMED_OUT)
279 CLog::Log(LOGERROR, "%s - IcmpSendEcho failed - %s", __FUNCTION__, WUSysMsg(lastErr).c_str());
281 IcmpCloseHandle (hIcmpFile);
285 PICMP_ECHO_REPLY pEchoReply = (PICMP_ECHO_REPLY)ReplyBuffer;
286 return (pEchoReply->Status == IP_SUCCESS);
291 bool CNetworkInterfaceWin32::GetHostMacAddress(unsigned long host, CStdString& mac)
293 IPAddr src_ip = inet_addr(GetCurrentIPAddress().c_str());
294 BYTE bPhysAddr[6]; // for 6-byte hardware addresses
295 ULONG PhysAddrLen = 6; // default to length of six bytes
297 memset(&bPhysAddr, 0xff, sizeof (bPhysAddr));
299 DWORD dwRetVal = SendARP(host, src_ip, &bPhysAddr, &PhysAddrLen);
300 if (dwRetVal == NO_ERROR)
302 if (PhysAddrLen == 6)
304 mac.Format("%02X:%02X:%02X:%02X:%02X:%02X",
305 bPhysAddr[0], bPhysAddr[1], bPhysAddr[2],
306 bPhysAddr[3], bPhysAddr[4], bPhysAddr[5]);
310 CLog::Log(LOGERROR, "%s - SendArp completed successfully, but mac address has length != 6 (%d)", __FUNCTION__, PhysAddrLen);
313 CLog::Log(LOGERROR, "%s - SendArp failed with error (%d)", __FUNCTION__, dwRetVal);
318 std::vector<NetworkAccessPoint> CNetworkInterfaceWin32::GetAccessPoints(void)
320 std::vector<NetworkAccessPoint> result;
321 #ifdef HAS_WIN32_WLAN_API
325 // According to Mozilla: "We could be executing on either Windows XP or Windows
326 // Vista, so use the lower version of the client WLAN API. It seems that the
327 // negotiated version is the Vista version irrespective of what we pass!"
328 // http://dxr.mozilla.org/mozilla-central/source/netwerk/wifi/nsWifiScannerWin.cpp#l51
329 static const int xpWlanClientVersion = 1;
330 DWORD negotiated_version;
332 HANDLE wlan_handle = NULL;
334 // Get the handle to the WLAN API
335 dwResult = WlanOpenHandle(xpWlanClientVersion, NULL, &negotiated_version, &wlan_handle);
336 if (dwResult != ERROR_SUCCESS || !wlan_handle)
338 CLog::Log(LOGERROR, "Could not load the client WLAN API");
342 // Get the list of interfaces (WlanEnumInterfaces allocates interface_list)
343 WLAN_INTERFACE_INFO_LIST *interface_list = NULL;
344 dwResult = WlanEnumInterfaces(wlan_handle, NULL, &interface_list);
345 if (dwResult != ERROR_SUCCESS || !interface_list)
347 WlanCloseHandle(wlan_handle, NULL);
348 CLog::Log(LOGERROR, "Failed to get the list of interfaces");
352 for (unsigned int i = 0; i < interface_list->dwNumberOfItems; ++i)
354 GUID guid = interface_list->InterfaceInfo[i].InterfaceGuid;
356 StringFromGUID2(guid, (LPOLESTR)&wcguid, 64);
357 CStdStringW strGuid = wcguid;
358 CStdStringW strAdaptername = m_adapter.AdapterName;
359 if (strGuid == strAdaptername)
361 WLAN_BSS_LIST *bss_list;
362 HRESULT rv = WlanGetNetworkBssList(wlan_handle,
363 &interface_list->InterfaceInfo[i].InterfaceGuid,
364 NULL, // Get all SSIDs
365 dot11_BSS_type_any, // unused
366 false, // bSecurityEnabled - unused
369 if (rv != ERROR_SUCCESS || !bss_list)
371 for (unsigned int j = 0; j < bss_list->dwNumberOfItems; ++j)
373 const WLAN_BSS_ENTRY bss_entry = bss_list->wlanBssEntries[j];
374 // Add the access point info to the list of results
375 CStdString essId((char*)bss_entry.dot11Ssid.ucSSID, (unsigned int)bss_entry.dot11Ssid.uSSIDLength);
376 CStdString macAddress;
377 // macAddress is big-endian, write in byte chunks
378 macAddress.Format("%02x-%02x-%02x-%02x-%02x-%02x",
379 bss_entry.dot11Bssid[0], bss_entry.dot11Bssid[1], bss_entry.dot11Bssid[2],
380 bss_entry.dot11Bssid[3], bss_entry.dot11Bssid[4], bss_entry.dot11Bssid[5]);
381 int signalLevel = bss_entry.lRssi;
382 EncMode encryption = ENC_NONE; // TODO
383 int channel = NetworkAccessPoint::FreqToChannel((float)bss_entry.ulChCenterFrequency * 1000);
384 result.push_back(NetworkAccessPoint(essId, macAddress, signalLevel, encryption, channel));
386 WlanFreeMemory(bss_list);
391 // Free the interface list
392 WlanFreeMemory(interface_list);
395 WlanCloseHandle(wlan_handle, NULL);
402 void CNetworkInterfaceWin32::GetSettings(NetworkAssignment& assignment, CStdString& ipAddress, CStdString& networkMask, CStdString& defaultGateway, CStdString& essId, CStdString& key, EncMode& encryptionMode)
404 ipAddress = "0.0.0.0";
405 networkMask = "0.0.0.0";
406 defaultGateway = "0.0.0.0";
409 encryptionMode = ENC_NONE;
410 assignment = NETWORK_DISABLED;
413 PIP_ADAPTER_INFO adapterInfo;
414 PIP_ADAPTER_INFO adapter = NULL;
416 ULONG ulOutBufLen = sizeof (IP_ADAPTER_INFO);
418 adapterInfo = (IP_ADAPTER_INFO *) malloc(sizeof (IP_ADAPTER_INFO));
419 if (adapterInfo == NULL)
422 if (GetAdaptersInfo(adapterInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW)
425 adapterInfo = (IP_ADAPTER_INFO *) malloc(ulOutBufLen);
426 if (adapterInfo == NULL)
428 OutputDebugString("Error allocating memory needed to call GetAdaptersinfo\n");
433 if ((GetAdaptersInfo(adapterInfo, &ulOutBufLen)) == NO_ERROR)
435 adapter = adapterInfo;
438 if(m_adapter.Index == adapter->Index)
440 ipAddress = adapter->IpAddressList.IpAddress.String;
441 networkMask = adapter->IpAddressList.IpMask.String;
442 defaultGateway = adapter->GatewayList.IpAddress.String;
443 if (adapter->DhcpEnabled)
444 assignment = NETWORK_DHCP;
446 assignment = NETWORK_STATIC;
449 adapter = adapter->Next;
454 #ifdef HAS_WIN32_WLAN_API
457 HANDLE hClientHdl = NULL;
460 PWLAN_CONNECTION_ATTRIBUTES pAttributes;
463 if(WlanOpenHandle(1,NULL,&dwVersion, &hClientHdl) == ERROR_SUCCESS)
465 PWLAN_INTERFACE_INFO_LIST ppInterfaceList;
466 if(WlanEnumInterfaces(hClientHdl,NULL, &ppInterfaceList ) == ERROR_SUCCESS)
468 for(unsigned int i=0; i<ppInterfaceList->dwNumberOfItems;i++)
470 GUID guid = ppInterfaceList->InterfaceInfo[i].InterfaceGuid;
472 StringFromGUID2(guid, (LPOLESTR)&wcguid, 64);
473 CStdStringW strGuid = wcguid;
474 CStdStringW strAdaptername = m_adapter.AdapterName;
475 if( strGuid == strAdaptername)
477 if(WlanQueryInterface(hClientHdl,&ppInterfaceList->InterfaceInfo[i].InterfaceGuid,wlan_intf_opcode_current_connection, NULL, &dwSize, (PVOID*)&pAttributes, NULL ) == ERROR_SUCCESS)
479 essId = (char*)pAttributes->wlanAssociationAttributes.dot11Ssid.ucSSID;
480 if(pAttributes->wlanSecurityAttributes.bSecurityEnabled)
482 switch(pAttributes->wlanSecurityAttributes.dot11AuthAlgorithm)
484 case DOT11_AUTH_ALGO_80211_SHARED_KEY:
485 encryptionMode = ENC_WEP;
487 case DOT11_AUTH_ALGO_WPA:
488 case DOT11_AUTH_ALGO_WPA_PSK:
489 encryptionMode = ENC_WPA;
491 case DOT11_AUTH_ALGO_RSNA:
492 case DOT11_AUTH_ALGO_RSNA_PSK:
493 encryptionMode = ENC_WPA2;
496 WlanFreeMemory((PVOID*)&pAttributes);
499 OutputDebugString("Can't query wlan interface\n");
503 WlanCloseHandle(&hClientHdl, NULL);
506 OutputDebugString("Can't open wlan handle\n");
508 // Todo: get the key (WlanGetProfile, CryptUnprotectData?)
512 void CNetworkInterfaceWin32::SetSettings(NetworkAssignment& assignment, CStdString& ipAddress, CStdString& networkMask, CStdString& defaultGateway, CStdString& essId, CStdString& key, EncMode& encryptionMode)
517 void CNetworkInterfaceWin32::WriteSettings(FILE* fw, NetworkAssignment assignment, CStdString& ipAddress, CStdString& networkMask, CStdString& defaultGateway, CStdString& essId, CStdString& key, EncMode& encryptionMode)