|
|
/*++
Copyright (c) 1999 Microsoft Corporation
Module Name:
ipconfig.c
Abstract:
updated clean version of ipconfig.
--*/
#include <precomp.h>
#include <shellapi.h>
#include <dhcpcapi.h>
#include <locale.h>
#include <dnslib.h>
int InitLogger(const char *fname); void DoneLogger(void);
DWORD WriteOutput( IN HANDLE hOut, IN LPWSTR String ) { DWORD Error = NO_ERROR, Unused; INT StringLen;
StringLen = wcslen(String); if( GetFileType(hOut) == FILE_TYPE_CHAR ) { if (StringLen < 0) { Error = ERROR_NOT_ENOUGH_MEMORY; } else { while (StringLen > 0) { if (!WriteConsoleW(hOut, String, min(StringLen, 1024), &Unused, 0 )) { Error = GetLastError(); break; } String += 1024; StringLen -= 1024; } }
if( ERROR_INVALID_HANDLE != Error ) return Error; }
//
// If it is not a console, this is good enough
//
printf("%ws", String ); return NO_ERROR; }
BOOL StringIsSame( IN LPWSTR Str1, IN LPWSTR Str2 ) { return CSTR_EQUAL == CompareString( LOCALE_USER_DEFAULT, NORM_IGNORECASE | NORM_IGNOREKANATYPE | NORM_IGNOREWIDTH, Str1, -1, Str2, -2 ); }
DWORD ParseCommandLine( IN OUT PCMD_ARGS Args, IN OUT LPWSTR *Argv, IN DWORD Argc, OUT LPWSTR *AdapterName, OUT LPWSTR *ClassId ) { Argc --; Argv ++;
if( Argc == 0 ) return NO_ERROR; if( **Argv != L'/' && **Argv != L'-' ) { return ERROR_INVALID_DATA; } (*Argv)++;
if( StringIsSame(Args->Debug, *Argv) ) { LocalFree( Args->Debug ); Args->Debug = NULL; Argv ++; Argc --;
if( **Argv != L'/' && **Argv != L'-' ) { return ERROR_INVALID_DATA; } (*Argv)++; }
if( Argc == 0 ) return NO_ERROR; if( Argc >= 2 ) (*AdapterName) = Argv[1]; if( Argc == 3 ) (*ClassId) = Argv[2];
if( **Argv == L'?' ) { return ERROR_INVALID_COMMAND_LINE; }
if( StringIsSame(Args->All, *Argv) ) { if( Argc != 1 ) return ERROR_INVALID_DATA; LocalFree( Args->All ); Args->All = NULL; return NO_ERROR; }
if( StringIsSame(Args->FlushDns, *Argv) ) { if( Argc != 1 ) return ERROR_INVALID_DATA; LocalFree( Args->FlushDns ); Args->FlushDns = NULL; return NO_ERROR; }
if( StringIsSame(Args->Register, *Argv) ) { if( Argc != 1 ) return ERROR_INVALID_DATA; LocalFree( Args->Register ); Args->Register = NULL; return NO_ERROR; }
if( StringIsSame(Args->DisplayDns, *Argv) ) { if( Argc != 1 ) return ERROR_INVALID_DATA; LocalFree( Args->DisplayDns ); Args->DisplayDns = NULL; return NO_ERROR; }
if( StringIsSame(Args->Renew, *Argv) ) { if( Argc > 2 ) return ERROR_INVALID_DATA; LocalFree( Args->Renew ); Args->Renew = NULL; return NO_ERROR; }
if( StringIsSame(Args->Release, *Argv) ) { if( Argc > 2 ) return ERROR_INVALID_DATA; LocalFree( Args->Release ); Args->Release = NULL; return NO_ERROR; }
if( StringIsSame(Args->ShowClassId, *Argv) ) { if( Argc != 2 ) return ERROR_INVALID_DATA; LocalFree( Args->ShowClassId ); Args->ShowClassId = NULL; return NO_ERROR; } if( StringIsSame(Args->SetClassId, *Argv) ) { if( Argc != 3 && Argc != 2 ) return ERROR_INVALID_DATA; LocalFree( Args->SetClassId ); Args->SetClassId = NULL; return NO_ERROR; }
return ERROR_INVALID_DATA; }
BOOL StringMatch( IN LPWSTR Str1, IN LPWSTR Str2, IN BOOL fCase ) { switch( *Str1 ){ case L'\0' : return ! *Str2 ; case L'*' : return StringMatch( Str1+1, Str2, fCase ) || *Str2 && StringMatch( Str1, Str2+1, fCase ); case L'?' : return *Str2 && StringMatch( Str1+1, Str2+1, fCase ); default : return (fCase ? *Str1 == *Str2 : toupper(*Str1) == toupper(*Str2)) && StringMatch( Str1+1, Str2+1, fCase ); } }
BOOL InterfaceMatch( IN PINTERFACE_NETWORK_INFO IfInfo, IN LPWSTR AdapterName ) { if( NULL == AdapterName ) return TRUE;
if( IfInfo->ConnectionName != NULL ) { return StringMatch( AdapterName, IfInfo->ConnectionName, FALSE ); }
return FALSE; }
DWORD DoRenew( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo, IN LPWSTR AdapterName ) { DWORD i, FindCount, Error, LastError; PINTERFACE_NETWORK_INFO IfInfo; BOOL fSucceeded;
fSucceeded = FALSE; FindCount = 0; LastError = ERROR_INTERNAL_ERROR; for( i = 0; i < NetInfo->nInterfaces; i ++ ) { IfInfo = NetInfo->IfInfo[i];
if( !InterfaceMatch(IfInfo, AdapterName) ) continue;
FindCount ++;
if( !IfInfo->EnableDhcp || 0 == IfInfo->nIpAddresses ) { if( NULL != AdapterName ) { DumpMessage( Buffer, BufSize, MSG_OPERATION_ON_NONDHCP, IfInfo->ConnectionName ); } else FindCount --; } else if( IfInfo->MediaDisconnected ) { DumpMessage( Buffer, BufSize, MSG_MEDIA_SENSE, IfInfo->ConnectionName ); } else { Error = DhcpAcquireParameters(IfInfo->DeviceGuidName); if( NO_ERROR == Error ) { fSucceeded = TRUE; } else if (ERROR_SEM_TIMEOUT == Error) { LastError = Error; DumpMessage( Buffer, BufSize, MSG_RENEW_FAILED_UNREACHABLE_DHCP, IfInfo->ConnectionName ); } else { LastError = Error; DumpMessageError( Buffer, BufSize, MSG_RENEW_FAILED, Error, IfInfo->ConnectionName ); } } }
if( fSucceeded ) return NO_ERROR; if( FindCount == 0 ) { DumpMessage( Buffer, BufSize, MSG_NO_MATCH ); }
return LastError; }
DWORD DoRelease( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo, IN LPWSTR AdapterName ) { DWORD i, FindCount, Error, LastError; PINTERFACE_NETWORK_INFO IfInfo; BOOL fSucceeded;
fSucceeded = FALSE; FindCount = 0; LastError = ERROR_INTERNAL_ERROR; for( i = 0; i < NetInfo->nInterfaces; i ++ ) { IfInfo = NetInfo->IfInfo[i];
if( !InterfaceMatch(IfInfo, AdapterName) ) continue;
FindCount ++;
if( !IfInfo->EnableDhcp || 0 == IfInfo->nIpAddresses ) { if( NULL != AdapterName ) { DumpMessage( Buffer, BufSize, MSG_OPERATION_ON_NONDHCP, IfInfo->ConnectionName ); } else FindCount --; } else if( IfInfo->MediaDisconnected ) { DumpMessage( Buffer, BufSize, MSG_MEDIA_SENSE, IfInfo->ConnectionName ); } else if( IfInfo->IpAddress[0] == INADDR_ANY ) { DumpMessage( Buffer, BufSize, MSG_ZERO_ADDRESS, IfInfo->ConnectionName ); } else { Error = DhcpReleaseParameters(IfInfo->DeviceGuidName); if( NO_ERROR == Error ) { fSucceeded = TRUE; } else { LastError = Error; DumpMessageError( Buffer, BufSize, MSG_RELEASE_FAILED, Error, IfInfo->ConnectionName ); } } }
if( fSucceeded ) return NO_ERROR; if( FindCount == 0 ) { DumpMessage( Buffer, BufSize, MSG_NO_MATCH ); }
return LastError; }
DWORD DoFlushDns( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo ) { DWORD Error; UNREFERENCED_PARAMETER( NetInfo);
Error = DnsFlushResolverCache(); if( FALSE == Error ) { Error = ERROR_FUNCTION_FAILED; DumpMessageError( Buffer, BufSize, MSG_FLUSHDNS_FAILED, Error, NULL ); } else { DumpMessage( Buffer, BufSize, MSG_FLUSHDNS_SUCCEEDED ); } return Error; }
extern DWORD DhcpStaticRefreshParams(IN LPWSTR Adapter);
DWORD DoRegisterDns( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo ) { DWORD Error; extern DWORD RegisterIPv6Dns(VOID); //
// Since IPv6 call is asynchronous, no error doesn't necessarily
// mean the success of DNS registration. So ignore it.
//
Error = RegisterIPv6Dns();
Error = DhcpStaticRefreshParams(NULL); if( NO_ERROR != Error ) { DumpMessageError( Buffer, BufSize, MSG_REGISTERDNS_FAILED, Error, NULL ); } else { DumpMessage( Buffer, BufSize, MSG_REGISTERDNS_SUCCEEDED ); }
UNREFERENCED_PARAMETER( NetInfo ); return Error; }
VOID PrintARecord ( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR IpAddr[20]; swprintf( IpAddr, L"%d.%d.%d.%d", ((BYTE *) &DnsRecord->Data.A.IpAddress)[0], ((BYTE *) &DnsRecord->Data.A.IpAddress)[1], ((BYTE *) &DnsRecord->Data.A.IpAddress)[2], ((BYTE *) &DnsRecord->Data.A.IpAddress)[3] );
DumpMessage( Buffer, BufSize, MSG_DNS_A_RECORD, IpAddr ); }
VOID PrintSOARecord ( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR SerialNo[20], Refresh[20], Retry[20]; WCHAR Expire[20], Ttl[20];
swprintf( SerialNo, L"%d", DnsRecord->Data.SOA.dwSerialNo ); swprintf( Refresh, L"%d", DnsRecord->Data.SOA.dwRefresh ); swprintf( Retry, L"%d", DnsRecord->Data.SOA.dwRetry ); swprintf( Expire, L"%d", DnsRecord->Data.SOA.dwExpire ); swprintf( Ttl, L"%d", DnsRecord->Data.SOA.dwDefaultTtl );
DumpMessage( Buffer, BufSize, MSG_DNS_SOA_RECORD, DnsRecord->Data.SOA.pNamePrimaryServer, DnsRecord->Data.SOA.pNameAdministrator, SerialNo, Refresh, Retry, Expire, Ttl ); }
VOID PrintPTRRecord ( IN ULONG MsgId, IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { DumpMessage( Buffer, BufSize, MsgId, DnsRecord->Data.PTR.pNameHost ); }
VOID PrintMXRecord ( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR Pref[20], Pad[20];
swprintf( Pref, L"%d", DnsRecord->Data.MX.wPreference ); swprintf( Pad, L"%d", DnsRecord->Data.MX.Pad );
DumpMessage( Buffer, BufSize, MSG_DNS_MX_RECORD, DnsRecord->Data.MX.pNameExchange, Pref, Pad ); }
VOID PrintAAAARecord( IN OUT PWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR stringIp6[IP6_ADDRESS_STRING_BUFFER_LENGTH]; Dns_Ip6AddressToString_W( stringIp6, & DnsRecord->Data.AAAA.Ip6Address );
DumpMessage( Buffer, BufSize, MSG_DNS_AAAA_RECORD, stringIp6 ); }
VOID PrintSRVRecord ( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR Priority[20], Weight[20], Port[20];
swprintf( Priority, L"%d", DnsRecord->Data.SRV.wPriority ); swprintf( Weight, L"%d", DnsRecord->Data.SRV.wWeight ); swprintf( Port, L"%d", DnsRecord->Data.SRV.wPort );
DumpMessage( Buffer, BufSize, MSG_DNS_SRV_RECORD, DnsRecord->Data.SRV.pNameTarget, Priority, Weight, Port ); }
VOID PrintATMARecord ( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR Type[20], Address[300];
swprintf( Type, L"%d", DnsRecord->Data.ATMA.AddressType );
if ( DnsRecord->Data.ATMA.Address && DnsRecord->Data.ATMA.AddressType == DNS_ATMA_FORMAT_E164 ) {
swprintf( Address, L"%S", DnsRecord->Data.ATMA.Address );
} else { DWORD iter;
for ( iter = 0; iter < DnsRecord->wDataLength; iter++ ) {
swprintf( &Address[iter*2], L"%02x", DnsRecord->Data.ATMA.Address[iter] ); } }
DumpMessage( Buffer, BufSize, MSG_DNS_ATMA_RECORD, Type, Address ); }
VOID PrintDnsRecord( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { WCHAR Type[20], Ttl[20], DataLen[20]; DWORD MsgId; if ( ! DnsRecord ) return;
swprintf( Type, L"%d", DnsRecord->wType ); swprintf( Ttl, L"%d", DnsRecord->dwTtl ); swprintf( DataLen, L"%d", DnsRecord->wDataLength ); DumpMessage( Buffer, BufSize, MSG_DNS_RECORD_HEADER, DnsRecord->pName, (LPWSTR)Type, (LPWSTR)Ttl, (LPWSTR)DataLen );
if ( DnsRecord->Flags.S.Section == DNSREC_QUESTION ) { MsgId = MSG_DNS_QUESTION_SECTION; } else if ( DnsRecord->Flags.S.Section == DNSREC_ANSWER ) { MsgId = MSG_DNS_ANSWER_SECTION; } else if ( DnsRecord->Flags.S.Section == DNSREC_AUTHORITY ) { MsgId = MSG_DNS_AUTHORITY_SECTION; } else { MsgId = MSG_DNS_ADDITIONAL_SECTION; }
DumpMessage( Buffer, BufSize, MsgId ); switch( DnsRecord->wType ) { case DNS_TYPE_A : PrintARecord( Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_SOA : PrintSOARecord( Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_PTR : PrintPTRRecord( MSG_DNS_PTR_RECORD, Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_NS : PrintPTRRecord( MSG_DNS_NS_RECORD, Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_CNAME : PrintPTRRecord( MSG_DNS_CNAME_RECORD, Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_MX : PrintMXRecord( Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_AAAA : PrintAAAARecord( Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_SRV : PrintSRVRecord( Buffer, BufSize, DnsRecord ); break;
case DNS_TYPE_ATMA : PrintATMARecord( Buffer, BufSize, DnsRecord ); break;
default : break; } } VOID PrintDnsRecords( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN PDNS_RECORD DnsRecord ) { PDNS_RECORD Tmp = DnsRecord;
while( Tmp ) { PrintDnsRecord( Buffer, BufSize, Tmp ); Tmp = Tmp->pNext; } }
BOOL GetDnsCachedData( IN OUT LPWSTR Buffer, IN ULONG BufSize, IN LPWSTR Name, IN WORD Type ) { PDNS_RECORD DnsRecord = NULL; DNS_STATUS DnsStatus;
DnsStatus = DnsQuery_W( Name, Type, DNS_QUERY_CACHE_ONLY, NULL, & DnsRecord, NULL );
if ( DnsStatus != NO_ERROR ) {
if ( DnsStatus == DNS_INFO_NO_RECORDS ) {
WCHAR wsTypeName[ 20 ];
Dns_WriteStringForType_W( wsTypeName, Type ); DumpMessage( Buffer, BufSize, MSG_DNS_ERR_NO_RECORDS, Name, wsTypeName );
} else if ( DnsStatus == DNS_ERROR_RCODE_NAME_ERROR ) { DumpMessage( Buffer, BufSize, MSG_DNS_ERR_NAME_ERROR, Name );
} else if ( DnsStatus == DNS_ERROR_RECORD_DOES_NOT_EXIST ) { //
// Don't print?
//
} else { DumpMessage( Buffer, BufSize, MSG_DNS_ERR_UNABLE_TO_DISPLAY, Name ); }
return FALSE; }
if ( DnsRecord ) { DumpMessage( Buffer, BufSize, MSG_DNS_RECORD_PREAMBLE, Name ); PrintDnsRecords( Buffer, BufSize, DnsRecord ); DnsRecordListFree( DnsRecord, TRUE ); }
return TRUE; }
VOID DisplayDnsCacheData( IN OUT LPWSTR Buffer, IN ULONG BufSize ) { PDNS_CACHE_TABLE pDNSCacheTable = NULL; PDNS_CACHE_TABLE pTempDNSCacheTable = NULL;
if( !DnsGetCacheDataTable( &pDNSCacheTable ) ) { DumpMessage( Buffer, BufSize, MSG_DISPLAYDNS_FAILED ); return; }
pTempDNSCacheTable = pDNSCacheTable;
while( pTempDNSCacheTable ) { PDNS_CACHE_TABLE pNext = pTempDNSCacheTable->pNext;
if( pTempDNSCacheTable->Type1 != DNS_TYPE_ZERO ) { GetDnsCachedData( Buffer, BufSize, pTempDNSCacheTable->Name, pTempDNSCacheTable->Type1 ); }
if( pTempDNSCacheTable->Type2 != DNS_TYPE_ZERO ) { GetDnsCachedData( Buffer, BufSize, pTempDNSCacheTable->Name, pTempDNSCacheTable->Type2 ); }
if( pTempDNSCacheTable->Type3 != DNS_TYPE_ZERO ) { GetDnsCachedData( Buffer, BufSize, pTempDNSCacheTable->Name, pTempDNSCacheTable->Type3 ); }
DnsFree( pTempDNSCacheTable->Name, DnsFreeFlat ); DnsFree( pTempDNSCacheTable, DnsFreeFlat );
pTempDNSCacheTable = pNext; } }
DWORD DoDisplayDns( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo ) { UNREFERENCED_PARAMETER( NetInfo);
DisplayDnsCacheData( Buffer, BufSize ); return NO_ERROR; }
DWORD DoShowClassId( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo, IN LPWSTR AdapterName ) { DWORD i, Error, Size, nClassesPrinted; PINTERFACE_NETWORK_INFO IfInfo; DHCP_CLASS_INFO *pClasses; IfInfo = NULL; for( i = 0; i < NetInfo->nInterfaces; i ++ ) { IfInfo = NetInfo->IfInfo[i];
if( !InterfaceMatch(IfInfo, AdapterName) ) continue;
if( !IfInfo->DeviceGuidName ) continue;
break; }
if( i == NetInfo->nInterfaces ) { DumpMessage( Buffer, BufSize, MSG_NO_MATCH ); return ERROR_NOT_FOUND; }
do { Size = 0; pClasses = NULL; Error = DhcpEnumClasses( 0, IfInfo->DeviceGuidName, &Size, pClasses );
if( ERROR_MORE_DATA != Error ) { if( NO_ERROR == Error ) { DumpMessage( Buffer, BufSize, MSG_NO_CLASSES, IfInfo->ConnectionName ); } else { IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("1. DhcpEnumClasses returns %d\n", Error)); } break; }
IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("DhcpEnumClasses asks for %d bytes\n", Size)); pClasses = LocalAlloc(LMEM_FIXED, Size); if( NULL == pClasses ) { Error = ERROR_NOT_ENOUGH_MEMORY; break; }
Error = DhcpEnumClasses( 0, IfInfo->DeviceGuidName, &Size, pClasses );
if( NO_ERROR != Error ) { IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("2. DhcpEnumClasses returns %d\n", Error)); LocalFree(pClasses); break; }
nClassesPrinted = Size;
if( nClassesPrinted ) { DumpMessage( Buffer, BufSize, MSG_CLASSES_LIST_HEADER, IfInfo->ConnectionName );
for( i = 0; i < nClassesPrinted ; i ++ ) { IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("%d. Class %ws Descr %ws\n", i + 1, pClasses[i].ClassName, pClasses[i].ClassDescr )); DumpMessage( Buffer, BufSize, MSG_CLASSID, pClasses[i].ClassName, pClasses[i].ClassDescr ); } } else { DumpMessage( Buffer, BufSize, MSG_NO_CLASSES, IfInfo->ConnectionName ); }
LocalFree(pClasses); } while ( 0 );
if( NO_ERROR != Error ) { DumpMessageError( Buffer, BufSize, MSG_CLASSID_FAILED, Error, IfInfo->ConnectionName ); }
return Error; }
DWORD DoSetClassId( IN LPWSTR Buffer, IN ULONG BufSize, IN PNETWORK_INFO NetInfo, IN LPWSTR AdapterName, IN LPWSTR ClassId ) { DWORD i, Error; PINTERFACE_NETWORK_INFO IfInfo; HKEY hKey; DHCP_PNP_CHANGE Changes;
IfInfo = NULL; for( i = 0; i < NetInfo->nInterfaces; i ++ ) { IfInfo = NetInfo->IfInfo[i];
if( !InterfaceMatch(IfInfo, AdapterName) ) continue;
if( !IfInfo->DeviceGuidName ) continue;
break; }
if( i == NetInfo->nInterfaces ) { DumpMessage( Buffer, BufSize, MSG_NO_MATCH ); return ERROR_NOT_FOUND; }
Error = OpenRegKey( IfInfo->DeviceGuidName, OpenTcpipKey, OpenKeyForWrite, &hKey ); if( NO_ERROR != Error ) { DumpErrorMessage( Buffer, BufSize, InterfaceOpenTcpipKeyReadFailure, Error ); return Error; }
if( NULL == ClassId ) ClassId = (LPWSTR)L""; RegSetValueExW( hKey, (LPWSTR) L"DhcpClassId", 0, REG_SZ, (LPBYTE)ClassId, sizeof(WCHAR)*(1+wcslen(ClassId)) );
RegCloseKey( hKey ); ZeroMemory(&Changes, sizeof(Changes)); Changes.ClassIdChanged = TRUE;
Error = DhcpHandlePnPEvent( 0, DHCP_CALLER_TCPUI, IfInfo->DeviceGuidName, &Changes, NULL );
if( NO_ERROR != Error ) { DumpMessageError( Buffer, BufSize, MSG_SETCLASSID_FAILED, Error, IfInfo->ConnectionName ); } else { DumpMessage( Buffer, BufSize, MSG_SETCLASSID_SUCCEEDED, IfInfo->ConnectionName ); }
return Error; } void _cdecl main(void) { WCHAR TmpBuffer[8000]; WCHAR *Buffer; DWORD LocalError, Error, InternalError, BufSize = 512; PNETWORK_INFO NetInfo; LPWSTR *Argv, CommandLine, AdapterName, ClassId; int Argc; CMD_ARGS Args; BOOL fVerbose; BOOL fDebug; BOOL fDisplay; UNREFERENCED_PARAMETER(InternalError); UNREFERENCED_PARAMETER(NetInfo);
//
// bug 211927: set the locale to the system default
//
setlocale ( LC_ALL, "" );
InitLogger("ipconfig.log"); CommandLine = GetCommandLineW(); if( NULL == CommandLine ) { DbgPrint("GetCommandLineW: 0x%x\n", GetLastError()); DoneLogger(); exit(1); }
Argv = CommandLineToArgvW(CommandLine, &Argc ); if( NULL == Argv ) { DbgPrint("CommandLineToArgvW: 0x%x\n", GetLastError()); DoneLogger(); exit(1); }
Error = GetCommandArgConstants( &Args ); if( NO_ERROR != Error ) { DbgPrint("GetCommandArgConstants: 0x%lx\n", Error ); DoneLogger(); exit(1); }
AdapterName = NULL; ClassId = NULL;
//
// The ParseCommandLine routine NULLs out the field in Args
// that was supplied by teh user... i.e ipconfig /renew would
// cause Args.Renew to be NULL on return
//
Error = ParseCommandLine( &Args, Argv, Argc, &AdapterName, &ClassId ); if( ERROR_INVALID_COMMAND_LINE == Error ) { WriteOutput( GetStdHandle( STD_OUTPUT_HANDLE), Args.Usage ); DoneLogger(); exit(1); } if( NO_ERROR != Error ) { WriteOutput( GetStdHandle( STD_OUTPUT_HANDLE), Args.UsageErr ); WriteOutput( GetStdHandle( STD_OUTPUT_HANDLE), Args.Usage ); DoneLogger(); exit(1); }
fVerbose = (Args.All == NULL); fDebug = (Args.Debug == NULL); fDisplay = FALSE; NetInfo = NULL; Error = GetNetworkInformation( &NetInfo, &InternalError );
//
// Calculate how much space it would take to display a full display..
//
do { Buffer = LocalAlloc( LPTR, sizeof(WCHAR)*BufSize ); if( Buffer == NULL ) break;
LocalError = FormatNetworkInfo( Buffer, BufSize, NetInfo, Error,InternalError,TRUE,fDebug);
LocalFree(Buffer); BufSize *= 2; } while( NO_ERROR != LocalError );
if( NULL != Buffer ) { Buffer = LocalAlloc( LPTR, sizeof(WCHAR)*BufSize ); } if( NULL == Buffer ) { Buffer = (LPWSTR)TmpBuffer; BufSize = sizeof(TmpBuffer)/sizeof(WCHAR); Error = ERROR_NOT_ENOUGH_MEMORY; }
DumpMessage( Buffer, BufSize, MSG_TITLE ); if( NO_ERROR != Error || Args.All == NULL ) { fDisplay = TRUE; } else if( Args.Renew == NULL ) { //
// Attempt to renew
//
Error = DoRenew( Buffer, BufSize, NetInfo, AdapterName); fDisplay = (NO_ERROR == Error);
if( fDisplay ) { FreeNetworkInfo( NetInfo ); Error = GetNetworkInformation( &NetInfo, &InternalError ); } } else if( Args.Release == NULL ) { //
// Attempt to renew
//
Error = DoRelease( Buffer, BufSize, NetInfo, AdapterName); fDisplay = (NO_ERROR == Error);
if( fDisplay ) { FreeNetworkInfo( NetInfo ); Error = GetNetworkInformation( &NetInfo, &InternalError ); } } else if( Args.FlushDns == NULL ) {
DoFlushDns( Buffer, BufSize, NetInfo ); } else if( Args.Register == NULL ) {
DoRegisterDns( Buffer, BufSize, NetInfo ); } else if( Args.DisplayDns == NULL ) {
DoDisplayDns( Buffer, BufSize, NetInfo ); } else if( Args.ShowClassId == NULL ) {
DoShowClassId( Buffer, BufSize, NetInfo, AdapterName ); } else if( Args.SetClassId == NULL ) {
DoSetClassId( Buffer, BufSize, NetInfo, AdapterName, ClassId ); } else { fDisplay = TRUE; } if( fDisplay ) { FormatNetworkInfo( Buffer, BufSize, NetInfo, Error, InternalError, fVerbose, fDebug ); }
Error = WriteOutput( GetStdHandle( STD_OUTPUT_HANDLE), Buffer ); if (Error != NO_ERROR) { Buffer[0] = L'\0'; DumpMessage(Buffer, BufSize, MSG_TITLE ); DumpErrorMessage(Buffer, BufSize, NoSpecificError, Error); WriteOutput( GetStdHandle( STD_OUTPUT_HANDLE ), Buffer ); } FreeNetworkInfo(NetInfo); GlobalFree( Argv ); LocalFree( Args.All ); LocalFree( Args.Renew ); LocalFree( Args.Release ); LocalFree( Args.FlushDns ); LocalFree( Args.Register ); LocalFree( Args.DisplayDns ); LocalFree( Args.ShowClassId ); LocalFree( Args.SetClassId ); LocalFree( Args.Debug ); LocalFree( Args.Usage );
DoneLogger(); exit(0); }
DWORD RegisterIPv6Dns(VOID) { DWORD dwErr = NO_ERROR; SC_HANDLE hcm = NULL; SC_HANDLE hSvc = NULL; SERVICE_STATUS status = {0}; #define SERVICE_CONTROL_6TO4_REGISER_DNS 128
do { hcm = OpenSCManager(NULL, NULL, SC_MANAGER_CONNECT); if (NULL == hcm) { dwErr = GetLastError(); IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("OpenSCManager returns 0x%lx (%d)\n", dwErr, dwErr)); break; }
hSvc = OpenService(hcm, L"6to4", SERVICE_USER_DEFINED_CONTROL); if (NULL == hSvc) { dwErr = GetLastError(); IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("OpenService returns 0x%lx (%d)\n", dwErr, dwErr)); break; }
if (!ControlService(hSvc, SERVICE_CONTROL_6TO4_REGISER_DNS, &status)) { dwErr = GetLastError(); IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("ControlService returns 0x%lx (%d)\n", dwErr, dwErr)); break; } } while (FALSE);
if (hSvc) { CloseServiceHandle(hSvc); }
if (hcm) { CloseServiceHandle(hcm); }
IPCFG_TRACE(IPCFG_TRACE_TCPIP, ("RegisterIPv6Dns returns 0x%lx (%d)\n", dwErr, dwErr)); return dwErr; }
#ifdef __IPCFG_ENABLE_LOG__
DWORD dwTraceFlag = 0xffffffffU; static FILE *hLogger = NULL; int InitLogger(const char *fname) { if (hLogger) { fclose(hLogger); } hLogger = fopen(fname, "w+"); return (hLogger == NULL)? (-1): 0; }
void DoneLogger(void) { if (hLogger) { fclose(hLogger); hLogger = NULL; } }
int TraceFunc(const char* fmt, ...) { va_list ap;
if (hLogger != NULL) { va_start(ap, fmt); vfprintf(hLogger, fmt, ap); va_end(ap); } return 0; } #else
int InitLogger(const char *fname) { UNREFERENCED_PARAMETER(fname); return 0; }
void DoneLogger(void) {}
int TraceFunc(const char* fmt, ...) { UNREFERENCED_PARAMETER(fmt); return 0; } #endif
|