|
|
//============================================================================
// Copyright (c) 1995, Microsoft Corporation
//
// File: work.c
//
// History:
// Abolade Gbadegesin August 31, 1995 Created
//
// Worker function implementation
//============================================================================
#include "pchbootp.h"
#define STRSAFE_NO_DEPRECATE
#include <strsafe.h>
//----------------------------------------------------------------------------
// Function: CallbackFunctionNetworkEvents
//
// This function runs in the context of the ntdll wait thread. Using
// QueueBootpWorker ensures that the bootp dll is running
//----------------------------------------------------------------------------
VOID CallbackFunctionNetworkEvents( PVOID pvContext, BOOLEAN NotUsed ) {
HANDLE WaitHandle;
if (!ENTER_BOOTP_API()) { return; }
//
// set the handle to NULL, so that Unregister wont be called
//
WaitHandle = InterlockedExchangePointer(&ig.IG_InputEventHandle, NULL); if (WaitHandle) { UnregisterWaitEx( WaitHandle, NULL ) ; }
QueueBootpWorker(WorkerFunctionNetworkEvents, pvContext);
LEAVE_BOOTP_API(); return; }
//----------------------------------------------------------------------------
// Function: WorkerFunctionNetworkEvents
//
// This function enumerates the input events on each interface and processes
// any incoming input packets. Queued by CallbackFunctionNetworkEvents
//----------------------------------------------------------------------------
VOID WorkerFunctionNetworkEvents( PVOID pvContextNotused ) {
DWORD i, dwErr; PIF_TABLE pTable; PIPBOOTP_IF_CONFIG pic; PIPBOOTP_IF_BINDING pib; PIPBOOTP_IP_ADDRESS paddr; PIF_TABLE_ENTRY pite; PLIST_ENTRY ple, phead; WSANETWORKEVENTS wsane;
if (!ENTER_BOOTP_WORKER()) { return; }
pTable = ig.IG_IfTable;
ACQUIRE_READ_LOCK(&pTable->IT_RWL);
//
// go through the list of active interfaces
// processing sockets which are read-ready
//
phead = &pTable->IT_ListByAddress;
for (ple = phead->Flink; ple != phead; ple = ple->Flink) {
pite = CONTAINING_RECORD(ple, IF_TABLE_ENTRY, ITE_LinkByAddress);
pic = pite->ITE_Config; pib = pite->ITE_Binding; paddr = IPBOOTP_IF_ADDRESS_TABLE(pib);
for (i = 0; i < pib->IB_AddrCount; i++, paddr++) { if (pite->ITE_Sockets[i] == INVALID_SOCKET) { continue; } //
// enumerate network events to see whether
// any packets have arrived on this interface
//
dwErr = WSAEnumNetworkEvents(pite->ITE_Sockets[i], NULL, &wsane); if (dwErr != NO_ERROR) { LPSTR lpszAddr = INET_NTOA(paddr->IA_Address); TRACE3( RECEIVE, "error %d checking for input on interface %d (%s)", dwErr, pite->ITE_Index, lpszAddr ); LOGWARN1(ENUM_NETWORK_EVENTS_FAILED, lpszAddr, dwErr); continue; } //
// see if the input bit is set
//
if (!(wsane.lNetworkEvents & FD_READ)) { continue; } //
// the input flag is set, now see if there was an error
//
if (wsane.iErrorCode[FD_READ_BIT] != NO_ERROR) { LPSTR lpszAddr = INET_NTOA(paddr->IA_Address); TRACE3( RECEIVE, "error %d in input record for interface %d (%s)", wsane.iErrorCode[FD_READ_BIT], pite->ITE_Index, lpszAddr ); LOGWARN1(INPUT_RECORD_ERROR, lpszAddr, dwErr); continue; } //
// there is no error, so process the socket
//
ProcessSocket(pite, i, pTable); } }
RELEASE_READ_LOCK(&pTable->IT_RWL);
//
// register the InputEvent with the NtdllWait Thread again (only if the
// dll is not stopping). I use this model of registering the event with
// ntdll every time, to prevent the worker function from being called for
// every packet received (when packets are received at the same time).
//
if (ig.IG_Status != IPBOOTP_STATUS_STOPPING) {
if (! RegisterWaitForSingleObject( &ig.IG_InputEventHandle, ig.IG_InputEvent, CallbackFunctionNetworkEvents, NULL, //null context
INFINITE, //no timeout
(WT_EXECUTEINWAITTHREAD|WT_EXECUTEONLYONCE) )) {
dwErr = GetLastError(); TRACE1( START, "error %d returned by RegisterWaitForSingleObjectEx", dwErr ); LOGERR0(REGISTER_WAIT_FAILED, dwErr); } }
LEAVE_BOOTP_WORKER(); return; }
//----------------------------------------------------------------------------
// Function: ProcessSocket
//
// This function processes a packet on an interface, queueing
// the packet for processing by a worker function after doing some
// basic validation.
//----------------------------------------------------------------------------
DWORD ProcessSocket( PIF_TABLE_ENTRY pite, DWORD dwAddrIndex, PIF_TABLE pTable ) {
BOOL bFreePacket; WORKERFUNCTION pwf; PINPUT_CONTEXT pwc; PIPBOOTP_IF_STATS pis; PIPBOOTP_IF_CONFIG pic; PIPBOOTP_IP_ADDRESS paddr; PIPBOOTP_PACKET pibp; PLIST_ENTRY ple; DWORD dwErr, dwInputSource; SOCKADDR_IN sinInputSource; PIPBOOTP_GLOBAL_CONFIG pigc; INT iInputLength, iAddrLength; PBYTE pInputPacket;
pigc = ig.IG_Config;
pis = &pite->ITE_Stats; paddr = IPBOOTP_IF_ADDRESS_TABLE(pite->ITE_Binding) + dwAddrIndex;
//
// the descriptor for this interface is set,
// so allocate space for the packet
//
pwc = BOOTP_ALLOC(sizeof(INPUT_CONTEXT)); if (pwc == NULL) {
dwErr = ERROR_NOT_ENOUGH_MEMORY; TRACE2( RECEIVE, "error %d allocating %d bytes for incoming packet", dwErr, sizeof(INPUT_CONTEXT) ); LOGERR0(HEAP_ALLOC_FAILED, dwErr);
return dwErr; }
pInputPacket = pwc->IC_InputPacket;
dwErr = NO_ERROR; bFreePacket = TRUE;
do {
CHAR szSource[20];
//
// receive the packet
//
iAddrLength = sizeof(SOCKADDR_IN);
iInputLength = recvfrom( pite->ITE_Sockets[dwAddrIndex], pInputPacket, MAX_PACKET_SIZE, 0, (PSOCKADDR)&sinInputSource, &iAddrLength );
if (iInputLength == 0 || iInputLength == SOCKET_ERROR) {
LPSTR lpszAddr;
dwErr = WSAGetLastError(); lpszAddr = INET_NTOA(paddr->IA_Address); TRACE3( RECEIVE, "error %d receiving on interface %d (%s)", dwErr, pite->ITE_Index, lpszAddr ); LOGERR1(RECVFROM_FAILED, lpszAddr, dwErr);
InterlockedIncrement(&pis->IS_ReceiveFailures); break; }
dwInputSource = sinInputSource.sin_addr.s_addr;
//
// filter out packets we sent ourselves
//
if (GetIfByAddress(pTable, dwInputSource, NULL)) { break; }
{ PCHAR pStr1, pStr2; pStr1 = INET_NTOA(dwInputSource); if (pStr1) lstrcpy(szSource, pStr1);
pStr2 = INET_NTOA(paddr->IA_Address);
if (pStr1 && pStr2) {
TRACE4( RECEIVE, "received %d-byte packet from %s on interface %d (%s)", iInputLength, szSource, pite->ITE_Index, pStr2 ); } }
//
// cast packet as a BOOTP message
//
pibp = (PIPBOOTP_PACKET)pInputPacket;
//
// consistency check 1: length of packet must exceed the length
// of the BOOTP header
//
if (iInputLength < sizeof(IPBOOTP_PACKET)) {
LPSTR lpszAddr = INET_NTOA(paddr->IA_Address); TRACE3( RECEIVE, "minimum BOOTP data is %d bytes, dropping %d byte packet from %s", sizeof(IPBOOTP_PACKET), iInputLength, szSource ); LOGWARN2(PACKET_TOO_SMALL, lpszAddr, szSource, 0);
break; }
//
// consistency check 2: op field must be either BOOTP_REQUEST
// or BOOTP_REPLY
//
if (pibp->IP_Operation != IPBOOTP_OPERATION_REQUEST && pibp->IP_Operation != IPBOOTP_OPERATION_REPLY) {
LPSTR lpszAddr = INET_NTOA(paddr->IA_Address); TRACE2( RECEIVE, "dropping packet from %s due to unknown operation field %d", szSource, pibp->IP_Operation ); LOGWARN2(PACKET_OPCODE_INVALID, lpszAddr, szSource, 0);
break; }
//
// update statistics on incoming packets
//
switch (pibp->IP_Operation) {
case IPBOOTP_OPERATION_REQUEST: InterlockedIncrement(&pis->IS_RequestsReceived); break;
case IPBOOTP_OPERATION_REPLY: InterlockedIncrement(&pis->IS_RepliesReceived); break; }
//
// finish initializing the work context
//
pwc->IC_InterfaceIndex = pite->ITE_Index; pwc->IC_AddrIndex = dwAddrIndex; pwc->IC_InputSource = sinInputSource; pwc->IC_InputLength = iInputLength;
//
// place the packet on the receive queue
//
ACQUIRE_READ_LOCK(&ig.IG_RWL); ACQUIRE_LIST_LOCK(ig.IG_RecvQueue);
dwErr = EnqueueRecvEntry( ig.IG_RecvQueue, (DWORD)pibp->IP_Operation, (PBYTE)pwc );
RELEASE_LIST_LOCK(ig.IG_RecvQueue); RELEASE_READ_LOCK(&ig.IG_RWL);
if (dwErr != NO_ERROR) {
LPSTR lpszAddr = INET_NTOA(paddr->IA_Address); TRACE4( RECEIVE, "error %d queueing packet from %s on interface %d (%s)", dwErr, szSource, pite->ITE_Index, lpszAddr ); LOGERR2(QUEUE_PACKET_FAILED, lpszAddr, szSource, dwErr);
break; }
//
// queue the function to handle the packet
//
dwErr = QueueBootpWorker(WorkerFunctionProcessInput, NULL);
if (dwErr != NO_ERROR) {
PLIST_ENTRY phead;
TRACE2( RECEIVE, "error %d queueing packet from %s for processing", dwErr, szSource ); LOGERR0(QUEUE_WORKER_FAILED, dwErr);
ACQUIRE_LIST_LOCK(ig.IG_RecvQueue);
phead = &ig.IG_RecvQueue->LL_Head; RemoveTailList(phead); ig.IG_RecvQueueSize -= sizeof(RECV_QUEUE_ENTRY); RELEASE_LIST_LOCK(ig.IG_RecvQueue);
break; }
//
// all went well, so we let the input processor free the packet
//
bFreePacket = FALSE;
} while(FALSE);
if (bFreePacket) { BOOTP_FREE(pwc); }
return dwErr; }
//----------------------------------------------------------------------------
// Function: WorkerFunctionProcessInput
//
// This function processes an incoming packet.
//----------------------------------------------------------------------------
VOID WorkerFunctionProcessInput( PVOID pContext ) {
PINPUT_CONTEXT pwc; DWORD dwErr, dwCommand;
if (!ENTER_BOOTP_WORKER()) { return; }
TRACE0(ENTER, "entering WorkerFunctionProcessInput");
do {
ACQUIRE_LIST_LOCK(ig.IG_RecvQueue); dwErr = DequeueRecvEntry(ig.IG_RecvQueue, &dwCommand, (PBYTE *)&pwc); RELEASE_LIST_LOCK(ig.IG_RecvQueue); if (dwErr != NO_ERROR) { TRACE1( RECEIVE, "error %d dequeueing packet from receive queue", dwErr ); break; }
switch (dwCommand) {
case IPBOOTP_OPERATION_REQUEST: ProcessRequest(pwc); break;
case IPBOOTP_OPERATION_REPLY: ProcessReply(pwc); break; }
} while(FALSE);
TRACE0(LEAVE, "leaving WorkerFunctionProcessInput");
LEAVE_BOOTP_WORKER();
}
//----------------------------------------------------------------------------
// Function: ProcessRequest
//
// This function handles the processing of BOOT_REQUEST messages
//----------------------------------------------------------------------------
VOID ProcessRequest( PVOID pContext ) {
INT iErr; PIF_TABLE pTable; PINPUT_CONTEXT pwc; SOCKADDR_IN sinsrv; PIPBOOTP_PACKET pibp; PIF_TABLE_ENTRY pite; PIPBOOTP_IF_STATS pis; PIPBOOTP_IF_CONFIG pic; PIPBOOTP_IP_ADDRESS paddr; PIPBOOTP_GLOBAL_CONFIG pigc; DWORD dwErr, dwIndex, dwDhcpInformServer; PDWORD pdwAddr, pdwEnd; PDHCP_PACKET pdp;
TRACE0(ENTER, "entering ProcessRequest");
pwc = (PINPUT_CONTEXT)pContext;
pTable = ig.IG_IfTable;
ACQUIRE_READ_LOCK(&pTable->IT_RWL);
do { // error breakout loop
//
// find the interface on which the input arrived
//
dwIndex = pwc->IC_InterfaceIndex; pite = GetIfByIndex(pTable, dwIndex); if (pite == NULL) {
TRACE1( REQUEST, "processing request: interface %d not found", dwIndex );
break; }
pis = &pite->ITE_Stats; pic = pite->ITE_Config;
//
// Check if interface still bound to an IP address
//
if (pite->ITE_Binding == NULL) {
TRACE1( REQUEST, "processing request: interface %d not bound", dwIndex );
break; } paddr = IPBOOTP_IF_ADDRESS_TABLE(pite->ITE_Binding) + pwc->IC_AddrIndex; //
// if we are not configured to relay, do nothing
//
if (pic->IC_RelayMode == IPBOOTP_RELAY_DISABLED) { break; }
pibp = (PIPBOOTP_PACKET)pwc->IC_InputPacket;
//
// check the hop-count field to see if it is over the max hop-count
// configured for this interface
//
if (pibp->IP_HopCount > IPBOOTP_MAX_HOP_COUNT || pibp->IP_HopCount > pic->IC_MaxHopCount) {
//
// discard and log
//
CHAR szHops[12], *lpszAddr = INET_NTOA(paddr->IA_Address);
_ltoa(pibp->IP_HopCount, szHops, 10); TRACE4( REQUEST, "dropping REQUEST with hop-count %d: max hop-count is %d on interface %d (%s)", pibp->IP_HopCount, pic->IC_MaxHopCount, dwIndex, lpszAddr ); LOGWARN2(HOP_COUNT_TOO_HIGH, lpszAddr, szHops, 0);
InterlockedIncrement(&pis->IS_RequestsDiscarded); break; }
//
// check the seconds threshold to make sure it is up to the minimum
//
if (pibp->IP_SecondsSinceBoot < pic->IC_MinSecondsSinceBoot) {
//
// discard and log
//
CHAR szSecs[12], *lpszAddr = INET_NTOA(paddr->IA_Address);
_ltoa(pibp->IP_SecondsSinceBoot, szSecs, 10); TRACE3( REQUEST, "dropping REQUEST with secs-since-boot %d on interface %d (%s)", pibp->IP_SecondsSinceBoot, dwIndex, lpszAddr ); LOGINFO2(SECS_SINCE_BOOT_TOO_LOW, lpszAddr, szSecs, 0);
InterlockedIncrement(&pis->IS_RequestsDiscarded); break; }
//
// increment the hop-count
//
++pibp->IP_HopCount;
//
// fill in relay agent IP address if it is empty
//
if (pibp->IP_AgentAddress == 0) { pibp->IP_AgentAddress = paddr->IA_Address; }
//
// if a dhcp-inform server has been set,
// and this packet is a dhcp inform packet,
// we will forward it to the dhcp-inform server.
//
pdp = (PDHCP_PACKET)(pibp + 1); if (!(dwDhcpInformServer = ig.IG_DhcpInformServer) || pwc->IC_InputLength < sizeof(IPBOOTP_PACKET) + sizeof(DHCP_PACKET) + 1 || *(DWORD UNALIGNED *)pdp->Cookie != DHCP_MAGIC_COOKIE || pdp->Tag != DHCP_TAG_MESSAGE_TYPE || pdp->Length != 1 || pdp->Option[0] != DHCP_MESSAGE_INFORM ) { dwDhcpInformServer = 0; }
//
// relay the request to all configured BOOTP servers
//
ACQUIRE_READ_LOCK(&ig.IG_RWL); pigc = ig.IG_Config; if (dwDhcpInformServer) { pdwAddr = &dwDhcpInformServer; pdwEnd = pdwAddr + 1; } else { pdwAddr = (PDWORD)((PBYTE)pigc + sizeof(IPBOOTP_GLOBAL_CONFIG)); pdwEnd = pdwAddr + pigc->GC_ServerCount; }
for ( ; pdwAddr < pdwEnd; pdwAddr++) {
sinsrv.sin_family = AF_INET; sinsrv.sin_port = htons(IPBOOTP_SERVER_PORT); sinsrv.sin_addr.s_addr = *pdwAddr;
iErr = sendto( pite->ITE_Sockets[pwc->IC_AddrIndex], pwc->IC_InputPacket, pwc->IC_InputLength, 0, (PSOCKADDR)&sinsrv, sizeof(SOCKADDR_IN) );
if (iErr == SOCKET_ERROR || iErr < (INT)pwc->IC_InputLength) {
CHAR szSrv[20], *lpszAddr;
dwErr = WSAGetLastError(); if ((lpszAddr = INET_NTOA(*pdwAddr)) != NULL) { lstrcpy(szSrv, lpszAddr); lpszAddr = INET_NTOA(paddr->IA_Address); if (lpszAddr != NULL) { TRACE4( REQUEST, "error %d relaying REQUEST to server %s on interface %d (%s)", dwErr, szSrv, dwIndex, lpszAddr ); LOGERR2(RELAY_REQUEST_FAILED, lpszAddr, szSrv, dwErr); } } InterlockedIncrement(&pis->IS_SendFailures); } }
RELEASE_READ_LOCK(&ig.IG_RWL);
} while(FALSE);
RELEASE_READ_LOCK(&pTable->IT_RWL);
BOOTP_FREE(pwc);
TRACE0(LEAVE, "leaving ProcessRequest");
return; }
//----------------------------------------------------------------------------
// Function: ProcessReply
//
// This function handles the relaying of BOOT_REPLY packets
//----------------------------------------------------------------------------
VOID ProcessReply( PVOID pContext ) {
INT iErr; PIF_TABLE pTable; BOOL bArpUpdated; SOCKADDR_IN sincli; PINPUT_CONTEXT pwc; PIPBOOTP_PACKET pibp; PIPBOOTP_IP_ADDRESS paddrin, paddrout; PIPBOOTP_IF_STATS pisin, pisout; PIF_TABLE_ENTRY pitein, piteout; DWORD dwErr, dwIndex, dwAddress, dwAddrIndexOut;
TRACE0(ENTER, "entering ProcessReply");
pwc = (PINPUT_CONTEXT)pContext;
pTable = ig.IG_IfTable;
ACQUIRE_READ_LOCK(&pTable->IT_RWL);
do { // error breakout loop
//
// get the interface on which the packet was received
//
dwIndex = pwc->IC_InterfaceIndex;
pitein = GetIfByIndex(pTable, dwIndex);
if (pitein == NULL) {
TRACE1(REPLY, "processing REPLY: interface %d not found", dwIndex);
break; }
if (pitein->ITE_Binding == NULL) { TRACE1(REPLY, "processing REPLY: interface %d not bound", dwIndex); break; } paddrin = IPBOOTP_IF_ADDRESS_TABLE(pitein->ITE_Binding) + pwc->IC_AddrIndex;
//
// if we are not configured t relay on this interface, do nothing
//
if (pitein->ITE_Config->IC_RelayMode == IPBOOTP_RELAY_DISABLED) {
TRACE2( REPLY, "dropping REPLY: relaying on interface %d (%s) is disabled", pitein->ITE_Index, INET_NTOA(paddrin->IA_Address) );
break; }
pisin = &pitein->ITE_Stats;
//
// place a template over the packet, and retrieve
// the AgentAddress field; this contains the address
// of the relay agent responsible for relaying this REPLY
//
pibp = (PIPBOOTP_PACKET)pwc->IC_InputPacket; dwAddress = pibp->IP_AgentAddress; //
// see if the address in the reply matches any local interface
//
piteout = GetIfByAddress(pTable, dwAddress, &dwAddrIndexOut);
if (piteout == NULL) {
CHAR szAddress[20]; PCHAR pStr1, pStr2; pStr1 = INET_NTOA(dwAddress); if (pStr1) lstrcpy(szAddress, pStr1);
pStr2 = INET_NTOA(paddrin->IA_Address);
if (pStr1 && pStr2) { TRACE3( REPLY, "dropping REPLY packet on interface %d (%s); no interfaces have address %s", pitein->ITE_Index, pStr2, szAddress ); } InterlockedIncrement(&pisin->IS_RepliesDiscarded); break; }
if (piteout->ITE_Binding == NULL) { TRACE1(REPLY, "processing REPLY: outgoing interface %d is not bound", dwIndex); break; } paddrout = IPBOOTP_IF_ADDRESS_TABLE(piteout->ITE_Binding) + dwAddrIndexOut;
//
// only relay if relay is enabled on the outgoing interface
//
if (piteout->ITE_Config->IC_RelayMode == IPBOOTP_RELAY_DISABLED) {
TRACE2( REPLY, "dropping REPLY: relaying on interface %d (%s) is disabled", piteout->ITE_Index, INET_NTOA(paddrout->IA_Address) );
break; }
pisout = &piteout->ITE_Stats;
//
// the message must be relayed on the interface whose address
// was specifed in the packet;
//
//
// if the broadcast bit is not set and the clients IP address
// is in the packet, add an entry to the ARP cache for the client
// and then relay the packet by unicast
//
sincli.sin_family = AF_INET; sincli.sin_port = htons(IPBOOTP_CLIENT_PORT);
if ((pibp->IP_Flags & htons(IPBOOTP_FLAG_BROADCAST)) != 0 || pibp->IP_OfferedAddress == 0) {
//
// the broadcast bit is set of the offered address is 0,
// which is not an address we can add to the ARP cache;
// in this case, send by broadcast
//
bArpUpdated = FALSE; sincli.sin_addr.s_addr = INADDR_BROADCAST; } else {
//
// attempt to seed the ARP cache with the address
// offered to the client in the packet we are about
// to send to the client.
//
dwErr = UpdateArpCache( piteout->ITE_Index, pibp->IP_OfferedAddress, (PBYTE)pibp->IP_MacAddr, pibp->IP_MacAddrLength, TRUE, ig.IG_FunctionTable );
if (dwErr == NO_ERROR) {
bArpUpdated = TRUE; sincli.sin_addr.s_addr = pibp->IP_OfferedAddress; } else {
//
// okay, that didn't work,
// so fall back on broadcasting the packet
//
TRACE3( REPLY, "error %d adding entry to ARP cache on interface %d (%s)", dwErr, piteout->ITE_Index, INET_NTOA(paddrout->IA_Address) );
bArpUpdated = FALSE; sincli.sin_addr.s_addr = INADDR_BROADCAST;
InterlockedIncrement(&pisout->IS_ArpUpdateFailures); } }
//
// relay the packet
//
iErr = sendto( piteout->ITE_Sockets[dwAddrIndexOut], pwc->IC_InputPacket, pwc->IC_InputLength, 0, (PSOCKADDR)&sincli, sizeof(SOCKADDR_IN) );
if (iErr == SOCKET_ERROR || iErr < (INT)pwc->IC_InputLength) {
INT i; BYTE *pb; CHAR szCli[64], *psz, *lpszAddr, szDigits[] = "0123456789ABCDEF";
dwErr = WSAGetLastError(); lpszAddr = INET_NTOA(paddrout->IA_Address);
//
// format the client's hardware address
//
for (i = 0, psz = szCli, pb = pibp->IP_MacAddr; i < 16 && i < pibp->IP_MacAddrLength; i++, pb++) { *psz++ = szDigits[*pb / 16]; *psz++ = szDigits[*pb % 16]; *psz++ = ':'; }
(psz == szCli) ? (*psz = '\0') : (*(psz - 1) = '\0');
TRACE4( REPLY, "error %d relaying REPLY to client %s on interface %d (%s)", dwErr, szCli, dwIndex, lpszAddr ); LOGERR2(RELAY_REPLY_FAILED, lpszAddr, szCli, dwErr);
InterlockedIncrement(&pisout->IS_SendFailures); }
//
// remove the ARP entry if one was added
//
if (bArpUpdated) {
dwErr = UpdateArpCache( piteout->ITE_Index, pibp->IP_OfferedAddress, (PBYTE)pibp->IP_MacAddr, pibp->IP_MacAddrLength, FALSE, ig.IG_FunctionTable );
if (dwErr != NO_ERROR) { InterlockedIncrement(&pisout->IS_ArpUpdateFailures); } }
} while(FALSE);
RELEASE_READ_LOCK(&pTable->IT_RWL);
BOOTP_FREE(pwc);
TRACE0(LEAVE, "leaving ProcessReply");
return; }
#define ClearScreen(h) { \
DWORD _dwin,_dwout; \ COORD _c = {0, 0}; \ CONSOLE_SCREEN_BUFFER_INFO _csbi; \ GetConsoleScreenBufferInfo(h,&_csbi); \ _dwin = _csbi.dwSize.X * _csbi.dwSize.Y; \ FillConsoleOutputCharacter(h,' ',_dwin,_c,&_dwout); \ }
VOID PrintGlobalConfig( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod );
VOID PrintIfConfig( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod );
VOID PrintIfBinding( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod );
VOID PrintIfStats( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod );
#if DBG
VOID CallbackFunctionMibDisplay( PVOID pContext, BOOLEAN NotUsed ) {
// enter/leaveBootpWorker not required as timer queue is persistent
QueueBootpWorker(WorkerFunctionMibDisplay, pContext);
return; }
VOID WorkerFunctionMibDisplay( PVOID pContext ) { COORD c; INT iErr; FD_SET fdsRead; HANDLE hConsole; TIMEVAL tvTimeout; DWORD dwErr, dwTraceID; DWORD dwExactSize, dwInSize, dwOutSize; IPBOOTP_MIB_GET_INPUT_DATA imgid; PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod;
if (!ENTER_BOOTP_WORKER()) { return; }
TraceGetConsole(ig.IG_MibTraceID, &hConsole); if (hConsole == NULL) { LEAVE_BOOTP_WORKER(); return; }
ClearScreen(hConsole);
c.X = c.Y = 0;
dwInSize = sizeof(imgid); imgid.IMGID_TypeID = IPBOOTP_GLOBAL_CONFIG_ID; dwOutSize = 0; pimgod = NULL;
dwErr = MibGetFirst(dwInSize, &imgid, &dwOutSize, pimgod);
if (dwErr == ERROR_INSUFFICIENT_BUFFER) {
pimgod = BOOTP_ALLOC(dwOutSize); if (pimgod) { dwErr = MibGetFirst(dwInSize, &imgid, &dwOutSize, pimgod); } }
while (dwErr == NO_ERROR) {
switch(pimgod->IMGOD_TypeID) { case IPBOOTP_GLOBAL_CONFIG_ID: PrintGlobalConfig(hConsole, &c, &imgid, pimgod); break; case IPBOOTP_IF_CONFIG_ID: PrintIfConfig(hConsole, &c, &imgid, pimgod); break; case IPBOOTP_IF_BINDING_ID: PrintIfBinding(hConsole, &c, &imgid, pimgod); break; case IPBOOTP_IF_STATS_ID: PrintIfStats(hConsole, &c, &imgid, pimgod); break; default: break; }
//
// move to next line
//
++c.Y;
dwOutSize = 0; if (pimgod) { BOOTP_FREE(pimgod); pimgod = NULL; }
dwErr = MibGetNext(dwInSize, &imgid, &dwOutSize, pimgod);
if (dwErr == ERROR_INSUFFICIENT_BUFFER) {
pimgod = BOOTP_ALLOC(dwOutSize); if (pimgod) { dwErr = MibGetNext(dwInSize, &imgid, &dwOutSize, pimgod); } } }
if (pimgod != NULL) { BOOTP_FREE(pimgod); }
LEAVE_BOOTP_WORKER();
return; } #endif //if DBG
#define WriteLine(h,c,fmt,arg) { \
DWORD _dw; \ CHAR _sz[200]; \ _dw = StringCchPrintf(_sz, 200, fmt, arg); \ if ( SUCCEEDED(_dw) ) { \ WriteConsoleOutputCharacter(h,_sz,lstrlen(_sz),c,&_dw); \ ++(c).Y; \ } \ }
VOID PrintGlobalConfig( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod ) {
PDWORD pdwsrv, pdwsrvend; PIPBOOTP_GLOBAL_CONFIG pgc;
pgc = (PIPBOOTP_GLOBAL_CONFIG)pimgod->IMGOD_Buffer;
WriteLine( hConsole, *pc, "Logging Level: %d", pgc->GC_LoggingLevel ); WriteLine( hConsole, *pc, "Max Receive Queue Size: %d", pgc->GC_MaxRecvQueueSize ); WriteLine( hConsole, *pc, "BOOTP Server Count: %d", pgc->GC_ServerCount ); pdwsrv = (PDWORD)(pgc + 1); pdwsrvend = pdwsrv + pgc->GC_ServerCount; for ( ; pdwsrv < pdwsrvend; pdwsrv++) { WriteLine( hConsole, *pc, "BOOTP Server: %s", INET_NTOA(*pdwsrv) ); }
pimgid->IMGID_TypeID = IPBOOTP_GLOBAL_CONFIG_ID; }
VOID PrintIfConfig( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod ) {
CHAR szMode[20]; PIPBOOTP_IF_CONFIG pic;
pic = (PIPBOOTP_IF_CONFIG)pimgod->IMGOD_Buffer;
switch (pic->IC_RelayMode) { case IPBOOTP_RELAY_ENABLED: strcpy(szMode, "enabled"); break; case IPBOOTP_RELAY_DISABLED: strcpy(szMode, "disabled"); break; default: break; }
WriteLine( hConsole, *pc, "Interface Index: %d", pimgod->IMGOD_IfIndex ); WriteLine( hConsole, *pc, "Relay Mode: %s", szMode ); WriteLine( hConsole, *pc, "Max Hop Count: %d", pic->IC_MaxHopCount ); WriteLine( hConsole, *pc, "Min Seconds Since Boot: %d", pic->IC_MinSecondsSinceBoot );
pimgid->IMGID_TypeID = IPBOOTP_IF_CONFIG_ID; pimgid->IMGID_IfIndex = pimgod->IMGOD_IfIndex; }
VOID PrintIfBinding( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod ) {
DWORD i; CHAR szAddr[64]; PIPBOOTP_IF_BINDING pib; PIPBOOTP_IP_ADDRESS paddr;
pib = (PIPBOOTP_IF_BINDING)pimgod->IMGOD_Buffer; paddr = IPBOOTP_IF_ADDRESS_TABLE(pib);
WriteLine( hConsole, *pc, "Interface Index: %d", pimgod->IMGOD_IfIndex ); WriteLine( hConsole, *pc, "Address Count: %d", pib->IB_AddrCount ); for (i = 0; i < pib->IB_AddrCount; i++, paddr++) { LPSTR szTemp; szTemp = INET_NTOA(paddr->IA_Address); if (szTemp != NULL) { lstrcpy(szAddr, szTemp); lstrcat(szAddr, " - "); szTemp = INET_NTOA(paddr->IA_Netmask); if ( szTemp != NULL ) { lstrcat(szAddr, szTemp); } WriteLine( hConsole, *pc, "Address Entry: %s", szAddr ); } }
pimgid->IMGID_TypeID = IPBOOTP_IF_BINDING_ID; pimgid->IMGID_IfIndex = pimgod->IMGOD_IfIndex; }
VOID PrintIfStats( HANDLE hConsole, PCOORD pc, PIPBOOTP_MIB_GET_INPUT_DATA pimgid, PIPBOOTP_MIB_GET_OUTPUT_DATA pimgod ) {
PIPBOOTP_IF_STATS pis;
pis = (PIPBOOTP_IF_STATS)pimgod->IMGOD_Buffer;
WriteLine( hConsole, *pc, "Interface Index: %d", pimgod->IMGOD_IfIndex ); WriteLine( hConsole, *pc, "Send Failures: %d", pis->IS_SendFailures ); WriteLine( hConsole, *pc, "Receive Failures: %d", pis->IS_ReceiveFailures ); WriteLine( hConsole, *pc, "ARP Cache Update Failures: %d", pis->IS_ArpUpdateFailures ); WriteLine( hConsole, *pc, "Requests Received: %d", pis->IS_RequestsReceived ); WriteLine( hConsole, *pc, "Requests Discarded: %d", pis->IS_RequestsDiscarded ); WriteLine( hConsole, *pc, "Replies Received: %d", pis->IS_RepliesReceived ); WriteLine( hConsole, *pc, "Replies Discarded: %d", pis->IS_RepliesDiscarded );
pimgid->IMGID_TypeID = IPBOOTP_IF_STATS_ID; pimgid->IMGID_IfIndex = pimgod->IMGOD_IfIndex; }
char * myinet_ntoa( struct in_addr in ) {
char *rv;
rv = inet_ntoa(in); return rv ? rv : UNKNOWN_ADDRESS_STR;
}
|