NT4/private/ntos/tdi/nbf/nbfndis.c
2020-09-30 17:12:29 +02:00

1957 lines
50 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
nbfndis.c
Abstract:
This module contains code which implements the routines used to interface
NBF and NDIS. All callback routines (except for Transfer Data,
Send Complete, and ReceiveIndication) are here, as well as those routines
called to initialize NDIS.
Author:
David Beaver (dbeaver) 13-Feb-1991
Environment:
Kernel mode
Revision History:
David Beaver (dbeaver) 1-July-1991
modify to use new TDI interface
--*/
#include "precomp.h"
#pragma hdrstop
#ifdef NBF_LOCKS // see spnlckdb.c
VOID
NbfFakeSendCompletionHandler(
IN NDIS_HANDLE ProtocolBindingContext,
IN PNDIS_PACKET NdisPacket,
IN NDIS_STATUS NdisStatus
);
VOID
NbfFakeTransferDataComplete (
IN NDIS_HANDLE BindingContext,
IN PNDIS_PACKET NdisPacket,
IN NDIS_STATUS NdisStatus,
IN UINT BytesTransferred
);
#endif
//
// This is a one-per-driver variable used in binding
// to the NDIS interface.
//
NDIS_HANDLE NbfNdisProtocolHandle = (NDIS_HANDLE)NULL;
NDIS_STATUS
NbfSubmitNdisRequest(
IN PDEVICE_CONTEXT DeviceContext,
IN PNDIS_REQUEST NdisRequest,
IN PNDIS_STRING AdapterName
);
VOID
NbfOpenAdapterComplete (
IN NDIS_HANDLE BindingContext,
IN NDIS_STATUS NdisStatus,
IN NDIS_STATUS OpenErrorStatus
);
VOID
NbfCloseAdapterComplete(
IN NDIS_HANDLE NdisBindingContext,
IN NDIS_STATUS Status
);
VOID
NbfResetComplete(
IN NDIS_HANDLE NdisBindingContext,
IN NDIS_STATUS Status
);
VOID
NbfRequestComplete (
IN NDIS_HANDLE BindingContext,
IN PNDIS_REQUEST NdisRequest,
IN NDIS_STATUS NdisStatus
);
VOID
NbfStatusIndication (
IN NDIS_HANDLE NdisBindingContext,
IN NDIS_STATUS NdisStatus,
IN PVOID StatusBuffer,
IN UINT StatusBufferLength
);
VOID
NbfProcessStatusClosing(
IN PVOID Parameter
);
VOID
NbfStatusComplete (
IN NDIS_HANDLE NdisBindingContext
);
#ifdef _PNP_POWER
VOID
NbfProtocolBindAdapter(
OUT PNDIS_STATUS NdisStatus,
IN NDIS_HANDLE BindContext,
IN PNDIS_STRING DeviceName,
IN PVOID SystemSpecific1,
IN PVOID SystemSpecific2
);
VOID
NbfProtocolUnbindAdapter(
OUT PNDIS_STATUS NdisStatus,
IN NDIS_HANDLE ProtocolBindContext,
IN PNDIS_HANDLE UnbindContext
);
#endif
#ifdef ALLOC_PRAGMA
#ifndef _PNP_POWER
#pragma alloc_text(INIT,NbfRegisterProtocol)
#pragma alloc_text(INIT,NbfSubmitNdisRequest)
#pragma alloc_text(INIT,NbfInitializeNdis)
#else // PNP_POWER
#pragma alloc_text(PAGE,NbfProtocolBindAdapter)
#pragma alloc_text(PAGE,NbfProtocolUnbindAdapter)
#pragma alloc_text(PAGE,NbfRegisterProtocol)
#pragma alloc_text(PAGE,NbfSubmitNdisRequest)
#pragma alloc_text(PAGE,NbfInitializeNdis)
#endif
#endif
NTSTATUS
NbfRegisterProtocol (
IN PUNICODE_STRING NameString
)
/*++
Routine Description:
This routine introduces this transport to the NDIS interface.
Arguments:
Irp - Pointer to the request packet representing the I/O request.
Return Value:
The function value is the status of the operation.
STATUS_SUCCESS if all goes well,
Failure status if we tried to register and couldn't,
STATUS_INSUFFICIENT_RESOURCES if we couldn't even try to register.
--*/
{
NDIS_STATUS ndisStatus;
PNDIS_PROTOCOL_CHARACTERISTICS ProtChars; // Used temporarily to register
ProtChars = ExAllocatePoolWithTag(
NonPagedPool,
#ifndef _PNP_POWER
sizeof(NDIS_PROTOCOL_CHARACTERISTICS) +
#else
sizeof(NDIS40_PROTOCOL_CHARACTERISTICS) +
#endif
NameString->MaximumLength,
' FBN');
if (ProtChars == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Set up the characteristics of this protocol
//
#ifndef _PNP_POWER
ProtChars->MajorNdisVersion = 3;
#else
ProtChars->MajorNdisVersion = 4;
ProtChars->ReceivePacketHandler = NULL;
ProtChars->TranslateHandler = NULL;
ProtChars->BindAdapterHandler = NbfProtocolBindAdapter; // FIX ME!!!
ProtChars->UnbindAdapterHandler = NbfProtocolUnbindAdapter; // FIX ME!!!
#endif
ProtChars->MinorNdisVersion = 0;
ProtChars->Name.Length = NameString->Length;
ProtChars->Name.Buffer = (PWCHAR)(ProtChars+1);
RtlCopyMemory (ProtChars->Name.Buffer, NameString->Buffer, NameString->Length);
ProtChars->Name.Buffer[NameString->Length/sizeof(WCHAR)] = UNICODE_NULL;
ProtChars->OpenAdapterCompleteHandler = NbfOpenAdapterComplete;
ProtChars->CloseAdapterCompleteHandler = NbfCloseAdapterComplete;
ProtChars->ResetCompleteHandler = NbfResetComplete;
ProtChars->RequestCompleteHandler = NbfRequestComplete;
#ifdef NBF_LOCKS
ProtChars->SendCompleteHandler = NbfFakeSendCompletionHandler;
ProtChars->TransferDataCompleteHandler = NbfFakeTransferDataComplete;
#else
ProtChars->SendCompleteHandler = NbfSendCompletionHandler;
ProtChars->TransferDataCompleteHandler = NbfTransferDataComplete;
#endif
ProtChars->ReceiveHandler = NbfReceiveIndication;
ProtChars->ReceiveCompleteHandler = NbfReceiveComplete;
ProtChars->StatusHandler = NbfStatusIndication;
ProtChars->StatusCompleteHandler = NbfStatusComplete;
NdisRegisterProtocol (
&ndisStatus,
&NbfNdisProtocolHandle,
ProtChars,
(UINT)sizeof(NDIS_PROTOCOL_CHARACTERISTICS) + NameString->MaximumLength);
ExFreePool (ProtChars);
if (ndisStatus != NDIS_STATUS_SUCCESS) {
#if DBG
IF_NBFDBG (NBF_DEBUG_RESOURCE) {
NbfPrint1("NbfInitialize: NdisRegisterProtocol failed: %s\n",
NbfGetNdisStatus(ndisStatus));
}
#endif
return (NTSTATUS)ndisStatus;
}
return STATUS_SUCCESS;
}
VOID
NbfDeregisterProtocol (
VOID
)
/*++
Routine Description:
This routine removes this transport to the NDIS interface.
Arguments:
None.
Return Value:
None.
--*/
{
NDIS_STATUS ndisStatus;
if (NbfNdisProtocolHandle != (NDIS_HANDLE)NULL) {
NdisDeregisterProtocol (
&ndisStatus,
NbfNdisProtocolHandle);
NbfNdisProtocolHandle = (NDIS_HANDLE)NULL;
}
}
NDIS_STATUS
NbfSubmitNdisRequest(
IN PDEVICE_CONTEXT DeviceContext,
IN PNDIS_REQUEST NdisRequest,
IN PNDIS_STRING AdapterString
)
/*++
Routine Description:
This routine passed an NDIS_REQUEST to the MAC and waits
until it has completed before returning the final status.
Arguments:
DeviceContext - Pointer to the device context for this driver.
NdisRequest - Pointer to the NDIS_REQUEST to submit.
AdapterString - The name of the adapter, in case an error needs
to be logged.
Return Value:
The function value is the status of the operation.
--*/
{
NDIS_STATUS NdisStatus;
NdisRequest(
&NdisStatus,
DeviceContext->NdisBindingHandle,
NdisRequest);
if (NdisStatus == NDIS_STATUS_PENDING) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("OID %lx pended.\n",
NdisRequest->DATA.QUERY_INFORMATION.Oid);
}
//
// The completion routine will set NdisRequestStatus.
//
KeWaitForSingleObject(
&DeviceContext->NdisRequestEvent,
Executive,
KernelMode,
TRUE,
(PLARGE_INTEGER)NULL
);
NdisStatus = DeviceContext->NdisRequestStatus;
KeResetEvent(
&DeviceContext->NdisRequestEvent
);
}
if (NdisStatus == STATUS_SUCCESS) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
if (NdisRequest->RequestType == NdisRequestSetInformation) {
NbfPrint1 ("Nbfdrvr: Set OID %lx succeeded.\n",
NdisRequest->DATA.SET_INFORMATION.Oid);
} else {
NbfPrint1 ("Nbfdrvr: Query OID %lx succeeded.\n",
NdisRequest->DATA.QUERY_INFORMATION.Oid);
}
}
} else {
#if DBG
if (NdisRequest->RequestType == NdisRequestSetInformation) {
NbfPrint2 ("Nbfdrvr: Set OID %lx failed: %s.\n",
NdisRequest->DATA.SET_INFORMATION.Oid, NbfGetNdisStatus(NdisStatus));
} else {
NbfPrint2 ("Nbfdrvr: Query OID %lx failed: %s.\n",
NdisRequest->DATA.QUERY_INFORMATION.Oid, NbfGetNdisStatus(NdisStatus));
}
#endif
NbfWriteOidErrorLog(
DeviceContext,
NdisRequest->RequestType == NdisRequestSetInformation ?
EVENT_TRANSPORT_SET_OID_FAILED : EVENT_TRANSPORT_QUERY_OID_FAILED,
NdisStatus,
AdapterString->Buffer,
NdisRequest->DATA.QUERY_INFORMATION.Oid);
}
return NdisStatus;
}
NTSTATUS
NbfInitializeNdis (
IN PDEVICE_CONTEXT DeviceContext,
IN PCONFIG_DATA NbfConfig,
IN UINT ConfigInfoNameIndex
)
/*++
Routine Description:
This routine introduces this transport to the NDIS interface and sets up
any necessary NDIS data structures (Buffer pools and such). It will be
called for each adapter opened by this transport.
Arguments:
DeviceObject - Pointer to the device object for this driver.
Irp - Pointer to the request packet representing the I/O request.
Return Value:
The function value is the status of the operation.
--*/
{
ULONG SendPacketReservedLength;
ULONG ReceivePacketReservedLen;
ULONG SendPacketPoolSize;
ULONG ReceivePacketPoolSize;
NDIS_STATUS NdisStatus;
NDIS_STATUS OpenErrorStatus;
NDIS_MEDIUM NbfSupportedMedia[] = { NdisMedium802_3, NdisMedium802_5, NdisMediumFddi, NdisMediumWan };
UINT SelectedMedium;
NDIS_REQUEST NbfRequest;
UCHAR NbfDataBuffer[6];
NDIS_OID NbfOid;
UCHAR WanProtocolId[6] = { 0x80, 0x00, 0x00, 0x00, 0x80, 0xd5 };
ULONG WanHeaderFormat = NdisWanHeaderEthernet;
ULONG MinimumLookahead = 128 + sizeof(DLC_FRAME) + sizeof(NBF_HDR_CONNECTIONLESS);
ULONG MacOptions;
PNDIS_STRING AdapterString;
//
// Initialize this adapter for NBF use through NDIS
//
//
// This event is used in case any of the NDIS requests
// pend; we wait until it is set by the completion
// routine, which also sets NdisRequestStatus.
//
KeInitializeEvent(
&DeviceContext->NdisRequestEvent,
NotificationEvent,
FALSE
);
DeviceContext->NdisBindingHandle = NULL;
AdapterString = (PNDIS_STRING)&NbfConfig->Names[ConfigInfoNameIndex];
NdisOpenAdapter (
&NdisStatus,
&OpenErrorStatus,
&DeviceContext->NdisBindingHandle,
&SelectedMedium,
NbfSupportedMedia,
sizeof (NbfSupportedMedia) / sizeof(NDIS_MEDIUM),
NbfNdisProtocolHandle,
(NDIS_HANDLE)DeviceContext,
AdapterString,
0,
NULL);
if (NdisStatus == NDIS_STATUS_PENDING) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("Adapter %S open pended.\n", AdapterString);
}
//
// The completion routine will set NdisRequestStatus.
//
KeWaitForSingleObject(
&DeviceContext->NdisRequestEvent,
Executive,
KernelMode,
TRUE,
(PLARGE_INTEGER)NULL
);
NdisStatus = DeviceContext->NdisRequestStatus;
KeResetEvent(
&DeviceContext->NdisRequestEvent
);
}
if (NdisStatus == NDIS_STATUS_SUCCESS) {
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("Adapter %S successfully opened.\n", AdapterString);
}
#endif
} else {
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint2 ("Adapter open %S failed, status: %s.\n",
AdapterString,
NbfGetNdisStatus (NdisStatus));
}
#endif
NbfWriteGeneralErrorLog(
DeviceContext,
EVENT_TRANSPORT_ADAPTER_NOT_FOUND,
807,
NdisStatus,
AdapterString->Buffer,
0,
NULL);
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Get the information we need about the adapter, based on
// the media type.
//
MacInitializeMacInfo(
NbfSupportedMedia[SelectedMedium],
(BOOLEAN)(NbfConfig->UseDixOverEthernet != 0),
&DeviceContext->MacInfo);
DeviceContext->MacInfo.QueryWithoutSourceRouting =
NbfConfig->QueryWithoutSourceRouting ? TRUE : FALSE;
DeviceContext->MacInfo.AllRoutesNameRecognized =
NbfConfig->AllRoutesNameRecognized ? TRUE : FALSE;
//
// Set the multicast/functional addresses first so we avoid windows where we
// receive only part of the addresses.
//
MacSetNetBIOSMulticast (
DeviceContext->MacInfo.MediumType,
DeviceContext->NetBIOSAddress.Address);
switch (DeviceContext->MacInfo.MediumType) {
case NdisMedium802_3:
case NdisMediumDix:
//
// Fill in the data for our multicast list.
//
RtlCopyMemory(NbfDataBuffer, DeviceContext->NetBIOSAddress.Address, 6);
//
// Now fill in the NDIS_REQUEST.
//
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.SET_INFORMATION.Oid = OID_802_3_MULTICAST_LIST;
NbfRequest.DATA.SET_INFORMATION.InformationBuffer = &NbfDataBuffer;
NbfRequest.DATA.SET_INFORMATION.InformationBufferLength = 6;
break;
case NdisMedium802_5:
//
// For token-ring, we pass the last four bytes of the
// Netbios functional address.
//
//
// Fill in the OVB for our functional address.
//
RtlCopyMemory(NbfDataBuffer, ((PUCHAR)(DeviceContext->NetBIOSAddress.Address)) + 2, 4);
//
// Now fill in the NDIS_REQUEST.
//
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.SET_INFORMATION.Oid = OID_802_5_CURRENT_FUNCTIONAL;
NbfRequest.DATA.SET_INFORMATION.InformationBuffer = &NbfDataBuffer;
NbfRequest.DATA.SET_INFORMATION.InformationBufferLength = 4;
break;
case NdisMediumFddi:
//
// Fill in the data for our multicast list.
//
RtlCopyMemory(NbfDataBuffer, DeviceContext->NetBIOSAddress.Address, 6);
//
// Now fill in the NDIS_REQUEST.
//
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.SET_INFORMATION.Oid = OID_FDDI_LONG_MULTICAST_LIST;
NbfRequest.DATA.SET_INFORMATION.InformationBuffer = &NbfDataBuffer;
NbfRequest.DATA.SET_INFORMATION.InformationBufferLength = 6;
break;
}
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
switch (DeviceContext->MacInfo.MediumType) {
case NdisMedium802_3:
case NdisMediumDix:
if (DeviceContext->MacInfo.MediumAsync) {
NbfOid = OID_WAN_CURRENT_ADDRESS;
} else {
NbfOid = OID_802_3_CURRENT_ADDRESS;
}
break;
case NdisMedium802_5:
NbfOid = OID_802_5_CURRENT_ADDRESS;
break;
case NdisMediumFddi:
NbfOid = OID_FDDI_LONG_CURRENT_ADDR;
break;
default:
NdisStatus = NDIS_STATUS_FAILURE;
break;
}
NbfRequest.RequestType = NdisRequestQueryInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = NbfOid;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = DeviceContext->LocalAddress.Address;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 6;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Set up the reserved Netbios address.
//
RtlZeroMemory(DeviceContext->ReservedNetBIOSAddress, 10);
RtlCopyMemory(&DeviceContext->ReservedNetBIOSAddress[10], DeviceContext->LocalAddress.Address, 6);
//
// Now query the maximum packet sizes.
//
NbfRequest.RequestType = NdisRequestQueryInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_MAXIMUM_FRAME_SIZE;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &(DeviceContext->MaxReceivePacketSize);
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
NbfRequest.RequestType = NdisRequestQueryInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_MAXIMUM_TOTAL_SIZE;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &(DeviceContext->MaxSendPacketSize);
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
DeviceContext->CurSendPacketSize = DeviceContext->MaxSendPacketSize;
//
// Now set the minimum lookahead size.
//
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_CURRENT_LOOKAHEAD;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &MinimumLookahead;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Now query the link speed for non-wan media
//
if (!DeviceContext->MacInfo.MediumAsync) {
NbfRequest.RequestType = NdisRequestQueryInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_LINK_SPEED;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &(DeviceContext->MediumSpeed);
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
DeviceContext->MediumSpeedAccurate = TRUE;
DeviceContext->MinimumT1Timeout = 8; // == 400 ms
} else {
//
// On an wan media, this isn't valid until we get an
// WAN_LINE_UP indication. Set the timeouts to
// low values for now.
//
DeviceContext->DefaultT1Timeout = 8;
DeviceContext->MinimumT1Timeout = 8;
DeviceContext->MediumSpeedAccurate = FALSE;
//
// Back off our connectionless timeouts to 2 seconds.
//
DeviceContext->NameQueryTimeout = 2 * SECONDS;
DeviceContext->AddNameQueryTimeout = 2 * SECONDS;
DeviceContext->GeneralTimeout = 2 * SECONDS;
//
// Use the WAN parameter for name query retries.
//
DeviceContext->NameQueryRetries = NbfConfig->WanNameQueryRetries;
//
// Use this until we know better.
//
DeviceContext->RecommendedSendWindow = 1;
}
//
// On media that use source routing, we double our name query
// retry count if we are configured to try both ways (with and
// without source routing).
//
if ((DeviceContext->MacInfo.QueryWithoutSourceRouting) &&
(DeviceContext->MacInfo.SourceRouting)) {
DeviceContext->NameQueryRetries *= 2;
}
//
// For wan, specify our protocol ID and header format.
// We don't query the medium subtype because we don't
// case (since we require ethernet emulation).
//
if (DeviceContext->MacInfo.MediumAsync) {
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_WAN_PROTOCOL_TYPE;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = WanProtocolId;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 6;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_WAN_HEADER_FORMAT;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &WanHeaderFormat;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
}
//
// Now query the MAC's optional characteristics.
//
NbfRequest.RequestType = NdisRequestQueryInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_MAC_OPTIONS;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &MacOptions;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
#if 1
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
#else
MacOptions = 0;
#endif
}
DeviceContext->MacInfo.CopyLookahead =
(BOOLEAN)((MacOptions & NDIS_MAC_OPTION_COPY_LOOKAHEAD_DATA) != 0);
DeviceContext->MacInfo.ReceiveSerialized =
(BOOLEAN)((MacOptions & NDIS_MAC_OPTION_RECEIVE_SERIALIZED) != 0);
DeviceContext->MacInfo.TransferSynchronous =
(BOOLEAN)((MacOptions & NDIS_MAC_OPTION_TRANSFERS_NOT_PEND) != 0);
DeviceContext->MacInfo.SingleReceive =
(BOOLEAN)(DeviceContext->MacInfo.ReceiveSerialized && DeviceContext->MacInfo.TransferSynchronous);
#if 0
//
// Now set our options if needed.
//
// Don't allow early indications because we can't determine
// if the CRC has been checked yet.
//
if ((DeviceContext->MacInfo.MediumType == NdisMedium802_3) ||
(DeviceContext->MacInfo.MediumType == NdisMediumDix)) {
ULONG ProtocolOptions = NDIS_PROT_OPTION_ESTIMATED_LENGTH;
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.QUERY_INFORMATION.Oid = OID_GEN_PROTOCOL_OPTIONS;
NbfRequest.DATA.QUERY_INFORMATION.InformationBuffer = &ProtocolOptions;
NbfRequest.DATA.QUERY_INFORMATION.InformationBufferLength = 4;
NdisStatus = NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
}
#endif
//
// Calculate the NDIS-related stuff.
//
SendPacketReservedLength = sizeof (SEND_PACKET_TAG);
ReceivePacketReservedLen = sizeof (RECEIVE_PACKET_TAG);
//
// The send packet pool is used for UI frames and regular packets.
//
SendPacketPoolSize = NbfConfig->SendPacketPoolSize;
//
// The receive packet pool is used in transfer data.
//
// For a MAC that will only have one receive active, we
// don't need multiple receive packets. Allow an extra
// one for loopback.
//
if (DeviceContext->MacInfo.SingleReceive) {
ReceivePacketPoolSize = 2;
} else {
ReceivePacketPoolSize = NbfConfig->ReceivePacketPoolSize;
}
// Allocate Packet pool descriptors for dynamic packet allocation.
DeviceContext->SendPacketPoolDesc = ExAllocatePoolWithTag(
NonPagedPool,
sizeof(NBF_POOL_LIST_DESC),
' FBN');
if (DeviceContext->SendPacketPoolDesc == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlZeroMemory(DeviceContext->SendPacketPoolDesc,
sizeof(NBF_POOL_LIST_DESC));
DeviceContext->SendPacketPoolDesc->NumElements =
DeviceContext->SendPacketPoolDesc->TotalElements = (USHORT)SendPacketPoolSize;
NdisAllocatePacketPool (
&NdisStatus,
&DeviceContext->SendPacketPoolDesc->PoolHandle,
SendPacketPoolSize,
SendPacketReservedLength);
if (NdisStatus == NDIS_STATUS_SUCCESS) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint0 ("NdisInitializePacketPool successful.\n");
}
} else {
#if DBG
NbfPrint1 ("NbfInitialize: NdisInitializePacketPool failed, reason: %s.\n",
NbfGetNdisStatus (NdisStatus));
#endif
NbfWriteResourceErrorLog(
DeviceContext,
EVENT_TRANSPORT_RESOURCE_POOL,
109,
SendPacketPoolSize,
0);
ExFreePool (DeviceContext->SendPacketPoolDesc);
DeviceContext->SendPacketPoolDesc = NULL;
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
DeviceContext->SendPacketPoolSize = SendPacketPoolSize;
DeviceContext->MemoryUsage +=
(SendPacketPoolSize *
(sizeof(NDIS_PACKET) + SendPacketReservedLength));
#if DBG
IF_NBFDBG (NBF_DEBUG_DYNAMIC) {
DbgPrint ("send pool %d hdr %d, %ld\n",
SendPacketPoolSize,
SendPacketReservedLength,
DeviceContext->MemoryUsage);
}
#endif
// Allocate Packet pool descriptors for dynamic packet allocation.
DeviceContext->ReceivePacketPoolDesc = ExAllocatePoolWithTag(
NonPagedPool,
sizeof(NBF_POOL_LIST_DESC),
' FBN');
if (DeviceContext->ReceivePacketPoolDesc == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
RtlZeroMemory(DeviceContext->ReceivePacketPoolDesc,
sizeof(NBF_POOL_LIST_DESC));
DeviceContext->ReceivePacketPoolDesc->NumElements =
DeviceContext->ReceivePacketPoolDesc->TotalElements = (USHORT)ReceivePacketPoolSize;
NdisAllocatePacketPool(
&NdisStatus,
&DeviceContext->ReceivePacketPoolDesc->PoolHandle,
ReceivePacketPoolSize,
ReceivePacketReservedLen);
if (NdisStatus == NDIS_STATUS_SUCCESS) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("NdisInitializePacketPool successful, Pool: %lx\n",
DeviceContext->ReceivePacketPoolDesc->PoolHandle);
}
} else {
#if DBG
NbfPrint1 ("NbfInitialize: NdisInitializePacketPool failed, reason: %s.\n",
NbfGetNdisStatus (NdisStatus));
#endif
ExFreePool (DeviceContext->SendPacketPoolDesc);
ExFreePool(DeviceContext->ReceivePacketPoolDesc);
DeviceContext->SendPacketPoolDesc = NULL;
DeviceContext->ReceivePacketPoolDesc = NULL;
NbfCloseNdis (DeviceContext);
NbfWriteResourceErrorLog(
DeviceContext,
EVENT_TRANSPORT_RESOURCE_POOL,
209,
ReceivePacketPoolSize,
0);
return STATUS_INSUFFICIENT_RESOURCES;
}
DeviceContext->ReceivePacketPoolSize = ReceivePacketPoolSize;
DeviceContext->MemoryUsage +=
(ReceivePacketPoolSize *
(sizeof(NDIS_PACKET) + ReceivePacketReservedLen));
#if DBG
IF_NBFDBG (NBF_DEBUG_DYNAMIC) {
DbgPrint ("receive pool %d hdr %d, %ld\n",
ReceivePacketPoolSize,
ReceivePacketReservedLen,
DeviceContext->MemoryUsage);
}
#endif
//
// Allocate the buffer pool; as an estimate, allocate
// one per send or receive packet.
//
NdisAllocateBufferPool (
&NdisStatus,
&DeviceContext->NdisBufferPool,
SendPacketPoolSize + ReceivePacketPoolSize);
if (NdisStatus == NDIS_STATUS_SUCCESS) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint0 ("NdisAllocateBufferPool successful.\n");
}
} else {
#if DBG
NbfPrint1 ("NbfInitialize: NdisAllocateBufferPool failed, reason: %s.\n",
NbfGetNdisStatus (NdisStatus));
#endif
ExFreePool(DeviceContext->SendPacketPoolDesc);
ExFreePool(DeviceContext->ReceivePacketPoolDesc);
DeviceContext->SendPacketPoolDesc = NULL;
DeviceContext->ReceivePacketPoolDesc = NULL;
DeviceContext->NdisBufferPool = NULL;
NbfCloseNdis (DeviceContext);
NbfWriteResourceErrorLog(
DeviceContext,
EVENT_TRANSPORT_RESOURCE_POOL,
309,
SendPacketPoolSize + ReceivePacketPoolSize,
0);
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Now that everything is set up, we enable the filter
// for packet reception.
//
//
// Fill in the OVB for packet filter.
//
switch (DeviceContext->MacInfo.MediumType) {
case NdisMedium802_3:
case NdisMediumDix:
case NdisMediumFddi:
RtlStoreUlong((PULONG)NbfDataBuffer,
(NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_MULTICAST));
break;
case NdisMedium802_5:
RtlStoreUlong((PULONG)NbfDataBuffer,
(NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_FUNCTIONAL));
break;
default:
NdisStatus = NDIS_STATUS_FAILURE;
break;
}
//
// Now fill in the NDIS_REQUEST.
//
NbfRequest.RequestType = NdisRequestSetInformation;
NbfRequest.DATA.SET_INFORMATION.Oid = OID_GEN_CURRENT_PACKET_FILTER;
NbfRequest.DATA.SET_INFORMATION.InformationBuffer = &NbfDataBuffer;
NbfRequest.DATA.SET_INFORMATION.InformationBufferLength = sizeof(ULONG);
NbfSubmitNdisRequest (DeviceContext, &NbfRequest, AdapterString);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
NbfCloseNdis (DeviceContext);
return STATUS_INSUFFICIENT_RESOURCES;
}
return STATUS_SUCCESS;
} /* NbfInitializeNdis */
VOID
NbfCloseNdis (
IN PDEVICE_CONTEXT DeviceContext
)
/*++
Routine Description:
This routine unbinds the transport from the NDIS interface and does
any other work required to undo what was done in NbfInitializeNdis.
It is written so that it can be called from within NbfInitializeNdis
if it fails partway through.
Arguments:
DeviceObject - Pointer to the device object for this driver.
Return Value:
The function value is the status of the operation.
--*/
{
NDIS_STATUS ndisStatus;
//
// Close the NDIS binding.
//
if (DeviceContext->NdisBindingHandle != (NDIS_HANDLE)NULL) {
//
// This event is used in case any of the NDIS requests
// pend; we wait until it is set by the completion
// routine, which also sets NdisRequestStatus.
//
KeInitializeEvent(
&DeviceContext->NdisRequestEvent,
NotificationEvent,
FALSE
);
NdisCloseAdapter(
&ndisStatus,
DeviceContext->NdisBindingHandle);
if (ndisStatus == NDIS_STATUS_PENDING) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint0 ("Adapter close pended.\n");
}
//
// The completion routine will set NdisRequestStatus.
//
KeWaitForSingleObject(
&DeviceContext->NdisRequestEvent,
Executive,
KernelMode,
TRUE,
(PLARGE_INTEGER)NULL
);
ndisStatus = DeviceContext->NdisRequestStatus;
KeResetEvent(
&DeviceContext->NdisRequestEvent
);
}
//
// We ignore ndisStatus.
//
}
if (DeviceContext->SendPacketPoolDesc != NULL &&
DeviceContext->SendPacketPoolDesc->PoolHandle != NULL) {
NdisFreePacketPool (DeviceContext->SendPacketPoolDesc->PoolHandle);
ExFreePool(DeviceContext->SendPacketPoolDesc);
DeviceContext->SendPacketPoolDesc = NULL;
}
if (DeviceContext->ReceivePacketPoolDesc != NULL &&
DeviceContext->ReceivePacketPoolDesc->PoolHandle != NULL) {
NdisFreePacketPool (DeviceContext->ReceivePacketPoolDesc->PoolHandle);
ExFreePool(DeviceContext->ReceivePacketPoolDesc);
DeviceContext->ReceivePacketPoolDesc = NULL;
}
if (DeviceContext->NdisBufferPool != NULL) {
NdisFreeBufferPool (DeviceContext->NdisBufferPool);
}
} /* NbfCloseNdis */
VOID
NbfOpenAdapterComplete (
IN NDIS_HANDLE BindingContext,
IN NDIS_STATUS NdisStatus,
IN NDIS_STATUS OpenErrorStatus
)
/*++
Routine Description:
This routine is called by NDIS to indicate that an open adapter
is complete. Since we only ever have one outstanding, and then only
during initialization, all we do is record the status and set
the event to signalled to unblock the initialization thread.
Arguments:
BindingContext - Pointer to the device object for this driver.
NdisStatus - The request completion code.
OpenErrorStatus - More status information.
Return Value:
None.
--*/
{
PDEVICE_CONTEXT DeviceContext = (PDEVICE_CONTEXT)BindingContext;
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("Nbfdrvr: NbfOpenAdapterCompleteNDIS Status: %s\n",
NbfGetNdisStatus (NdisStatus));
}
#endif
ENTER_NBF;
DeviceContext->NdisRequestStatus = NdisStatus;
KeSetEvent(
&DeviceContext->NdisRequestEvent,
0L,
FALSE);
LEAVE_NBF;
return;
}
VOID
NbfCloseAdapterComplete (
IN NDIS_HANDLE BindingContext,
IN NDIS_STATUS NdisStatus
)
/*++
Routine Description:
This routine is called by NDIS to indicate that a close adapter
is complete. Currently we don't close adapters, so this is not
a problem.
Arguments:
BindingContext - Pointer to the device object for this driver.
NdisStatus - The request completion code.
Return Value:
None.
--*/
{
PDEVICE_CONTEXT DeviceContext = (PDEVICE_CONTEXT)BindingContext;
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("Nbfdrvr: NbfCloseAdapterCompleteNDIS Status: %s\n",
NbfGetNdisStatus (NdisStatus));
}
#endif
ENTER_NBF;
DeviceContext->NdisRequestStatus = NdisStatus;
KeSetEvent(
&DeviceContext->NdisRequestEvent,
0L,
FALSE);
LEAVE_NBF;
return;
}
VOID
NbfResetComplete (
IN NDIS_HANDLE BindingContext,
IN NDIS_STATUS NdisStatus
)
/*++
Routine Description:
This routine is called by NDIS to indicate that a reset adapter
is complete. Currently we don't reset adapters, so this is not
a problem.
Arguments:
BindingContext - Pointer to the device object for this driver.
NdisStatus - The request completion code.
Return Value:
None.
--*/
{
UNREFERENCED_PARAMETER(BindingContext);
UNREFERENCED_PARAMETER(NdisStatus);
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint1 ("Nbfdrvr: NbfResetCompleteNDIS Status: %s\n",
NbfGetNdisStatus (NdisStatus));
}
#endif
return;
}
VOID
NbfRequestComplete (
IN NDIS_HANDLE BindingContext,
IN PNDIS_REQUEST NdisRequest,
IN NDIS_STATUS NdisStatus
)
/*++
Routine Description:
This routine is called by NDIS to indicate that a request is complete.
Since we only ever have one request outstanding, and then only
during initialization, all we do is record the status and set
the event to signalled to unblock the initialization thread.
Arguments:
BindingContext - Pointer to the device object for this driver.
NdisRequest - The object describing the request.
NdisStatus - The request completion code.
Return Value:
None.
--*/
{
PDEVICE_CONTEXT DeviceContext = (PDEVICE_CONTEXT)BindingContext;
#if DBG
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint2 ("Nbfdrvr: NbfRequestComplete request: %i, NDIS Status: %s\n",
NdisRequest->RequestType,NbfGetNdisStatus (NdisStatus));
}
#endif
ENTER_NBF;
DeviceContext->NdisRequestStatus = NdisStatus;
KeSetEvent(
&DeviceContext->NdisRequestEvent,
0L,
FALSE);
LEAVE_NBF;
return;
}
VOID
NbfStatusIndication (
IN NDIS_HANDLE NdisBindingContext,
IN NDIS_STATUS NdisStatus,
IN PVOID StatusBuffer,
IN UINT StatusBufferSize
)
{
PDEVICE_CONTEXT DeviceContext;
PNDIS_WAN_LINE_UP LineUp;
KIRQL oldirql;
PTP_LINK Link;
DeviceContext = (PDEVICE_CONTEXT)NdisBindingContext;
KeRaiseIrql (DISPATCH_LEVEL, &oldirql);
switch (NdisStatus) {
case NDIS_STATUS_WAN_LINE_UP:
//
// A wan line is connected.
//
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->SpinLock);
//
// If this happens before we are ready, then make
// a note of it, otherwise make the device ready.
//
DeviceContext->MediumSpeedAccurate = TRUE;
LineUp = (PNDIS_WAN_LINE_UP)StatusBuffer;
//
// See if this is a new lineup for this protocol type
//
if (LineUp->ProtocolType == 0x80D5) {
NDIS_HANDLE TransportHandle;
*((ULONG UNALIGNED *)(&TransportHandle)) =
*((ULONG UNALIGNED *)(&LineUp->LocalAddress[2]));
//
// See if this is a new lineup
//
if (TransportHandle == NULL) {
*((ULONG UNALIGNED *)(&LineUp->LocalAddress[2])) = *((ULONG UNALIGNED *)(&DeviceContext));
// ETH_COPY_NETWORK_ADDRESS(DeviceContext->LocalAddress.Address, LineUp->LocalAddress);
// ETH_COPY_NETWORK_ADDRESS(&DeviceContext->ReservedNetBIOSAddress[10], DeviceContext->LocalAddress.Address);
}
//
// Calculate minimum link timeouts based on the speed,
// which is passed in StatusBuffer.
//
// The formula is (max_frame_size * 2) / speed + 0.4 sec.
// This expands to
//
// MFS (bytes) * 2 8 bits
// ------------------- x ------ == timeout (sec),
// speed (100 bits/sec) byte
//
// which is (MFS * 16 / 100) / speed. We then convert it into
// the 50 ms units that NBF uses and add 8 (which is
// 0.4 seconds in 50 ms units).
//
// As a default timeout we use the min + 0.2 seconds
// unless the configured default is more.
//
if (LineUp->LinkSpeed > 0) {
DeviceContext->MediumSpeed = LineUp->LinkSpeed;
}
if (LineUp->MaximumTotalSize > 0) {
#if DBG
if (LineUp->MaximumTotalSize > DeviceContext->MaxSendPacketSize) {
DbgPrint ("Nbf: Bad LINE_UP size, %d (> %d)\n",
LineUp->MaximumTotalSize, DeviceContext->MaxSendPacketSize);
}
if (LineUp->MaximumTotalSize < 128) {
DbgPrint ("NBF: Bad LINE_UP size, %d (< 128)\n",
LineUp->MaximumTotalSize);
}
#endif
DeviceContext->CurSendPacketSize = LineUp->MaximumTotalSize;
}
if (LineUp->SendWindow == 0) {
DeviceContext->RecommendedSendWindow = 3;
} else {
DeviceContext->RecommendedSendWindow = LineUp->SendWindow + 1;
}
DeviceContext->MinimumT1Timeout =
((((DeviceContext->CurSendPacketSize * 16) / 100) / DeviceContext->MediumSpeed) *
((1 * SECONDS) / (50 * MILLISECONDS))) + 8;
if (DeviceContext->DefaultT1Timeout < DeviceContext->MinimumT1Timeout) {
DeviceContext->DefaultT1Timeout = DeviceContext->MinimumT1Timeout + 4;
}
}
RELEASE_DPC_SPIN_LOCK (&DeviceContext->SpinLock);
break;
case NDIS_STATUS_WAN_LINE_DOWN:
//
// An wan line is disconnected.
//
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->SpinLock);
DeviceContext->MediumSpeedAccurate = FALSE;
//
// Set the timeouts to small values (0.4 seconds)
//
DeviceContext->DefaultT1Timeout = 8;
DeviceContext->MinimumT1Timeout = 8;
RELEASE_DPC_SPIN_LOCK (&DeviceContext->SpinLock);
//
// Stop the link on this device context (there
// will only be one).
//
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
if (DeviceContext->LinkTreeElements > 0) {
Link = (PTP_LINK)DeviceContext->LinkTreeRoot;
if ((Link->DeferredFlags & LINK_FLAGS_DEFERRED_DELETE) == 0) {
NbfReferenceLink ("Wan line down", Link, LREF_TREE);
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
//
// Put the link in ADM to shut it down.
//
ACQUIRE_DPC_SPIN_LOCK (&Link->SpinLock);
if (Link->State != LINK_STATE_ADM) {
Link->State = LINK_STATE_ADM;
RELEASE_DPC_SPIN_LOCK (&Link->SpinLock);
NbfDereferenceLinkSpecial ("Wan line down", Link, LREF_NOT_ADM);
} else {
RELEASE_DPC_SPIN_LOCK (&Link->SpinLock);
}
//
// Now stop it to destroy all connections on it.
//
NbfStopLink (Link);
NbfDereferenceLink ("Wan line down", Link, LREF_TREE);
} else {
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
}
} else {
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
}
break;
case NDIS_STATUS_WAN_FRAGMENT:
//
// A fragment has been received on the wan line.
// Send a reject back to him.
//
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
if (DeviceContext->LinkTreeElements > 0) {
Link = (PTP_LINK)DeviceContext->LinkTreeRoot;
NbfReferenceLink ("Async line down", Link, LREF_TREE);
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
ACQUIRE_DPC_SPIN_LOCK (&Link->SpinLock);
NbfSendRej (Link, FALSE, FALSE); // release lock
NbfDereferenceLink ("Async line down", Link, LREF_TREE);
} else {
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
}
break;
case NDIS_STATUS_CLOSING:
//
// The adapter is shutting down. We queue a worker
// thread to handle this.
//
ExInitializeWorkItem(
&DeviceContext->StatusClosingQueueItem,
NbfProcessStatusClosing,
(PVOID)DeviceContext);
ExQueueWorkItem(&DeviceContext->StatusClosingQueueItem, DelayedWorkQueue);
break;
default:
break;
}
KeLowerIrql (oldirql);
}
VOID
NbfProcessStatusClosing(
IN PVOID Parameter
)
/*++
Routine Description:
This is the thread routine which restarts packetizing
that has been delayed on WAN to allow RRs to come in.
This is very similar to PacketizeConnections.
Arguments:
Parameter - A pointer to the device context.
Return Value:
None.
--*/
{
PDEVICE_CONTEXT DeviceContext;
PLIST_ENTRY p;
#if 0
PTP_ADDRESS Address;
#endif
PTP_LINK Link;
PTP_REQUEST Request;
NDIS_STATUS ndisStatus;
KIRQL oldirql;
DeviceContext = (PDEVICE_CONTEXT)Parameter;
//
// Prevent new activity on the connection.
//
DeviceContext->State = DEVICECONTEXT_STATE_DOWN;
#if 0
//
// Stop all the addresses.
//
while ((p = ExInterlockedRemoveHeadList(
&DeviceContext->AddressDatabase,
&DeviceContext->SpinLock)) != NULL) {
Address = CONTAINING_RECORD (p, TP_ADDRESS, Linkage);
InitializeListHead(p);
NbfStopAddress (Address);
}
#endif
//
// To speed things along, stop all the links too.
//
KeRaiseIrql (DISPATCH_LEVEL, &oldirql);
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
DeviceContext->LastLink = NULL;
while (DeviceContext->LinkTreeRoot != NULL) {
Link = (PTP_LINK)DeviceContext->LinkTreeRoot;
DeviceContext->LinkTreeRoot = RtlDelete ((PRTL_SPLAY_LINKS)Link);
DeviceContext->LinkTreeElements--;
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->TimerSpinLock);
if (Link->OnShortList) {
RemoveEntryList (&Link->ShortList);
}
if (Link->OnLongList) {
RemoveEntryList (&Link->LongList);
}
RELEASE_DPC_SPIN_LOCK (&DeviceContext->TimerSpinLock);
ACQUIRE_DPC_SPIN_LOCK (&Link->SpinLock);
if (Link->State != LINK_STATE_ADM) {
Link->State = LINK_STATE_ADM;
NbfSendDm (Link, FALSE); // send DM/0, release lock
// moving to ADM, remove reference
NbfDereferenceLinkSpecial("Expire T1 in CONNECTING mode", Link, LREF_NOT_ADM);
} else {
RELEASE_DPC_SPIN_LOCK (&Link->SpinLock);
}
NbfStopLink (Link);
ACQUIRE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
}
RELEASE_DPC_SPIN_LOCK (&DeviceContext->LinkSpinLock);
KeLowerIrql (oldirql);
//
// Shutdown the control channel.
//
while ((p = ExInterlockedRemoveHeadList(
&DeviceContext->QueryIndicationQueue,
&DeviceContext->SpinLock)) != NULL) {
Request = CONTAINING_RECORD (p, TP_REQUEST, Linkage);
NbfCompleteRequest (Request, STATUS_INVALID_DEVICE_STATE, 0);
}
while ((p = ExInterlockedRemoveHeadList(
&DeviceContext->DatagramIndicationQueue,
&DeviceContext->SpinLock)) != NULL) {
Request = CONTAINING_RECORD (p, TP_REQUEST, Linkage);
NbfCompleteRequest (Request, STATUS_INVALID_DEVICE_STATE, 0);
}
while ((p = ExInterlockedRemoveHeadList(
&DeviceContext->StatusQueryQueue,
&DeviceContext->SpinLock)) != NULL) {
Request = CONTAINING_RECORD (p, TP_REQUEST, Linkage);
NbfCompleteRequest (Request, STATUS_INVALID_DEVICE_STATE, 0);
}
while ((p = ExInterlockedRemoveHeadList(
&DeviceContext->FindNameQueue,
&DeviceContext->SpinLock)) != NULL) {
Request = CONTAINING_RECORD (p, TP_REQUEST, Linkage);
NbfCompleteRequest (Request, STATUS_INVALID_DEVICE_STATE, 0);
}
//
// Close the NDIS binding.
//
KeInitializeEvent(
&DeviceContext->NdisRequestEvent,
NotificationEvent,
FALSE
);
NdisCloseAdapter(
&ndisStatus,
DeviceContext->NdisBindingHandle);
if (ndisStatus == NDIS_STATUS_PENDING) {
IF_NBFDBG (NBF_DEBUG_NDIS) {
NbfPrint0 ("Adapter close pended.\n");
}
//
// The completion routine will set NdisRequestStatus.
//
KeWaitForSingleObject(
&DeviceContext->NdisRequestEvent,
Executive,
KernelMode,
TRUE,
(PLARGE_INTEGER)NULL
);
ndisStatus = DeviceContext->NdisRequestStatus;
KeResetEvent(
&DeviceContext->NdisRequestEvent
);
}
DeviceContext->NdisBindingHandle = NULL;
//
// We ignore ndisStatus.
//
#if 0
//
// Remove all the storage associated with the device.
//
NbfFreeResources (DeviceContext);
NdisFreePacketPool (DeviceContext->SendPacketPoolHandle);
NdisFreePacketPool (DeviceContext->ReceivePacketPoolHandle);
NdisFreeBufferPool (DeviceContext->NdisBufferPoolHandle);
#endif
//
// And remove the creation reference from the device
// context.
//
NbfDereferenceDeviceContext ("Unload", DeviceContext, DCREF_CREATION);
} /* NbfProcessStatusClosing */
VOID
NbfStatusComplete (
IN NDIS_HANDLE NdisBindingContext
)
{
UNREFERENCED_PARAMETER (NdisBindingContext);
}
#if DBG
PUCHAR
NbfGetNdisStatus(
NDIS_STATUS GeneralStatus
)
/*++
Routine Description:
This routine returns a pointer to the string describing the NDIS error
denoted by GeneralStatus.
Arguments:
GeneralStatus - the status you wish to make readable.
Return Value:
None.
--*/
{
static NDIS_STATUS Status[] = {
NDIS_STATUS_SUCCESS,
NDIS_STATUS_PENDING,
NDIS_STATUS_ADAPTER_NOT_FOUND,
NDIS_STATUS_ADAPTER_NOT_OPEN,
NDIS_STATUS_ADAPTER_NOT_READY,
NDIS_STATUS_ADAPTER_REMOVED,
NDIS_STATUS_BAD_CHARACTERISTICS,
NDIS_STATUS_BAD_VERSION,
NDIS_STATUS_CLOSING,
NDIS_STATUS_DEVICE_FAILED,
NDIS_STATUS_FAILURE,
NDIS_STATUS_INVALID_DATA,
NDIS_STATUS_INVALID_LENGTH,
NDIS_STATUS_INVALID_OID,
NDIS_STATUS_INVALID_PACKET,
NDIS_STATUS_MULTICAST_FULL,
NDIS_STATUS_NOT_INDICATING,
NDIS_STATUS_NOT_RECOGNIZED,
NDIS_STATUS_NOT_RESETTABLE,
NDIS_STATUS_NOT_SUPPORTED,
NDIS_STATUS_OPEN_FAILED,
NDIS_STATUS_OPEN_LIST_FULL,
NDIS_STATUS_REQUEST_ABORTED,
NDIS_STATUS_RESET_IN_PROGRESS,
NDIS_STATUS_RESOURCES,
NDIS_STATUS_UNSUPPORTED_MEDIA
};
static PUCHAR String[] = {
"SUCCESS",
"PENDING",
"ADAPTER_NOT_FOUND",
"ADAPTER_NOT_OPEN",
"ADAPTER_NOT_READY",
"ADAPTER_REMOVED",
"BAD_CHARACTERISTICS",
"BAD_VERSION",
"CLOSING",
"DEVICE_FAILED",
"FAILURE",
"INVALID_DATA",
"INVALID_LENGTH",
"INVALID_OID",
"INVALID_PACKET",
"MULTICAST_FULL",
"NOT_INDICATING",
"NOT_RECOGNIZED",
"NOT_RESETTABLE",
"NOT_SUPPORTED",
"OPEN_FAILED",
"OPEN_LIST_FULL",
"REQUEST_ABORTED",
"RESET_IN_PROGRESS",
"RESOURCES",
"UNSUPPORTED_MEDIA"
};
static UCHAR BadStatus[] = "UNDEFINED";
#define StatusCount (sizeof(Status)/sizeof(NDIS_STATUS))
INT i;
for (i=0; i<StatusCount; i++)
if (GeneralStatus == Status[i])
return String[i];
return BadStatus;
#undef StatusCount
}
#endif