windows-nt/Source/XPSP1/NT/net/dlc/driver/llcmain.c
2020-09-26 16:20:57 +08:00

306 lines
8.7 KiB
C
Raw Permalink 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:
llcmain.c
Abstract:
This module implements data link object for NDIS 3.0.
Data link driver provides 802.2 LLC Class I and II services
and also interface to send and receive direct network data.
This module may later be used as general packet router
for the protocol modules => smaller system overhead, when
the NDIS packet header is checked only once. LLC interface
is also much easier to use than NDIS.
Its main tasks are:
- implement simple and realiable LLC Class II
connection-oriented services for the other drivers
- provide network independent interface 802.2 interface
- remove the system overhead of the packet and indication
routing to all protocol modules
- provide transmit packet queueing for the protocol drivers
- provide 802.2 compatible connect, disconnect and close
services, that calls back, when the transmit queue is
empty.
The services provides the module are somewhat extended
802.2 services, because the protocol engine is a IBM version
of ISO-HDLC ABM. LlcDisconnet and LlcConnect primitives
implements both Request and Confirm of Connect/Disconnect
Data link driver does not make any buffering for the data.
All data buffer except the LLC header and lan header with the
I-frames must be provided by the calling protocol driver.
*********************** SPINLOCK RULES *************************
DataLink driver uses extensively several spin locks to make it
as re-entrant as possible with multiple processors and to
minimize unnecessary spin locking calls.
The main problem with multiple spin locks are the dead locks.
The spin locks must always be acquired the same order (and
released in a reverse order):
This is the order used with DataLink spin locks:
1. Adapter->ObjectDataBase
Protects objects from close/delete. This is always locked in
the receive indication routine.
2. Link->SpinLock
Protects the link. This lock is needed to prevent to upper
protocol to call the link station, when we are changing it state.
SendSpinLock cannot protect it, because it must be switched off,
when the pending commands are executed. (But could we keep
SendSpinLock locked, when the transmit commands are completed).
There will occure a dead lock in any way, if the upper protocol
is waiting the last transmit to complete before it disconnects
a link station.
3. Adapter->SendSpinLock // protects queues and packet pools
---
Timer spin locks is used only by timer servives:
4. TimerSpinLock // protects the timer queues
****************************************************************
Contents:
LlcInitialize
LlcTerminate
Author:
Antti Saarenheimo (o-anttis) 17-MAY-1991
Revision History:
--*/
//
// This define enables the private DLC function prototypes
// We don't want to export our data types to the dlc layer.
// MIPS compiler doesn't accept hiding of the internal data
// structures by a PVOID in the function prototype.
// i386 will check the type defines
//
#ifndef i386
#define LLC_PUBLIC_NDIS_PROTOTYPES
#endif
#include <llc.h>
#include "dbgmsg.h"
//
// These are the defaults defined in the IBM LAN Architecture reference,
// this may be set by a DLC application, if it want to change the defaults.
//
DLC_LINK_PARAMETERS DefaultParameters = {
3, // T1 600 milliseconds (3 * 5 * 40)
2, // T2 80 milliseconds (2 * 1 * 40)
10, // Ti 25 seconds ((10 - 5) * 125 * 4)
127, // TW: maximum transmit window size
127, // RW: The maximum receive window size
20, // Nw: number of LPDUs acknowledged, before Ww is incr.)
5, // N2: Number of retires allowed (both Polls and I LPDSs)
0, // lowest proirity by default
600 // default for info field length by default
};
KMUTEX NdisAccessMutex;
KSEMAPHORE OpenAdapterSemaphore;
#define LLC_PROTOCOL_NAME L"DLC"
#ifdef NDIS40
extern ULONG gWaitForAdapter;
#endif // NDIS40
DLC_STATUS
LlcInitialize(
VOID
)
/*++
Routine Description:
The routines initializes the protocol module and
does the minimal stuff, that must be done in the
serialized initialization routine.
Arguments:
None.
Return Value:
NDIS_STATUS
--*/
{
NDIS_STATUS Status;
NDIS_PROTOCOL_CHARACTERISTICS LlcChars;
ASSUME_IRQL(PASSIVE_LEVEL);
//
// We must build a MDL for the XID information used
// when the link level takes care of XID handling.
//
pXidMdl = IoAllocateMdl(&Ieee802Xid,
sizeof(Ieee802Xid),
FALSE,
FALSE,
NULL
);
if (pXidMdl == NULL) {
return DLC_STATUS_NO_MEMORY;
}
#ifdef NDIS40
//
// Event to signal all adapters have been bound - OK for LlcOpenAdapter
// to bind.
//
NdisInitializeEvent(&PnPBindsComplete);
NdisResetEvent(&PnPBindsComplete);
#endif // NDIS40
#if LLC_DBG
ALLOCATE_SPIN_LOCK(&MemCheckLock);
#endif
MmBuildMdlForNonPagedPool(pXidMdl);
LlcInitializeTimerSystem();
NdisZeroMemory(&LlcChars, sizeof(LlcChars));
NdisInitUnicodeString(&LlcChars.Name, LLC_PROTOCOL_NAME);
#ifdef NDIS40
LlcChars.MajorNdisVersion = 4;
LlcChars.MinorNdisVersion = 0;
LlcChars.OpenAdapterCompleteHandler = LlcNdisOpenAdapterComplete;
LlcChars.CloseAdapterCompleteHandler = LlcNdisCloseComplete;
LlcChars.SendCompleteHandler = LlcNdisSendComplete;
LlcChars.TransferDataCompleteHandler = LlcNdisTransferDataComplete;
LlcChars.ResetCompleteHandler = LlcNdisResetComplete;
LlcChars.RequestCompleteHandler = LlcNdisRequestComplete;
LlcChars.ReceiveHandler = LlcNdisReceiveIndication;
LlcChars.ReceiveCompleteHandler = LlcNdisReceiveComplete;
LlcChars.StatusHandler = NdisStatusHandler;
LlcChars.StatusCompleteHandler = LlcNdisReceiveComplete;
// DLC supports bind/unbind/pnp, but not unload.
LlcChars.UnloadHandler = NULL;
LlcChars.PnPEventHandler = LlcPnPEventHandler;
LlcChars.BindAdapterHandler = LlcBindAdapterHandler;
LlcChars.UnbindAdapterHandler = LlcUnbindAdapterHandler;
//
// Need to get value for waiting on uninitialized adapters.
//
if (!NT_SUCCESS(GetAdapterWaitTimeout(&gWaitForAdapter)))
{
ASSERT(FALSE);
gWaitForAdapter = 15; // Default.
}
DEBUGMSG(DBG_WARN, (TEXT("WaitForAdapter delay = %d sec\n"), gWaitForAdapter));
#else // NDIS40
LlcChars.MajorNdisVersion = 3;
LlcChars.MinorNdisVersion = 0;
LlcChars.OpenAdapterCompleteHandler = LlcNdisOpenAdapterComplete;
LlcChars.CloseAdapterCompleteHandler = LlcNdisCloseComplete;
LlcChars.SendCompleteHandler = LlcNdisSendComplete;
LlcChars.TransferDataCompleteHandler = LlcNdisTransferDataComplete;
LlcChars.ResetCompleteHandler = LlcNdisResetComplete;
LlcChars.RequestCompleteHandler = LlcNdisRequestComplete;
LlcChars.ReceiveHandler = LlcNdisReceiveIndication;
LlcChars.ReceiveCompleteHandler = LlcNdisReceiveComplete;
LlcChars.StatusHandler = NdisStatusHandler;
LlcChars.StatusCompleteHandler = LlcNdisReceiveComplete;
#endif // !NDIS40
NdisRegisterProtocol(&Status,
&LlcProtocolHandle,
&LlcChars,
sizeof(LlcChars));
KeInitializeSpinLock(&LlcSpinLock);
ASSUME_IRQL(PASSIVE_LEVEL);
KeInitializeMutex(&NdisAccessMutex, 1);
//
// We use the OpenAdapterSemaphore in the LlcOpenAdapter function. We really
// want a mutex, but a mutex causes us problems on a checked build if we
// make a call into NTOS. In either case, we just need a mechanism to ensure
// only one thread is creating the ADAPTER_CONTEXT & opening the adapter at
// NDIS level
//
KeInitializeSemaphore(&OpenAdapterSemaphore, 1, 1);
if (Status != STATUS_SUCCESS) {
IoFreeMdl(pXidMdl);
}
return Status;
}
VOID
LlcTerminate(
VOID
)
/*++
Routine Description:
The routines terminates the LLC protocol module and frees its global
resources. This assumes all adapter bindings to be closed.
Arguments:
None.
Return Value:
None.
--*/
{
NDIS_STATUS Status;
ASSUME_IRQL(PASSIVE_LEVEL);
DEBUGMSG(DBG_INIT, (TEXT("+LlcTerminate()\n")));
LlcTerminateTimerSystem();
#ifdef NDIS40
CloseAllAdapters();
#endif // NDIS40
NdisDeregisterProtocol(&Status, LlcProtocolHandle);
IoFreeMdl(pXidMdl);
}