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.
883 lines
20 KiB
883 lines
20 KiB
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
misc.c
|
|
|
|
Abstract:
|
|
|
|
Miscellaneous management functions
|
|
|
|
Author:
|
|
|
|
Stefan Solomon 03/13/1995
|
|
|
|
Revision History:
|
|
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
|
|
UCHAR bcastnode[6] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
|
|
|
|
VOID
|
|
SetAdapterBindingInfo(PIPX_ADAPTER_BINDING_INFO abip,
|
|
PACB acbp);
|
|
|
|
VOID
|
|
RMCreateLocalRoute(PICB icbp);
|
|
|
|
VOID
|
|
RMDeleteLocalRoute(PICB icbp);
|
|
|
|
VOID
|
|
ExternalBindInterfaceToAdapter(PICB icbp);
|
|
|
|
VOID
|
|
ExternalUnbindInterfaceFromAdapter(ULONG InterfaceIndex);
|
|
|
|
/*++
|
|
|
|
Function: GetTocEntry
|
|
Descr: Returns a pointer to the specified table of contents entry
|
|
in the interface info block.
|
|
|
|
--*/
|
|
|
|
PIPX_TOC_ENTRY
|
|
GetTocEntry(PIPX_INFO_BLOCK_HEADER InterfaceInfop,
|
|
ULONG InfoEntryType)
|
|
{
|
|
PIPX_TOC_ENTRY tocep;
|
|
UINT i;
|
|
|
|
for(i=0, tocep = InterfaceInfop->TocEntry;
|
|
i<InterfaceInfop->TocEntriesCount;
|
|
i++, tocep++) {
|
|
|
|
if(tocep->InfoType == InfoEntryType) {
|
|
|
|
return tocep;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*++
|
|
|
|
Function: GetInfoEntry
|
|
Descr: Returns a pointer to the specified info entry in the interface
|
|
control block. If more then one entries, returns a pointer to
|
|
the first one.
|
|
|
|
--*/
|
|
|
|
LPVOID
|
|
GetInfoEntry(PIPX_INFO_BLOCK_HEADER InterfaceInfop,
|
|
ULONG InfoEntryType)
|
|
{
|
|
PIPX_TOC_ENTRY tocep;
|
|
|
|
if(tocep = GetTocEntry(InterfaceInfop, InfoEntryType)) {
|
|
|
|
return((LPVOID)((PUCHAR)InterfaceInfop + tocep->Offset));
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
Function: UpdateStaticIfEntries
|
|
Descr: Compares the entries in the interface info block with the
|
|
stored static entries. Deletes the entries not present
|
|
in the interface info blolck and adds the new entries
|
|
|
|
--*/
|
|
|
|
|
|
DWORD
|
|
UpdateStaticIfEntries(
|
|
PICB icbp,
|
|
HANDLE EnumHandle, // handle for the get next enumeration
|
|
ULONG StaticEntrySize,
|
|
ULONG NewStaticEntriesCount, // number of new static entries
|
|
LPVOID NewStaticEntry, // start of the new entries array
|
|
ULONG (*GetNextStaticEntry)(HANDLE EnumHandle, LPVOID entry),
|
|
ULONG (*DeleteStaticEntry)(ULONG IfIndex, LPVOID entry),
|
|
ULONG (*CreateStaticEntry)(PICB icbp, LPVOID entry))
|
|
{
|
|
PUCHAR EntryIsNew, nsep, OldStaticEntry;
|
|
BOOL found;
|
|
UINT i;
|
|
|
|
// delete non-present entries and add the new entries
|
|
|
|
// array of flags to mark the new entries
|
|
if((EntryIsNew = GlobalAlloc(GPTR, NewStaticEntriesCount)) == NULL) {
|
|
|
|
return 1;
|
|
}
|
|
|
|
memset(EntryIsNew, 1, NewStaticEntriesCount);
|
|
|
|
if((OldStaticEntry = GlobalAlloc(GPTR, StaticEntrySize)) == NULL) {
|
|
|
|
GlobalFree(EntryIsNew);
|
|
|
|
return 1;
|
|
}
|
|
|
|
if(EnumHandle) {
|
|
|
|
while(!GetNextStaticEntry(EnumHandle, OldStaticEntry))
|
|
{
|
|
|
|
// compare it with each new static static entry until we find a match
|
|
found = FALSE;
|
|
for(i = 0, nsep = NewStaticEntry;
|
|
i<NewStaticEntriesCount;
|
|
i++, nsep+= StaticEntrySize) {
|
|
|
|
if(!memcmp(OldStaticEntry, nsep, StaticEntrySize)) {
|
|
|
|
// match - set the flags to OLD
|
|
EntryIsNew[i] = 0;
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!found) {
|
|
|
|
// non present old entry -> delete it
|
|
DeleteStaticEntry(icbp->InterfaceIndex, OldStaticEntry);
|
|
}
|
|
}
|
|
}
|
|
|
|
// all compared and old non-present ones deleted
|
|
// now, add all the new ones
|
|
|
|
for(i=0, nsep = NewStaticEntry;
|
|
i<NewStaticEntriesCount;
|
|
i++, nsep+= StaticEntrySize) {
|
|
|
|
if(EntryIsNew[i]) {
|
|
|
|
CreateStaticEntry(icbp, nsep);
|
|
}
|
|
}
|
|
|
|
GlobalFree(EntryIsNew);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*++
|
|
|
|
Function: GetInterfaceAnsiName
|
|
|
|
Arguments:
|
|
|
|
AnsiInterfaceNameBuffer - buffer of IPX_INTERFACE_ANSI_NAME_LEN
|
|
|
|
UnicodeInterfaceNameBuffer -
|
|
|
|
Descr:
|
|
|
|
--*/
|
|
|
|
VOID
|
|
GetInterfaceAnsiName(PUCHAR AnsiInterfaceNameBuffer,
|
|
PWSTR UnicodeInterfaceNameBuffer)
|
|
{
|
|
UNICODE_STRING UnicodeInterfaceName;
|
|
ANSI_STRING AnsiInterfaceName;
|
|
NTSTATUS ntStatus;
|
|
|
|
// init a unicode string with the interface name string
|
|
RtlInitUnicodeString(&UnicodeInterfaceName, UnicodeInterfaceNameBuffer);
|
|
|
|
// make the interface name unicode string an ansi string
|
|
// in an rtl allocated buffer
|
|
ntStatus = RtlUnicodeStringToAnsiString(&AnsiInterfaceName,
|
|
&UnicodeInterfaceName,
|
|
TRUE // allocate the ansi buffer
|
|
);
|
|
if (ntStatus != STATUS_SUCCESS)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// copy the interface name into the supplied buffer, up to the
|
|
// argument buffer max size
|
|
memcpy(AnsiInterfaceNameBuffer,
|
|
AnsiInterfaceName.Buffer,
|
|
min(AnsiInterfaceName.MaximumLength, IPX_INTERFACE_ANSI_NAME_LEN));
|
|
|
|
// free the rtl allocated buffer
|
|
RtlFreeAnsiString(&AnsiInterfaceName);
|
|
}
|
|
|
|
/*++
|
|
|
|
Function: BindInterfaceToAdapter
|
|
|
|
Descr: Binds the interface to the adapter in the router manager and in
|
|
all the other modules and creates a local route for the interface
|
|
in the RTM
|
|
|
|
--*/
|
|
|
|
VOID
|
|
BindInterfaceToAdapter(PICB icbp,
|
|
PACB acbp)
|
|
{
|
|
DWORD rc;
|
|
|
|
Trace(BIND_TRACE, "BindInterfaceToAdapter: Bind interface # %d to adapter # %d",
|
|
icbp->InterfaceIndex,
|
|
acbp->AdapterIndex);
|
|
|
|
if(icbp->acbp != NULL)
|
|
{
|
|
Trace(BIND_TRACE, "BindInterfaceToAdapter: interface # %d already bound !!!",
|
|
icbp->InterfaceIndex);
|
|
|
|
//SS_ASSERT(FALSE);
|
|
|
|
return;
|
|
}
|
|
|
|
// Make sure that the adapter is not currently claimed by any
|
|
// interface either.
|
|
//
|
|
if ((acbp->icbp) && (acbp->icbp->acbp == acbp))
|
|
{
|
|
Trace(
|
|
BIND_TRACE,
|
|
"BindInterfaceToAdapter: adapter # %d already bound to int # %d!!",
|
|
acbp->AdapterIndex,
|
|
acbp->icbp->InterfaceIndex);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
// internal bind the adapter control block and the interface control block
|
|
icbp->acbp = acbp;
|
|
acbp->icbp = icbp;
|
|
|
|
// if a connection was requested on this if, mark that it has been done
|
|
if(icbp->ConnectionRequestPending) {
|
|
|
|
icbp->ConnectionRequestPending = FALSE;
|
|
}
|
|
|
|
if(!icbp->InterfaceReachable) {
|
|
|
|
// we should never hit this code path in normal operation
|
|
// However, should anybody atempt and succeed a dial (manually?) on an
|
|
// interface marked unreachable, we should reset our state
|
|
|
|
// icbp->InterfaceReachable = TRUE;
|
|
|
|
if(icbp->AdminState == ADMIN_STATE_ENABLED) {
|
|
|
|
// enable all static routes for this interface
|
|
EnableStaticRoutes(icbp->InterfaceIndex);
|
|
|
|
// enable external interfaces. Implicitly, this will enable static services
|
|
// bound to this interface to be advertised
|
|
ExternalEnableInterface(icbp->InterfaceIndex);
|
|
}
|
|
}
|
|
|
|
if (icbp->AdminState==ADMIN_STATE_ENABLED) {
|
|
icbp->OperState = OPER_STATE_UP;
|
|
// create a local route entry in RTM for the connected interface
|
|
RMCreateLocalRoute(icbp);
|
|
}
|
|
|
|
|
|
ExternalBindInterfaceToAdapter(icbp);
|
|
|
|
// if the interface is a local client type (i.e. host doing manual dial from
|
|
// the local machine, try to update the internal routing table
|
|
if(icbp->MIBInterfaceType == IF_TYPE_ROUTER_WORKSTATION_DIALOUT) {
|
|
|
|
if((rc = RtProtRequestRoutesUpdate(icbp->InterfaceIndex)) == NO_ERROR) {
|
|
|
|
icbp->UpdateReq.RoutesReqStatus = UPDATE_PENDING;
|
|
}
|
|
else
|
|
{
|
|
Trace(UPDATE_TRACE, "BindInterfaceToAdapter: Routing Update is Disabled");
|
|
}
|
|
|
|
if((rc = RtProtRequestServicesUpdate(icbp->InterfaceIndex)) == NO_ERROR) {
|
|
|
|
icbp->UpdateReq.ServicesReqStatus = UPDATE_PENDING;
|
|
}
|
|
else
|
|
{
|
|
Trace(UPDATE_TRACE, "BindInterfaceToAdapter: Services Update is Disabled");
|
|
}
|
|
}
|
|
}
|
|
|
|
/*++
|
|
|
|
Function: UnbindInterfaceFromAdapter
|
|
|
|
Descr: Unbind the Rip, Sap and Forwarder interfaces with this index from
|
|
the respective adapter
|
|
|
|
--*/
|
|
|
|
VOID
|
|
UnbindInterfaceFromAdapter(PICB icbp)
|
|
{
|
|
PACB acbp;
|
|
ULONG new_if_oper_state;
|
|
|
|
|
|
acbp = icbp->acbp;
|
|
|
|
if (acbp==NULL) {
|
|
Trace(BIND_TRACE, "UnbindInterfaceFromAdapter:Interface # %d is not bound to any adapter",
|
|
icbp->InterfaceIndex);
|
|
return;
|
|
}
|
|
|
|
Trace(BIND_TRACE, "UnbindInterfaceFromAdapter: Unbind interface # %d from adapter # %d",
|
|
icbp->InterfaceIndex,
|
|
acbp->AdapterIndex);
|
|
|
|
switch(icbp->MIBInterfaceType) {
|
|
|
|
case IF_TYPE_PERSONAL_WAN_ROUTER:
|
|
case IF_TYPE_WAN_WORKSTATION:
|
|
case IF_TYPE_WAN_ROUTER:
|
|
case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
|
|
|
|
if (icbp->AdminState==ADMIN_STATE_ENABLED) {
|
|
icbp->OperState = OPER_STATE_SLEEPING;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
icbp->OperState = OPER_STATE_DOWN;
|
|
break;
|
|
|
|
}
|
|
|
|
if (icbp->AdminState==ADMIN_STATE_ENABLED) {
|
|
|
|
// delete local route and ext unbind
|
|
RMDeleteLocalRoute(icbp);
|
|
|
|
}
|
|
|
|
ExternalUnbindInterfaceFromAdapter(icbp->InterfaceIndex);
|
|
|
|
// if there were updates going on they will get cancelled automatically
|
|
// by the respective routing protocols.
|
|
// We just have to reset the update state in the ICB
|
|
ResetUpdateRequest(icbp);
|
|
|
|
// Now we can unbind the adapter from interface
|
|
acbp->icbp = NULL;
|
|
icbp->acbp = NULL;
|
|
}
|
|
|
|
/*++
|
|
|
|
Function: GetNextInterfaceIndex
|
|
|
|
Descr: Returns the next available interface index. There are a number of
|
|
policies to consider here. The one we'll use is to keep the
|
|
interface index a small number and to return the first unused
|
|
interface index between 1 and MAX_INTERFACE_INDEX.
|
|
|
|
Note: Called with database locked
|
|
|
|
--*/
|
|
|
|
ULONG
|
|
GetNextInterfaceIndex(VOID)
|
|
{
|
|
PICB icbp;
|
|
PLIST_ENTRY lep;
|
|
ULONG i;
|
|
|
|
if((icbp = GetInterfaceByIndex(1)) == NULL) {
|
|
|
|
return 1;
|
|
}
|
|
|
|
lep = icbp->IndexListLinkage.Flink;
|
|
i = 2;
|
|
|
|
while(lep != &IndexIfList)
|
|
{
|
|
icbp = CONTAINING_RECORD(lep, ICB, IndexListLinkage);
|
|
if(i < icbp->InterfaceIndex) {
|
|
|
|
return i;
|
|
}
|
|
|
|
i = icbp->InterfaceIndex + 1;
|
|
|
|
if(i == MAX_INTERFACE_INDEX) {
|
|
|
|
// abort
|
|
SS_ASSERT(FALSE);
|
|
|
|
return i;
|
|
}
|
|
|
|
lep = icbp->IndexListLinkage.Flink;
|
|
}
|
|
|
|
SS_ASSERT(i < MAX_INTERFACE_INDEX);
|
|
|
|
return i;
|
|
}
|
|
|
|
|
|
|
|
|
|
VOID
|
|
SetAdapterBindingInfo(PIPX_ADAPTER_BINDING_INFO abip,
|
|
PACB acbp)
|
|
{
|
|
abip->AdapterIndex = acbp->AdapterIndex;
|
|
memcpy(abip->Network, acbp->AdapterInfo.Network, 4);
|
|
memcpy(abip->LocalNode, acbp->AdapterInfo.LocalNode, 6);
|
|
if(acbp->AdapterInfo.NdisMedium != NdisMediumWan) {
|
|
|
|
memcpy(abip->RemoteNode, bcastnode, 6);
|
|
}
|
|
else
|
|
{
|
|
memcpy(abip->RemoteNode, acbp->AdapterInfo.RemoteNode, 6);
|
|
}
|
|
abip->MaxPacketSize = acbp->AdapterInfo.MaxPacketSize;
|
|
abip->LinkSpeed = acbp->AdapterInfo.LinkSpeed;
|
|
}
|
|
|
|
|
|
VOID
|
|
ExternalBindInterfaceToAdapter(PICB icbp)
|
|
{
|
|
PACB acbp;
|
|
IPX_ADAPTER_BINDING_INFO abi;
|
|
|
|
acbp = icbp->acbp;
|
|
|
|
SetAdapterBindingInfo(&abi, acbp);
|
|
FwBindFwInterfaceToAdapter(icbp->InterfaceIndex, &abi);
|
|
BindRoutingProtocolsIfsToAdapter(icbp->InterfaceIndex, &abi);
|
|
}
|
|
|
|
VOID
|
|
ExternalUnbindInterfaceFromAdapter(ULONG InterfaceIndex)
|
|
{
|
|
UnbindRoutingProtocolsIfsFromAdapter(InterfaceIndex);
|
|
FwUnbindFwInterfaceFromAdapter(InterfaceIndex);
|
|
}
|
|
|
|
VOID
|
|
ExternalEnableInterface(ULONG InterfaceIndex)
|
|
{
|
|
RoutingProtocolsEnableIpxInterface(InterfaceIndex);
|
|
FwEnableFwInterface(InterfaceIndex);
|
|
}
|
|
|
|
VOID
|
|
ExternalDisableInterface(ULONG InterfaceIndex)
|
|
{
|
|
FwDisableFwInterface(InterfaceIndex);
|
|
RoutingProtocolsDisableIpxInterface(InterfaceIndex);
|
|
}
|
|
|
|
VOID
|
|
RMCreateLocalRoute(PICB icbp)
|
|
{
|
|
PADAPTER_INFO aip;
|
|
|
|
// check if a network number has been assigned to this interface
|
|
aip = &(icbp->acbp->AdapterInfo);
|
|
|
|
if(!memcmp(aip->Network, nullnet, 4)) {
|
|
|
|
// no net number
|
|
return;
|
|
}
|
|
|
|
// if the interface is a remote workstation and global wan net exists,
|
|
// we are done.
|
|
if((icbp->MIBInterfaceType == IF_TYPE_WAN_WORKSTATION) &&
|
|
EnableGlobalWanNet &&
|
|
!LanOnlyMode) {
|
|
|
|
SS_ASSERT(!memcmp(aip->Network, GlobalWanNet, 4));
|
|
return;
|
|
}
|
|
|
|
CreateLocalRoute(icbp);
|
|
}
|
|
|
|
VOID
|
|
RMDeleteLocalRoute(PICB icbp)
|
|
{
|
|
PADAPTER_INFO aip;
|
|
|
|
// check if a network number has been assigned to this interface
|
|
aip = &(icbp->acbp->AdapterInfo);
|
|
|
|
if(!memcmp(aip->Network, nullnet, 4)) {
|
|
|
|
// no net number
|
|
return;
|
|
}
|
|
|
|
// if the interface is a remote workstation and global wan net exists,
|
|
// we are done.
|
|
if((icbp->MIBInterfaceType == IF_TYPE_WAN_WORKSTATION) &&
|
|
EnableGlobalWanNet &&
|
|
!LanOnlyMode) {
|
|
|
|
SS_ASSERT(!memcmp(aip->Network, GlobalWanNet, 4));
|
|
return;
|
|
}
|
|
|
|
DeleteLocalRoute(icbp);
|
|
}
|
|
|
|
VOID
|
|
AdminEnable(PICB icbp)
|
|
{
|
|
PACB acbp;
|
|
IPX_ADAPTER_BINDING_INFO aii;
|
|
|
|
if(icbp->AdminState == ADMIN_STATE_ENABLED) {
|
|
|
|
return;
|
|
}
|
|
|
|
icbp->AdminState = ADMIN_STATE_ENABLED;
|
|
|
|
// XP 497242. AdminEnable and AdminDisable are always called with the
|
|
// database lock held. But locking is order is supposed to be from
|
|
// DDM to the router managers. So when we callback to DDM, we must
|
|
// first release our lock.
|
|
//
|
|
RELEASE_DATABASE_LOCK;
|
|
|
|
InterfaceEnabled (icbp->hDIMInterface, PID_IPX, TRUE);
|
|
|
|
ACQUIRE_DATABASE_LOCK;
|
|
|
|
if(icbp->acbp != NULL) {
|
|
|
|
// bound to adapter
|
|
icbp->OperState = OPER_STATE_UP;
|
|
|
|
RMCreateLocalRoute(icbp);
|
|
}
|
|
else {
|
|
switch(icbp->MIBInterfaceType) {
|
|
|
|
case IF_TYPE_PERSONAL_WAN_ROUTER:
|
|
case IF_TYPE_WAN_WORKSTATION:
|
|
case IF_TYPE_WAN_ROUTER:
|
|
case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
|
|
icbp->OperState = OPER_STATE_SLEEPING;
|
|
break;
|
|
default:
|
|
|
|
icbp->OperState = OPER_STATE_DOWN;
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
// if REACHABLE, resume advertising routes and services
|
|
if(icbp->InterfaceReachable) {
|
|
|
|
// enable all static routes for this interface
|
|
EnableStaticRoutes(icbp->InterfaceIndex);
|
|
|
|
// enable external interfaces. Implicitly, this will enable static services
|
|
// bound to this interface to be advertised
|
|
ExternalEnableInterface(icbp->InterfaceIndex);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
AdminDisable(PICB icbp)
|
|
{
|
|
if(icbp->AdminState == ADMIN_STATE_DISABLED) {
|
|
|
|
return;
|
|
}
|
|
|
|
icbp->AdminState = ADMIN_STATE_DISABLED;
|
|
|
|
// XP 497242. AdminEnable and AdminDisable are always called with the
|
|
// database lock held. But locking is order is supposed to be from
|
|
// DDM to the router managers. So when we callback to DDM, we must
|
|
// first release our lock.
|
|
//
|
|
RELEASE_DATABASE_LOCK;
|
|
|
|
InterfaceEnabled (icbp->hDIMInterface, PID_IPX, FALSE);
|
|
|
|
ACQUIRE_DATABASE_LOCK;
|
|
|
|
icbp->OperState = OPER_STATE_DOWN;
|
|
|
|
if(icbp->acbp != NULL)
|
|
RMDeleteLocalRoute(icbp);
|
|
|
|
// disable all static routes for this interface
|
|
DisableStaticRoutes(icbp->InterfaceIndex);
|
|
|
|
// disable external interfaces. Implicitly, static services bound to this
|
|
// interface will stop being advertised.
|
|
ExternalDisableInterface(icbp->InterfaceIndex);
|
|
}
|
|
|
|
|
|
NET_INTERFACE_TYPE
|
|
MapIpxToNetInterfaceType(PICB icbp)
|
|
{
|
|
NET_INTERFACE_TYPE NetInterfaceType;
|
|
|
|
switch(icbp->MIBInterfaceType) {
|
|
|
|
case IF_TYPE_WAN_ROUTER:
|
|
case IF_TYPE_PERSONAL_WAN_ROUTER:
|
|
|
|
NetInterfaceType = DEMAND_DIAL;
|
|
break;
|
|
|
|
case IF_TYPE_ROUTER_WORKSTATION_DIALOUT:
|
|
|
|
NetInterfaceType = LOCAL_WORKSTATION_DIAL;
|
|
break;
|
|
|
|
case IF_TYPE_WAN_WORKSTATION:
|
|
|
|
NetInterfaceType = REMOTE_WORKSTATION_DIAL;
|
|
break;
|
|
|
|
default:
|
|
|
|
NetInterfaceType = PERMANENT;
|
|
break;
|
|
}
|
|
|
|
return NetInterfaceType;
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
Function: I_SetFilters
|
|
|
|
Descr: Internal parses the traffic filter info block and sets the filter
|
|
driver info.
|
|
|
|
--*/
|
|
/*
|
|
DWORD
|
|
I_SetFilters(ULONG InterfaceIndex,
|
|
ULONG FilterMode, // inbound or outbound
|
|
LPVOID FilterInfop)
|
|
|
|
{
|
|
PIPX_TRAFFIC_FILTER_GLOBAL_INFO gip;
|
|
PIPX_TOC_ENTRY tocep;
|
|
LPVOID FilterDriverInfop;
|
|
ULONG FilterDriverInfoSize;
|
|
DWORD rc;
|
|
|
|
if(FilterInfop == NULL) {
|
|
|
|
// remove all filters
|
|
rc = SetFilters(InterfaceIndex,
|
|
FilterMode, // in or outbound,
|
|
0,
|
|
0,
|
|
NULL,
|
|
0);
|
|
|
|
return rc;
|
|
}
|
|
|
|
gip = GetInfoEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
|
|
IPX_TRAFFIC_FILTER_GLOBAL_INFO_TYPE);
|
|
|
|
if(gip == NULL) {
|
|
|
|
return ERROR_CAN_NOT_COMPLETE;
|
|
}
|
|
|
|
FilterDriverInfop = GetInfoEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
|
|
IPX_TRAFFIC_FILTER_INFO_TYPE);
|
|
|
|
if(FilterDriverInfop == NULL) {
|
|
|
|
rc = SetFilters(InterfaceIndex,
|
|
FilterMode, // in or outbound,
|
|
0, // pass or don't pass
|
|
0, // filter size
|
|
NULL,
|
|
0);
|
|
|
|
return rc;
|
|
}
|
|
|
|
tocep = GetTocEntry((PIPX_INFO_BLOCK_HEADER)FilterInfop,
|
|
IPX_TRAFFIC_FILTER_INFO_TYPE);
|
|
|
|
FilterDriverInfoSize = tocep->Count * tocep->InfoSize;
|
|
|
|
rc = SetFilters(InterfaceIndex,
|
|
FilterMode, // in or outbound,
|
|
gip->FilterAction, // pass or don't pass
|
|
tocep->InfoSize, // filter size
|
|
FilterDriverInfop,
|
|
FilterDriverInfoSize);
|
|
|
|
return rc;
|
|
}
|
|
*/
|
|
/*++
|
|
|
|
Function: I_GetFilters
|
|
|
|
Descr: Internal builds the traffic filters info block from the filter driver
|
|
information.
|
|
|
|
--*/
|
|
/*
|
|
typedef struct _FILTERS_INFO_HEADER {
|
|
|
|
IPX_INFO_BLOCK_HEADER Header;
|
|
IPX_TOC_ENTRY TocEntry;
|
|
IPX_TRAFFIC_FILTER_GLOBAL_INFO GlobalInfo;
|
|
|
|
} FILTERS_INFO_HEADER, *PFILTERS_INFO_HEADER;
|
|
|
|
DWORD
|
|
I_GetFilters(ULONG InterfaceIndex,
|
|
ULONG FilterMode,
|
|
LPVOID FilterInfop,
|
|
PULONG FilterInfoSize)
|
|
{
|
|
DWORD rc;
|
|
ULONG FilterAction;
|
|
ULONG FilterSize;
|
|
PFILTERS_INFO_HEADER fhp;
|
|
LPVOID FilterDriverInfop;
|
|
ULONG FilterDriverInfoSize = 0;
|
|
PIPX_TOC_ENTRY tocep;
|
|
|
|
if((FilterInfop == NULL) || (*FilterInfoSize == 0)) {
|
|
|
|
// we are asked for size
|
|
rc = GetFilters(InterfaceIndex,
|
|
FilterMode,
|
|
&FilterAction,
|
|
&FilterSize,
|
|
NULL,
|
|
&FilterDriverInfoSize);
|
|
|
|
if((rc != NO_ERROR) && (rc != ERROR_INSUFFICIENT_BUFFER)) {
|
|
|
|
return ERROR_CAN_NOT_COMPLETE;
|
|
}
|
|
|
|
if(FilterDriverInfoSize) {
|
|
|
|
// there are filters
|
|
*FilterInfoSize = sizeof(FILTERS_INFO_HEADER) + FilterDriverInfoSize;
|
|
return ERROR_INSUFFICIENT_BUFFER;
|
|
}
|
|
else
|
|
{
|
|
// NO filters exist
|
|
*FilterInfoSize = 0;
|
|
return NO_ERROR;
|
|
}
|
|
}
|
|
|
|
if(*FilterInfoSize <= sizeof(FILTERS_INFO_HEADER)) {
|
|
|
|
return ERROR_INSUFFICIENT_BUFFER;
|
|
}
|
|
|
|
FilterDriverInfoSize = *FilterInfoSize - sizeof(FILTERS_INFO_HEADER);
|
|
|
|
FilterDriverInfop = (LPVOID)((PUCHAR)FilterInfop + sizeof(FILTERS_INFO_HEADER));
|
|
|
|
rc = GetFilters(InterfaceIndex,
|
|
FilterMode,
|
|
&FilterAction,
|
|
&FilterSize,
|
|
FilterDriverInfop,
|
|
&FilterDriverInfoSize);
|
|
|
|
if(rc != NO_ERROR) {
|
|
|
|
if(rc == ERROR_MORE_DATA) {
|
|
|
|
*FilterInfoSize = sizeof(FILTERS_INFO_HEADER) + FilterDriverInfoSize;
|
|
return ERROR_INSUFFICIENT_BUFFER;
|
|
}
|
|
else
|
|
{
|
|
return ERROR_CAN_NOT_COMPLETE;
|
|
}
|
|
}
|
|
|
|
// got it
|
|
fhp = (PFILTERS_INFO_HEADER)FilterInfop;
|
|
fhp->Header.Version = IPX_ROUTER_VERSION_1;
|
|
fhp->Header.Size = *FilterInfoSize;
|
|
fhp->Header.TocEntriesCount = 2;
|
|
|
|
tocep = fhp->Header.TocEntry;
|
|
|
|
tocep->InfoType = IPX_TRAFFIC_FILTER_GLOBAL_INFO_TYPE;
|
|
tocep->InfoSize = sizeof(IPX_TRAFFIC_FILTER_GLOBAL_INFO);
|
|
tocep->Count = 1;
|
|
tocep->Offset = (ULONG)((PUCHAR)&(fhp->GlobalInfo) - (PUCHAR)FilterInfop);
|
|
|
|
tocep++;
|
|
|
|
tocep->InfoType = IPX_TRAFFIC_FILTER_INFO_TYPE;
|
|
tocep->InfoSize = FilterSize;
|
|
tocep->Count = FilterDriverInfoSize / FilterSize;
|
|
tocep->Offset = sizeof(FILTERS_INFO_HEADER);
|
|
|
|
fhp->GlobalInfo.FilterAction = FilterAction;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
*/
|