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.
4088 lines
98 KiB
4088 lines
98 KiB
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
wanarp\adapter.c
|
|
|
|
Abstract:
|
|
|
|
|
|
The file contains the section interface of WANARP to the TCP/IP stack
|
|
that is involved with Binding Notification and Querying/Setting
|
|
information for interfaces
|
|
|
|
Revision History:
|
|
|
|
AmritanR
|
|
|
|
--*/
|
|
|
|
#define __FILE_SIG__ ADAPTER_SIG
|
|
|
|
#include "inc.h"
|
|
#pragma hdrstop
|
|
|
|
|
|
NDIS_HANDLE g_nhUnbindContext;
|
|
NDIS_HANDLE g_nhBindContext;
|
|
NDIS_STATUS g_nsStatus = 0;
|
|
NDIS_STATUS g_nsError = 0;
|
|
|
|
BOOLEAN g_bRemovalInProgress;
|
|
WORK_QUEUE_ITEM g_wqiWorkItem;
|
|
WORK_QUEUE_ITEM g_wqiNdisWorkItem;
|
|
|
|
const GUID ServerInterfaceGuid = {0x6e06f030, 0x7526, 0x11d2, {0xba, 0xf4, 0x00, 0x60, 0x08, 0x15, 0xa4, 0xbd}};
|
|
|
|
extern
|
|
VOID
|
|
IPDelayedNdisReEnumerateBindings(
|
|
CTEEvent *Event,
|
|
PVOID Context
|
|
);
|
|
|
|
BOOLEAN
|
|
IsBindingPresent(
|
|
IN GUID *pGuid
|
|
);
|
|
|
|
VOID
|
|
WanNdisBindAdapter(
|
|
PNDIS_STATUS pnsRetStatus,
|
|
NDIS_HANDLE nhBindContext,
|
|
PNDIS_STRING nsAdapterName,
|
|
PVOID pvSystemSpecific1,
|
|
PVOID pvSystemSpecific2
|
|
)
|
|
{
|
|
RtAssert(FALSE);
|
|
}
|
|
|
|
INT
|
|
WanIpBindAdapter(
|
|
IN PNDIS_STATUS pnsRetStatus,
|
|
IN NDIS_HANDLE nhBindContext,
|
|
IN PNDIS_STRING pnsAdapterName,
|
|
IN PVOID pvSS1,
|
|
IN PVOID pvSS2
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Called by IP to bind an adapter. We open the binding and then do
|
|
all the other work in complete handler
|
|
|
|
Locks:
|
|
|
|
The routine acquires the global adapter list lock, so it is not
|
|
PAGEABLE.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsStatus;
|
|
KIRQL kiIrql;
|
|
LONG lResult;
|
|
|
|
#if DBG
|
|
|
|
ANSI_STRING asTempAnsiString;
|
|
|
|
#endif
|
|
|
|
TraceEnter(ADPT, "WanIpBindAdapter");
|
|
|
|
//
|
|
// Serialize Bind, Unbind code
|
|
//
|
|
|
|
WanpAcquireResource(&g_wrBindMutex);
|
|
|
|
RtAcquireSpinLock(&g_rlStateLock,
|
|
&kiIrql);
|
|
|
|
//
|
|
// See if we have already received a BIND call or are in the process
|
|
// of servicing a bind
|
|
//
|
|
|
|
if(g_lBindRcvd isnot 0)
|
|
{
|
|
//
|
|
// We have already received a bind call
|
|
//
|
|
|
|
RtAssert(g_lBindRcvd is 1);
|
|
|
|
RtReleaseSpinLock(&g_rlStateLock,
|
|
kiIrql);
|
|
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
|
|
Trace(ADPT,ERROR,
|
|
("WanIpBindAdapter: Duplicate bind call\n"));
|
|
|
|
*pnsRetStatus = NDIS_STATUS_SUCCESS;
|
|
|
|
TraceLeave(ADPT, "WanIpBindAdapter");
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Set BindRcvd to 1 so that we dont service any binds till we
|
|
// get done
|
|
//
|
|
|
|
g_lBindRcvd = 1;
|
|
|
|
RtReleaseSpinLock(&g_rlStateLock,
|
|
kiIrql);
|
|
|
|
Trace(ADPT, INFO,
|
|
("WanIpBindAdapter: IP called to bind to adapter %S\n",
|
|
pnsAdapterName->Buffer));
|
|
|
|
//
|
|
// We need to open this adapter as our NDISWAN binding
|
|
//
|
|
|
|
g_nhBindContext = nhBindContext;
|
|
|
|
nsStatus = WanpOpenNdisWan(pnsAdapterName,
|
|
(PNDIS_STRING)pvSS1);
|
|
|
|
|
|
if((nsStatus isnot NDIS_STATUS_SUCCESS) and
|
|
(nsStatus isnot NDIS_STATUS_PENDING))
|
|
{
|
|
RtAcquireSpinLock(&g_rlStateLock,
|
|
&kiIrql);
|
|
|
|
g_lBindRcvd = 0;
|
|
|
|
RtReleaseSpinLock(&g_rlStateLock,
|
|
kiIrql);
|
|
|
|
Trace(ADPT, ERROR,
|
|
("WanIpBindAdapter: Error %x opening NDISWAN on %S\n",
|
|
nsStatus,
|
|
pnsAdapterName->Buffer));
|
|
|
|
}
|
|
|
|
*pnsRetStatus = nsStatus;
|
|
|
|
//
|
|
// At this point we are done. If the operation finished synchronously
|
|
// then our OpenAdapterComplete handler has already been called
|
|
// Otherwise stuff will happen later
|
|
//
|
|
|
|
TraceLeave(ADPT, "WanIpBindAdapter");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#pragma alloc_text(PAGE, WanpOpenNdisWan)
|
|
|
|
|
|
NDIS_STATUS
|
|
WanpOpenNdisWan(
|
|
PNDIS_STRING pnsAdapterName,
|
|
PNDIS_STRING pnsSystemSpecific1
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine opens the ndiswan adapter. It also stores the SS1 string
|
|
in a global for use by the completion routine
|
|
|
|
Locks:
|
|
|
|
Called with the g_wrBindMutex held exclusively
|
|
|
|
Arguments:
|
|
|
|
pusBindName Name of the bindings
|
|
pnsSystemSpecific1 The SS1 passed in the bind call from IP
|
|
|
|
Return Value:
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsStatus, nsError;
|
|
UINT i;
|
|
NDIS_MEDIUM MediaArray[] = {NdisMediumWan};
|
|
PWCHAR pwcNameBuffer, pwcStringBuffer;
|
|
PBYTE pbyBuffer;
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
TraceEnter(ADPT,"OpenNdiswan");
|
|
|
|
//
|
|
// Save a copy of the NDIS binding name
|
|
// Increase the length by 1 so that we can NULL terminate it for
|
|
// easy printing
|
|
//
|
|
|
|
pnsAdapterName->Length += sizeof(WCHAR);
|
|
|
|
g_usNdiswanBindName.Buffer = RtAllocate(NonPagedPool,
|
|
pnsAdapterName->Length,
|
|
WAN_STRING_TAG);
|
|
|
|
if(g_usNdiswanBindName.Buffer is NULL)
|
|
{
|
|
Trace(GLOBAL, ERROR,
|
|
("OpenNdiswan: Unable to allocate %d bytes\n",
|
|
pnsAdapterName->Length));
|
|
|
|
g_nhBindContext = NULL;
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
g_usNdiswanBindName.MaximumLength = pnsAdapterName->Length;
|
|
|
|
RtlUpcaseUnicodeString(&g_usNdiswanBindName,
|
|
pnsAdapterName,
|
|
FALSE);
|
|
|
|
pnsAdapterName->Length -= sizeof(WCHAR);
|
|
g_usNdiswanBindName.Length -= sizeof(WCHAR);
|
|
|
|
//
|
|
// NULL terminate
|
|
//
|
|
|
|
g_usNdiswanBindName.Buffer[g_usNdiswanBindName.MaximumLength/sizeof(WCHAR) - 1] = UNICODE_NULL;
|
|
|
|
//
|
|
// Save a copy of SS1
|
|
//
|
|
|
|
pwcStringBuffer = RtAllocate(NonPagedPool,
|
|
pnsSystemSpecific1->MaximumLength,
|
|
WAN_STRING_TAG);
|
|
|
|
|
|
if(pwcStringBuffer is NULL)
|
|
{
|
|
RtFree(g_usNdiswanBindName.Buffer);
|
|
|
|
g_usNdiswanBindName.Buffer = NULL;
|
|
g_usNdiswanBindName.MaximumLength = 0;
|
|
g_usNdiswanBindName.Length = 0;
|
|
|
|
g_nhBindContext = NULL;
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
return NDIS_STATUS_RESOURCES;
|
|
}
|
|
|
|
RtlZeroMemory(pwcStringBuffer,
|
|
pnsSystemSpecific1->MaximumLength);
|
|
|
|
//
|
|
// Save a copy of the SS1 string
|
|
//
|
|
|
|
g_nsSystemSpecific1.MaximumLength = pnsSystemSpecific1->MaximumLength;
|
|
g_nsSystemSpecific1.Length = pnsSystemSpecific1->Length;
|
|
g_nsSystemSpecific1.Buffer = pwcStringBuffer;
|
|
|
|
RtlCopyMemory(g_nsSystemSpecific1.Buffer,
|
|
pnsSystemSpecific1->Buffer,
|
|
pnsSystemSpecific1->Length);
|
|
|
|
//
|
|
// Open the NDIS adapter.
|
|
//
|
|
|
|
Trace(ADPT, INFO,
|
|
("OpenNdiswan: Opening %S\n", g_usNdiswanBindName.Buffer));
|
|
|
|
//
|
|
// Medium index
|
|
//
|
|
|
|
i = 0;
|
|
|
|
#define MAX_MEDIA 1
|
|
|
|
NdisOpenAdapter(&nsStatus,
|
|
&nsError,
|
|
&g_nhNdiswanBinding,
|
|
&i,
|
|
MediaArray,
|
|
MAX_MEDIA,
|
|
g_nhWanarpProtoHandle,
|
|
(NDIS_HANDLE)0x0CABB1E5,
|
|
pnsAdapterName,
|
|
0,
|
|
NULL);
|
|
|
|
|
|
#undef MAX_MEDIA
|
|
|
|
if(nsStatus isnot NDIS_STATUS_PENDING)
|
|
{
|
|
//
|
|
// we always return PENDING to NDIS because we need to
|
|
// do a switch to system thread when we are done setting OIDS to
|
|
// NDISWAN. So call out completion routine here
|
|
//
|
|
|
|
WanNdisOpenAdapterComplete((NDIS_HANDLE)0x0CABB1E5,
|
|
nsStatus,
|
|
nsError);
|
|
|
|
nsStatus = NDIS_STATUS_PENDING;
|
|
}
|
|
|
|
TraceLeave(ADPT, "OpenNdiswan");
|
|
|
|
return nsStatus;
|
|
}
|
|
|
|
#pragma alloc_text (PAGE, WanNdisOpenAdapterComplete)
|
|
|
|
|
|
VOID
|
|
WanNdisOpenAdapterComplete(
|
|
NDIS_HANDLE nhHandle,
|
|
NDIS_STATUS nsStatus,
|
|
NDIS_STATUS nsError
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Completion routine called by NDIS when it is done opening the NDISWAN
|
|
binding
|
|
If the open was successful, we start the train of requests
|
|
|
|
Locks:
|
|
|
|
Called either synchronously or off a worker thread with the g_wrBindMutex
|
|
This routine needs to release the resource and set g_lBindRcvd
|
|
|
|
Arguments:
|
|
|
|
nhHandle
|
|
nsStatus
|
|
nsError
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsResult;
|
|
BYTE rgbyProtocolId[ARP_802_ADDR_LENGTH] = {0x80,
|
|
0x00,
|
|
0x00,
|
|
0x00,
|
|
0x08,
|
|
0x00};
|
|
|
|
PWANARP_NDIS_REQUEST_CONTEXT pRequestContext;
|
|
|
|
TraceEnter(ADPT,"NdisOpenAdapterComplete");
|
|
|
|
PAGED_CODE();
|
|
|
|
RtAssert(nhHandle is (NDIS_HANDLE)0x0CABB1E5);
|
|
|
|
//
|
|
// If we couldnt open the NDIS wan binding, free all the resources
|
|
// and return
|
|
//
|
|
|
|
if(nsStatus isnot NDIS_STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("OpenAdapterComplete: Status %x\n", nsStatus));
|
|
|
|
RtAssert(g_nhBindContext isnot NULL);
|
|
|
|
NdisCompleteBindAdapter(g_nhBindContext,
|
|
nsStatus,
|
|
nsError);
|
|
|
|
g_nhBindContext = NULL;
|
|
|
|
WanpFreeBindResourcesAndReleaseLock();
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// From here on, the NDISWAN adapter is opened, so to cleanup resources
|
|
// we must call CloseNdisWan()
|
|
//
|
|
|
|
pRequestContext = RtAllocate(NonPagedPool,
|
|
sizeof(WANARP_NDIS_REQUEST_CONTEXT),
|
|
WAN_REQUEST_TAG);
|
|
|
|
if(pRequestContext is NULL)
|
|
{
|
|
RtAssert(g_nhBindContext isnot NULL);
|
|
#if 0
|
|
|
|
NdisCompleteBindAdapter(g_nhBindContext,
|
|
nsStatus,
|
|
nsError);
|
|
|
|
|
|
g_nhBindContext = NULL;
|
|
#endif
|
|
|
|
g_nsStatus = g_nsError = NDIS_STATUS_RESOURCES;
|
|
|
|
ExInitializeWorkItem(&g_wqiNdisWorkItem,
|
|
WanpCloseNdisWan,
|
|
NULL);
|
|
|
|
ExQueueWorkItem(&g_wqiNdisWorkItem,
|
|
DelayedWorkQueue);
|
|
|
|
// WanpCloseNdisWan(NULL);
|
|
|
|
Trace(ADPT, ERROR,
|
|
("NdisOpenAdapterComplete: Couldnt allocate memory for request context\n"));
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Set protocol type
|
|
//
|
|
|
|
RtlCopyMemory(pRequestContext->rgbyProtocolId,
|
|
rgbyProtocolId,
|
|
sizeof(rgbyProtocolId));
|
|
|
|
nsResult = WanpDoNdisRequest(NdisRequestSetInformation,
|
|
OID_WAN_PROTOCOL_TYPE,
|
|
pRequestContext->rgbyProtocolId,
|
|
sizeof(pRequestContext->rgbyProtocolId),
|
|
pRequestContext,
|
|
WanpSetProtocolTypeComplete);
|
|
|
|
if(nsResult isnot NDIS_STATUS_PENDING)
|
|
{
|
|
//
|
|
// No need to call the completion routine, since DoNdisRequest
|
|
// always calls it
|
|
//
|
|
|
|
Trace(ADPT, ERROR,
|
|
("NdisOpenAdapterComplete: %x from OID_WAN_PROTOCOL_TYPE\n",
|
|
nsResult));
|
|
}
|
|
|
|
|
|
TraceLeave(ADPT,"NdisOpenAdapterComplete");
|
|
}
|
|
|
|
|
|
VOID
|
|
WanpSetProtocolTypeComplete(
|
|
NDIS_HANDLE nhHandle,
|
|
struct _WANARP_NDIS_REQUEST_CONTEXT *pRequestContext,
|
|
NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Completion handler called when we are done with setting
|
|
OID_WAN_PROTOCOL_TYPE
|
|
|
|
Locks:
|
|
|
|
None
|
|
|
|
Arguments:
|
|
|
|
nhHandle Binding for the adapter to which the request was sent
|
|
pRequestContext Pointer to the request context
|
|
nsStatus Result of the request
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsResult;
|
|
|
|
TraceEnter(ADPT,"SetProtocolTypeComplete");
|
|
|
|
if(nsStatus isnot NDIS_STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetProtocolTypeComplete: Status %x\n", nsStatus));
|
|
|
|
WanpLastOidComplete(nhHandle,
|
|
pRequestContext,
|
|
nsStatus);
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Set lookahead size
|
|
//
|
|
|
|
pRequestContext->ulLookahead = WANARP_LOOKAHEAD_SIZE;
|
|
|
|
nsResult = WanpDoNdisRequest(NdisRequestSetInformation,
|
|
OID_GEN_CURRENT_LOOKAHEAD,
|
|
&(pRequestContext->ulLookahead),
|
|
sizeof(pRequestContext->ulLookahead),
|
|
pRequestContext,
|
|
WanpSetLookaheadComplete);
|
|
|
|
if(nsResult isnot NDIS_STATUS_PENDING)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetProtocolTypeComplete: %x from OID_GEN_CURRENT_LOOKAHEAD\n",
|
|
nsResult));
|
|
}
|
|
|
|
TraceLeave(ADPT,"SetProtocolTypeComplete");
|
|
}
|
|
|
|
VOID
|
|
WanpSetLookaheadComplete(
|
|
NDIS_HANDLE nhHandle,
|
|
struct _WANARP_NDIS_REQUEST_CONTEXT *pRequestContext,
|
|
NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Completion handler called when we are done with setting
|
|
OID_GEN_CURRENT_LOOKAHEAD
|
|
|
|
Locks:
|
|
|
|
None
|
|
|
|
Arguments:
|
|
|
|
nhHandle Binding for the adapter to which the request was sent
|
|
pRequestContext Pointer to the request context
|
|
nsStatus Result of the request
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsResult;
|
|
|
|
TraceEnter(ADPT,"SetLookaheadComplete");
|
|
|
|
if(nsStatus isnot NDIS_STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetLookaheadComplete: Status %x\n", nsStatus));
|
|
|
|
WanpLastOidComplete(nhHandle,
|
|
pRequestContext,
|
|
nsStatus);
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Set packet filters
|
|
//
|
|
|
|
pRequestContext->ulPacketFilter =
|
|
NDIS_PACKET_TYPE_BROADCAST | NDIS_PACKET_TYPE_DIRECTED;
|
|
|
|
nsResult = WanpDoNdisRequest(NdisRequestSetInformation,
|
|
OID_GEN_CURRENT_PACKET_FILTER,
|
|
&(pRequestContext->ulPacketFilter),
|
|
sizeof(pRequestContext->ulPacketFilter),
|
|
pRequestContext,
|
|
WanpSetPacketFilterComplete);
|
|
|
|
if(nsResult isnot NDIS_STATUS_PENDING)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetLookaheadComplete: %x from OID_GEN_CURRENT_PACKET_FILTER\n",
|
|
nsResult));
|
|
|
|
}
|
|
|
|
TraceLeave(ADPT,"SetLookaheadComplete");
|
|
}
|
|
|
|
VOID
|
|
WanpSetPacketFilterComplete(
|
|
NDIS_HANDLE nhHandle,
|
|
struct _WANARP_NDIS_REQUEST_CONTEXT *pRequestContext,
|
|
NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Completion handler called when we are done with setting
|
|
OID_GEN_CURRENT_PACKET_FILTER
|
|
|
|
Locks:
|
|
|
|
None
|
|
|
|
Arguments:
|
|
|
|
nhHandle Binding for the adapter to which the request was sent
|
|
pRequestContext Pointer to the request context
|
|
nsStatus Result of the request
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsResult;
|
|
BYTE byHeaderSize;
|
|
|
|
TraceEnter(ADPT,"SetPacketFilterComplete");
|
|
|
|
if(nsStatus isnot NDIS_STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetPacketFilterComplete: Status %x\n", nsStatus));
|
|
|
|
WanpLastOidComplete(nhHandle,
|
|
pRequestContext,
|
|
nsStatus);
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Set packet filters
|
|
//
|
|
|
|
pRequestContext->TransportHeaderOffset.ProtocolType =
|
|
NDIS_PROTOCOL_ID_TCP_IP;
|
|
|
|
pRequestContext->TransportHeaderOffset.HeaderOffset =
|
|
(USHORT)sizeof(ETH_HEADER);
|
|
|
|
nsResult = WanpDoNdisRequest(NdisRequestSetInformation,
|
|
OID_GEN_TRANSPORT_HEADER_OFFSET,
|
|
&(pRequestContext->TransportHeaderOffset),
|
|
sizeof(pRequestContext->TransportHeaderOffset),
|
|
pRequestContext,
|
|
WanpLastOidComplete);
|
|
|
|
if(nsResult isnot NDIS_STATUS_PENDING)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("SetPacketFilterComplete: %x from OID_GEN_TRANSPORT_HEADER_OFFSET\n",
|
|
nsResult));
|
|
|
|
}
|
|
|
|
TraceLeave(ADPT,"SetPacketFilterComplete");
|
|
}
|
|
|
|
VOID
|
|
WanpLastOidComplete(
|
|
NDIS_HANDLE nhHandle,
|
|
struct _WANARP_NDIS_REQUEST_CONTEXT *pRequestContext,
|
|
NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Completion handler called when we are done with setting the last OID
|
|
Currently that is OID_GEN_TRANSPORT_HEADER_OFFSET,
|
|
If everything went, well we initialize our adapters
|
|
|
|
Locks:
|
|
|
|
None
|
|
|
|
Arguments:
|
|
|
|
nhHandle Binding for the adapter to which the request was sent
|
|
pRequestContext Pointer to the request context
|
|
nsStatus Result of the request
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsResult;
|
|
|
|
TraceEnter(ADPT,"LastOidComplete");
|
|
|
|
RtFree(pRequestContext);
|
|
|
|
RtAssert(g_wrBindMutex.lWaitCount >= 1);
|
|
|
|
if(nsStatus isnot STATUS_SUCCESS)
|
|
{
|
|
RtAssert(g_nhBindContext isnot NULL);
|
|
|
|
#if 0
|
|
//
|
|
// BindContext being NON-NULL means we are being called
|
|
// asynchronously, hence call NdisCompleteBindAdapter
|
|
//
|
|
|
|
NdisCompleteBindAdapter(g_nhBindContext,
|
|
nsStatus,
|
|
nsStatus);
|
|
g_nhBindContext = NULL;
|
|
|
|
#endif
|
|
|
|
Trace(ADPT, ERROR,
|
|
("LastOidComplete: Status %x\n", nsStatus));
|
|
|
|
g_nsStatus = g_nsError = nsStatus;
|
|
|
|
ExInitializeWorkItem(&g_wqiNdisWorkItem,
|
|
WanpCloseNdisWan,
|
|
NULL);
|
|
|
|
ExQueueWorkItem(&g_wqiNdisWorkItem,
|
|
DelayedWorkQueue);
|
|
|
|
// WanpCloseNdisWan(NULL);
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// This needs to be put in when the multiple adapters stuff is done
|
|
//
|
|
|
|
//
|
|
// The OpenNdiswan call would have setup the system specific
|
|
// Since the WanpInitializeAdapters must be called at PASSIVE, we
|
|
// fire of a work item here to call the function.
|
|
//
|
|
|
|
ExInitializeWorkItem(&g_wqiWorkItem,
|
|
WanpInitializeAdapters,
|
|
NULL);
|
|
|
|
ExQueueWorkItem(&g_wqiWorkItem,
|
|
DelayedWorkQueue);
|
|
|
|
//
|
|
// Dont release the mutex here
|
|
//
|
|
|
|
TraceLeave(ADPT,"LastOidComplete");
|
|
}
|
|
|
|
//
|
|
// When this is used, remember NOT to use NdisUnbindAdapter as a clean
|
|
// up routine because of the way the resource is held
|
|
//
|
|
|
|
#pragma alloc_text(PAGE, WanpInitializeAdapters)
|
|
|
|
NTSTATUS
|
|
WanpInitializeAdapters(
|
|
PVOID pvContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine uses the stored SS1 and opens the IP config for the
|
|
binding. It then loops through the multi_sz and creates one adapter
|
|
for every sz.
|
|
|
|
Locks:
|
|
|
|
Called with the bind mutex held.
|
|
This function releases the mutex
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STRING IPConfigName = NDIS_STRING_CONST("IPConfig");
|
|
NDIS_STATUS nsStatus;
|
|
NDIS_HANDLE nhConfigHandle;
|
|
USHORT i;
|
|
PADAPTER pNewAdapter;
|
|
NTSTATUS nStatus, nReturnStatus;
|
|
PWCHAR pwszDeviceBuffer;
|
|
|
|
PNDIS_CONFIGURATION_PARAMETER pParam;
|
|
|
|
TraceEnter(ADPT, "InitializeAdapters");
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER(pvContext);
|
|
|
|
RtAssert(g_nhBindContext isnot NULL);
|
|
|
|
do
|
|
{
|
|
//
|
|
// Open the key for this "adapter"
|
|
//
|
|
|
|
NdisOpenProtocolConfiguration(&nsStatus,
|
|
&nhConfigHandle,
|
|
&g_nsSystemSpecific1);
|
|
|
|
if(nsStatus isnot NDIS_STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: Unable to Open Protocol Config %x\n",
|
|
nsStatus));
|
|
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Read in the IPConfig string. If this is not present,
|
|
// fail this call.
|
|
//
|
|
|
|
NdisReadConfiguration(&nsStatus,
|
|
&pParam,
|
|
nhConfigHandle,
|
|
&IPConfigName,
|
|
NdisParameterMultiString);
|
|
|
|
if((nsStatus isnot NDIS_STATUS_SUCCESS) or
|
|
(pParam->ParameterType isnot NdisParameterMultiString))
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: Unable to read IPConfig. Status %x \n",
|
|
nsStatus));
|
|
|
|
NdisCloseConfiguration(nhConfigHandle);
|
|
|
|
nsStatus = STATUS_UNSUCCESSFUL;
|
|
|
|
break;
|
|
}
|
|
|
|
|
|
//
|
|
// Allocate memory for the max length device name - used later
|
|
//
|
|
|
|
pwszDeviceBuffer =
|
|
RtAllocate(NonPagedPool,
|
|
(WANARP_MAX_DEVICE_NAME_LEN + 1) * sizeof(WCHAR),
|
|
WAN_STRING_TAG);
|
|
|
|
if(pwszDeviceBuffer is NULL)
|
|
{
|
|
NdisCloseConfiguration(nhConfigHandle);
|
|
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: Couldnt alloc %d bytes for dev name\n",
|
|
(WANARP_MAX_DEVICE_NAME_LEN + 1) * sizeof(WCHAR)));
|
|
|
|
|
|
nsStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
break;
|
|
}
|
|
|
|
}while(FALSE);
|
|
|
|
if(nsStatus isnot STATUS_SUCCESS)
|
|
{
|
|
//
|
|
// Tell NDIS that the bind is done but it failed
|
|
//
|
|
|
|
g_nsStatus = g_nsError = nsStatus;
|
|
|
|
#if 0
|
|
NdisCompleteBindAdapter(g_nhBindContext,
|
|
nsStatus,
|
|
nsStatus);
|
|
|
|
g_nhBindContext = NULL;
|
|
|
|
#endif
|
|
|
|
WanpCloseNdisWan(NULL);
|
|
TraceLeave(ADPT, "InitializeAdapters");
|
|
|
|
return nsStatus;
|
|
}
|
|
|
|
//
|
|
// Now walk through the each of the string
|
|
// Since the string must end with two \0's we only walk Len/2 - 1
|
|
//
|
|
|
|
i = 0;
|
|
|
|
nReturnStatus = STATUS_UNSUCCESSFUL;
|
|
|
|
RtAssert((pParam->ParameterData.StringData.Length % sizeof(WCHAR)) is 0);
|
|
|
|
//
|
|
// The IPConfigString is a MULTI_SZ with each SZ being the relative key
|
|
// for an interfaces IP configuration. It is thus something like
|
|
// TCPIP\PARAMETERS\INTERFACES\<device_name>
|
|
// Being a MULTI_SZ it is terminated by 2 NULL characters. So we
|
|
// walk till the last but one character while parsing out the SZ from
|
|
// the multi_sz
|
|
//
|
|
|
|
while(i < (pParam->ParameterData.StringData.Length/sizeof(WCHAR)) - 1)
|
|
{
|
|
USHORT j, usConfigLen, usDevLen;
|
|
UNICODE_STRING usTempString, usUpcaseString, usDevString;
|
|
GUID Guid;
|
|
|
|
if(pParam->ParameterData.StringData.Buffer[i] is UNICODE_NULL)
|
|
{
|
|
if(pParam->ParameterData.StringData.Buffer[i + 1] is UNICODE_NULL)
|
|
{
|
|
//
|
|
// Two nulls - end of string
|
|
//
|
|
|
|
break;
|
|
}
|
|
|
|
//
|
|
// only one null, just move on to the next character
|
|
//
|
|
|
|
i++;
|
|
|
|
continue;
|
|
}
|
|
|
|
RtAssert(pParam->ParameterData.StringData.Buffer[i] isnot UNICODE_NULL);
|
|
|
|
//
|
|
// Now i points to the start of an SZ. Figure out the length of this
|
|
//
|
|
|
|
usConfigLen =
|
|
(USHORT)wcslen(&(pParam->ParameterData.StringData.Buffer[i]));
|
|
|
|
|
|
//
|
|
// Make usTempString point to this config. Also increase
|
|
// the size by one so one can print this nicely
|
|
//
|
|
|
|
usTempString.MaximumLength = (usConfigLen + 1) * sizeof(WCHAR);
|
|
usTempString.Length = (usConfigLen + 1) * sizeof(WCHAR);
|
|
|
|
usTempString.Buffer = &(pParam->ParameterData.StringData.Buffer[i]);
|
|
|
|
|
|
usUpcaseString.Buffer = RtAllocate(NonPagedPool,
|
|
usTempString.MaximumLength,
|
|
WAN_STRING_TAG);
|
|
|
|
if(usUpcaseString.Buffer is NULL)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: Unable to allocate %d bytes\n",
|
|
usTempString.MaximumLength));
|
|
|
|
usUpcaseString.MaximumLength = 0;
|
|
usUpcaseString.Length = 0;
|
|
|
|
i += usConfigLen;
|
|
|
|
continue;
|
|
}
|
|
|
|
usUpcaseString.MaximumLength = usTempString.MaximumLength;
|
|
|
|
RtlUpcaseUnicodeString(&usUpcaseString,
|
|
&usTempString,
|
|
FALSE);
|
|
|
|
//
|
|
// Set the last wchar to null
|
|
//
|
|
|
|
usUpcaseString.Buffer[usConfigLen] = UNICODE_NULL;
|
|
|
|
//
|
|
// Set the length back to what it was
|
|
//
|
|
|
|
usUpcaseString.Length -= sizeof(WCHAR);
|
|
|
|
RtAssert((usUpcaseString.Length % sizeof(WCHAR)) is 0);
|
|
|
|
//
|
|
// The device name is the last thing in the '\' separated path.
|
|
// So walk the SZ backwards looking for '\'
|
|
// NOTE one could use wcsrchr, but it would be inefficient since
|
|
// it doesnt know the end of the buffer
|
|
//
|
|
|
|
for(j = usUpcaseString.Length/sizeof(WCHAR) - 1;
|
|
j >= 0;
|
|
j--)
|
|
{
|
|
if(usUpcaseString.Buffer[j] is L'\\')
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// So at this point j is the index to the last '\'
|
|
//
|
|
|
|
//
|
|
// First off, make this a GUID
|
|
//
|
|
|
|
nStatus = ConvertStringToGuid(&(usUpcaseString.Buffer[j + 1]),
|
|
wcslen(&(usUpcaseString.Buffer[j + 1])) * sizeof(WCHAR),
|
|
&Guid);
|
|
|
|
if(nStatus isnot STATUS_SUCCESS)
|
|
{
|
|
//
|
|
// Hmm - not a GUID?
|
|
//
|
|
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: String %S is not a GUID\n",
|
|
&(usUpcaseString.Buffer[j + 1])));
|
|
|
|
//
|
|
// Free the upcase string and move to the next SZ in IPConfig
|
|
//
|
|
|
|
RtFree(usUpcaseString.Buffer);
|
|
|
|
usUpcaseString.Buffer = NULL;
|
|
usUpcaseString.MaximumLength = 0;
|
|
usUpcaseString.Length = 0;
|
|
|
|
i += usConfigLen;
|
|
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Make sure this binding is not present
|
|
//
|
|
|
|
if(IsBindingPresent(&Guid))
|
|
{
|
|
Trace(ADPT, WARN,
|
|
("InitializeAdapters: %S already present\n",
|
|
&(usUpcaseString.Buffer[j + 1])));
|
|
|
|
//
|
|
// Free the upcase string and move to the next SZ in IPConfig
|
|
//
|
|
|
|
RtFree(usUpcaseString.Buffer);
|
|
|
|
usUpcaseString.Buffer = NULL;
|
|
usUpcaseString.MaximumLength = 0;
|
|
usUpcaseString.Length = 0;
|
|
|
|
i += usConfigLen;
|
|
|
|
continue;
|
|
}
|
|
|
|
|
|
//
|
|
// We need to tack on a \DEVICE in front of Guid string
|
|
//
|
|
|
|
usDevLen = wcslen(TCPIP_IF_PREFIX) +
|
|
wcslen(&(usUpcaseString.Buffer[j]));
|
|
|
|
//
|
|
// make sure we can fit in the space we have allocated
|
|
//
|
|
|
|
RtAssert(usDevLen <= WANARP_MAX_DEVICE_NAME_LEN);
|
|
|
|
pwszDeviceBuffer[usDevLen] = UNICODE_NULL;
|
|
|
|
usDevString.MaximumLength = usDevLen * sizeof(WCHAR);
|
|
usDevString.Length = usDevLen * sizeof(WCHAR);
|
|
usDevString.Buffer = pwszDeviceBuffer;
|
|
|
|
//
|
|
// Copy over the \Device part
|
|
//
|
|
|
|
RtlCopyMemory(pwszDeviceBuffer,
|
|
TCPIP_IF_PREFIX,
|
|
wcslen(TCPIP_IF_PREFIX) * sizeof(WCHAR));
|
|
|
|
//
|
|
// Cat the \<Name> part
|
|
//
|
|
|
|
RtlCopyMemory(&(pwszDeviceBuffer[wcslen(TCPIP_IF_PREFIX)]),
|
|
&usUpcaseString.Buffer[j],
|
|
wcslen(&(usUpcaseString.Buffer[j])) * sizeof(WCHAR));
|
|
|
|
|
|
//
|
|
// Create an adapter with this name and config
|
|
//
|
|
|
|
pNewAdapter = NULL;
|
|
|
|
Trace(ADPT, INFO,
|
|
("InitializeAdapters: Calling create adapter for %S %S\n",
|
|
usUpcaseString.Buffer,
|
|
usDevString.Buffer));
|
|
|
|
nStatus = WanpCreateAdapter(&Guid,
|
|
&usUpcaseString,
|
|
&usDevString,
|
|
&pNewAdapter);
|
|
|
|
if(nStatus isnot STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("InitializeAdapters: Err %x creating adapter for %S (%S)\n",
|
|
nStatus,
|
|
usUpcaseString.Buffer,
|
|
pwszDeviceBuffer));
|
|
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// If even one succeeds, we return success
|
|
//
|
|
|
|
nReturnStatus = STATUS_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// Done with the upcased string
|
|
//
|
|
|
|
RtFree(usUpcaseString.Buffer);
|
|
|
|
usUpcaseString.Buffer = NULL;
|
|
usUpcaseString.MaximumLength = 0;
|
|
usUpcaseString.Length = 0;
|
|
|
|
//
|
|
// Go to the next SZ in the MULTI_SZ
|
|
//
|
|
|
|
i += usConfigLen;
|
|
}
|
|
|
|
NdisCloseConfiguration(nhConfigHandle);
|
|
|
|
RtFree(pwszDeviceBuffer);
|
|
|
|
NdisCompleteBindAdapter(g_nhBindContext,
|
|
nReturnStatus,
|
|
nReturnStatus);
|
|
|
|
g_nhBindContext = NULL;
|
|
|
|
if(nReturnStatus isnot STATUS_SUCCESS)
|
|
{
|
|
WanpCloseNdisWan(NULL);
|
|
}
|
|
else
|
|
{
|
|
g_lBindRcvd = 1;
|
|
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
}
|
|
|
|
|
|
TraceLeave(ADPT, "InitializeAdapters");
|
|
|
|
return nReturnStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
WanpCreateAdapter(
|
|
IN GUID *pAdapterGuid,
|
|
IN PUNICODE_STRING pusConfigName,
|
|
IN PUNICODE_STRING pusDeviceName,
|
|
OUT ADAPTER **ppNewAdapter
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates and initializes an ADAPTER structure
|
|
If this is the first adapter, it is used as the server adapter.
|
|
For that case, it also creates and initializes the server interface.
|
|
The server adapter is added to IP
|
|
|
|
Locks:
|
|
|
|
Called at passive, however it acquires the adapter list lock since it
|
|
sets g_pServerAdapter/Interface
|
|
|
|
Arguments:
|
|
|
|
pusConfigName Name of config key
|
|
pusDeviceName The device name for the adapter
|
|
ppNewAdapter Pointer to storage for the pointer to newly created adapter
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS
|
|
STATUS_INSUFFICIENT_RESOURCES
|
|
|
|
--*/
|
|
|
|
{
|
|
ULONG ulSize;
|
|
PVOID pvBuffer;
|
|
PADAPTER pAdapter;
|
|
USHORT i;
|
|
NTSTATUS nStatus;
|
|
KIRQL kiIrql;
|
|
|
|
PASSIVE_ENTRY();
|
|
|
|
TraceEnter(ADPT, "CreateAdapter");
|
|
|
|
Trace(ADPT, TRACE,
|
|
("CreateAdapter: \n\t%S\n\t%S\n",
|
|
pusConfigName->Buffer,
|
|
pusDeviceName->Buffer));
|
|
|
|
*ppNewAdapter = NULL;
|
|
|
|
//
|
|
// The size that one needs is the size of the adapter + the length of the
|
|
// name. Add a WCHAR to get a easily printtable string.
|
|
// Align everything on a 4 byte boundary
|
|
//
|
|
|
|
ulSize = ALIGN_UP(sizeof(ADAPTER), ULONG) +
|
|
ALIGN_UP((pusConfigName->Length + sizeof(WCHAR)), ULONG) +
|
|
ALIGN_UP((pusDeviceName->Length + sizeof(WCHAR)), ULONG);
|
|
|
|
|
|
#if DBG
|
|
|
|
//
|
|
// For debug code we also store the adapter name in ANSI
|
|
// We use 2 * sizeof(CHAR) because RtlUnicodeToAnsiString needs
|
|
// MaximumLength + 1
|
|
//
|
|
|
|
ulSize += pusDeviceName->Length/sizeof(WCHAR) + (2 * sizeof(CHAR));
|
|
|
|
#endif
|
|
|
|
pAdapter = RtAllocate(NonPagedPool,
|
|
ulSize,
|
|
WAN_ADAPTER_TAG);
|
|
|
|
if(pAdapter is NULL)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("CreateAdapter: Failed to allocate memory\n"));
|
|
|
|
TraceLeave(ADPT, "CreateAdapter");
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
//
|
|
// Clear all the fields out
|
|
//
|
|
|
|
RtlZeroMemory(pAdapter,
|
|
ulSize);
|
|
|
|
//
|
|
// The config name buffer starts at the end of the adapter structure.
|
|
//
|
|
|
|
pvBuffer = (PVOID)((ULONG_PTR)pAdapter + sizeof(ADAPTER));
|
|
|
|
//
|
|
// We DWORD align it for better compare/copy
|
|
//
|
|
|
|
pvBuffer = ALIGN_UP_POINTER(pvBuffer, ULONG);
|
|
|
|
pAdapter->usConfigKey.Length = pusConfigName->Length;
|
|
pAdapter->usConfigKey.MaximumLength = pusConfigName->Length;
|
|
pAdapter->usConfigKey.Buffer = (PWCHAR)pvBuffer;
|
|
|
|
RtlCopyMemory(pAdapter->usConfigKey.Buffer,
|
|
pusConfigName->Buffer,
|
|
pusConfigName->Length);
|
|
|
|
//
|
|
// The device name comes after this
|
|
//
|
|
|
|
|
|
pvBuffer = (PVOID)((ULONG_PTR)pvBuffer +
|
|
pusConfigName->Length +
|
|
sizeof(WCHAR));
|
|
|
|
pvBuffer = ALIGN_UP_POINTER(pvBuffer, ULONG);
|
|
|
|
pAdapter->usDeviceNameW.Length = pusDeviceName->Length;
|
|
pAdapter->usDeviceNameW.MaximumLength = pusDeviceName->Length;
|
|
pAdapter->usDeviceNameW.Buffer = (PWCHAR)pvBuffer;
|
|
|
|
RtlCopyMemory(pAdapter->usDeviceNameW.Buffer,
|
|
pusDeviceName->Buffer,
|
|
pusDeviceName->Length);
|
|
|
|
#if DBG
|
|
|
|
//
|
|
// The debug string comes after the UNICODE device name buffer
|
|
//
|
|
|
|
pvBuffer = (PVOID)((ULONG_PTR)pvBuffer +
|
|
pusDeviceName->Length +
|
|
sizeof(WCHAR));
|
|
|
|
pvBuffer = ALIGN_UP_POINTER(pvBuffer, ULONG);
|
|
|
|
pAdapter->asDeviceNameA.Buffer = (PCHAR)pvBuffer;
|
|
|
|
//
|
|
// Apparently, the unicode to ansi function wants length + 1
|
|
//
|
|
|
|
pAdapter->asDeviceNameA.MaximumLength = pusDeviceName->Length/sizeof(WCHAR) + 1;
|
|
|
|
RtlUnicodeStringToAnsiString(&pAdapter->asDeviceNameA,
|
|
&pAdapter->usDeviceNameW,
|
|
FALSE);
|
|
|
|
#endif // DBG
|
|
|
|
//
|
|
// Structure copy
|
|
//
|
|
|
|
pAdapter->Guid = *pAdapterGuid;
|
|
|
|
//
|
|
// Must be set to INVALID so that GetEntityList can work
|
|
//
|
|
|
|
pAdapter->dwATInstance = INVALID_ENTITY_INSTANCE;
|
|
pAdapter->dwIfInstance = INVALID_ENTITY_INSTANCE;
|
|
|
|
//
|
|
// Set the state
|
|
//
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
|
|
//
|
|
// This hardware index is needed to generate the Unique ID that
|
|
// DHCP uses.
|
|
// NOTE - we dont have an index so all hardware addrs will be the same
|
|
//
|
|
|
|
BuildHardwareAddrFromIndex(pAdapter->rgbyHardwareAddr,
|
|
pAdapter->dwAdapterIndex);
|
|
|
|
//
|
|
// Initialize the lock for the adapter
|
|
//
|
|
|
|
RtInitializeSpinLock(&(pAdapter->rlLock));
|
|
|
|
InitializeListHead(&(pAdapter->lePendingPktList));
|
|
InitializeListHead(&(pAdapter->lePendingHdrList));
|
|
InitializeListHead(&(pAdapter->leEventList));
|
|
|
|
//
|
|
// Set the Refcount to 1 because it will either lie on some list
|
|
// or be pointed to by g_pServerAdapter
|
|
//
|
|
|
|
InitAdapterRefCount(pAdapter);
|
|
|
|
if(g_pServerAdapter is NULL)
|
|
{
|
|
PUMODE_INTERFACE pInterface;
|
|
|
|
//
|
|
// We use the first adapter adapter as the server adapter
|
|
// We also create the server interface right here
|
|
//
|
|
|
|
RtAssert(g_pServerInterface is NULL);
|
|
|
|
pInterface = RtAllocate(NonPagedPool,
|
|
sizeof(UMODE_INTERFACE),
|
|
WAN_INTERFACE_TAG);
|
|
|
|
|
|
if(pInterface is NULL)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("CreateAdapter: Couldnt allocate %d bytes for server i/f\n",
|
|
sizeof(UMODE_INTERFACE)));
|
|
|
|
RtFree(pAdapter);
|
|
|
|
*ppNewAdapter = NULL;
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
RtlZeroMemory(pInterface,
|
|
sizeof(UMODE_INTERFACE));
|
|
|
|
//
|
|
// Get an index from IP
|
|
// If this fails, it sets the value to INVALID_IF_INDEX
|
|
//
|
|
|
|
nStatus = WanpGetNewIndex(&(pInterface->dwRsvdAdapterIndex));
|
|
|
|
if(nStatus isnot STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("CreateAdapter: Couldnt get index for server interface\n"));
|
|
|
|
RtFree(pAdapter);
|
|
|
|
RtFree(pInterface);
|
|
|
|
*ppNewAdapter = NULL;
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
}
|
|
|
|
Trace(ADPT, TRACE,
|
|
("CreateAdapter: Server Index is %d\n",
|
|
pInterface->dwRsvdAdapterIndex));
|
|
|
|
RtInitializeSpinLock(&(pInterface->rlLock));
|
|
InitInterfaceRefCount(pInterface);
|
|
|
|
pInterface->dwIfIndex = INVALID_IF_INDEX;
|
|
pInterface->dwAdminState = IF_ADMIN_STATUS_UP;
|
|
pInterface->duUsage = DU_CALLIN;
|
|
|
|
//
|
|
// Structure copy
|
|
//
|
|
|
|
pInterface->Guid = ServerInterfaceGuid;
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
pInterface->dwOperState = IF_OPER_STATUS_DISCONNECTED;
|
|
pInterface->dwLastChange = GetTimeTicks();
|
|
|
|
//
|
|
// We dont cross ref the structures here. That is done
|
|
// we have added the server interface to IP.
|
|
//
|
|
|
|
g_pServerAdapter = pAdapter;
|
|
g_pServerInterface = pInterface;
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
Trace(ADPT, INFO,
|
|
("CreateAdapter: %S is server adapter\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
}
|
|
else
|
|
{
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
InsertHeadList(&g_leFreeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumFreeAdapters++;
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
}
|
|
|
|
InterlockedIncrement(&g_ulNumAdapters);
|
|
|
|
*ppNewAdapter = pAdapter;
|
|
|
|
TraceLeave(ADPT, "CreateAdapter");
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
#pragma alloc_text(PAGE, WanNdisUnbindAdapter)
|
|
|
|
|
|
VOID
|
|
WanNdisUnbindAdapter(
|
|
PNDIS_STATUS pnsRetStatus,
|
|
NDIS_HANDLE nhProtocolContext,
|
|
NDIS_HANDLE nhUnbindContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Called by NDIS to unbind an open adapter. Since we only open one adapter
|
|
we get this only once.
|
|
|
|
Locks:
|
|
|
|
The function acquires the g_wrBindMutex. This is released in
|
|
WanpFreeBindResourcesAndReleaseLock
|
|
|
|
Arguments:
|
|
|
|
pnsRetStatus
|
|
nhProtocolContext
|
|
nhUnbindContext
|
|
|
|
Return Value:
|
|
|
|
NDIS_STATUS_PENDING
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsStatus;
|
|
|
|
TraceEnter(ADPT,"NdisUnbindAdapter");
|
|
|
|
PAGED_CODE();
|
|
|
|
RtAssert(nhProtocolContext is (NDIS_HANDLE)0x0CABB1E5);
|
|
|
|
//
|
|
// Get into the critical section
|
|
//
|
|
|
|
WanpAcquireResource(&g_wrBindMutex);
|
|
|
|
|
|
//
|
|
// Mark all the adapters as down so that we dont process anything
|
|
// from IP on them
|
|
//
|
|
|
|
//
|
|
// Close the adapter if we havent already done so.
|
|
// This will stop us getting stuff from NDISWAN
|
|
//
|
|
|
|
if(g_nhNdiswanBinding isnot NULL)
|
|
{
|
|
g_nhUnbindContext = nhUnbindContext;
|
|
WanpCloseNdisWan(NULL);
|
|
|
|
//
|
|
// If we call CloseNdisWan then we DONT release the resource here.
|
|
// It is released by CloseAdapterComplete
|
|
//
|
|
|
|
*pnsRetStatus = NDIS_STATUS_PENDING;
|
|
}
|
|
else
|
|
{
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
|
|
*pnsRetStatus = NDIS_STATUS_SUCCESS;
|
|
}
|
|
|
|
TraceLeave(ADPT, "NdisUnbindAdapter");
|
|
|
|
}
|
|
|
|
VOID
|
|
WanpCloseNdisWan(
|
|
PVOID pvContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function cleans all the resources of an open adapter
|
|
|
|
Locks:
|
|
|
|
The function is called with the g_wrBindMutex held.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS nsStatus;
|
|
|
|
TraceEnter(ADPT, "WanpCloseNdisWan");
|
|
|
|
RtAssert(g_nhNdiswanBinding isnot NULL);
|
|
|
|
RtAssert(g_wrBindMutex.lWaitCount >= 1);
|
|
|
|
NdisCloseAdapter(&nsStatus,
|
|
g_nhNdiswanBinding);
|
|
|
|
g_nhNdiswanBinding = NULL;
|
|
|
|
//
|
|
// If our completion routine will not be called, do it here
|
|
//
|
|
|
|
if(nsStatus isnot NDIS_STATUS_PENDING)
|
|
{
|
|
WanNdisCloseAdapterComplete((NDIS_HANDLE)0x0CABB1E5,
|
|
nsStatus);
|
|
}
|
|
|
|
//
|
|
// We DONT release the resource here. It is released by
|
|
// CloseAdapterComplete
|
|
//
|
|
|
|
TraceLeave(ADPT, "WanpCloseNdisWan");
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
#pragma alloc_text(PAGE, WanNdisCloseAdapterComplete)
|
|
|
|
VOID
|
|
WanNdisCloseAdapterComplete(
|
|
NDIS_HANDLE nhBindHandle,
|
|
NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Called by NDIS when it is done closing the NDISWAN adapter
|
|
The close adapter can be done either because we failed somewhere (after
|
|
successfully opening our adapter) and in the process of cleanup are
|
|
closing our adapter (in which case the unbind context will be null) or
|
|
because we got an unbind and were cleaning up all adapter related stuff
|
|
|
|
Locks:
|
|
|
|
Called the g_wrBindMutex held.
|
|
The resource was acquired either in the Unbind handler or was acquired
|
|
before the failure code was called
|
|
|
|
Arguments:
|
|
|
|
nhBindHandle
|
|
nsStatus
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
KIRQL kiIrql;
|
|
NDIS_HANDLE nhUnbind;
|
|
NDIS_HANDLE nhBind;
|
|
NDIS_STATUS tnsStatus, tnsError;
|
|
|
|
TraceEnter(ADPT, "NdisCloseAdapterComplete");
|
|
|
|
PAGED_CODE();
|
|
|
|
nhUnbind = g_nhUnbindContext;
|
|
nhBind = g_nhBindContext;
|
|
tnsStatus = g_nsStatus;
|
|
tnsError = g_nsError;
|
|
|
|
RtAssert(g_wrBindMutex.lWaitCount >= 1);
|
|
|
|
g_nhUnbindContext = NULL;
|
|
g_nhBindContext = NULL;
|
|
|
|
WanpFreeBindResourcesAndReleaseLock();
|
|
|
|
if(nhBind isnot NULL)
|
|
{
|
|
RtAssert(nhUnbind == NULL);
|
|
|
|
//
|
|
// Tell NDIS that the bind is done but it failed
|
|
//
|
|
NdisCompleteBindAdapter(nhBind,
|
|
tnsStatus,
|
|
tnsError);
|
|
}
|
|
|
|
//
|
|
// If this is being triggered from an unbind..
|
|
//
|
|
if(nhUnbind isnot NULL)
|
|
{
|
|
RtAssert(nhBind == NULL);
|
|
NdisCompleteUnbindAdapter(nhUnbind,
|
|
NDIS_STATUS_SUCCESS);
|
|
}
|
|
|
|
}
|
|
|
|
VOID
|
|
WanpFreeBindResourcesAndReleaseLock(
|
|
VOID
|
|
)
|
|
{
|
|
RtAssert(g_wrBindMutex.lWaitCount >= 1);
|
|
|
|
RtFree(g_usNdiswanBindName.Buffer);
|
|
|
|
g_usNdiswanBindName.Buffer = NULL;
|
|
|
|
RtFree(g_nsSystemSpecific1.Buffer);
|
|
|
|
g_nsSystemSpecific1.Buffer = NULL;
|
|
|
|
WanpRemoveAllAdapters();
|
|
|
|
//
|
|
// There could be turds in the connection table if this was not
|
|
// a clean shutdown
|
|
//
|
|
|
|
WanpRemoveAllConnections();
|
|
|
|
//
|
|
// Done, set the global event
|
|
//
|
|
|
|
KeSetEvent(&g_keCloseEvent,
|
|
0,
|
|
FALSE);
|
|
|
|
|
|
//
|
|
// Finally done, set g_lBindRcvd and release the resource
|
|
//
|
|
|
|
g_lBindRcvd = 0;
|
|
|
|
WanpReleaseResource(&g_wrBindMutex);
|
|
|
|
PASSIVE_EXIT();
|
|
}
|
|
|
|
|
|
PADAPTER
|
|
WanpFindAdapterToMap(
|
|
IN DIAL_USAGE duUsage,
|
|
OUT PKIRQL pkiIrql,
|
|
IN DWORD dwAdapterIndex,
|
|
IN PUNICODE_STRING pusNewIfName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Finds a free adapter to map to an interface.
|
|
The adapter, if found is locked and referenced. Its state is set to
|
|
AS_MAPPING and it is put on the mapped list.
|
|
|
|
Locks:
|
|
|
|
Must be called at PASSIVE
|
|
Acquires g_rwlAdapterLock as WRITER
|
|
If it returns a mapped adapter, then the function returns at DPC and
|
|
the original IRQL is in pkiIrql
|
|
|
|
Arguments:
|
|
|
|
duUsage The usage for which the adapter needs to be found
|
|
pkiIrql
|
|
dwAdapterIndex
|
|
pusNewIfName
|
|
|
|
Return Value:
|
|
|
|
Pointer to adapter if successful
|
|
|
|
--*/
|
|
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL kiIrql;
|
|
PLIST_ENTRY pleNode;
|
|
NTSTATUS nStatus;
|
|
KEVENT keChangeEvent;
|
|
|
|
WAN_EVENT_NODE TempEvent;
|
|
|
|
//
|
|
// Find a free adapter
|
|
//
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
if(duUsage is DU_CALLIN)
|
|
{
|
|
if(g_pServerAdapter is NULL)
|
|
{
|
|
Trace(CONN, ERROR,
|
|
("FindAdapterToMap: No server adapter\n"));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
RtAssert(g_pServerInterface);
|
|
|
|
//
|
|
// Lock out the adapter
|
|
//
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(g_pServerAdapter->rlLock));
|
|
|
|
//
|
|
// Reference it because we are going to return it. Also we need the
|
|
// reference when we let go of the lock if we need to wait
|
|
//
|
|
|
|
ReferenceAdapter(g_pServerAdapter);
|
|
|
|
//
|
|
// See if the adapter has been added to IP
|
|
// rasiphlp has to call us to map the server adapter before
|
|
// doing a line up
|
|
//
|
|
|
|
if(g_pServerAdapter->byState isnot AS_MAPPED)
|
|
{
|
|
RtAssert(g_pServerAdapter->byState is AS_ADDING);
|
|
|
|
if(g_pServerAdapter->byState is AS_ADDING)
|
|
{
|
|
KeInitializeEvent(&(TempEvent.keEvent),
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
InsertTailList(&(g_pServerAdapter->leEventList),
|
|
&(TempEvent.leEventLink));
|
|
|
|
//
|
|
// Let go of the lock and wait on the event
|
|
//
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(g_pServerAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
nStatus = KeWaitForSingleObject(&(TempEvent.keEvent),
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(g_pServerAdapter->rlLock));
|
|
|
|
RemoveEntryList(&(TempEvent.leEventLink));
|
|
}
|
|
}
|
|
|
|
if(g_pServerAdapter->byState isnot AS_MAPPED)
|
|
{
|
|
PADAPTER pTempAdapter;
|
|
|
|
//
|
|
// Couldnt add the adapter because of some reason,
|
|
//
|
|
|
|
pTempAdapter = g_pServerAdapter;
|
|
|
|
Trace(CONN, ERROR,
|
|
("FindAdapterToMap: Unable to get mapped server adapter\n"));
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(g_pServerAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
DereferenceAdapter(pTempAdapter);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
RtAssert(g_pServerAdapter->byState is AS_MAPPED);
|
|
|
|
ExitWriterFromDpcLevel(&g_rwlAdapterLock);
|
|
|
|
*pkiIrql = kiIrql;
|
|
|
|
return g_pServerAdapter;
|
|
}
|
|
|
|
//
|
|
// Non dial in case. First try and see if there is an adapter with the
|
|
// index we want already added to IP
|
|
//
|
|
|
|
pAdapter = NULL;
|
|
|
|
for(pleNode = g_leAddedAdapterList.Flink;
|
|
pleNode isnot &g_leAddedAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PADAPTER pTempAdapter;
|
|
|
|
pTempAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAssert(pTempAdapter->byState is AS_ADDED);
|
|
|
|
if(pTempAdapter->dwAdapterIndex is dwAdapterIndex)
|
|
{
|
|
RemoveEntryList(&(pTempAdapter->leAdapterLink));
|
|
|
|
g_ulNumAddedAdapters--;
|
|
|
|
pAdapter = pTempAdapter;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(pAdapter isnot NULL)
|
|
{
|
|
//
|
|
// So we found an added adapter (it has been removed from the
|
|
// added adapter list). Add it to the mapped list
|
|
//
|
|
|
|
InsertTailList(&g_leMappedAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumMappedAdapters++;
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriterFromDpcLevel(&g_rwlAdapterLock);
|
|
|
|
ReferenceAdapter(pAdapter);
|
|
|
|
pAdapter->byState = AS_MAPPING;
|
|
|
|
*pkiIrql = kiIrql;
|
|
|
|
#if DBG
|
|
|
|
Trace(CONN, INFO,
|
|
("FindAdapterToMap: Found %s already added\n",
|
|
pAdapter->asDeviceNameA.Buffer));
|
|
|
|
#endif
|
|
|
|
return pAdapter;
|
|
}
|
|
|
|
//
|
|
// Didnt find an added adapter
|
|
// See if the adapter is on the change list. We want to wait for this
|
|
// because IP doesnt like to have to interfaces with the same index, even
|
|
// if one is in the process of being removed
|
|
//
|
|
|
|
for(pleNode = g_leChangeAdapterList.Flink;
|
|
pleNode isnot &g_leChangeAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PADAPTER pTempAdapter;
|
|
|
|
pTempAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
if(pTempAdapter->dwAdapterIndex is dwAdapterIndex)
|
|
{
|
|
//
|
|
// Wait for the change to complete
|
|
//
|
|
|
|
KeInitializeEvent(&(TempEvent.keEvent),
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pTempAdapter->rlLock));
|
|
|
|
InsertTailList(&(pTempAdapter->leEventList),
|
|
&(TempEvent.leEventLink));
|
|
|
|
//
|
|
// Reference the adapter, let go of the lock and wait on the event
|
|
//
|
|
|
|
ReferenceAdapter(pTempAdapter);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pTempAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
nStatus = KeWaitForSingleObject(&(TempEvent.keEvent),
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pTempAdapter->rlLock));
|
|
|
|
RemoveEntryList(&(TempEvent.leEventLink));
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pTempAdapter->rlLock));
|
|
|
|
DereferenceAdapter(pTempAdapter);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!IsListEmpty(&g_leFreeAdapterList))
|
|
{
|
|
pleNode = RemoveHeadList(&g_leFreeAdapterList);
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
g_ulNumFreeAdapters--;
|
|
|
|
#if DBG
|
|
|
|
Trace(CONN, INFO,
|
|
("FindAdapterToMap: Found free adapter %s\n",
|
|
pAdapter->asDeviceNameA.Buffer));
|
|
|
|
#endif // DBG
|
|
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Couldnt find an adapter. That is bad
|
|
// TODO - if the added adapter list is not empty, we could call
|
|
// RemoveSomeAddedAdapter... at this point
|
|
//
|
|
|
|
Trace(ADPT, ERROR,
|
|
("FindAdapterToMap: Couldnt find free adapter\n"));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Lock and refcount the adapter.
|
|
//
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ReferenceAdapter(pAdapter);
|
|
|
|
//
|
|
// The adapter better not be mapped
|
|
//
|
|
|
|
RtAssert(pAdapter->pInterface is NULL);
|
|
RtAssert(pAdapter->byState is AS_FREE);
|
|
|
|
//
|
|
// Set the state to ADDING
|
|
//
|
|
|
|
pAdapter->byState = AS_ADDING;
|
|
|
|
//
|
|
// Since we are changing the state, no one else should be also
|
|
// changing the state
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent is NULL);
|
|
|
|
KeInitializeEvent(&keChangeEvent,
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
pAdapter->pkeChangeEvent = &keChangeEvent;
|
|
|
|
//
|
|
// Insert into the change list
|
|
//
|
|
|
|
InsertTailList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
//
|
|
// Add the adapter to IP
|
|
//
|
|
|
|
nStatus = WanpAddAdapterToIp(pAdapter,
|
|
FALSE,
|
|
dwAdapterIndex,
|
|
pusNewIfName,
|
|
IF_TYPE_PPP,
|
|
IF_ACCESS_POINTTOPOINT,
|
|
IF_CONNECTION_DEMAND);
|
|
|
|
if(nStatus isnot STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("FindAdapterToMap: %x adding %x to IP\n",
|
|
nStatus, pAdapter));
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
#if DBG
|
|
|
|
RtAssert(IsEntryOnList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
pAdapter->pkeChangeEvent = NULL;
|
|
|
|
RemoveEntryList(&(pAdapter->leAdapterLink));
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
|
|
InsertTailList(&g_leFreeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumFreeAdapters++;
|
|
|
|
//
|
|
// If anyone is waiting on a state change, notify them
|
|
//
|
|
|
|
for(pleNode = pAdapter->leEventList.Flink;
|
|
pleNode isnot &(pAdapter->leEventList);
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PWAN_EVENT_NODE pTempEvent;
|
|
|
|
pTempEvent = CONTAINING_RECORD(pleNode,
|
|
WAN_EVENT_NODE,
|
|
leEventLink);
|
|
|
|
KeSetEvent(&(pTempEvent->keEvent),
|
|
0,
|
|
FALSE);
|
|
}
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Wait till the OpenAdapter is called
|
|
//
|
|
|
|
nStatus = KeWaitForSingleObject(&keChangeEvent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
Trace(ADPT, TRACE,
|
|
("FindAdapterToMap: IPAddInterface succeeded for adapter %w\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
//
|
|
// Lock the adapter, insert into the added list
|
|
//
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
#if DBG
|
|
|
|
RtAssert(IsEntryOnList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
RemoveEntryList(&(pAdapter->leAdapterLink));
|
|
|
|
pAdapter->pkeChangeEvent = NULL;
|
|
|
|
InsertHeadList(&g_leMappedAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumMappedAdapters++;
|
|
|
|
//
|
|
// If anyone is waiting on a state change, notify them
|
|
//
|
|
|
|
for(pleNode = pAdapter->leEventList.Flink;
|
|
pleNode isnot &(pAdapter->leEventList);
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PWAN_EVENT_NODE pTempEvent;
|
|
|
|
pTempEvent = CONTAINING_RECORD(pleNode,
|
|
WAN_EVENT_NODE,
|
|
leEventLink);
|
|
|
|
KeSetEvent(&(pTempEvent->keEvent),
|
|
0,
|
|
FALSE);
|
|
}
|
|
|
|
ExitWriterFromDpcLevel(&g_rwlAdapterLock);
|
|
|
|
*pkiIrql = kiIrql;
|
|
|
|
return pAdapter;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
WanpAddAdapterToIp(
|
|
IN PADAPTER pAdapter,
|
|
IN BOOLEAN bServerAdapter,
|
|
IN DWORD dwAdapterIndex, OPTIONAL
|
|
IN PUNICODE_STRING pusNewIfName, OPTIONAL
|
|
IN DWORD dwMediaType,
|
|
IN BYTE byAccessType,
|
|
IN BYTE byConnectionType
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine adds the interface to IP
|
|
|
|
Locks:
|
|
|
|
MUST BE CALLED AT PASSIVE, but acquires a spinlock so cant be paged out
|
|
|
|
Arguments:
|
|
|
|
pAdapter The adapter to add
|
|
bServerAdapter Set to TRUE if this is a server adapter
|
|
dwAdapterIndex
|
|
pusNewIfName
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS
|
|
|
|
--*/
|
|
|
|
{
|
|
LLIPBindInfo BindInfo;
|
|
IP_STATUS IPStatus;
|
|
UNICODE_STRING usName;
|
|
NTSTATUS nStatus;
|
|
KIRQL kiIrql;
|
|
|
|
PASSIVE_ENTRY();
|
|
|
|
TraceEnter(ADPT, "AddAdapterToIp");
|
|
|
|
RtlZeroMemory(&BindInfo, sizeof(LLIPBindInfo));
|
|
|
|
BindInfo.lip_context = pAdapter;
|
|
BindInfo.lip_mss = WANARP_DEFAULT_MTU;
|
|
BindInfo.lip_speed = WANARP_DEFAULT_SPEED;
|
|
BindInfo.lip_txspace = sizeof(ETH_HEADER);
|
|
|
|
BindInfo.lip_transmit = WanIpTransmit;
|
|
BindInfo.lip_transfer = WanIpTransferData;
|
|
BindInfo.lip_returnPkt = WanIpReturnPacket;
|
|
BindInfo.lip_close = WanIpCloseAdapter;
|
|
BindInfo.lip_addaddr = WanIpAddAddress;
|
|
BindInfo.lip_deladdr = WanIpDeleteAddress;
|
|
BindInfo.lip_invalidate = WanIpInvalidateRce;
|
|
BindInfo.lip_open = WanIpOpenAdapter;
|
|
BindInfo.lip_qinfo = WanIpQueryInfo;
|
|
BindInfo.lip_setinfo = WanIpSetInfo;
|
|
BindInfo.lip_getelist = WanIpGetEntityList;
|
|
BindInfo.lip_flags = LIP_COPY_FLAG;
|
|
|
|
if(bServerAdapter)
|
|
{
|
|
BindInfo.lip_flags |= (LIP_P2MP_FLAG | LIP_NOLINKBCST_FLAG);
|
|
|
|
BindInfo.lip_closelink = WanIpCloseLink;
|
|
}
|
|
else
|
|
{
|
|
BindInfo.lip_flags |= (LIP_P2P_FLAG | LIP_NOIPADDR_FLAG);
|
|
}
|
|
|
|
BindInfo.lip_addrlen = ARP_802_ADDR_LENGTH;
|
|
BindInfo.lip_addr = pAdapter->rgbyHardwareAddr;
|
|
|
|
BindInfo.lip_OffloadFlags = 0;
|
|
BindInfo.lip_dowakeupptrn = NULL;
|
|
BindInfo.lip_pnpcomplete = NULL;
|
|
BindInfo.lip_arpflushate = NULL;
|
|
BindInfo.lip_arpflushallate = NULL;
|
|
|
|
BindInfo.lip_setndisrequest = WanIpSetRequest;
|
|
|
|
RtlInitUnicodeString(&usName,
|
|
pAdapter->usConfigKey.Buffer);
|
|
|
|
#if DBG
|
|
|
|
//
|
|
// Only set while adding or deleting the interface, which is serialized
|
|
//
|
|
|
|
pAdapter->dwRequestedIndex = dwAdapterIndex;
|
|
|
|
#endif
|
|
|
|
IPStatus = g_pfnIpAddInterface(&(pAdapter->usDeviceNameW),
|
|
pusNewIfName,
|
|
&(pAdapter->usConfigKey),
|
|
NULL,
|
|
pAdapter,
|
|
WanIpDynamicRegister,
|
|
&BindInfo,
|
|
dwAdapterIndex,
|
|
dwMediaType,
|
|
byAccessType,
|
|
byConnectionType);
|
|
|
|
|
|
if(IPStatus isnot IP_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("AddAdapterToIp: IPAddInterface failed for adapter %w\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
TraceLeave(ADPT, "AddAdapterToIp");
|
|
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
TraceLeave(ADPT, "AddAdapterToIp");
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
VOID
|
|
WanIpOpenAdapter(
|
|
IN PVOID pvContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Called by IP when the adapter from its IPAddInterface() call
|
|
|
|
Locks:
|
|
|
|
|
|
Arguments:
|
|
|
|
pvContext Pointer to the ADAPTER structure
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL kiIrql;
|
|
PLIST_ENTRY pleNode;
|
|
|
|
TraceEnter(ADPT, "WanOpenAdapter");
|
|
|
|
pAdapter = (PADAPTER)pvContext;
|
|
|
|
RtAcquireSpinLock(&(pAdapter->rlLock),
|
|
&kiIrql);
|
|
|
|
//
|
|
// Set the state to added
|
|
//
|
|
|
|
pAdapter->byState = AS_ADDED;
|
|
|
|
//
|
|
// One reference for the fact that we added the adapter to IP
|
|
//
|
|
|
|
ReferenceAdapter(pAdapter);
|
|
|
|
//
|
|
// Wake up the thread that caused this
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent);
|
|
|
|
KeSetEvent(pAdapter->pkeChangeEvent,
|
|
0,
|
|
FALSE);
|
|
|
|
|
|
RtReleaseSpinLock(&(pAdapter->rlLock),
|
|
kiIrql);
|
|
|
|
TraceLeave(ADPT, "WanOpenAdapter");
|
|
}
|
|
|
|
VOID
|
|
WanpUnmapAdapter(
|
|
PADAPTER pAdapter
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function moves an adapter from the mapped list and puts it on
|
|
the free list
|
|
|
|
Locks:
|
|
|
|
Called with no locks held, but not necessarily at PASSIVE.
|
|
The adapter and adapter list locks are acquired
|
|
|
|
Arguments:
|
|
|
|
pAdapter
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
KIRQL kiIrql;
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
#if DBG
|
|
|
|
//
|
|
// Make sure the adapter is on the mapped list
|
|
//
|
|
|
|
RtAssert(IsEntryOnList(&g_leMappedAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
RemoveEntryList(&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumMappedAdapters--;
|
|
|
|
InsertHeadList(&g_leAddedAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumAddedAdapters++;
|
|
|
|
pAdapter->byState = AS_ADDED;
|
|
pAdapter->pInterface = NULL;
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
//
|
|
// Queue a work item if there is none queued, and we are not
|
|
// shutting down
|
|
//
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&g_rlStateLock);
|
|
|
|
if((g_bRemovalInProgress isnot TRUE) and
|
|
(g_dwDriverState is DRIVER_STARTED))
|
|
{
|
|
g_bRemovalInProgress = TRUE;
|
|
|
|
ExInitializeWorkItem(&g_wqiWorkItem,
|
|
WanpRemoveSomeAddedAdaptersFromIp,
|
|
(PVOID)FALSE);
|
|
|
|
ExQueueWorkItem(&g_wqiWorkItem,
|
|
DelayedWorkQueue);
|
|
}
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&g_rlStateLock);
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
VOID
|
|
WanpRemoveSomeAddedAdaptersFromIp(
|
|
PVOID pvContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
We queue this function to a work item to delete adapter(s) from IP when
|
|
the connection is torn down. Unlike the next function, this
|
|
(i) only removes added (but unmapped) adapters and
|
|
(ii) removes the adapters one at a time.
|
|
This allows another connection to move an adapter from the added list
|
|
to the free list while we are deleting another adapter from IP
|
|
|
|
Locks:
|
|
|
|
Must be called at passive
|
|
Acquires the adapter list lock as WRITER and the lock for each adapter
|
|
|
|
Arguments:
|
|
|
|
pvContext
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL kiIrql;
|
|
PLIST_ENTRY pleNode;
|
|
NTSTATUS nStatus;
|
|
KEVENT keChangeEvent;
|
|
|
|
PASSIVE_ENTRY();
|
|
|
|
UNREFERENCED_PARAMETER(pvContext);
|
|
|
|
KeInitializeEvent(&keChangeEvent,
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
while(!IsListEmpty(&g_leAddedAdapterList))
|
|
{
|
|
pleNode = RemoveHeadList(&g_leAddedAdapterList);
|
|
|
|
g_ulNumAddedAdapters--;
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
pAdapter->byState = AS_REMOVING;
|
|
|
|
//
|
|
// Since we are changing the state, no one else should be also
|
|
// changing the state
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent is NULL);
|
|
|
|
pAdapter->pkeChangeEvent = &keChangeEvent;
|
|
|
|
//
|
|
// Insert it into the change list
|
|
//
|
|
|
|
InsertHeadList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
//
|
|
// Delete from IP but dont clear the index
|
|
//
|
|
|
|
g_pfnIpDeleteInterface(pAdapter->pvIpContext,
|
|
FALSE);
|
|
|
|
//
|
|
// Wait till the CloseAdapter completes
|
|
//
|
|
|
|
nStatus = KeWaitForSingleObject(&keChangeEvent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
Trace(ADPT, INFO,
|
|
("RemoveSomeAddedAdaptersFromIp: Removed %S from Ip\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
//
|
|
// Remove from the change list and put on the free list
|
|
//
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
#if DBG
|
|
|
|
RtAssert(IsEntryOnList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
RemoveEntryList(&(pAdapter->leAdapterLink));
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
|
|
pAdapter->pkeChangeEvent = NULL;
|
|
|
|
InsertHeadList(&g_leFreeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumFreeAdapters++;
|
|
|
|
//
|
|
// If anyone is waiting on a state change, notify them
|
|
//
|
|
|
|
for(pleNode = pAdapter->leEventList.Flink;
|
|
pleNode isnot &(pAdapter->leEventList);
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PWAN_EVENT_NODE pTempEvent;
|
|
|
|
pTempEvent = CONTAINING_RECORD(pleNode,
|
|
WAN_EVENT_NODE,
|
|
leEventLink);
|
|
|
|
KeSetEvent(&(pTempEvent->keEvent),
|
|
0,
|
|
FALSE);
|
|
}
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
}
|
|
|
|
//
|
|
// We are done with the APC
|
|
//
|
|
|
|
g_bRemovalInProgress = FALSE;
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
return;
|
|
}
|
|
|
|
VOID
|
|
WanpRemoveAllAdaptersFromIp(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is a cleanup function called to remove all adapters (added and
|
|
mapped) from IP.
|
|
|
|
Locks:
|
|
|
|
Acquired the g_rwlAdapterLock as WRITER
|
|
Also acquire each adapter's lock when changing state
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL kiIrql;
|
|
LIST_ENTRY leTempHead, *pleNode;
|
|
NTSTATUS nStatus;
|
|
KEVENT keChangeEvent;
|
|
|
|
WAN_EVENT_NODE TempEvent;
|
|
|
|
PASSIVE_ENTRY();
|
|
|
|
KeInitializeEvent(&keChangeEvent,
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
KeInitializeEvent(&(TempEvent.keEvent),
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
InitializeListHead(&leTempHead);
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
//
|
|
// First wait on any adapter in the change list to get out
|
|
// of the change state. Once that is done it is either added to ip
|
|
// or not, and we can deal with those two cases. This is to handle
|
|
// the case where we get an unbind in the middle of a status indication
|
|
//
|
|
|
|
while(!IsListEmpty(&g_leChangeAdapterList))
|
|
{
|
|
pleNode = g_leChangeAdapterList.Flink;
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
InsertTailList(&(pAdapter->leEventList),
|
|
&(TempEvent.leEventLink));
|
|
|
|
ReferenceAdapter(pAdapter);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
nStatus = KeWaitForSingleObject(&(TempEvent.keEvent),
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
#if DBG
|
|
|
|
RtAssert(!IsEntryOnList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
RemoveEntryList(&(TempEvent.leEventLink));
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
}
|
|
|
|
//
|
|
// Set the state of all the adapters to removing...
|
|
//
|
|
|
|
while(!IsListEmpty(&g_leMappedAdapterList))
|
|
{
|
|
pleNode = RemoveHeadList(&g_leMappedAdapterList);
|
|
|
|
g_ulNumMappedAdapters--;
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
pAdapter->byState = AS_REMOVING;
|
|
|
|
InsertHeadList(&leTempHead,
|
|
pleNode);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
}
|
|
|
|
RtAssert(IsListEmpty(&g_leMappedAdapterList));
|
|
|
|
while(!IsListEmpty(&g_leAddedAdapterList))
|
|
{
|
|
pleNode = RemoveHeadList(&g_leAddedAdapterList);
|
|
|
|
g_ulNumAddedAdapters--;
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
pAdapter->byState = AS_REMOVING;
|
|
|
|
InsertHeadList(&leTempHead,
|
|
pleNode);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
}
|
|
|
|
RtAssert(IsListEmpty(&g_leAddedAdapterList));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
//
|
|
// No go through the list of adapters to remove from IP
|
|
//
|
|
|
|
while(!IsListEmpty(&leTempHead))
|
|
{
|
|
pleNode = RemoveHeadList(&leTempHead);
|
|
|
|
pAdapter = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
//
|
|
// Since we are changing the state, no one else should be also
|
|
// changing the state
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent is NULL);
|
|
|
|
pAdapter->pkeChangeEvent = &keChangeEvent;
|
|
|
|
//
|
|
// Insert it into the change list
|
|
//
|
|
|
|
InsertHeadList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
//
|
|
// Delete from IP but dont clear the index
|
|
//
|
|
|
|
g_pfnIpDeleteInterface(pAdapter->pvIpContext,
|
|
FALSE);
|
|
|
|
//
|
|
// Wait till the CloseAdapter completes
|
|
//
|
|
|
|
nStatus = KeWaitForSingleObject(&keChangeEvent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
Trace(ADPT, INFO,
|
|
("RemoveAllAdaptersFromIp: Removed %S from Ip\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
//
|
|
// Remove from the change list and put on the free list
|
|
//
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
#if DBG
|
|
|
|
RtAssert(IsEntryOnList(&g_leChangeAdapterList,
|
|
&(pAdapter->leAdapterLink)));
|
|
|
|
#endif
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
RemoveEntryList(&(pAdapter->leAdapterLink));
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
|
|
pAdapter->pkeChangeEvent = NULL;
|
|
|
|
InsertHeadList(&g_leFreeAdapterList,
|
|
&(pAdapter->leAdapterLink));
|
|
|
|
g_ulNumFreeAdapters++;
|
|
|
|
//
|
|
// If anyone is waiting on a state change, notify them
|
|
//
|
|
|
|
for(pleNode = pAdapter->leEventList.Flink;
|
|
pleNode isnot &(pAdapter->leEventList);
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PWAN_EVENT_NODE pTempEvent;
|
|
|
|
pTempEvent = CONTAINING_RECORD(pleNode,
|
|
WAN_EVENT_NODE,
|
|
leEventLink);
|
|
|
|
KeSetEvent(&(pTempEvent->keEvent),
|
|
0,
|
|
FALSE);
|
|
}
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
}
|
|
|
|
//
|
|
// The write is atomic so we dont need to
|
|
// acquire a lock
|
|
//
|
|
|
|
g_bRemovalInProgress = FALSE;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
WanIpCloseAdapter(
|
|
IN PVOID pvContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Called by IP when it wants to close an adapter. Currently this is done
|
|
from CloseNets() and IPDelInterface().
|
|
|
|
Locks:
|
|
|
|
Acquires the adapter lock
|
|
|
|
Arguments:
|
|
|
|
pvContext Pointer to the ADAPTER
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL kiIrql;
|
|
PLIST_ENTRY pleNode;
|
|
|
|
TraceEnter(ADPT, "IpCloseAdapter");
|
|
|
|
pAdapter = (PADAPTER)pvContext;
|
|
|
|
RtAcquireSpinLock(&(pAdapter->rlLock),
|
|
&kiIrql);
|
|
|
|
pAdapter->pvIpContext = NULL;
|
|
|
|
//
|
|
// Wake up the thread that caused this
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent);
|
|
|
|
KeSetEvent(pAdapter->pkeChangeEvent,
|
|
0,
|
|
FALSE);
|
|
|
|
RtReleaseSpinLock(&(pAdapter->rlLock),
|
|
kiIrql);
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
|
|
TraceLeave(ADPT, "IpCloseAdapter");
|
|
}
|
|
|
|
VOID
|
|
WanpRemoveAllAdapters(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
|
|
Locks:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
|
|
--*/
|
|
|
|
{
|
|
LIST_ENTRY leTempHead, *pleNode;
|
|
KIRQL kiIrql;
|
|
NTSTATUS nStatus;
|
|
|
|
TraceEnter(ADPT, "RemoveAllAdapters");
|
|
|
|
//
|
|
// We can have mapped adapters.
|
|
//
|
|
|
|
//RtAssert(IsListEmpty(&g_leMappedAdapterList));
|
|
|
|
//
|
|
// Just call RemoveAllAdaptersFromIp()
|
|
//
|
|
|
|
WanpRemoveAllAdaptersFromIp();
|
|
|
|
//
|
|
// At this point the movement from one list to another is frozen
|
|
// because we dont accept LinkUp's from Ndiswan
|
|
//
|
|
|
|
RtAssert(IsListEmpty(&g_leAddedAdapterList));
|
|
//RtAssert(IsListEmpty(&g_leMappedAdapterList));
|
|
//RtAssert(IsListEmpty(&g_leChangeAdapterList));
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
//
|
|
// First clean out the free adapters
|
|
//
|
|
|
|
while(!IsListEmpty(&g_leFreeAdapterList))
|
|
{
|
|
PADAPTER pAdapterTmp;
|
|
|
|
pleNode = RemoveHeadList(&g_leFreeAdapterList);
|
|
|
|
g_ulNumFreeAdapters--;
|
|
|
|
pAdapterTmp = CONTAINING_RECORD(pleNode,
|
|
ADAPTER,
|
|
leAdapterLink);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapterTmp->rlLock));
|
|
|
|
RtAssert(pAdapterTmp->byState is AS_FREE);
|
|
|
|
pAdapterTmp->byState = 0xFF;
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapterTmp->rlLock));
|
|
|
|
DereferenceAdapter(pAdapterTmp);
|
|
}
|
|
|
|
//
|
|
// At the end of this, also remove the server adapter
|
|
//
|
|
|
|
if(g_pServerAdapter)
|
|
{
|
|
PADAPTER pAdapter;
|
|
PUMODE_INTERFACE pInterface;
|
|
BOOLEAN bCrossRefed;
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(g_pServerAdapter->rlLock));
|
|
RtAcquireSpinLockAtDpcLevel(&(g_pServerInterface->rlLock));
|
|
|
|
pAdapter = g_pServerAdapter;
|
|
pInterface = g_pServerInterface;
|
|
|
|
if(g_pServerAdapter->byState is AS_MAPPED)
|
|
{
|
|
RtAssert(g_pServerAdapter->pInterface is g_pServerInterface);
|
|
RtAssert(g_pServerInterface->pAdapter is g_pServerAdapter);
|
|
|
|
bCrossRefed = TRUE;
|
|
|
|
}
|
|
else
|
|
{
|
|
RtAssert(g_pServerAdapter->pInterface is NULL);
|
|
RtAssert(g_pServerInterface->pAdapter is NULL);
|
|
|
|
RtAssert(g_pServerAdapter->byState is AS_FREE);
|
|
|
|
bCrossRefed = FALSE;
|
|
}
|
|
|
|
//
|
|
// Remove the global pointers
|
|
//
|
|
|
|
g_pServerInterface = NULL;
|
|
g_pServerAdapter = NULL;
|
|
|
|
//
|
|
// Remove the mapping from interface
|
|
//
|
|
|
|
pInterface->pAdapter = NULL;
|
|
pAdapter->pInterface = NULL;
|
|
pAdapter->byState = AS_REMOVING;
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pInterface->rlLock));
|
|
|
|
if(bCrossRefed)
|
|
{
|
|
//
|
|
// Deref because the cross mapping is removed
|
|
//
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
DereferenceInterface(pInterface);
|
|
}
|
|
|
|
if(pAdapter->pvIpContext)
|
|
{
|
|
KEVENT keChangeEvent;
|
|
|
|
KeInitializeEvent(&keChangeEvent,
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
//
|
|
// Since we are changing the state, no one else should be also
|
|
// changing the state
|
|
//
|
|
|
|
RtAssert(pAdapter->pkeChangeEvent is NULL);
|
|
|
|
pAdapter->pkeChangeEvent = &keChangeEvent;
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
//
|
|
// Delete from IP, but dont clear the index
|
|
//
|
|
|
|
g_pfnIpDeleteInterface(pAdapter->pvIpContext,
|
|
FALSE);
|
|
|
|
nStatus = KeWaitForSingleObject(&keChangeEvent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
|
|
RtAssert(nStatus is STATUS_SUCCESS);
|
|
|
|
Trace(ADPT, INFO,
|
|
("RemoveAllAdapters: Removed %S (server adapter) from Ip\n",
|
|
pAdapter->usDeviceNameW.Buffer));
|
|
|
|
EnterWriter(&g_rwlAdapterLock,
|
|
&kiIrql);
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pAdapter->rlLock));
|
|
|
|
pAdapter->byState = AS_FREE;
|
|
|
|
pAdapter->pkeChangeEvent = NULL;
|
|
|
|
//
|
|
// If anyone is waiting on a state change, notify them
|
|
//
|
|
|
|
for(pleNode = pAdapter->leEventList.Flink;
|
|
pleNode isnot &(pAdapter->leEventList);
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
PWAN_EVENT_NODE pTempEvent;
|
|
|
|
pTempEvent = CONTAINING_RECORD(pleNode,
|
|
WAN_EVENT_NODE,
|
|
leEventLink);
|
|
|
|
KeSetEvent(&(pTempEvent->keEvent),
|
|
0,
|
|
FALSE);
|
|
}
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
} else {
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&(pAdapter->rlLock));
|
|
}
|
|
|
|
//
|
|
// Deref because the global pointers are null
|
|
//
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
DereferenceInterface(pInterface);
|
|
}
|
|
|
|
ExitWriter(&g_rwlAdapterLock,
|
|
kiIrql);
|
|
|
|
TraceLeave(ADPT, "RemoveAllAdapters");
|
|
}
|
|
|
|
|
|
INT
|
|
WanIpDynamicRegister(
|
|
IN PNDIS_STRING InterfaceName,
|
|
IN PVOID pvIpInterfaceContext,
|
|
IN struct _IP_HANDLERS * IpHandlers,
|
|
IN struct LLIPBindInfo * ARPBindInfo,
|
|
IN UINT uiInterfaceNumber
|
|
)
|
|
{
|
|
PADAPTER pAdapter;
|
|
KIRQL irql;
|
|
|
|
|
|
TraceEnter(ADPT, "IpDynamicRegister");
|
|
|
|
|
|
pAdapter = (PADAPTER)(ARPBindInfo->lip_context);
|
|
|
|
RtAcquireSpinLock(&(pAdapter->rlLock),
|
|
&irql);
|
|
|
|
#if DBG
|
|
|
|
Trace(ADPT, INFO,
|
|
("IpDynamicRegister: IP called out to dynamically register %s\n",
|
|
pAdapter->asDeviceNameA.Buffer));
|
|
|
|
#endif
|
|
|
|
pAdapter->pvIpContext = pvIpInterfaceContext;
|
|
pAdapter->dwAdapterIndex = uiInterfaceNumber;
|
|
|
|
#if DBG
|
|
|
|
RtAssert(pAdapter->dwAdapterIndex is pAdapter->dwRequestedIndex);
|
|
|
|
#endif
|
|
|
|
if(g_pfnIpRcv is NULL)
|
|
{
|
|
g_pfnIpRcv = IpHandlers->IpRcvHandler;
|
|
g_pfnIpRcvComplete = IpHandlers->IpRcvCompleteHandler;
|
|
g_pfnIpSendComplete = IpHandlers->IpTxCompleteHandler;
|
|
g_pfnIpTDComplete = IpHandlers->IpTransferCompleteHandler;
|
|
g_pfnIpStatus = IpHandlers->IpStatusHandler;
|
|
g_pfnIpRcvPkt = IpHandlers->IpRcvPktHandler;
|
|
g_pfnIpPnp = IpHandlers->IpPnPHandler;
|
|
}
|
|
|
|
RtReleaseSpinLock(&(pAdapter->rlLock),
|
|
irql);
|
|
|
|
TraceLeave(ADPT, "IpDynamicRegister");
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
NDIS_STATUS
|
|
WanpDoNdisRequest(
|
|
IN NDIS_REQUEST_TYPE RequestType,
|
|
IN NDIS_OID Oid,
|
|
IN PVOID pvInfo,
|
|
IN UINT uiInBufferLen,
|
|
IN PWANARP_NDIS_REQUEST_CONTEXT pRequestContext,
|
|
IN PFNWANARP_REQUEST_COMPLETION_HANDLER pfnCompletionHandler OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Wrapper for the NdisRequest call. We create a request context and submit
|
|
the call to NDIS. If it returns synchronously, we fake asynchronous
|
|
behaviour by call the completion routine.
|
|
|
|
Locks:
|
|
|
|
None. The ndiswan binding must be valid for the duration of the call
|
|
|
|
Arguments:
|
|
|
|
RequestType Type of NDIS request
|
|
Oid Ndis OID to set or get
|
|
pvInfo Opaque OID specific info
|
|
pRequestContext A context for this request
|
|
pfnCompletionHandler Completion handler for this request
|
|
|
|
Return Value:
|
|
|
|
STATUS_INVALID_PARAMETER
|
|
NDIS_STATUS_RESOURCES
|
|
NDIS_STATUS_PENDING
|
|
|
|
other error
|
|
|
|
--*/
|
|
|
|
{
|
|
PNDIS_REQUEST pRequest;
|
|
NDIS_STATUS nsStatus;
|
|
|
|
if((RequestType isnot NdisRequestSetInformation) and
|
|
(RequestType isnot NdisRequestQueryInformation))
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
|
|
pRequestContext->pfnCompletionRoutine = pfnCompletionHandler;
|
|
|
|
pRequest = &(pRequestContext->NdisRequest);
|
|
|
|
pRequest->RequestType = RequestType;
|
|
|
|
if(RequestType is NdisRequestSetInformation)
|
|
{
|
|
pRequest->DATA.SET_INFORMATION.Oid = Oid;
|
|
pRequest->DATA.SET_INFORMATION.InformationBuffer = pvInfo;
|
|
pRequest->DATA.SET_INFORMATION.InformationBufferLength = uiInBufferLen;
|
|
}
|
|
else
|
|
{
|
|
pRequest->DATA.QUERY_INFORMATION.Oid = Oid;
|
|
pRequest->DATA.QUERY_INFORMATION.InformationBuffer = pvInfo;
|
|
pRequest->DATA.QUERY_INFORMATION.InformationBufferLength = uiInBufferLen;
|
|
}
|
|
|
|
//
|
|
// Submit the request.
|
|
//
|
|
|
|
NdisRequest(&nsStatus,
|
|
g_nhNdiswanBinding,
|
|
pRequest);
|
|
|
|
|
|
if(nsStatus isnot NDIS_STATUS_PENDING)
|
|
{
|
|
//
|
|
// If it finished synchronously, call the handler
|
|
//
|
|
|
|
if(pfnCompletionHandler)
|
|
{
|
|
(pfnCompletionHandler)(g_nhNdiswanBinding,
|
|
pRequestContext,
|
|
nsStatus);
|
|
}
|
|
|
|
|
|
if(nsStatus is NDIS_STATUS_SUCCESS)
|
|
{
|
|
//
|
|
// Always make this look asynchronous
|
|
//
|
|
|
|
nsStatus = NDIS_STATUS_PENDING;
|
|
}
|
|
}
|
|
|
|
return nsStatus;
|
|
}
|
|
|
|
VOID
|
|
WanNdisRequestComplete(
|
|
IN NDIS_HANDLE nhHandle,
|
|
IN PNDIS_REQUEST pRequest,
|
|
IN NDIS_STATUS nsStatus
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Handler called by NDIS when it is done processing our request
|
|
|
|
Locks:
|
|
|
|
None
|
|
|
|
Arguments:
|
|
|
|
nhHandle NdisHandle of the adapter to which the request was submitted
|
|
pRequest The original request
|
|
nsStatus The status returned by the adapter
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PWANARP_NDIS_REQUEST_CONTEXT pRequestContext;
|
|
|
|
//
|
|
// Get a pointer to the context
|
|
//
|
|
|
|
pRequestContext = CONTAINING_RECORD(pRequest,
|
|
WANARP_NDIS_REQUEST_CONTEXT,
|
|
NdisRequest);
|
|
|
|
if(pRequestContext->pfnCompletionRoutine is NULL)
|
|
{
|
|
//
|
|
// No more handlers to call, we are done
|
|
//
|
|
|
|
RtFree(pRequestContext);
|
|
|
|
return;
|
|
}
|
|
|
|
//
|
|
// If the request is OID_GEN_TRANSPORT_HEADER_OFFSET, ignore the error
|
|
//
|
|
|
|
RtAssert(pRequest is &(pRequestContext->NdisRequest));
|
|
|
|
if((pRequest->RequestType is NdisRequestSetInformation) and
|
|
(pRequest->DATA.SET_INFORMATION.Oid is OID_GEN_TRANSPORT_HEADER_OFFSET))
|
|
{
|
|
nsStatus = NDIS_STATUS_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// Call the request complete handler
|
|
//
|
|
|
|
(pRequestContext->pfnCompletionRoutine)(nhHandle,
|
|
pRequestContext,
|
|
nsStatus);
|
|
|
|
}
|
|
|
|
PUMODE_INTERFACE
|
|
WanpFindInterfaceGivenIndex(
|
|
DWORD dwIfIndex
|
|
)
|
|
{
|
|
PLIST_ENTRY pleNode;
|
|
PUMODE_INTERFACE pIf;
|
|
|
|
for(pleNode = g_leIfList.Flink;
|
|
pleNode isnot &g_leIfList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pIf = CONTAINING_RECORD(pleNode,
|
|
UMODE_INTERFACE,
|
|
leIfLink);
|
|
|
|
if(pIf->dwIfIndex is dwIfIndex)
|
|
{
|
|
RtAcquireSpinLockAtDpcLevel(&(pIf->rlLock));
|
|
|
|
ReferenceInterface(pIf);
|
|
|
|
return pIf;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
VOID
|
|
WanpDeleteAdapter(
|
|
IN PADAPTER pAdapter
|
|
)
|
|
{
|
|
|
|
#if DBG
|
|
|
|
Trace(ADPT, TRACE,
|
|
("DeleteAdapter: Deleting %x %s\n",
|
|
pAdapter, pAdapter->asDeviceNameA.Buffer));
|
|
|
|
#else
|
|
|
|
Trace(ADPT, TRACE,
|
|
("DeleteAdapter: Deleting %x\n",
|
|
pAdapter));
|
|
|
|
#endif
|
|
|
|
InterlockedDecrement(&g_ulNumAdapters);
|
|
|
|
RtFree(pAdapter);
|
|
}
|
|
|
|
NDIS_STATUS
|
|
WanNdisIpPnPEventHandler(
|
|
IN PNET_PNP_EVENT pNetPnPEvent
|
|
)
|
|
|
|
{
|
|
PWANARP_RECONFIGURE_INFO pInfo;
|
|
NTSTATUS nStatus = NDIS_STATUS_FAILURE;
|
|
|
|
RtAssert(pNetPnPEvent->NetEvent is NetEventReconfigure);
|
|
|
|
pInfo = (PWANARP_RECONFIGURE_INFO) pNetPnPEvent->Buffer;
|
|
|
|
switch (pInfo->wrcOperation)
|
|
{
|
|
case WRC_TCP_WINDOW_SIZE_UPDATE:
|
|
{
|
|
PLIST_ENTRY pleNode;
|
|
KIRQL kirql;
|
|
PADAPTER pAdapter=NULL;
|
|
|
|
RtAssert(pInfo->ulNumInterfaces is 1);
|
|
|
|
if(pInfo->ulNumInterfaces isnot 1)
|
|
{
|
|
break;
|
|
}
|
|
|
|
EnterReader(&g_rwlAdapterLock, &kirql);
|
|
|
|
for(pleNode = g_leMappedAdapterList.Flink;
|
|
pleNode != &g_leMappedAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pAdapter = CONTAINING_RECORD(pleNode, ADAPTER, leAdapterLink);
|
|
|
|
if(IsEqualGUID(&(pAdapter->Guid),
|
|
&pInfo->rgInterfaces[0]))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((pleNode is &g_leMappedAdapterList) || pAdapter==NULL)
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kirql);
|
|
break;
|
|
}
|
|
|
|
if( (pAdapter->byState is AS_MAPPED)
|
|
&& (pAdapter->pInterface)
|
|
&& (pAdapter->pInterface->duUsage is DU_CALLOUT))
|
|
{
|
|
NET_PNP_EVENT pnpEvent;
|
|
IP_PNP_RECONFIG_REQUEST Request;
|
|
|
|
RtlZeroMemory(&pnpEvent, sizeof(pnpEvent));
|
|
RtlZeroMemory(&Request, sizeof(Request));
|
|
Request.version = IP_PNP_RECONFIG_VERSION;
|
|
Request.Flags = IP_PNP_FLAG_INTERFACE_TCP_PARAMETER_UPDATE;
|
|
pnpEvent.NetEvent = NetEventReconfigure;
|
|
pnpEvent.Buffer = &Request;
|
|
pnpEvent.BufferLength = sizeof(Request);
|
|
ReferenceAdapter(pAdapter);
|
|
|
|
ExitReader(&g_rwlAdapterLock, kirql);
|
|
|
|
nStatus = g_pfnIpPnp(pAdapter->pvIpContext,
|
|
&pnpEvent);
|
|
|
|
DereferenceAdapter(pAdapter);
|
|
}
|
|
else
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kirql);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
RtAssert(FALSE);
|
|
nStatus = NDIS_STATUS_NOT_RECOGNIZED;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return nStatus;
|
|
}
|
|
|
|
//
|
|
// Misc ndis interface functions
|
|
//
|
|
|
|
NDIS_STATUS
|
|
WanNdisPnPEvent(
|
|
IN NDIS_HANDLE nhProtocolBindingContext,
|
|
IN PNET_PNP_EVENT pNetPnPEvent
|
|
)
|
|
{
|
|
ULONG ulNumGuids, i;
|
|
NTSTATUS nStatus, nRetStatus;
|
|
|
|
PWANARP_RECONFIGURE_INFO pInfo;
|
|
|
|
TraceEnter(ADPT, "NdisPnPEvent");
|
|
|
|
if(nhProtocolBindingContext isnot (NDIS_HANDLE)0x0CABB1E5)
|
|
{
|
|
return NDIS_STATUS_NOT_RECOGNIZED;
|
|
}
|
|
|
|
if(pNetPnPEvent->NetEvent isnot NetEventReconfigure)
|
|
{
|
|
return NDIS_STATUS_SUCCESS;
|
|
}
|
|
|
|
pInfo = (PWANARP_RECONFIGURE_INFO)(pNetPnPEvent->Buffer);
|
|
|
|
if(pNetPnPEvent->BufferLength < FIELD_OFFSET(WANARP_RECONFIGURE_INFO, rgInterfaces))
|
|
{
|
|
return NDIS_STATUS_BUFFER_TOO_SHORT;
|
|
}
|
|
|
|
pInfo = (PWANARP_RECONFIGURE_INFO)(pNetPnPEvent->Buffer);
|
|
|
|
if(pInfo->wrcOperation isnot WRC_ADD_INTERFACES)
|
|
{
|
|
nStatus = WanNdisIpPnPEventHandler(pNetPnPEvent);
|
|
return nStatus;
|
|
}
|
|
|
|
//
|
|
// Make sure the length and matches with whats in the info
|
|
//
|
|
|
|
ulNumGuids = pNetPnPEvent->BufferLength -
|
|
FIELD_OFFSET(WANARP_RECONFIGURE_INFO, rgInterfaces);
|
|
|
|
ulNumGuids /= sizeof(GUID);
|
|
|
|
if(ulNumGuids < pInfo->ulNumInterfaces)
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Now go through each guid, and if it isnt present, add it
|
|
// If we get a list of only duplicates, return SUCCESS (334575)
|
|
//
|
|
|
|
nRetStatus = STATUS_SUCCESS;
|
|
|
|
for(i = 0; i < pInfo->ulNumInterfaces; i++)
|
|
{
|
|
WCHAR rgwcDeviceBuffer[GUID_STR_LEN + 7 + 2];
|
|
WCHAR rgwcConfigBuffer[GUID_STR_LEN + 29 + 2];
|
|
|
|
PADAPTER pNewAdapter;
|
|
UNICODE_STRING usDevice;
|
|
UNICODE_STRING usConfig;
|
|
|
|
|
|
if(IsBindingPresent(&(pInfo->rgInterfaces[i])))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
RtlZeroMemory(rgwcDeviceBuffer,
|
|
sizeof(rgwcDeviceBuffer));
|
|
|
|
RtlZeroMemory(rgwcConfigBuffer,
|
|
sizeof(rgwcConfigBuffer));
|
|
|
|
//
|
|
// Make sure no problems with people changing the string
|
|
//
|
|
|
|
RtAssert(((wcslen(TCPIP_IF_PREFIX) + 1 + GUID_STR_LEN) * sizeof(WCHAR)) < sizeof(rgwcDeviceBuffer));
|
|
|
|
RtAssert(((wcslen(TCPIP_REG_PREFIX) + 1 + GUID_STR_LEN) * sizeof(WCHAR)) < sizeof(rgwcConfigBuffer));
|
|
|
|
|
|
//
|
|
// Copy over the \Device part
|
|
//
|
|
|
|
RtlCopyMemory(rgwcDeviceBuffer,
|
|
TCPIP_IF_PREFIX,
|
|
wcslen(TCPIP_IF_PREFIX) * sizeof(WCHAR));
|
|
|
|
//
|
|
// Tack on a '\'
|
|
//
|
|
|
|
rgwcDeviceBuffer[wcslen(TCPIP_IF_PREFIX)] = L'\\';
|
|
|
|
//
|
|
// Convert the guid to a string. Just pass the buffer starting
|
|
// after the '\' we just catted. The conversion function returns
|
|
// an upcase string - so thats good.
|
|
//
|
|
|
|
ConvertGuidToString(&(pInfo->rgInterfaces[i]),
|
|
&(rgwcDeviceBuffer[wcslen(TCPIP_IF_PREFIX) + 1]));
|
|
|
|
//
|
|
// Create the config
|
|
//
|
|
|
|
RtlCopyMemory(rgwcConfigBuffer,
|
|
TCPIP_REG_PREFIX,
|
|
wcslen(TCPIP_REG_PREFIX) * sizeof(WCHAR));
|
|
|
|
//
|
|
// Put the device guid at the end
|
|
//
|
|
|
|
ConvertGuidToString(&(pInfo->rgInterfaces[i]),
|
|
&(rgwcConfigBuffer[wcslen(TCPIP_REG_PREFIX)]));
|
|
|
|
//
|
|
// Set up the strings
|
|
//
|
|
|
|
usDevice.Length = wcslen(rgwcDeviceBuffer) * sizeof(WCHAR);
|
|
usDevice.MaximumLength = usDevice.Length;
|
|
usDevice.Buffer = rgwcDeviceBuffer;
|
|
|
|
usConfig.Length = wcslen(rgwcConfigBuffer) * sizeof(WCHAR);
|
|
usConfig.MaximumLength = usConfig.Length;
|
|
usConfig.Buffer = rgwcConfigBuffer;
|
|
|
|
//
|
|
// Create an adapter with this name and config
|
|
//
|
|
|
|
pNewAdapter = NULL;
|
|
|
|
Trace(ADPT, INFO,
|
|
("NdisPnPEvent: Calling create adapter for %S %S\n",
|
|
usConfig.Buffer,
|
|
usDevice.Buffer));
|
|
|
|
nStatus = WanpCreateAdapter(&(pInfo->rgInterfaces[i]),
|
|
&usConfig,
|
|
&usDevice,
|
|
&pNewAdapter);
|
|
|
|
if(nStatus isnot STATUS_SUCCESS)
|
|
{
|
|
Trace(ADPT, ERROR,
|
|
("NdisPnPEvent: Err %x creating adapter for %S (%S)\n",
|
|
nStatus,
|
|
usConfig.Buffer,
|
|
usDevice.Buffer));
|
|
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// If even one succeeds, we return success
|
|
//
|
|
|
|
nRetStatus = STATUS_SUCCESS;
|
|
}
|
|
|
|
}
|
|
|
|
return nRetStatus;
|
|
}
|
|
|
|
VOID
|
|
WanNdisResetComplete(
|
|
NDIS_HANDLE Handle,
|
|
NDIS_STATUS Status
|
|
)
|
|
{
|
|
// We dont do anything here.
|
|
}
|
|
|
|
|
|
|
|
BOOLEAN
|
|
IsBindingPresent(
|
|
IN GUID *pGuid
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Code to catch duplicate bind notifications
|
|
|
|
Locks:
|
|
|
|
acquires adapter list lock
|
|
|
|
Arguments:
|
|
|
|
pGuid Guid of the adapter
|
|
|
|
Return Value:
|
|
|
|
NO_ERROR
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOLEAN bFound;
|
|
PADAPTER pAdapter;
|
|
PLIST_ENTRY pleNode;
|
|
KIRQL kiIrql;
|
|
|
|
|
|
EnterReader(&g_rwlAdapterLock, &kiIrql);
|
|
|
|
for(pleNode = g_leAddedAdapterList.Flink;
|
|
pleNode != &g_leAddedAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pAdapter = CONTAINING_RECORD(pleNode, ADAPTER, leAdapterLink);
|
|
|
|
if(IsEqualGUID(&(pAdapter->Guid),
|
|
pGuid))
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
return TRUE;
|
|
|
|
}
|
|
}
|
|
|
|
for(pleNode = g_leMappedAdapterList.Flink;
|
|
pleNode != &g_leMappedAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pAdapter = CONTAINING_RECORD(pleNode, ADAPTER, leAdapterLink);
|
|
|
|
if(IsEqualGUID(&(pAdapter->Guid),
|
|
pGuid))
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
return TRUE;
|
|
|
|
}
|
|
}
|
|
|
|
for(pleNode = g_leFreeAdapterList.Flink;
|
|
pleNode != &g_leFreeAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pAdapter = CONTAINING_RECORD(pleNode, ADAPTER, leAdapterLink);
|
|
|
|
if(IsEqualGUID(&(pAdapter->Guid),
|
|
pGuid))
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
return TRUE;
|
|
|
|
}
|
|
}
|
|
|
|
for(pleNode = g_leChangeAdapterList.Flink;
|
|
pleNode != &g_leChangeAdapterList;
|
|
pleNode = pleNode->Flink)
|
|
{
|
|
pAdapter = CONTAINING_RECORD(pleNode, ADAPTER, leAdapterLink);
|
|
|
|
if(IsEqualGUID(&(pAdapter->Guid),
|
|
pGuid))
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
return TRUE;
|
|
|
|
}
|
|
}
|
|
|
|
if(g_pServerAdapter)
|
|
{
|
|
if(IsEqualGUID(&(g_pServerAdapter->Guid),
|
|
pGuid))
|
|
{
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
ExitReader(&g_rwlAdapterLock, kiIrql);
|
|
|
|
return FALSE;
|
|
}
|