2020-09-30 16:53:55 +02:00

872 lines
21 KiB
C

/*++
Copyright (C) 1999-2002 Microsoft Corporation
Module Name:
isdhcp.c
Abstract:
test program to see if a DHCP server is around or not.
Environment:
Win2K+
History:
Code provided by JRuan on May 8, 2002 and integrated into
CYS by JeffJon
--*/
#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include <winsock2.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <dhcpcapi.h>
#include <iprtrmib.h>
#include <iphlpapi.h>
#include <stdio.h>
#define OPTION_PAD 0
#define OPTION_HOST_NAME 12
#define OPTION_MESSAGE_TYPE 53
#define OPTION_SERVER_IDENTIFIER 54
#define OPTION_PARAMETER_REQUEST_LIST 55
#define OPTION_CLIENT_ID 61
#define OPTION_END 255
#define DHCP_CLIENT_PORT 68
#define DHCP_SERVR_PORT 67
BYTE HardwareAddress[16];
BYTE HardwareAddressLength = 6;
#define SOCKET_RECEIVE_BUFFER_SIZE 1024 * 4 // 4K max.
#define AUTH_SERVERS_MAX 64
#define SMALL_BUFFER_SIZE 32
#define ALERT_INTERVAL 5 * 60 // 5 mins
#define ALERT_MESSAGE_LENGTH 256
#define MAX_ALERT_NAMES 256
#define BOOT_REQUEST 1
#define DHCP_BROADCAST 0x8000
#define DHCP_DISCOVER_MESSAGE 1
#define DHCP_INFORM_MESSAGE 8
#define DHCP_MESSAGE_SIZE 576
#define DHCP_RECV_MESSAGE_SIZE 4096
#define BOOT_FILE_SIZE 128
#define BOOT_SERVER_SIZE 64
#define DHCP_MAGIC_COOKIE_BYTE1 99
#define DHCP_MAGIC_COOKIE_BYTE2 130
#define DHCP_MAGIC_COOKIE_BYTE3 83
#define DHCP_MAGIC_COOKIE_BYTE4 99
#include <packon.h>
typedef struct _OPTION {
BYTE OptionType;
BYTE OptionLength;
BYTE OptionValue[1];
} OPTION, *POPTION, *LPOPTION;
typedef struct _DHCP_MESSAGE {
BYTE Operation;
BYTE HardwareAddressType;
BYTE HardwareAddressLength;
BYTE HopCount;
DWORD TransactionID;
WORD SecondsSinceBoot;
WORD Reserved;
ULONG ClientIpAddress;
ULONG YourIpAddress;
ULONG BootstrapServerAddress;
ULONG RelayAgentIpAddress;
BYTE HardwareAddress[16];
BYTE HostName[ BOOT_SERVER_SIZE ];
BYTE BootFileName[BOOT_FILE_SIZE];
OPTION Option;
} DHCP_MESSAGE, *PDHCP_MESSAGE, *LPDHCP_MESSAGE;
#include <packoff.h>
LPOPTION
DhcpAppendOption(
LPOPTION Option,
BYTE OptionType,
PVOID OptionValue,
ULONG OptionLength,
LPBYTE OptionEnd
)
/*++
Routine Description:
This function writes a DHCP option to message buffer.
Arguments:
Option - A pointer to a message buffer.
OptionType - The option number to append.
OptionValue - A pointer to the option data.
OptionLength - The length, in bytes, of the option data.
OptionEnd - End of Option Buffer.
Return Value:
A pointer to the end of the appended option.
--*/
{
if (!Option)
{
return Option;
}
if ( OptionType == OPTION_END ) {
//
// we should alway have atleast one BYTE space in the buffer
// to append this option.
//
Option->OptionType = OPTION_END;
return( (LPOPTION) ((LPBYTE)(Option) + 1) );
}
if ( OptionType == OPTION_PAD ) {
//
// add this option only iff we have enough space in the buffer.
//
if(((LPBYTE)Option + 1) < (OptionEnd - 1) ) {
Option->OptionType = OPTION_PAD;
return( (LPOPTION) ((LPBYTE)(Option) + 1) );
}
return Option;
}
//
// add this option only iff we have enough space in the buffer.
//
if(((LPBYTE)Option + 2 + OptionLength) >= (OptionEnd - 1) ) {
return Option;
}
if( OptionLength <= 0xFF ) {
// simple option.. no need to use OPTION_MSFT_CONTINUED
Option->OptionType = OptionType;
Option->OptionLength = (BYTE)OptionLength;
memcpy( Option->OptionValue, OptionValue, OptionLength );
return( (LPOPTION) ((LPBYTE)(Option) + Option->OptionLength + 2) );
}
// option size is > 0xFF --> need to continue it using multiple ones..
// there are OptionLenght / 0xFF occurances using 0xFF+2 bytes + one
// using 2 + (OptionLength % 0xFF ) space..
// check to see if we have the space first..
if( 2 + (OptionLength%0xFF) + 0x101*(OptionLength/0xFF)
+ (LPBYTE)Option >= (OptionEnd - 1) ) {
return Option;
}
return Option;
}
LPBYTE
DhcpAppendMagicCookie(
LPBYTE Option,
LPBYTE OptionEnd
)
/*++
Routine Description:
This routine appends magic cookie to a DHCP message.
Arguments:
Option - A pointer to the place to append the magic cookie.
OptionEnd - End of Option buffer.
Return Value:
A pointer to the end of the appended cookie.
Note : The magic cookie is :
--------------------
| 99 | 130 | 83 | 99 |
--------------------
--*/
{
if( (Option + 4) < (OptionEnd - 1) ) {
*Option++ = (BYTE)DHCP_MAGIC_COOKIE_BYTE1;
*Option++ = (BYTE)DHCP_MAGIC_COOKIE_BYTE2;
*Option++ = (BYTE)DHCP_MAGIC_COOKIE_BYTE3;
*Option++ = (BYTE)DHCP_MAGIC_COOKIE_BYTE4;
}
return( Option );
}
LPOPTION
DhcpAppendClientIDOption(
LPOPTION Option,
BYTE ClientHWType,
LPBYTE ClientHWAddr,
BYTE ClientHWAddrLength,
LPBYTE OptionEnd
)
/*++
Routine Description:
This routine appends client ID option to a DHCP message.
History:
8/26/96 Frankbee Removed 16 byte limitation on the hardware
address
Arguments:
Option - A pointer to the place to append the option request.
ClientHWType - Client hardware type.
ClientHWAddr - Client hardware address
ClientHWAddrLength - Client hardware address length.
OptionEnd - End of Option buffer.
Return Value:
A pointer to the end of the newly appended option.
Note : The client ID option will look like as below in the message:
-----------------------------------------------------------------
| OpNum | Len | HWType | HWA1 | HWA2 | ..... | HWAn |
-----------------------------------------------------------------
--*/
{
struct _CLIENT_ID {
BYTE bHardwareAddressType;
BYTE pbHardwareAddress[1];
} *pClientID;
LPOPTION lpNewOption = 0;
pClientID = LocalAlloc(LMEM_FIXED, sizeof( struct _CLIENT_ID ) + ClientHWAddrLength);
//
// currently there is no way to indicate failure. simply return unmodified option
// list
//
if ( !pClientID )
return Option;
pClientID->bHardwareAddressType = ClientHWType;
memcpy( pClientID->pbHardwareAddress, ClientHWAddr, ClientHWAddrLength );
lpNewOption = DhcpAppendOption(
Option,
OPTION_CLIENT_ID,
(LPBYTE)pClientID,
(BYTE)(ClientHWAddrLength + sizeof(BYTE)),
OptionEnd );
LocalFree( pClientID );
return lpNewOption;
}
DWORD
OpenSocket(
SOCKET *Socket,
unsigned long IpAddress,
unsigned short Port
)
{
DWORD Error = 0;
SOCKET Sock;
DWORD OptValue = TRUE;
struct sockaddr_in SocketName;
//
// Create a socket
//
Sock = socket( PF_INET, SOCK_DGRAM, IPPROTO_UDP );
if ( Sock == INVALID_SOCKET ) {
Error = WSAGetLastError();
goto error;
}
//
// Make the socket share-able
//
Error = setsockopt(
Sock,
SOL_SOCKET,
SO_REUSEADDR,
(char*)&OptValue,
sizeof(OptValue) );
if ( Error != ERROR_SUCCESS ) {
Error = WSAGetLastError();
goto error;
}
OptValue = TRUE;
Error = setsockopt(
Sock,
SOL_SOCKET,
SO_BROADCAST,
(char*)&OptValue,
sizeof(OptValue) );
if ( Error != ERROR_SUCCESS ) {
Error = WSAGetLastError();
goto error;
}
OptValue = SOCKET_RECEIVE_BUFFER_SIZE;
Error = setsockopt(
Sock,
SOL_SOCKET,
SO_RCVBUF,
(char*)&OptValue,
sizeof(OptValue) );
if ( Error != ERROR_SUCCESS ) {
Error = WSAGetLastError();
goto error;
}
SocketName.sin_family = PF_INET;
SocketName.sin_port = Port;
SocketName.sin_addr.s_addr = IpAddress;
RtlZeroMemory( SocketName.sin_zero, 8);
//
// Bind this socket to the DHCP server port
//
Error = bind(
Sock,
(struct sockaddr FAR *)&SocketName,
sizeof( SocketName )
);
if ( Error != ERROR_SUCCESS ) {
Error = WSAGetLastError();
goto error;
}
*Socket = Sock;
Error = ERROR_SUCCESS;
error:
if( Error != ERROR_SUCCESS ) {
//
// if we aren't successful, close the socket if it is opened.
//
if( Sock != INVALID_SOCKET ) {
closesocket( Sock );
}
}
return( Error );
}
DWORD
SendInformOrDiscover(
SOCKET Sock,
ULONG uClientIp,
BYTE ucMessageType,
PBYTE pMessageBuffer,
ULONG uMessageBufferSize,
ULONG DestIp,
PULONG puXid
)
{
DWORD Error = 0;
PDHCP_MESSAGE dhcpMessage = (PDHCP_MESSAGE)pMessageBuffer;
LPOPTION option = 0;
LPBYTE OptionEnd = 0;
BYTE value = 0;
ULONG uXid = 0;
LPSTR HostName = "detective";
ULONG uNumOfRequestOptions = 0;
UCHAR ucRequestOptions[256];
struct sockaddr_in socketName;
DWORD i;
uXid = (rand() & 0xff);
uXid <<= 8;
uXid |= (rand() & 0xff);
uXid <<= 8;
uXid |= (rand() & 0xff);
uXid <<= 8;
uXid |= (rand() & 0xff);
HardwareAddressLength = 6;
for (i = 0; i < HardwareAddressLength; i++) {
HardwareAddress[i] = (BYTE)(rand() & 0xff);
}
//
// prepare message.
//
RtlZeroMemory( dhcpMessage, uMessageBufferSize );
dhcpMessage->Operation = BOOT_REQUEST;
dhcpMessage->ClientIpAddress = uClientIp;
dhcpMessage->HardwareAddressType = 1;
dhcpMessage->SecondsSinceBoot = 60; // random value ??
dhcpMessage->Reserved = htons(DHCP_BROADCAST);
dhcpMessage->TransactionID = uXid;
*puXid = uXid;
memcpy(
dhcpMessage->HardwareAddress,
HardwareAddress,
HardwareAddressLength
);
dhcpMessage->HardwareAddressLength = (BYTE)HardwareAddressLength;
option = &dhcpMessage->Option;
OptionEnd = (LPBYTE)dhcpMessage + uMessageBufferSize;
//
// always add magic cookie first
//
option = (LPOPTION) DhcpAppendMagicCookie( (LPBYTE) option, OptionEnd );
value = ucMessageType;
option = DhcpAppendOption(
option,
OPTION_MESSAGE_TYPE,
&value,
1,
OptionEnd );
//
// Add client ID Option.
//
option = DhcpAppendClientIDOption(
option,
1,
HardwareAddress,
HardwareAddressLength,
OptionEnd );
//
// add Host name and comment options.
//
option = DhcpAppendOption(
option,
OPTION_HOST_NAME,
(LPBYTE)HostName,
(BYTE)((strlen(HostName) + 1) * sizeof(CHAR)),
OptionEnd );
//
// Add requested option
//
uNumOfRequestOptions = 0;
ucRequestOptions[uNumOfRequestOptions++] = 3;
option = DhcpAppendOption(
option,
OPTION_PARAMETER_REQUEST_LIST,
ucRequestOptions,
uNumOfRequestOptions,
OptionEnd
);
//
// Add END option.
//
option = DhcpAppendOption( option, OPTION_END, NULL, 0, OptionEnd );
//
// Initialize the outgoing address.
//
socketName.sin_family = PF_INET;
socketName.sin_port = htons( DHCP_SERVR_PORT );
socketName.sin_addr.s_addr = DestIp;
for ( i = 0; i < 8 ; i++ ) {
socketName.sin_zero[i] = 0;
}
Error = sendto(
Sock,
(char*)pMessageBuffer,
(int)(((PBYTE)option) - pMessageBuffer),
0,
(struct sockaddr *)&socketName,
sizeof( struct sockaddr )
);
if ( Error == SOCKET_ERROR ) {
Error = WSAGetLastError();
return( Error );
}
return( ERROR_SUCCESS );
}
DWORD
GetSpecifiedMessage(
SOCKET Sock,
PBYTE pMessageBuffer,
ULONG uMessageBufferSize,
ULONG uXid,
ULONG uTimeout,
ULONG * pServerIpAddress
)
{
DWORD Error = 0;
fd_set readSocketSet;
struct timeval timeout;
struct sockaddr socketName;
int socketNameSize = sizeof( socketName );
PDHCP_MESSAGE dhcpMessage = (PDHCP_MESSAGE)pMessageBuffer;
time_t start_time = 0, now = 0;
PUCHAR pucOption = NULL;
ULONG uBytesRemain = 0;
ULONG uOptionSize = 0;
BOOL bWellFormedPacket = FALSE;
ULONG uRemainingTime = 0;
BOOL continueLooping = TRUE;
BOOL continueInternalLoop = TRUE;
BOOL continueInternalLoop2 = FALSE;
BYTE ReqdCookie[] = {
(BYTE)DHCP_MAGIC_COOKIE_BYTE1,
(BYTE)DHCP_MAGIC_COOKIE_BYTE2,
(BYTE)DHCP_MAGIC_COOKIE_BYTE3,
(BYTE)DHCP_MAGIC_COOKIE_BYTE4
};
time(&now);
start_time = now;
*pServerIpAddress = 0;
while (continueLooping) {
time(&now);
uRemainingTime = uTimeout - (ULONG)(now - start_time);
FD_ZERO( &readSocketSet );
// FD_SET( Sock, &readSocketSet );
// Had to inline the macro because the compiler was complaining
// about the while(0) that was present in FD_SET
do {
u_int __i;
for (__i = 0; __i < ((fd_set FAR *)(&readSocketSet))->fd_count; __i++) {
if (((fd_set FAR *)(&readSocketSet))->fd_array[__i] == (Sock)) {
continueInternalLoop2 = FALSE;
break;
}
}
if (__i == ((fd_set FAR *)(&readSocketSet))->fd_count) {
if (((fd_set FAR *)(&readSocketSet))->fd_count < FD_SETSIZE) {
((fd_set FAR *)(&readSocketSet))->fd_array[__i] = (Sock);
((fd_set FAR *)(&readSocketSet))->fd_count++;
}
}
} while(continueInternalLoop2);
timeout.tv_sec = uRemainingTime;
timeout.tv_usec = 0;
Error = select(1, &readSocketSet, NULL, NULL, &timeout);
if (Error == 0) {
Error = ERROR_SEM_TIMEOUT;
continueLooping = FALSE;
break;
}
//
// receive available message.
//
Error = recvfrom(
Sock,
(char*)pMessageBuffer,
uMessageBufferSize,
0,
&socketName,
&socketNameSize
);
if ( Error == SOCKET_ERROR ) {
Error = WSAGetLastError();
//
// Don't bail out here.
//
continue;
}
//
// Some sanity check
//
if (Error < sizeof(DHCP_MESSAGE)) {
continue;
}
if (dhcpMessage->HardwareAddressLength != HardwareAddressLength) {
continue;
}
if (memcmp(dhcpMessage->HardwareAddress, HardwareAddress, HardwareAddressLength) != 0) {
continue;
}
if (dhcpMessage->TransactionID != uXid) {
continue;
}
//
// Make sure the option part is well-formed
// +--------------+----------+----------+-------------+
// | magic cookie | Option 1 | Length 1 | Option Data 1 ...
// +--------------+----------+----------+-------------+
//
pucOption = (PUCHAR)(&dhcpMessage->Option);
uBytesRemain = Error - (ULONG)(pucOption - ((PUCHAR)dhcpMessage));
if (uBytesRemain < sizeof(ReqdCookie)) {
continue;
}
if (0 != memcmp(pucOption, ReqdCookie, sizeof(ReqdCookie))) {
continue;
}
pucOption += sizeof(ReqdCookie);
uBytesRemain -= sizeof(ReqdCookie);
bWellFormedPacket = FALSE;
while (continueInternalLoop) {
//
// Make sure pucOption[0] is readable
//
if (uBytesRemain < 1) {
continueInternalLoop = FALSE;
break;
}
if (pucOption[0] == OPTION_PAD) {
pucOption++;
uBytesRemain--;
continue;
}
if (pucOption[0] == OPTION_END) {
//
// See the OPTION_END. This is a well-formed packet
//
bWellFormedPacket = TRUE;
continueInternalLoop = FALSE;
break;
}
//
// Make sure pucOption[1] is readable
//
if (uBytesRemain < 2) {
continueInternalLoop = FALSE;
break;
}
uOptionSize = pucOption[1];
//
// Make sure there is enough bytes for the option data
//
if (uBytesRemain < uOptionSize) {
continueInternalLoop = FALSE;
break;
}
if (pucOption[0] == OPTION_SERVER_IDENTIFIER) {
if (uOptionSize != sizeof(ULONG)) {
continueInternalLoop = FALSE;
break;
}
memcpy(pServerIpAddress, pucOption + 2, sizeof(ULONG));
}
//
// Skip the option head and option data and move
// to the next option
//
uBytesRemain -= uOptionSize + 2;
pucOption += uOptionSize + 2;
}
if (bWellFormedPacket) {
Error = ERROR_SUCCESS;
continueLooping = FALSE;
break;
}
}
return( Error );
}
// This will first attempt a DHCP_INFORM to detect a DHCP server.
// If that fails it will attempt a DHCP_DISCOVER.
DWORD
AnyDHCPServerRunning(
ULONG uClientIp,
ULONG * pServerIp
)
{
CHAR MessageBuffer[DHCP_RECV_MESSAGE_SIZE];
SOCKET Sock = INVALID_SOCKET;
ULONG DestIp = htonl(INADDR_BROADCAST);
DWORD dwError = ERROR_SUCCESS;
ULONG uXid = 0;
ULONG uMessageBufferSize = sizeof(MessageBuffer);
ULONG uTimeout = 4;
int retries = 0;
if (!pServerIp)
{
dwError = ERROR_INVALID_PARAMETER;
goto error;
}
dwError = OpenSocket(
&Sock,
uClientIp,
htons(DHCP_CLIENT_PORT)
);
if( dwError != ERROR_SUCCESS ) {
goto error;
}
for (retries = 0; retries < 3; retries++) {
//
// Try inform
//
dwError =
SendInformOrDiscover(
Sock,
uClientIp,
DHCP_INFORM_MESSAGE,
(PBYTE)MessageBuffer,
uMessageBufferSize,
DestIp,
&uXid);
if (dwError != ERROR_SUCCESS) {
goto error;
}
dwError = GetSpecifiedMessage(
Sock,
(PBYTE)MessageBuffer,
uMessageBufferSize,
uXid,
uTimeout,
pServerIp
);
if (dwError != ERROR_SEM_TIMEOUT && *pServerIp != htonl(INADDR_ANY) && *pServerIp != htonl(INADDR_BROADCAST)) {
break;
}
//
// Try discover
//
dwError =
SendInformOrDiscover(
Sock,
uClientIp,
DHCP_DISCOVER_MESSAGE,
(PBYTE)MessageBuffer,
uMessageBufferSize,
DestIp,
&uXid);
if (dwError != ERROR_SUCCESS) {
goto error;
}
dwError = GetSpecifiedMessage(
Sock,
(PBYTE)MessageBuffer,
uMessageBufferSize,
uXid,
uTimeout,
pServerIp
);
if (dwError != ERROR_SEM_TIMEOUT && *pServerIp != htonl(INADDR_ANY) && *pServerIp != htonl(INADDR_BROADCAST)) {
break;
}
}
error:
if (Sock != INVALID_SOCKET) {
closesocket(Sock);
}
return dwError;
}