Windows2003-3790/admin/cys/lib/networkinterface.cpp

1340 lines
36 KiB
C++
Raw Normal View History

2001-01-01 00:00:00 +01:00
// Copyright (c) 2001 Microsoft Corporation
//
// File: NetworkInterface.cpp
//
// Synopsis: Defines a NetworkInterface
// This object has the knowledge of an
// IP enabled network connection including
// IP address, DHCP information, etc.
//
// History: 03/01/2001 JeffJon Created
#include "pch.h"
#include "NetworkInterface.h"
#include <iphlpstk.h> // SetIpForwardEntryToStack
#include <IPIfCons.h> // Adapter type info
#include <raserror.h> // RAS error codes
#define MPR50 1 // Needed in order to include routprot.h
#include <routprot.h> // Router protocols
#define CYS_WMIPROP_IPADDRESS L"IPAddress"
#define CYS_WMIPROP_IPSUBNET L"IPSubnet"
#define CYS_WMIPROP_DHCPENABLED L"DHCPEnabled"
#define CYS_WMIPROP_DESCRIPTION L"Description"
#define CYS_WMIPROP_DNSSERVERS L"DNSServerSearchOrder"
#define CYS_WMIPROP_INDEX L"Index"
// An array of strings to ease the logging of the adapter type
static PCWSTR adapterTypes[] =
{
L"IF_TYPE_OTHER", // 1 None of the below
L"IF_TYPE_REGULAR_1822", // 2
L"IF_TYPE_HDH_1822", // 3
L"IF_TYPE_DDN_X25", // 4
L"IF_TYPE_RFC877_X25", // 5
L"IF_TYPE_ETHERNET_CSMACD", // 6
L"IF_TYPE_IS088023_CSMACD", // 7
L"IF_TYPE_ISO88024_TOKENBUS", // 8
L"IF_TYPE_ISO88025_TOKENRING", // 9
L"IF_TYPE_ISO88026_MAN", // 10
L"IF_TYPE_STARLAN", // 11
L"IF_TYPE_PROTEON_10MBIT", // 12
L"IF_TYPE_PROTEON_80MBIT", // 13
L"IF_TYPE_HYPERCHANNEL", // 14
L"IF_TYPE_FDDI", // 15
L"IF_TYPE_LAP_B", // 16
L"IF_TYPE_SDLC", // 17
L"IF_TYPE_DS1", // 18 DS1-MIB
L"IF_TYPE_E1", // 19 Obsolete; see DS1-MIB
L"IF_TYPE_BASIC_ISDN", // 20
L"IF_TYPE_PRIMARY_ISDN", // 21
L"IF_TYPE_PROP_POINT2POINT_SERIAL", // 22 proprietary serial
L"IF_TYPE_PPP", // 23
L"IF_TYPE_SOFTWARE_LOOPBACK", // 24
L"IF_TYPE_EON", // 25 CLNP over IP
L"IF_TYPE_ETHERNET_3MBIT", // 26
L"IF_TYPE_NSIP", // 27 XNS over IP
L"IF_TYPE_SLIP", // 28 Generic Slip
L"IF_TYPE_ULTRA", // 29 ULTRA Technologies
L"IF_TYPE_DS3", // 30 DS3-MIB
L"IF_TYPE_SIP", // 31 SMDS, coffee
L"IF_TYPE_FRAMERELAY", // 32 DTE only
L"IF_TYPE_RS232", // 33
L"IF_TYPE_PARA", // 34 Parallel port
L"IF_TYPE_ARCNET", // 35
L"IF_TYPE_ARCNET_PLUS", // 36
L"IF_TYPE_ATM", // 37 ATM cells
L"IF_TYPE_MIO_X25", // 38
L"IF_TYPE_SONET", // 39 SONET or SDH
L"IF_TYPE_X25_PLE", // 40
L"IF_TYPE_ISO88022_LLC", // 41
L"IF_TYPE_LOCALTALK", // 42
L"IF_TYPE_SMDS_DXI", // 43
L"IF_TYPE_FRAMERELAY_SERVICE", // 44 FRNETSERV-MIB
L"IF_TYPE_V35", // 45
L"IF_TYPE_HSSI", // 46
L"IF_TYPE_HIPPI", // 47
L"IF_TYPE_MODEM", // 48 Generic Modem
L"IF_TYPE_AAL5", // 49 AAL5 over ATM
L"IF_TYPE_SONET_PATH", // 50
L"IF_TYPE_SONET_VT", // 51
L"IF_TYPE_SMDS_ICIP", // 52 SMDS InterCarrier Interface
L"IF_TYPE_PROP_VIRTUAL", // 53 Proprietary virtual/internal
L"IF_TYPE_PROP_MULTIPLEXOR", // 54 Proprietary multiplexing
L"IF_TYPE_IEEE80212", // 55 100BaseVG
L"IF_TYPE_FIBRECHANNEL", // 56
L"IF_TYPE_HIPPIINTERFACE", // 57
L"IF_TYPE_FRAMERELAY_INTERCONNECT", // 58 Obsolete, use 32 or 44
L"IF_TYPE_AFLANE_8023", // 59 ATM Emulated LAN for 802.3
L"IF_TYPE_AFLANE_8025", // 60 ATM Emulated LAN for 802.5
L"IF_TYPE_CCTEMUL", // 61 ATM Emulated circuit
L"IF_TYPE_FASTETHER", // 62 Fast Ethernet (100BaseT)
L"IF_TYPE_ISDN", // 63 ISDN and X.25
L"IF_TYPE_V11", // 64 CCITT V.11/X.21
L"IF_TYPE_V36", // 65 CCITT V.36
L"IF_TYPE_G703_64K", // 66 CCITT G703 at 64Kbps
L"IF_TYPE_G703_2MB", // 67 Obsolete; see DS1-MIB
L"IF_TYPE_QLLC", // 68 SNA QLLC
L"IF_TYPE_FASTETHER_FX", // 69 Fast Ethernet (100BaseFX)
L"IF_TYPE_CHANNEL", // 70
L"IF_TYPE_IEEE80211", // 71 Radio spread spectrum
L"IF_TYPE_IBM370PARCHAN", // 72 IBM System 360/370 OEMI Channel
L"IF_TYPE_ESCON", // 73 IBM Enterprise Systems Connection
L"IF_TYPE_DLSW", // 74 Data Link Switching
L"IF_TYPE_ISDN_S", // 75 ISDN S/T interface
L"IF_TYPE_ISDN_U", // 76 ISDN U interface
L"IF_TYPE_LAP_D", // 77 Link Access Protocol D
L"IF_TYPE_IPSWITCH", // 78 IP Switching Objects
L"IF_TYPE_RSRB", // 79 Remote Source Route Bridging
L"IF_TYPE_ATM_LOGICAL", // 80 ATM Logical Port
L"IF_TYPE_DS0", // 81 Digital Signal Level 0
L"IF_TYPE_DS0_BUNDLE", // 82 Group of ds0s on the same ds1
L"IF_TYPE_BSC", // 83 Bisynchronous Protocol
L"IF_TYPE_ASYNC", // 84 Asynchronous Protocol
L"IF_TYPE_CNR", // 85 Combat Net Radio
L"IF_TYPE_ISO88025R_DTR", // 86 ISO 802.5r DTR
L"IF_TYPE_EPLRS", // 87 Ext Pos Loc Report Sys
L"IF_TYPE_ARAP", // 88 Appletalk Remote Access Protocol
L"IF_TYPE_PROP_CNLS", // 89 Proprietary Connectionless Proto
L"IF_TYPE_HOSTPAD", // 90 CCITT-ITU X.29 PAD Protocol
L"IF_TYPE_TERMPAD", // 91 CCITT-ITU X.3 PAD Facility
L"IF_TYPE_FRAMERELAY_MPI", // 92 Multiproto Interconnect over FR
L"IF_TYPE_X213", // 93 CCITT-ITU X213
L"IF_TYPE_ADSL", // 94 Asymmetric Digital Subscrbr Loop
L"IF_TYPE_RADSL", // 95 Rate-Adapt Digital Subscrbr Loop
L"IF_TYPE_SDSL", // 96 Symmetric Digital Subscriber Loop
L"IF_TYPE_VDSL", // 97 Very H-Speed Digital Subscrb Loop
L"IF_TYPE_ISO88025_CRFPRINT", // 98 ISO 802.5 CRFP
L"IF_TYPE_MYRINET", // 99 Myricom Myrinet
L"IF_TYPE_VOICE_EM", // 100 Voice recEive and transMit
L"IF_TYPE_VOICE_FXO", // 101 Voice Foreign Exchange Office
L"IF_TYPE_VOICE_FXS", // 102 Voice Foreign Exchange Station
L"IF_TYPE_VOICE_ENCAP", // 103 Voice encapsulation
L"IF_TYPE_VOICE_OVERIP", // 104 Voice over IP encapsulation
L"IF_TYPE_ATM_DXI", // 105 ATM DXI
L"IF_TYPE_ATM_FUNI", // 106 ATM FUNI
L"IF_TYPE_ATM_IMA", // 107 ATM IMA
L"IF_TYPE_PPPMULTILINKBUNDLE", // 108 PPP Multilink Bundle
L"IF_TYPE_IPOVER_CDLC", // 109 IBM ipOverCdlc
L"IF_TYPE_IPOVER_CLAW", // 110 IBM Common Link Access to Workstn
L"IF_TYPE_STACKTOSTACK", // 111 IBM stackToStack
L"IF_TYPE_VIRTUALIPADDRESS", // 112 IBM VIPA
L"IF_TYPE_MPC", // 113 IBM multi-proto channel support
L"IF_TYPE_IPOVER_ATM", // 114 IBM ipOverAtm
L"IF_TYPE_ISO88025_FIBER", // 115 ISO 802.5j Fiber Token Ring
L"IF_TYPE_TDLC", // 116 IBM twinaxial data link control
L"IF_TYPE_GIGABITETHERNET", // 117
L"IF_TYPE_HDLC", // 118
L"IF_TYPE_LAP_F", // 119
L"IF_TYPE_V37", // 120
L"IF_TYPE_X25_MLP", // 121 Multi-Link Protocol
L"IF_TYPE_X25_HUNTGROUP", // 122 X.25 Hunt Group
L"IF_TYPE_TRANSPHDLC", // 123
L"IF_TYPE_INTERLEAVE", // 124 Interleave channel
L"IF_TYPE_FAST", // 125 Fast channel
L"IF_TYPE_IP", // 126 IP (for APPN HPR in IP networks)
L"IF_TYPE_DOCSCABLE_MACLAYER", // 127 CATV Mac Layer
L"IF_TYPE_DOCSCABLE_DOWNSTREAM", // 128 CATV Downstream interface
L"IF_TYPE_DOCSCABLE_UPSTREAM", // 129 CATV Upstream interface
L"IF_TYPE_A12MPPSWITCH", // 130 Avalon Parallel Processor
L"IF_TYPE_TUNNEL", // 131 Encapsulation interface
L"IF_TYPE_COFFEE", // 132 Coffee pot
L"IF_TYPE_CES", // 133 Circuit Emulation Service
L"IF_TYPE_ATM_SUBINTERFACE", // 134 ATM Sub Interface
L"IF_TYPE_L2_VLAN", // 135 Layer 2 Virtual LAN using 802.1Q
L"IF_TYPE_L3_IPVLAN", // 136 Layer 3 Virtual LAN using IP
L"IF_TYPE_L3_IPXVLAN", // 137 Layer 3 Virtual LAN using IPX
L"IF_TYPE_DIGITALPOWERLINE", // 138 IP over Power Lines
L"IF_TYPE_MEDIAMAILOVERIP", // 139 Multimedia Mail over IP
L"IF_TYPE_DTM", // 140 Dynamic syncronous Transfer Mode
L"IF_TYPE_DCN", // 141 Data Communications Network
L"IF_TYPE_IPFORWARD", // 142 IP Forwarding Interface
L"IF_TYPE_MSDSL", // 143 Multi-rate Symmetric DSL
L"IF_TYPE_IEEE1394", // 144 IEEE1394 High Perf Serial Bus
L"IF_TYPE_RECEIVE_ONLY", // 145 TV adapter type
};
// macro that uses the string table above to log the adapter type
#define LOG_ADAPTER_TYPE(type) \
if (type >= 145 || type <= 0) \
{ \
LOG(String::format(L"adapterType = %1", adapterTypes[0])); \
} \
else \
{ \
LOG(String::format(L"adapterType = %1", adapterTypes[type-1])); \
}
NetworkInterface::NetworkInterface()
: initialized(false),
dhcpEnabled(false),
dhcpServerAvailable(false),
index(0)
{
LOG_CTOR(NetworkInterface);
}
NetworkInterface::~NetworkInterface()
{
LOG_DTOR(NetworkInterface);
if (!ipaddresses.empty())
{
ipaddresses.clear();
}
if (!subnetMasks.empty())
{
subnetMasks.clear();
}
}
NetworkInterface::NetworkInterface(const NetworkInterface &nic)
{
LOG_CTOR2(NetworkInterface, L"Copy constructor");
if (this == &nic)
{
return;
}
name = nic.name;
description = nic.description;
initialized = nic.initialized;
dhcpEnabled = nic.dhcpEnabled;
index = nic.index;
ipaddressStringList = nic.ipaddressStringList;
subnetMaskStringList = nic.subnetMaskStringList;
dnsServerSearchOrder = nic.dnsServerSearchOrder;
// Make a copy of the ipaddress array
ipaddresses = nic.ipaddresses;
subnetMasks = nic.subnetMasks;
}
NetworkInterface&
NetworkInterface::operator=(const NetworkInterface& rhs)
{
LOG_FUNCTION(NetworkInterface::operator=);
if (this == &rhs)
{
return *this;
}
name = rhs.name;
description = rhs.description;
initialized = rhs.initialized;
dhcpEnabled = rhs.dhcpEnabled;
index = rhs.index;
ipaddressStringList = rhs.ipaddressStringList;
subnetMaskStringList = rhs.subnetMaskStringList;
dnsServerSearchOrder = rhs.dnsServerSearchOrder;
// Make a copy of the ipaddress array
ipaddresses = rhs.ipaddresses;
subnetMasks = rhs.subnetMasks;
return *this;
}
HRESULT
NetworkInterface::Initialize(const IP_ADAPTER_INFO& adapterInfo)
{
LOG_FUNCTION(NetworkInterface::Initialize);
HRESULT hr = S_OK;
do
{
if (initialized)
{
ASSERT(!initialized);
hr = E_UNEXPECTED;
}
else
{
// Get the name
name = adapterInfo.AdapterName;
LOG(String::format(
L"name = %1",
name.c_str()));
// the description
description = adapterInfo.Description;
LOG(String::format(
L"description = %1",
description.c_str()));
// the type
type = adapterInfo.Type;
LOG_ADAPTER_TYPE(type);
// the index
index = adapterInfo.Index;
LOG(String::format(
L"index = %1!d!",
index));
// Is DHCP enabled?
dhcpEnabled = (adapterInfo.DhcpEnabled != 0);
LOG_BOOL(dhcpEnabled);
hr = SetIPList(adapterInfo.IpAddressList);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to set the IP and subnet mask: hr = 0x%1!x!",
hr));
break;
}
// Now retrieve the rest of the info from the registry
// Note: this has to be done after getting the name from
// the Adapter Info
hr = RetrieveAdapterInfoFromRegistry();
if (FAILED(hr))
{
LOG(String::format(
L"Failed to retrieve adapter info from registry: hr = 0x%1!x!",
hr));
break;
}
}
} while (false);
// If we succeeded in retrieving the data we need,
// mark the object initialized
if (SUCCEEDED(hr))
{
initialized = true;
}
LOG_HRESULT(hr);
return hr;
}
HRESULT
NetworkInterface::RetrieveAdapterInfoFromRegistry()
{
LOG_FUNCTION(NetworkInterface::RetrieveAdapterInfoFromRegistry);
HRESULT hr = S_OK;
do
{
String keyName = CYS_NETWORK_INTERFACES_KEY;
keyName += String(name);
RegistryKey key;
hr = key.Open(
HKEY_LOCAL_MACHINE,
keyName,
KEY_READ);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to read the interfaces regkey: hr = 0x%1!x!",
hr));
break;
}
// Read the NameServer key
String nameServers;
hr = key.GetValue(
CYS_NETWORK_NAME_SERVERS,
nameServers);
if (SUCCEEDED(hr))
{
LOG(String::format(
L"nameServers = %1",
nameServers.c_str()));
// Adds the name servers to the dnsServerSearchOrder member
nameServers.tokenize(std::back_inserter(dnsServerSearchOrder));
}
else
{
LOG(String::format(
L"Failed to read the NameServer regkey: hr = 0x%1!x!",
hr));
// This is not a breaking condition. We still can try the
// DhcpNameServer key
}
// Read the DhcpNameServer key
String dhcpNameServers;
hr = key.GetValue(
CYS_NETWORK_DHCP_NAME_SERVERS,
dhcpNameServers);
if (SUCCEEDED(hr))
{
LOG(String::format(
L"dhcpNameServers = %1",
dhcpNameServers.c_str()));
// Adds the name servers to the dnsServerSearchOrder member
dhcpNameServers.tokenize(std::back_inserter(dnsServerSearchOrder));
}
else
{
LOG(String::format(
L"Failed to read the DhcpNameServer regkey: hr = 0x%1!x!",
hr));
}
// It doesn't matter if we were not able to retrieve the name servers
// These are just used as suggestions for DNS forwarding.
hr = S_OK;
} while (false);
LOG_HRESULT(hr);
return hr;
}
HRESULT
NetworkInterface::SetIPList(
const IP_ADDR_STRING& ipList)
{
LOG_FUNCTION(NetworkInterface::SetIPList);
HRESULT hr = S_OK;
// if the list already contains some entries, delete them and start over
if (!ipaddresses.empty())
{
ipaddresses.erase(ipaddresses.begin());
}
if (!subnetMasks.empty())
{
subnetMasks.erase(subnetMasks.begin());
}
const IP_ADDR_STRING* current = &ipList;
while (current)
{
// IP Address - convert and add
String ipAddress(current->IpAddress.String);
DWORD newAddress = StringToIPAddress(ipAddress);
ASSERT(newAddress != INADDR_NONE);
// StringToIPAddress returns an address of 1.2.3.4 as 04030201. The UI
// controls return the same address as 01020304. So to make
// things consistent convert to the UI way
ipaddresses.push_back(ConvertIPAddressOrder(newAddress));
// also add it to the string list
ipaddressStringList.push_back(ipAddress);
LOG(String::format(
L"Adding address: %1",
ipAddress.c_str()));
// Subnet Mask - convert and add
String subnetMask(current->IpMask.String);
DWORD newMask = StringToIPAddress(subnetMask);
// NTBUG#NTRAID-561163-2002/03/19-JeffJon
// Do not assert the subnet mask is not INADDR_NONE because
// if there is an RRAS connection then the subnet mask may be
// 255.255.255.255 which is the same as INADDR_NONE. The appropriate
// fix would be to use WSAStringToAddress inside StringToIPAddress
// but that would require rearchitecting all of CYS since IP addresses
// have a different structure for use to with that API
// ASSERT(newMask != INADDR_NONE);
// StringToIPAddress returns an address of 1.2.3.4 as 04030201. The UI
// controls return the same address as 01020304. So to make
// things consistent convert to the UI way
subnetMasks.push_back(ConvertIPAddressOrder(newMask));
// also add it to the string list
subnetMaskStringList.push_back(subnetMask);
LOG(String::format(
L"Adding subnet: %1",
subnetMask.c_str()));
current = current->Next;
}
LOG_HRESULT(hr);
return hr;
}
DWORD
NetworkInterface::GetIPAddress(DWORD addressIndex) const
{
LOG_FUNCTION2(
NetworkInterface::GetIPAddress,
String::format(
L"%1!d!",
addressIndex));
ASSERT(initialized);
DWORD result = 0;
if (addressIndex < ipaddresses.size())
{
result = ipaddresses[addressIndex];
}
LOG(IPAddressToString(result));
return result;
}
String
NetworkInterface::GetStringIPAddress(DWORD addressIndex) const
{
LOG_FUNCTION2(
NetworkInterface::GetStringIPAddress,
String::format(L"%1!d!", addressIndex));
ASSERT(addressIndex < ipaddressStringList.size());
String result;
if (addressIndex < ipaddressStringList.size())
{
result = ipaddressStringList[addressIndex];
}
LOG(result);
return result;
}
DWORD
NetworkInterface::GetSubnetMask(DWORD addressIndex) const
{
LOG_FUNCTION2(
NetworkInterface::GetSubnetMask,
String::format(
L"%1!d!",
addressIndex));
ASSERT(initialized);
DWORD result = 0;
if (addressIndex < subnetMasks.size())
{
result = subnetMasks[addressIndex];
}
LOG(IPAddressToString(result));
return result;
}
String
NetworkInterface::GetStringSubnetMask(DWORD addressIndex) const
{
LOG_FUNCTION2(
NetworkInterface::GetStringSubnetMask,
String::format(L"%1!d!", addressIndex));
ASSERT(addressIndex < subnetMaskStringList.size());
String result;
if (addressIndex < subnetMaskStringList.size())
{
result = subnetMaskStringList[addressIndex];
}
LOG(result);
return result;
}
String
NetworkInterface::GetName() const
{
LOG_FUNCTION(NetworkInterface::GetName);
ASSERT(initialized);
LOG(name);
return name;
}
String
NetworkInterface::GetFriendlyName(
const String defaultName) const
{
LOG_FUNCTION(NetworkInterface::GetFriendlyName);
DWORD dwRet = 0;
HANDLE hMprConfig = 0;
static const unsigned friendlyNameLength = 128;
wchar_t wszFriendlyName[friendlyNameLength];
ZeroMemory(wszFriendlyName, sizeof(wchar_t) * friendlyNameLength);
String result;
String guidName = GetName();
dwRet = MprConfigServerConnect(0, &hMprConfig);
if (NO_ERROR == dwRet)
{
dwRet =
MprConfigGetFriendlyName(
hMprConfig,
const_cast<wchar_t*>(guidName.c_str()),
wszFriendlyName,
sizeof(wchar_t) * friendlyNameLength);
if (NO_ERROR != dwRet)
{
LOG(String::format(
L"MprConfigGetFriendlyName() failed: error = %1!x!",
dwRet));
*wszFriendlyName = 0;
}
else
{
LOG(String::format(
L"MprConfigGetFriendlyName() failed: error = 0x%1!x!",
dwRet));
}
}
else
{
LOG(String::format(
L"MprConfigServerConnect() failed: error = 0x%1!x!",
dwRet));
}
MprConfigServerDisconnect(hMprConfig);
if (!*wszFriendlyName)
{
// we failed to get a friendly name, so use the default one
result = defaultName;
}
else
{
result = wszFriendlyName;
}
LOG(result);
return result;
}
HRESULT
NetworkInterface::GetNameAsGUID(GUID& guid) const
{
LOG_FUNCTION(NetworkInterface::GetNameAsGUID);
ASSERT(initialized);
LPOLESTR oleString = 0;
HRESULT hr = name.as_OLESTR(oleString);
if (SUCCEEDED(hr))
{
hr = ::CLSIDFromString(
oleString,
&guid);
ASSERT(SUCCEEDED(hr));
::CoTaskMemFree(oleString);
}
LOG_HRESULT(hr);
return hr;
}
String
NetworkInterface::GetDescription() const
{
LOG_FUNCTION(NetworkInterface::GetDescription);
ASSERT(initialized);
LOG(description);
return description;
}
UINT
NetworkInterface::GetType() const
{
LOG_FUNCTION(NetworkInterface::GetType);
UINT result = type;
LOG_ADAPTER_TYPE(result);
return result;
}
DWORD
NetworkInterface::GetIndex() const
{
LOG_FUNCTION(NetworkInterface::GetIndex);
DWORD result = index;
LOG(String::format(L"index = %1!d!", result));
return result;
}
void
NetworkInterface::SetIPAddress(DWORD address, String addressString)
{
LOG_FUNCTION2(
NetworkInterface::SetIPAddress,
addressString);
DWORD newIPAddress = ConvertIPAddressOrder(address);
LOG(IPAddressToString(newIPAddress));
// Clear out the old values
if (!ipaddresses.empty())
{
ipaddresses.clear();
}
if (!ipaddressStringList.empty())
{
ipaddressStringList.clear();
}
// Now add the new values
ipaddresses.push_back(newIPAddress);
ipaddressStringList.push_back(addressString);
}
void
NetworkInterface::SetSubnetMask(DWORD address, String addressString)
{
LOG_FUNCTION2(
NetworkInterface::SetSubnetMask,
addressString);
LOG(IPAddressToString(address));
// Clear out the old values
if (!subnetMasks.empty())
{
subnetMasks.clear();
}
if (!subnetMaskStringList.empty())
{
subnetMaskStringList.clear();
}
// Now add the new values
subnetMasks.push_back(address);
subnetMaskStringList.push_back(addressString);
}
String
NetworkInterface::GetDNSServerString(DWORD index)
{
LOG_FUNCTION2(
NetworkInterface::GetDNSServerString,
String::format(
L"%1!d!",
index));
String dnsServer;
if (dnsServerSearchOrder.empty())
{
}
if (index < dnsServerSearchOrder.size())
{
dnsServer = dnsServerSearchOrder[index];
}
else
{
LOG(String::format(
L"Index to large for dnsServerSearchOrder vector: index = %1!d!, size = %2!d!",
index,
dnsServerSearchOrder.size()));
}
LOG(dnsServer);
return dnsServer;
}
void
NetworkInterface::GetDNSServers(IPAddressList& servers)
{
LOG_FUNCTION(NetworkInterface::GetDNSServers);
// Note: this will read the values from WMI if they
// haven't already been retrieved
String server = GetDNSServerString(0);
if (!dnsServerSearchOrder.empty())
{
for (StringVector::iterator itr = dnsServerSearchOrder.begin();
itr != dnsServerSearchOrder.end();
++itr)
{
server = *itr;
if (!server.empty())
{
DWORD newAddress = StringToIPAddress(server);
if (newAddress != INADDR_NONE)
{
// Don't add the current IP address of this server
DWORD newInorderAddress = ConvertIPAddressOrder(newAddress);
if (newInorderAddress != GetIPAddress(0))
{
LOG(String::format(
L"Adding server: %1",
IPAddressToString(newInorderAddress).c_str()));
servers.push_back(newInorderAddress);
}
}
}
}
}
}
bool
NetworkInterface::IsDHCPAvailable() const
{
LOG_FUNCTION(NetworkInterface::IsDHCPAvailable);
LOG_BOOL(dhcpServerAvailable);
return dhcpServerAvailable;
}
bool
NetworkInterface::CanDetectDHCPServer()
{
LOG_FUNCTION(NetworkInterface::CanDetectDHCPServer);
bool result = false;
do
{
if (!IsConnected())
{
// Since the NIC isn't connected there
// is no reason to make the check
break;
}
ULONG serverIPAddress = 0;
DWORD interfaceIPAddress =
ConvertIPAddressOrder(GetIPAddress(0));
// This will perform a DHCP_INFORM to try
// to detect the DHCP server, if that fails
// it will attemp a DHCP_DISCOVER to attempt
// to detect the DHCP server.
DWORD error =
AnyDHCPServerRunning(
interfaceIPAddress,
&serverIPAddress);
if (error == ERROR_SUCCESS)
{
// DHCP server found
result = true;
LOG(
String::format(
L"DHCP server found at the following IP: %1",
IPAddressToString(
ConvertIPAddressOrder(serverIPAddress)).c_str()));
}
else
{
LOG(
String::format(
L"DHCP server not found: error = 0x%1!x!",
error));
}
} while (false);
dhcpServerAvailable = result;
LOG_BOOL(result);
return result;
}
bool
NetworkInterface::IsConnected() const
{
LOG_FUNCTION(NetworkInterface::IsConnected);
bool result = true;
do
{
// Convert the name to a GUID
GUID guid;
ZeroMemory(&guid, sizeof(GUID));
HRESULT hr = GetNameAsGUID(guid);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to get name as guid: hr = 0x%1!x!",
hr));
result = false;
break;
}
// Different adapter types require different
// ways of detecting connection state
if (IsModem())
{
// We only know about modems if they are connected
result = true;
}
else
{
result = IsStandardAdapterConnected(guid);
}
} while(false);
LOG_BOOL(result);
return result;
}
bool
NetworkInterface::IsStandardAdapterConnected(const GUID& guid) const
{
LOG_FUNCTION(NetworkInterface::IsStandardAdapterConnected);
bool result = true;
do
{
// Now get the status using the GUID
NETCON_STATUS status = NCS_CONNECTED;
HRESULT hr = HrGetPnpDeviceStatus(&guid, &status);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to get device status: hr = 0x%1!x!",
hr));
result = false;
break;
}
// The status values are defined in netcon.h
LOG(String::format(
L"Device status = %1!d!",
status));
if (status == NCS_DISCONNECTED ||
status == NCS_DISCONNECTING ||
status == NCS_HARDWARE_NOT_PRESENT ||
status == NCS_HARDWARE_DISABLED ||
status == NCS_HARDWARE_MALFUNCTION ||
status == NCS_MEDIA_DISCONNECTED)
{
result = false;
}
} while (false);
LOG_BOOL(result);
return result;
}
bool
NetworkInterface::IsModem() const
{
LOG_FUNCTION(NetworkInterface::IsModem);
bool isModem = false;
switch (GetType())
{
case IF_TYPE_PPP:
isModem = true;
break;
default:
isModem = false;
}
LOG_BOOL(isModem);
return isModem;
}
DWORD
NetworkInterface::GetNextAvailableIPAddress(
DWORD startAddress,
DWORD subnetMask)
{
LOG_FUNCTION2(
NetworkInterface::GetNextAvailableIPAddress,
IPAddressToString(startAddress));
DWORD result = startAddress;
DWORD currentAddress = startAddress;
bool isIPInUse = false;
do
{
isIPInUse =
IsIPAddressInUse(
currentAddress,
subnetMask);
if (!isIPInUse)
{
break;
}
++currentAddress;
if ((currentAddress & subnetMask) != (startAddress & subnetMask))
{
// REVIEW_JEFFJON : what should the behavior be if there are
// no available addresses? Is this likely to happen?
// Since we couldn't find an available address in this subnet
// use the start address
currentAddress = startAddress;
break;
}
} while (isIPInUse);
result = currentAddress;
LOG(IPAddressToString(result));
return result;
}
bool
NetworkInterface::IsIPAddressInUse(
DWORD ipaddress,
DWORD subnetMask)
{
LOG_FUNCTION2(
NetworkInterface::IsIPAddressInUse,
IPAddressToString(ipaddress));
bool result = false;
do
{
if (!IsConnected())
{
break;
}
// Before calling SendARP we have to set a route
// on the interface for the subnet we are interested
// in so that SendARP will work even if the interface
// IP address is completely different than the IP
// address we are looking for.
SetRoute(
ipaddress,
subnetMask);
// Use the interface IP address to tell TCP which interface
// to send the ARP on
DWORD interfaceIPAddress = GetIPAddress(0);
if (interfaceIPAddress == ipaddress)
{
// If the IP address is in use by the NIC then
// of course it is in use on the network
result = true;
break;
}
// MSDN doesn't mention whether or not the buffer needs to be
// there but it appears that SendArp() is now failing if we
// don't pass it a buffer even though we ignore the macAddress.
// The size was taken from the example in MSDN.
ULONG macAddress[2];
ULONG macLength = sizeof(macAddress);
memset(macAddress, 0xff, macLength);
DWORD error =
SendARP(
ConvertIPAddressOrder(ipaddress),
ConvertIPAddressOrder(interfaceIPAddress),
macAddress,
&macLength);
LOG(
String::format(
L"SendARP returned: error = 0x%1!x!",
error));
if (error == NO_ERROR)
{
result = true;
}
// Now that we are done with the SendARP
// remove the route so that the TCP stack
// reverts to normal behavior
RemoveRoute(
ipaddress,
subnetMask);
} while (false);
LOG_BOOL(result);
return result;
}
void
NetworkInterface::SetRoute(
DWORD ipaddress,
DWORD subnetMask)
{
LOG_FUNCTION2(
NetworkInterface::SetRoute,
IPAddressToString(ipaddress));
LOG(
String::format(
L"subnetMask = %1",
IPAddressToString(subnetMask).c_str()));
MIB_IPFORWARDROW routerTableEntry;
ZeroMemory(&routerTableEntry, sizeof(MIB_IPFORWARDROW));
// Destination
routerTableEntry.dwForwardDest =
ConvertIPAddressOrder(ipaddress) & ConvertIPAddressOrder(subnetMask);
LOG(
String::format(
L"dwForwardDest = %1",
IPAddressToString(ipaddress & subnetMask).c_str()));
// net mask
routerTableEntry.dwForwardMask = ConvertIPAddressOrder(subnetMask);
LOG(
String::format(
L"dwForwardMask = %1",
IPAddressToString(subnetMask).c_str()));
// Interface index
routerTableEntry.dwForwardIfIndex = GetIndex();
LOG(
String::format(
L"dwForwardIfIndex = %1!d!",
GetIndex()));
// Gateway
routerTableEntry.dwForwardNextHop = ConvertIPAddressOrder(GetIPAddress(0));
LOG(
String::format(
L"dwForwardNextHop = %1",
IPAddressToString(GetIPAddress(0)).c_str()));
// Protocol generator (must be PROTO_IP_NETMGMT according to MSDN)
routerTableEntry.dwForwardProto = PROTO_IP_NETMGMT;
// Taking these from %sdxroot%\net\rras\cm\customactions\cmroute\cmroute.cpp
// since I can't get the API to succeed without them
routerTableEntry.dwForwardType = 3;
routerTableEntry.dwForwardAge = INFINITE;
routerTableEntry.dwForwardMetric1 = 1;
routerTableEntry.dwForwardMetric2 = 0xFFFFFFFF;
routerTableEntry.dwForwardMetric3 = 0xFFFFFFFF;
routerTableEntry.dwForwardMetric4 = 0xFFFFFFFF;
routerTableEntry.dwForwardMetric5 = 0xFFFFFFFF;
// Create the table entry
// NTRAID#NTBUG9-667088-2002/09/25-JeffJon
// Do not use CreateIpForwardEntry here because if
// RRAS is running the call will become asynchronous
// and the route may not be in the table by the time
// we call SendARP()
DWORD error =
SetIpForwardEntryToStack(&routerTableEntry);
if (error != NO_ERROR)
{
LOG(
String::format(
L"SetIpForwardEntryToStack failed: error = 0x%1!x!",
error));
}
}
void
NetworkInterface::RemoveRoute(
DWORD ipaddress,
DWORD subnetMask)
{
LOG_FUNCTION2(
NetworkInterface::RemoveRoute,
IPAddressToString(ipaddress));
LOG(
String::format(
L"subnetMask = %1",
IPAddressToString(subnetMask).c_str()));
MIB_IPFORWARDROW routerTableEntry;
ZeroMemory(&routerTableEntry, sizeof(MIB_IPFORWARDROW));
// Destination
routerTableEntry.dwForwardDest =
ConvertIPAddressOrder(ipaddress) & ConvertIPAddressOrder(subnetMask);
LOG(
String::format(
L"dwForwardDest = %1",
IPAddressToString(ipaddress & subnetMask).c_str()));
// net mask
routerTableEntry.dwForwardMask = ConvertIPAddressOrder(subnetMask);
LOG(
String::format(
L"dwForwardMask = %1",
IPAddressToString(subnetMask).c_str()));
// Interface index
routerTableEntry.dwForwardIfIndex = GetIndex();
LOG(
String::format(
L"dwForwardIfIndex = %1!d!",
GetIndex()));
// Gateway
routerTableEntry.dwForwardNextHop = ConvertIPAddressOrder(GetIPAddress(0));
LOG(
String::format(
L"dwForwardNextHop = %1",
IPAddressToString(GetIPAddress(0)).c_str()));
// Delete the table entry
DWORD error =
DeleteIpForwardEntry(&routerTableEntry);
if (error != NO_ERROR)
{
LOG(
String::format(
L"DeleteIpForwardEntry failed: error = 0x%1!x!",
error));
}
}