You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1186 lines
23 KiB
1186 lines
23 KiB
/*++
|
|
|
|
Copyright (c) 1998, Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
rmdhcp.c
|
|
|
|
Abstract:
|
|
|
|
This module contains routines for the DHCP allocator module's interface
|
|
to the IP router-manager. (See ROUTPROT.H for details).
|
|
|
|
Author:
|
|
|
|
Abolade Gbadegesin (aboladeg) 4-Mar-1998
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
COMPONENT_REFERENCE DhcpComponentReference;
|
|
|
|
PCHAR DhcpDomainName = NULL;
|
|
|
|
PIP_AUTO_DHCP_GLOBAL_INFO DhcpGlobalInfo;
|
|
|
|
CRITICAL_SECTION DhcpGlobalInfoLock;
|
|
|
|
HANDLE DhcpNotificationEvent;
|
|
|
|
ULONG DhcpProtocolStopped = 0;
|
|
|
|
const MPR_ROUTING_CHARACTERISTICS DhcpRoutingCharacteristics =
|
|
{
|
|
MS_ROUTER_VERSION,
|
|
MS_IP_DHCP_ALLOCATOR,
|
|
RF_ROUTING|RF_ADD_ALL_INTERFACES,
|
|
DhcpRmStartProtocol,
|
|
DhcpRmStartComplete,
|
|
DhcpRmStopProtocol,
|
|
DhcpRmGetGlobalInfo,
|
|
DhcpRmSetGlobalInfo,
|
|
NULL,
|
|
NULL,
|
|
DhcpRmAddInterface,
|
|
DhcpRmDeleteInterface,
|
|
DhcpRmInterfaceStatus,
|
|
DhcpRmGetInterfaceInfo,
|
|
DhcpRmSetInterfaceInfo,
|
|
DhcpRmGetEventMessage,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
DhcpRmMibCreate,
|
|
DhcpRmMibDelete,
|
|
DhcpRmMibGet,
|
|
DhcpRmMibSet,
|
|
DhcpRmMibGetFirst,
|
|
DhcpRmMibGetNext,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
IP_AUTO_DHCP_STATISTICS DhcpStatistics;
|
|
SUPPORT_FUNCTIONS DhcpSupportFunctions;
|
|
|
|
extern "C"
|
|
LPSTR WINAPI
|
|
DnsGetPrimaryDomainName_A(
|
|
VOID
|
|
);
|
|
|
|
|
|
VOID
|
|
DhcpCleanupModule(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to cleanup the DHCP module.
|
|
|
|
Arguments:
|
|
|
|
none.
|
|
|
|
Return Value:
|
|
|
|
none.
|
|
|
|
Environment:
|
|
|
|
Invoked from within a 'DllMain' routine on 'DLL_PROCESS_DETACH'.
|
|
|
|
--*/
|
|
|
|
{
|
|
DeleteCriticalSection(&DhcpGlobalInfoLock);
|
|
DhcpShutdownInterfaceManagement();
|
|
DeleteComponentReference(&DhcpComponentReference);
|
|
|
|
} // DhcpCleanupModule
|
|
|
|
|
|
VOID
|
|
DhcpCleanupProtocol(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to cleanup the DHCP protocol-component
|
|
after a 'StopProtocol'. It runs when the last reference to the
|
|
DHCP component is released. (See 'COMPREF.H').
|
|
|
|
Arguments:
|
|
|
|
none.
|
|
|
|
Return Value:
|
|
|
|
none.
|
|
|
|
Environment:
|
|
|
|
Invoked from within an arbitrary context with no locks held.
|
|
|
|
--*/
|
|
|
|
{
|
|
PROFILE("DhcpCleanupProtocol");
|
|
|
|
#if 1
|
|
if (DhcpDomainName) {
|
|
DnsFree(DhcpDomainName, DnsFreeFlat);
|
|
DhcpDomainName = NULL;
|
|
}
|
|
#else
|
|
if (DhcpDomainName) {
|
|
NH_FREE(DhcpDomainName);
|
|
DhcpDomainName = NULL;
|
|
}
|
|
#endif
|
|
if (DhcpGlobalInfo) { NH_FREE(DhcpGlobalInfo); DhcpGlobalInfo = NULL; }
|
|
InterlockedExchange(reinterpret_cast<LPLONG>(&DhcpProtocolStopped), 1);
|
|
SetEvent(DhcpNotificationEvent);
|
|
ResetComponentReference(&DhcpComponentReference);
|
|
|
|
NhStopEventLog();
|
|
} // DhcpCleanupProtocol
|
|
|
|
|
|
BOOLEAN
|
|
DhcpInitializeModule(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to initialize the DHCP module.
|
|
|
|
Arguments:
|
|
|
|
none.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN - TRUE if initialization succeeded, FALSE otherwise
|
|
|
|
Environment:
|
|
|
|
Invoked in the context of a 'DllMain' routine on 'DLL_PROCESS_ATTACH'.
|
|
|
|
--*/
|
|
|
|
{
|
|
if (InitializeComponentReference(
|
|
&DhcpComponentReference, DhcpCleanupProtocol
|
|
)) {
|
|
return FALSE;
|
|
} else if (DhcpInitializeInterfaceManagement()) {
|
|
DeleteComponentReference(&DhcpComponentReference);
|
|
return FALSE;
|
|
} else {
|
|
__try {
|
|
InitializeCriticalSection(&DhcpGlobalInfoLock);
|
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
|
DeleteComponentReference(&DhcpComponentReference);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
} // DhcpInitializeModule
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmStartProtocol(
|
|
HANDLE NotificationEvent,
|
|
PSUPPORT_FUNCTIONS SupportFunctions,
|
|
PVOID GlobalInfo,
|
|
ULONG StructureVersion,
|
|
ULONG StructureSize,
|
|
ULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to indicate the component's operation should begin.
|
|
|
|
Arguments:
|
|
|
|
NotificationEvent - event on which we notify the router-manager
|
|
about asynchronous occurrences
|
|
|
|
SupportFunctions - functions for initiating router-related operations
|
|
|
|
GlobalInfo - configuration for the component
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error = NO_ERROR;
|
|
ULONG Size;
|
|
|
|
PROFILE("DhcpRmStartProtocol");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
if (!GlobalInfo) { DEREFERENCE_DHCP_AND_RETURN(ERROR_INVALID_PARAMETER); }
|
|
|
|
NhStartEventLog();
|
|
|
|
do {
|
|
|
|
//
|
|
// Create a copy of the global configuration
|
|
//
|
|
|
|
Size =
|
|
sizeof(*DhcpGlobalInfo) +
|
|
((PIP_AUTO_DHCP_GLOBAL_INFO)GlobalInfo)->ExclusionCount *
|
|
sizeof(ULONG);
|
|
|
|
DhcpGlobalInfo =
|
|
reinterpret_cast<PIP_AUTO_DHCP_GLOBAL_INFO>(NH_ALLOCATE(Size));
|
|
|
|
if (!DhcpGlobalInfo) {
|
|
NhTrace(
|
|
TRACE_FLAG_INIT,
|
|
"DhcpRmStartProtocol: cannot allocate global info"
|
|
);
|
|
NhErrorLog(
|
|
IP_AUTO_DHCP_LOG_ALLOCATION_FAILED,
|
|
0,
|
|
"%d",
|
|
Size
|
|
);
|
|
Error = ERROR_NOT_ENOUGH_MEMORY;
|
|
break;
|
|
}
|
|
|
|
CopyMemory(DhcpGlobalInfo, GlobalInfo, Size);
|
|
|
|
//
|
|
// Save the notification event
|
|
//
|
|
|
|
DhcpNotificationEvent = NotificationEvent;
|
|
|
|
//
|
|
// Save the support functions
|
|
//
|
|
|
|
if (!SupportFunctions) {
|
|
ZeroMemory(&DhcpSupportFunctions, sizeof(DhcpSupportFunctions));
|
|
}
|
|
else {
|
|
CopyMemory(
|
|
&DhcpSupportFunctions,
|
|
SupportFunctions,
|
|
sizeof(*SupportFunctions)
|
|
);
|
|
}
|
|
|
|
InterlockedExchange(reinterpret_cast<LPLONG>(&DhcpProtocolStopped), 0);
|
|
|
|
} while (FALSE);
|
|
|
|
if (NO_ERROR != Error) {
|
|
NhStopEventLog();
|
|
}
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmStartProtocol
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmStartComplete(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked when the router has finished adding the initial
|
|
configuration
|
|
|
|
Arguments:
|
|
|
|
none.
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
return NO_ERROR;
|
|
} // DhcpRmStartComplete
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmStopProtocol(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to stop the protocol.
|
|
|
|
Arguments:
|
|
|
|
none.
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
PROFILE("DhcpStopProtocol");
|
|
|
|
//
|
|
// Reference the module to make sure it's running
|
|
//
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
//
|
|
// Drop the initial reference to cause a cleanup
|
|
//
|
|
|
|
ReleaseInitialComponentReference(&DhcpComponentReference);
|
|
|
|
return DEREFERENCE_DHCP() ? NO_ERROR : ERROR_PROTOCOL_STOP_PENDING;
|
|
|
|
} // DhcpRmStopProtocol
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmAddInterface(
|
|
PWCHAR Name,
|
|
ULONG Index,
|
|
NET_INTERFACE_TYPE Type,
|
|
ULONG MediaType,
|
|
USHORT AccessType,
|
|
USHORT ConnectionType,
|
|
PVOID InterfaceInfo,
|
|
ULONG StructureVersion,
|
|
ULONG StructureSize,
|
|
ULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to add an interface to the component.
|
|
|
|
Arguments:
|
|
|
|
Name - the name of the interface (unused)
|
|
|
|
Index - the index of the interface
|
|
|
|
Type - the type of the interface
|
|
|
|
InterfaceInfo - the configuration information for the interface
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmAddInterface");
|
|
|
|
if (Type != PERMANENT) { return NO_ERROR; }
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpCreateInterface(
|
|
Index,
|
|
Type,
|
|
(PIP_AUTO_DHCP_INTERFACE_INFO)InterfaceInfo,
|
|
NULL
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmAddInterface
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmDeleteInterface(
|
|
ULONG Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to delete an interface from the component.
|
|
|
|
Arguments:
|
|
|
|
Index - the index of the interface
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmDeleteInterface");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpDeleteInterface(
|
|
Index
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmDeleteInterface
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmGetEventMessage(
|
|
OUT ROUTING_PROTOCOL_EVENTS* Event,
|
|
OUT MESSAGE* Result
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to retrieve an event message from the component.
|
|
The only event message we generate is the 'ROUTER_STOPPED' message.
|
|
|
|
Arguments:
|
|
|
|
Event - receives the generated event
|
|
|
|
Result - receives the associated result
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
--*/
|
|
|
|
{
|
|
PROFILE("DhcpRmGetEventMessage");
|
|
|
|
if (InterlockedExchange(reinterpret_cast<LPLONG>(&DhcpProtocolStopped), 0)) {
|
|
*Event = ROUTER_STOPPED;
|
|
return NO_ERROR;
|
|
}
|
|
|
|
return ERROR_NO_MORE_ITEMS;
|
|
|
|
} // DhcpRmGetEventMessage
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmGetInterfaceInfo(
|
|
ULONG Index,
|
|
PVOID InterfaceInfo,
|
|
IN OUT PULONG InterfaceInfoSize,
|
|
IN OUT PULONG StructureVersion,
|
|
IN OUT PULONG StructureSize,
|
|
IN OUT PULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to retrieve the component's per-interface
|
|
configuration.
|
|
|
|
Arguments:
|
|
|
|
Index - the index of the interface to be queried
|
|
|
|
InterfaceInfo - receives the query results
|
|
|
|
InterfaceInfoSize - receives the amount of data retrieved
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmGetInterfaceInfo");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpQueryInterface(
|
|
Index,
|
|
(PIP_AUTO_DHCP_INTERFACE_INFO)InterfaceInfo,
|
|
InterfaceInfoSize
|
|
);
|
|
|
|
*StructureSize = *InterfaceInfoSize;
|
|
if (StructureCount) {*StructureCount = 1;}
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmGetInterfaceInfo
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmSetInterfaceInfo(
|
|
ULONG Index,
|
|
PVOID InterfaceInfo,
|
|
ULONG StructureVersion,
|
|
ULONG StructureSize,
|
|
ULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to change the component's per-interface
|
|
configuration.
|
|
|
|
Arguments:
|
|
|
|
Index - the index of the interface to be updated
|
|
|
|
InterfaceInfo - supplies the new configuration
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmSetInterfaceInfo");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpConfigureInterface(
|
|
Index,
|
|
(PIP_AUTO_DHCP_INTERFACE_INFO)InterfaceInfo
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmSetInterfaceInfo
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmInterfaceStatus(
|
|
ULONG Index,
|
|
BOOL InterfaceActive,
|
|
ULONG StatusType,
|
|
PVOID StatusInfo
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to bind/unbind, enable/disable an interface
|
|
|
|
Arguments:
|
|
|
|
Index - the interface to be bound
|
|
|
|
InterfaceActive - whether the interface is active
|
|
|
|
StatusType - type of status being changed (bind or enabled)
|
|
|
|
StatusInfo - Info pertaining to the state being changed
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 Status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error = NO_ERROR;
|
|
|
|
switch(StatusType) {
|
|
case RIS_INTERFACE_ADDRESS_CHANGE: {
|
|
PIP_ADAPTER_BINDING_INFO BindInfo =
|
|
(PIP_ADAPTER_BINDING_INFO)StatusInfo;
|
|
|
|
if (BindInfo->AddressCount) {
|
|
Error = DhcpRmBindInterface(Index, StatusInfo);
|
|
} else {
|
|
Error = DhcpRmUnbindInterface(Index);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case RIS_INTERFACE_ENABLED: {
|
|
Error = DhcpRmEnableInterface(Index);
|
|
break;
|
|
}
|
|
|
|
case RIS_INTERFACE_DISABLED: {
|
|
Error = DhcpRmDisableInterface(Index);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return Error;
|
|
|
|
} // DhcpRmInterfaceStatus
|
|
|
|
|
|
ULONG
|
|
DhcpRmBindInterface(
|
|
ULONG Index,
|
|
PVOID BindingInfo
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to bind an interface to its IP address(es).
|
|
|
|
Arguments:
|
|
|
|
Index - the interface to be bound
|
|
|
|
BindingInfo - the addressing information
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
PCHAR DomainName;
|
|
ULONG Error;
|
|
NTSTATUS status;
|
|
|
|
PROFILE("DhcpRmBindInterface");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpBindInterface(
|
|
Index,
|
|
(PIP_ADAPTER_BINDING_INFO)BindingInfo
|
|
);
|
|
|
|
//
|
|
// Re-read the domain name in case it changed
|
|
//
|
|
|
|
EnterCriticalSection(&DhcpGlobalInfoLock);
|
|
|
|
#if 1
|
|
DomainName = DnsGetPrimaryDomainName_A();
|
|
if (DomainName) {
|
|
if (DhcpDomainName && lstrcmpiA(DomainName, DhcpDomainName) == 0) {
|
|
DnsFree(DomainName, DnsFreeFlat);
|
|
} else {
|
|
if (DhcpDomainName) { DnsFree(DhcpDomainName, DnsFreeFlat); }
|
|
DhcpDomainName = DomainName;
|
|
}
|
|
}
|
|
#else
|
|
status = NhQueryDomainName(&DomainName);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
if (DhcpDomainName && lstrcmpiA(DomainName, DhcpDomainName) == 0) {
|
|
NH_FREE(DomainName);
|
|
} else {
|
|
NH_FREE(DhcpDomainName);
|
|
if (DhcpDomainName) { NH_FREE(DhcpDomainName); }
|
|
DhcpDomainName = DomainName;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
LeaveCriticalSection(&DhcpGlobalInfoLock);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmBindInterface
|
|
|
|
|
|
ULONG
|
|
DhcpRmUnbindInterface(
|
|
ULONG Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to unbind an interface from its IP address(es).
|
|
|
|
Arguments:
|
|
|
|
Index - the interface to be unbound
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmUnbindInterface");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpUnbindInterface(
|
|
Index
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmUnbindInterface
|
|
|
|
|
|
ULONG
|
|
DhcpRmEnableInterface(
|
|
ULONG Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to enable operation on an interface.
|
|
|
|
Arguments:
|
|
|
|
Index - the interface to be enabled.
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmEnableInterface");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpEnableInterface(
|
|
Index
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmEnableInterface
|
|
|
|
|
|
ULONG
|
|
DhcpRmDisableInterface(
|
|
ULONG Index
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to disable operation on an interface.
|
|
|
|
Arguments:
|
|
|
|
Index - the interface to be disabled.
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PROFILE("DhcpRmDisableInterface");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
Error =
|
|
DhcpDisableInterface(
|
|
Index
|
|
);
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
|
|
} // DhcpRmDisableInterface
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmGetGlobalInfo(
|
|
PVOID GlobalInfo,
|
|
IN OUT PULONG GlobalInfoSize,
|
|
PULONG StructureVersion,
|
|
IN OUT PULONG StructureSize,
|
|
IN OUT PULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to retrieve the configuration for the component.
|
|
|
|
Arguments:
|
|
|
|
GlobalInfo - receives the configuration
|
|
|
|
GlobalInfoSize - receives the size of the configuration
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Size;
|
|
PROFILE("DhcpRmGetGlobalInfo");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
if (!GlobalInfoSize || (*GlobalInfoSize && !GlobalInfo)) {
|
|
DEREFERENCE_DHCP_AND_RETURN(ERROR_INVALID_PARAMETER);
|
|
}
|
|
|
|
EnterCriticalSection(&DhcpGlobalInfoLock);
|
|
Size =
|
|
sizeof(*DhcpGlobalInfo) +
|
|
DhcpGlobalInfo->ExclusionCount * sizeof(ULONG);
|
|
if (*GlobalInfoSize < Size) {
|
|
LeaveCriticalSection(&DhcpGlobalInfoLock);
|
|
*StructureSize = *GlobalInfoSize = Size;
|
|
if (StructureCount) {*StructureCount = 1;}
|
|
DEREFERENCE_DHCP_AND_RETURN(ERROR_INSUFFICIENT_BUFFER);
|
|
}
|
|
CopyMemory(GlobalInfo, DhcpGlobalInfo, Size);
|
|
LeaveCriticalSection(&DhcpGlobalInfoLock);
|
|
*StructureSize = *GlobalInfoSize = Size;
|
|
if (StructureCount) {*StructureCount = 1;}
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(NO_ERROR);
|
|
|
|
} // DhcpRmGetGlobalInfo
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmSetGlobalInfo(
|
|
PVOID GlobalInfo,
|
|
ULONG StructureVersion,
|
|
ULONG StructureSize,
|
|
ULONG StructureCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is invoked to change the configuration for the component.
|
|
|
|
Arguments:
|
|
|
|
GlobalInfo - the new configuration
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code
|
|
|
|
Environment:
|
|
|
|
The routine runs in the context of an IP router-manager thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
PIP_AUTO_DHCP_GLOBAL_INFO NewInfo;
|
|
ULONG NewScope;
|
|
ULONG OldScope;
|
|
ULONG Size;
|
|
PROFILE("DhcpRmSetGlobalInfo");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
if (!GlobalInfo) { DEREFERENCE_DHCP_AND_RETURN(ERROR_INVALID_PARAMETER); }
|
|
|
|
Size =
|
|
sizeof(*DhcpGlobalInfo) +
|
|
((PIP_AUTO_DHCP_GLOBAL_INFO)GlobalInfo)->ExclusionCount * sizeof(ULONG);
|
|
NewInfo = reinterpret_cast<PIP_AUTO_DHCP_GLOBAL_INFO>(NH_ALLOCATE(Size));
|
|
if (!NewInfo) {
|
|
NhTrace(
|
|
TRACE_FLAG_INIT,
|
|
"DhcpRmSetGlobalInfo: error reallocating global info"
|
|
);
|
|
NhErrorLog(
|
|
IP_AUTO_DHCP_LOG_ALLOCATION_FAILED,
|
|
0,
|
|
"%d",
|
|
Size
|
|
);
|
|
DEREFERENCE_DHCP_AND_RETURN(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
CopyMemory(NewInfo, GlobalInfo, Size);
|
|
|
|
EnterCriticalSection(&DhcpGlobalInfoLock);
|
|
OldScope = DhcpGlobalInfo->ScopeNetwork & DhcpGlobalInfo->ScopeMask;
|
|
NH_FREE(DhcpGlobalInfo);
|
|
DhcpGlobalInfo = NewInfo;
|
|
NewScope = DhcpGlobalInfo->ScopeNetwork & DhcpGlobalInfo->ScopeMask;
|
|
LeaveCriticalSection(&DhcpGlobalInfoLock);
|
|
|
|
if (OldScope != NewScope) {
|
|
DhcpReactivateEveryInterface();
|
|
}
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(NO_ERROR);
|
|
|
|
} // DhcpRmSetGlobalInfo
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibCreate(
|
|
ULONG InputDataSize,
|
|
PVOID InputData
|
|
)
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibDelete(
|
|
ULONG InputDataSize,
|
|
PVOID InputData
|
|
)
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibGet(
|
|
ULONG InputDataSize,
|
|
PVOID InputData,
|
|
OUT PULONG OutputDataSize,
|
|
OUT PVOID OutputData
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
The DHCP allocator only exposes one item to the MIB; its statistics.
|
|
|
|
Arguments:
|
|
|
|
InputDataSize - the MIB query data size
|
|
|
|
InputData - specifies the MIB object to be retrieved
|
|
|
|
OutputDataSize - the MIB response data size
|
|
|
|
OutputData - receives the MIB object retrieved
|
|
|
|
Return Value:
|
|
|
|
ULONG - Win32 status code.
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG Error;
|
|
PIP_AUTO_DHCP_MIB_QUERY Oidp;
|
|
|
|
PROFILE("DhcpRmMibGet");
|
|
|
|
REFERENCE_DHCP_OR_RETURN(ERROR_CAN_NOT_COMPLETE);
|
|
|
|
if (InputDataSize < sizeof(*Oidp) || !InputData|| !OutputDataSize) {
|
|
Error = ERROR_INVALID_PARAMETER;
|
|
}
|
|
else {
|
|
Oidp = (PIP_AUTO_DHCP_MIB_QUERY)InputData;
|
|
switch(Oidp->Oid) {
|
|
case IP_AUTO_DHCP_STATISTICS_OID: {
|
|
if (*OutputDataSize < sizeof(*Oidp) + sizeof(DhcpStatistics)) {
|
|
*OutputDataSize = sizeof(*Oidp) + sizeof(DhcpStatistics);
|
|
Error = ERROR_INSUFFICIENT_BUFFER;
|
|
}
|
|
else if (!OutputData) {
|
|
Error = ERROR_INVALID_PARAMETER;
|
|
}
|
|
else {
|
|
*OutputDataSize = sizeof(*Oidp) + sizeof(DhcpStatistics);
|
|
Oidp = (PIP_AUTO_DHCP_MIB_QUERY)OutputData;
|
|
Oidp->Oid = IP_AUTO_DHCP_STATISTICS_OID;
|
|
CopyMemory(
|
|
Oidp->Data,
|
|
&DhcpStatistics,
|
|
sizeof(DhcpStatistics)
|
|
);
|
|
Error = NO_ERROR;
|
|
}
|
|
break;
|
|
}
|
|
default: {
|
|
NhTrace(
|
|
TRACE_FLAG_DHCP,
|
|
"DhcpRmMibGet: oid %d invalid",
|
|
Oidp->Oid
|
|
);
|
|
Error = ERROR_INVALID_PARAMETER;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
DEREFERENCE_DHCP_AND_RETURN(Error);
|
|
}
|
|
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibSet(
|
|
ULONG InputDataSize,
|
|
PVOID InputData
|
|
)
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibGetFirst(
|
|
ULONG InputDataSize,
|
|
PVOID InputData,
|
|
OUT PULONG OutputDataSize,
|
|
OUT PVOID OutputData
|
|
)
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
|
|
ULONG
|
|
APIENTRY
|
|
DhcpRmMibGetNext(
|
|
ULONG InputDataSize,
|
|
PVOID InputData,
|
|
OUT PULONG OutputDataSize,
|
|
OUT PVOID OutputData
|
|
)
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
|