Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1870 lines
49 KiB

/*++
Copyright (c) 1991 Microsoft Corporation
Copyright (c) 1991 Nokia Data Systems AB
Module Name:
dlcreq.c
Abstract:
This module handles the miscellaneous DLC requests (set & query information)
Contents:
DlcBufferFree
DlcBufferGet
DlcBufferCreate
DlcConnectStation
DlcFlowControl
ResetLocalBusyBufferStates
DlcReallocate
DlcReset
DirSetExceptionFlags
CompleteAsyncCommand
GetLinkStation
GetSapStation
GetStation
DlcReadCancel
DirOpenAdapter
DirCloseAdapter
CompleteDirCloseAdapter
DlcCompleteCommand
Author:
Antti Saarenheimo 22-Jul-1991 (o-anttis)
Environment:
Kernel mode
Revision History:
--*/
#include <dlc.h>
#include "dlcdebug.h"
#if 0
//
// if DLC and LLC share the same driver then we can use macros to access fields
// in the BINDING_CONTEXT and ADAPTER_CONTEXT structures
//
#if DLC_AND_LLC
#ifndef i386
#define LLC_PRIVATE_PROTOTYPES
#endif
#include "llcdef.h"
#include "llctyp.h"
#include "llcapi.h"
#endif
#endif
NTSTATUS
DlcBufferFree(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure simply releases the given user buffers.
Arguments:
pIrp - current io request packet
pFileContext - DLC address object
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
STATUS_SUCCESS
DLC_STATUS_INADEQUATE_BUFFERS - buffer pool does not exist
DLC_STATUS_INVALID_STATION_ID -
DLC_STATUS_INVALID_BUFFER_LENGTH -
NOTE!!! BUFFER.FREE does not return error, if the
given buffer is invalid, or released twice!!!
--*/
{
NTSTATUS status = STATUS_SUCCESS;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
if (!pFileContext->hBufferPool) {
return DLC_STATUS_INADEQUATE_BUFFERS;
}
//
// The parameter list is a DLC desriptor array.
// Get the number of descriptor elements in the array.
//
if (InputBufferLength != (sizeof(NT_DLC_BUFFER_FREE_PARMS)
- sizeof(LLC_TRANSMIT_DESCRIPTOR)
+ pDlcParms->BufferFree.BufferCount
* sizeof(LLC_TRANSMIT_DESCRIPTOR))) {
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
//
// We refernce the buffer pool, because otherwise it may disappear
// immeadiately after DLC_LEAVE (when the adapter is closed)
//
ReferenceBufferPool(pFileContext);
//
// Don't try to allocate 0 buffers, it will fail.
//
if (pDlcParms->BufferFree.BufferCount) {
LEAVE_DLC(pFileContext);
RELEASE_DRIVER_LOCK();
status = BufferPoolDeallocate(pFileContext->hBufferPool,
pDlcParms->BufferFree.BufferCount,
pDlcParms->BufferFree.DlcBuffer
);
ACQUIRE_DRIVER_LOCK();
ENTER_DLC(pFileContext);
//
// Reset the local busy states, if there is now enough
// buffers the receive the expected stuff.
//
if (!IsListEmpty(&pFileContext->FlowControlQueue)
&& BufGetUncommittedSpace(pFileContext->hBufferPool) >= 0) {
ResetLocalBusyBufferStates(pFileContext);
}
#if LLC_DBG
cFramesReleased++;
#endif
}
pDlcParms->BufferFree.cBuffersLeft = (USHORT)BufferPoolCount(pFileContext->hBufferPool);
DereferenceBufferPool(pFileContext);
return status;
}
NTSTATUS
DlcBufferGet(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure allocates the requested number or size of DLC buffers
and returns them back to user in a single entry link list..
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
STATUS_SUCCESS
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
UINT SegmentSize;
UINT SizeIndex;
UINT BufferSize;
UINT PrevBufferSize;
UINT cBuffersToGet;
PDLC_BUFFER_HEADER pBufferHeader = NULL;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
if (pFileContext->hBufferPool == NULL) {
return DLC_STATUS_INADEQUATE_BUFFERS;
}
//
// If the segment size is 0, then we return the optimal mix
// of buffer for the requested size. Non null buffer count defines
// how many buffers having the requested size is returned.
//
cBuffersToGet = pDlcParms->BufferGet.cBuffersToGet;
/*******************************************************************************
#if PAGE_SIZE == 8192
if (cBuffersToGet == 0) {
cBuffersToGet = 1;
SegmentSize = pDlcParms->BufferGet.cbBufferSize;
SizeIndex = (UINT)(-1);
} else if (pDlcParms->BufferGet.cbBufferSize <= 256) {
SegmentSize = 256 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 5;
} else if (pDlcParms->BufferGet.cbBufferSize <= 512) {
SegmentSize = 512 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 4;
} else if (pDlcParms->BufferGet.cbBufferSize <= 1024) {
SegmentSize = 1024 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 3;
} else if (pDlcParms->BufferGet.cbBufferSize <= 2048) {
SegmentSize = 2048 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 2;
} else if (pDlcParms->BufferGet.cbBufferSize <= 4096) {
SegmentSize = 4096 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 1;
} else if (pDlcParms->BufferGet.cbBufferSize <= 8192) {
SegmentSize = 8192 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 0;
} else {
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
#elif PAGE_SIZE == 4096
if (cBuffersToGet == 0) {
cBuffersToGet = 1;
SegmentSize = pDlcParms->BufferGet.cbBufferSize;
SizeIndex = (UINT)(-1);
} else if (pDlcParms->BufferGet.cbBufferSize <= 256) {
SegmentSize = 256 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 4;
} else if (pDlcParms->BufferGet.cbBufferSize <= 512) {
SegmentSize = 512 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 3;
} else if (pDlcParms->BufferGet.cbBufferSize <= 1024) {
SegmentSize = 1024 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 2;
} else if (pDlcParms->BufferGet.cbBufferSize <= 2048) {
SegmentSize = 2048 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 1;
} else if (pDlcParms->BufferGet.cbBufferSize <= 4096) {
SegmentSize = 4096 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 0;
} else {
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
#else
#error "Target machine page size not 4096 or 8192"
#endif
*******************************************************************************/
#if defined(ALPHA)
if (cBuffersToGet == 0) {
cBuffersToGet = 1;
SegmentSize = pDlcParms->BufferGet.cbBufferSize;
SizeIndex = (UINT)(-1);
} else if (pDlcParms->BufferGet.cbBufferSize <= 256) {
SegmentSize = 256 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 5;
} else if (pDlcParms->BufferGet.cbBufferSize <= 512) {
SegmentSize = 512 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 4;
} else if (pDlcParms->BufferGet.cbBufferSize <= 1024) {
SegmentSize = 1024 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 3;
} else if (pDlcParms->BufferGet.cbBufferSize <= 2048) {
SegmentSize = 2048 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 2;
} else if (pDlcParms->BufferGet.cbBufferSize <= 4096) {
SegmentSize = 4096 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 1;
} else if (pDlcParms->BufferGet.cbBufferSize <= 8192) {
SegmentSize = 8192 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 0;
} else {
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
#else
if (cBuffersToGet == 0) {
cBuffersToGet = 1;
SegmentSize = pDlcParms->BufferGet.cbBufferSize;
SizeIndex = (UINT)(-1);
} else if (pDlcParms->BufferGet.cbBufferSize <= 256) {
SegmentSize = 256 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 4;
} else if (pDlcParms->BufferGet.cbBufferSize <= 512) {
SegmentSize = 512 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 3;
} else if (pDlcParms->BufferGet.cbBufferSize <= 1024) {
SegmentSize = 1024 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 2;
} else if (pDlcParms->BufferGet.cbBufferSize <= 2048) {
SegmentSize = 2048 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 1;
} else if (pDlcParms->BufferGet.cbBufferSize <= 4096) {
SegmentSize = 4096 - sizeof(NEXT_DLC_SEGMENT);
SizeIndex = 0;
} else {
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
#endif
//
// We refernce the buffer pool, because otherwise it may disappear
// immeadiately after DLC_LEAVE (when the adapter is closed)
//
ReferenceBufferPool(pFileContext);
//
// We don't need to initialize the LAN and DLC header sizes
// in the buffer header and we allocate the requested
// frame as a single buffer.
//
BufferSize = SegmentSize * cBuffersToGet;
if (BufferSize != 0) {
pBufferHeader = NULL;
PrevBufferSize = 0;
LEAVE_DLC(pFileContext);
do {
//
// We must again do this interlocked to avoid the buffer
// pool to be deleted while we are allocating buffers.
//
Status = BufferPoolAllocate(
#if DBG
pFileContext,
#endif
(PDLC_BUFFER_POOL)pFileContext->hBufferPool,
BufferSize,
0, // FrameHeaderSize,
0, // UserDataSize,
0, // frame length
SizeIndex, // fixed segment size set
&pBufferHeader,
&BufferSize
);
#if DBG
BufferPoolExpand(pFileContext, (PDLC_BUFFER_POOL)pFileContext->hBufferPool);
#else
BufferPoolExpand((PDLC_BUFFER_POOL)pFileContext->hBufferPool);
#endif
//
// Don't try to expand buffer pool any more, if it doesn't help!
//
if (BufferSize == PrevBufferSize) {
break;
}
PrevBufferSize = BufferSize;
} while (Status == DLC_STATUS_EXPAND_BUFFER_POOL);
ENTER_DLC(pFileContext);
if (pBufferHeader != NULL) {
pBufferHeader->FrameBuffer.BufferState = BUF_USER;
}
if (Status == STATUS_SUCCESS) {
pDlcParms->BufferGet.pFirstBuffer = (PLLC_XMIT_BUFFER)
((PUCHAR)pBufferHeader->FrameBuffer.pParent->Header.pLocalVa +
MIN_DLC_BUFFER_SEGMENT * pBufferHeader->FrameBuffer.Index);
} else {
BufferPoolDeallocateList(pFileContext->hBufferPool, pBufferHeader);
}
}
pDlcParms->BufferGet.cBuffersLeft = (USHORT)BufferPoolCount(pFileContext->hBufferPool);
DereferenceBufferPool(pFileContext);
return Status;
}
NTSTATUS
DlcBufferCreate(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure creates a new buffer pool and allocates the initial
space for it.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
Success - STATUS_SUCCESS
Failure - DLC_STATUS_DUPLICATE_COMMAND
--*/
{
NTSTATUS status;
PVOID newBufferAddress;
ULONG newBufferSize;
PVOID hExternalBufferPool;
PVOID hBufferPool;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
//
// if we already have a buffer pool defined for this handle, fail the
// request
//
if (pFileContext->hBufferPool) {
return DLC_STATUS_DUPLICATE_COMMAND;
}
hExternalBufferPool = pFileContext->hExternalBufferPool;
LEAVE_DLC(pFileContext);
#if DBG
status = BufferPoolCreate(pFileContext,
#else
status = BufferPoolCreate(
#endif
pDlcParms->BufferCreate.pBuffer,
pDlcParms->BufferCreate.cbBufferSize,
pDlcParms->BufferCreate.cbMinimumSizeThreshold,
&hExternalBufferPool,
&newBufferAddress,
&newBufferSize
);
ENTER_DLC(pFileContext);
pFileContext->hExternalBufferPool = hExternalBufferPool;
if (status == STATUS_SUCCESS) {
//
// The reference count keeps the buffer pool alive
// when it is used (and simultaneously deleted by another
// thread)
//
pFileContext->BufferPoolReferenceCount = 1;
hBufferPool = pFileContext->hBufferPool;
LEAVE_DLC(pFileContext);
status = BufferPoolReference(hExternalBufferPool,
&hBufferPool
);
ENTER_DLC(pFileContext);
pFileContext->hBufferPool = hBufferPool;
pDlcParms->BufferCreate.hBufferPool = pFileContext->hExternalBufferPool;
}
// ENTER_DLC(pFileContext);
return status;
}
NTSTATUS
DlcConnectStation(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure connects an local link station to a remote node
or accepts a remote connection request.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
STATUS_SUCCESS
--*/
{
PDLC_OBJECT pLinkStation;
NTSTATUS Status;
PUCHAR pSourceRouting = NULL;
PDLC_COMMAND pPacket;
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
//
// Procedure checks the sap and link station ids and
// returns the requested link station.
// The error status indicates a wrong sap or station id.
//
Status = GetLinkStation(pFileContext,
pDlcParms->Async.Parms.DlcConnectStation.StationId,
&pLinkStation
);
if (Status != STATUS_SUCCESS) {
return Status;
}
pPacket = ALLOCATE_PACKET_DLC_PKT(pFileContext->hPacketPool);
if (pPacket == NULL) {
return DLC_STATUS_NO_MEMORY;
}
pPacket->pIrp = pIrp;
//
// IBM LAN Tech. Ref p 3-48 (DLC.CONNECT.STATION) states that ROUTING_ADDR
// field is ignored if the link was created due to receipt of a SABME from
// the remote station, EVEN IF THE ADDRESS IS NON-ZERO
//
if (pDlcParms->Async.Parms.DlcConnectStation.RoutingInformationLength != 0) {
pSourceRouting = pDlcParms->Async.Parms.DlcConnectStation.aRoutingInformation;
}
pLinkStation->PendingLlcRequests++;
ReferenceLlcObject(pLinkStation);
LEAVE_DLC(pFileContext);
//
// LlcConnect returns the maximum information field,
// through the tr bridges!
//
LlcConnectStation(pLinkStation->hLlcObject,
(PLLC_PACKET)pPacket,
pSourceRouting,
&pLinkStation->u.Link.MaxInfoFieldLength
);
ENTER_DLC(pFileContext);
DereferenceLlcObject(pLinkStation);
return STATUS_PENDING;
}
NTSTATUS
DlcFlowControl(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure sets or resets the loacl busy state on the given link station
or on all link stations of a sap station.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
Success - STATUS_SUCCESS
Failure - DLC_STATUS_NO_MEMORY
--*/
{
NTSTATUS Status;
PDLC_OBJECT pDlcObject;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
//
// Procedure checks the sap and link station ids and
// returns the requested link station.
// The error status indicates a wrong sap or station id.
//
Status = GetStation(pFileContext,
pDlcParms->DlcFlowControl.StationId,
&pDlcObject
);
if (Status != STATUS_SUCCESS) {
return Status;
}
//
// We will queue all reset local busy buffer commands
// given to the link stations
//
if (((pDlcParms->DlcFlowControl.FlowControlOption & LLC_RESET_LOCAL_BUSY_BUFFER) == LLC_RESET_LOCAL_BUSY_BUFFER)
&& (pDlcObject->Type == DLC_LINK_OBJECT)) {
PDLC_RESET_LOCAL_BUSY_CMD pClearCmd;
pClearCmd = (PDLC_RESET_LOCAL_BUSY_CMD)ALLOCATE_PACKET_DLC_PKT(pFileContext->hPacketPool);
if (pClearCmd == NULL) {
return DLC_STATUS_NO_MEMORY;
}
pClearCmd->StationId = pDlcParms->DlcFlowControl.StationId;
pClearCmd->RequiredBufferSpace = 0;
LlcInsertTailList(&pFileContext->FlowControlQueue, pClearCmd);
ResetLocalBusyBufferStates(pFileContext);
} else {
ReferenceLlcObject(pDlcObject);
LEAVE_DLC(pFileContext);
Status = LlcFlowControl(pDlcObject->hLlcObject,
pDlcParms->DlcFlowControl.FlowControlOption
);
ENTER_DLC(pFileContext);
DereferenceLlcObject(pDlcObject);
}
return Status;
}
VOID
ResetLocalBusyBufferStates(
IN PDLC_FILE_CONTEXT pFileContext
)
/*++
Routine Description:
Procedure executes the pending busy state resets when there is
enough memory in the buffer pool to receive the expected data.
Arguments:
pFileContext - DLC adapter context
Return Value:
None
--*/
{
NTSTATUS Status;
PDLC_OBJECT pDlcObject;
PDLC_RESET_LOCAL_BUSY_CMD pClearCmd;
ASSUME_IRQL(DISPATCH_LEVEL);
//
// We cannot reset anything, if the buffer pool is not yet
// defined.
//
if (pFileContext->hBufferPool == NULL) {
return;
}
ReferenceBufferPool(pFileContext);
while (!IsListEmpty(&pFileContext->FlowControlQueue)) {
pClearCmd = LlcRemoveHeadList(&pFileContext->FlowControlQueue);
Status = GetLinkStation(pFileContext,
pClearCmd->StationId,
&pDlcObject
);
//
// All commands having an invalid station id will be just removed
// from the queue. The local busy state can be reset only
// for the existing link stations
//
if (Status == STATUS_SUCCESS) {
//
// The required space is nul, when a mew packet is checked
// in the first time, the non-null value just prevents
// us to check the commited memory in the second time.
//
if (pClearCmd->RequiredBufferSpace == 0) {
//
// We must also remove the old uncommited space,
// otherwise the same buffer size could be
// committed several times, but uncommitted only once
//
if (pDlcObject->CommittedBufferSpace != 0) {
BufUncommitBuffers(pFileContext->hBufferPool,
pDlcObject->CommittedBufferSpace
);
}
pDlcObject->CommittedBufferSpace =
pClearCmd->RequiredBufferSpace = LlcGetCommittedSpace(pDlcObject->hLlcObject);
BufCommitBuffers(pFileContext->hBufferPool,
pDlcObject->CommittedBufferSpace
);
}
//
// We are be removing a local buffer busy state =>
// we must expand the buffer pools before the local busy
// is removed, but only if we are not calling this
// from a DPC level.
//
if (BufGetUncommittedSpace(pFileContext->hBufferPool) < 0) {
LEAVE_DLC(pFileContext);
#if DBG
BufferPoolExpand(pFileContext, pFileContext->hBufferPool);
#else
BufferPoolExpand(pFileContext->hBufferPool);
#endif
ENTER_DLC(pFileContext);
}
//
// Now we have expanded the buffer pool for the new
// flow control command, check if we have now enough
// memory to receive the commited size of data.
//
if (BufGetUncommittedSpace(pFileContext->hBufferPool) >= 0
&& pDlcObject->hLlcObject != NULL) {
ReferenceLlcObject(pDlcObject);
LEAVE_DLC(pFileContext);
Status = LlcFlowControl(pDlcObject->hLlcObject,
LLC_RESET_LOCAL_BUSY_BUFFER
);
ENTER_DLC(pFileContext);
DereferenceLlcObject(pDlcObject);
} else {
//
// We must exit this loop when there is not enough available
// space in the buffer pool, but we must return
// the command back to the head of the list
//
LlcInsertHeadList(&pFileContext->FlowControlQueue, pClearCmd);
break;
}
}
DEALLOCATE_PACKET_DLC_PKT(pFileContext->hPacketPool, pClearCmd);
}
DereferenceBufferPool(pFileContext);
}
NTSTATUS
DlcReallocate(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure changes the number of link stations allocated to a SAP
without closing or reopening the sap station.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength
Return Value:
NTSTATUS:
STATUS_SUCCESS
--*/
{
PDLC_OBJECT pSap;
UCHAR ExtraStations;
UCHAR StationCount;
NTSTATUS Status;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
//
// Procedure checks the sap and returns the requested sap station.
// The error status indicates an invalid sap station id.
//
Status = GetSapStation(pFileContext,
pDlcParms->DlcReallocate.usStationId,
&pSap
);
if (Status != STATUS_SUCCESS) {
return Status;
}
//
// The new link station count must be more than current number
// of open link stations but less than the available number
// of link stations for the file context
//
StationCount = pDlcParms->DlcReallocate.uchStationCount;
if (StationCount != 0 && Status == STATUS_SUCCESS) {
//
// Bit7 set in options => decrease the number of available
// stations by the given station count. Otherwise we increase it.
//
if (pDlcParms->DlcReallocate.uchOption & 0x80) {
ExtraStations = pSap->u.Sap.MaxStationCount - pSap->u.Sap.LinkStationCount;
if (StationCount > ExtraStations) {
StationCount = ExtraStations;
Status = DLC_STATUS_INADEQUATE_LINKS;
}
pFileContext->LinkStationCount += StationCount;
pSap->u.Sap.MaxStationCount -= StationCount;
} else {
if (pFileContext->LinkStationCount < StationCount) {
StationCount = pFileContext->LinkStationCount;
Status = DLC_STATUS_INADEQUATE_LINKS;
}
pFileContext->LinkStationCount -= StationCount;
pSap->u.Sap.MaxStationCount += StationCount;
}
}
//
// Set the return parameters even if there would be an error
// (inadequate stations is a non fatal error)
//
pDlcParms->DlcReallocate.uchStationsAvailOnAdapter = pFileContext->LinkStationCount;
pDlcParms->DlcReallocate.uchStationsAvailOnSap = pSap->u.Sap.MaxStationCount - pSap->u.Sap.LinkStationCount;
pDlcParms->DlcReallocate.uchTotalStationsOnAdapter = MAX_LINK_STATIONS;
pDlcParms->DlcReallocate.uchTotalStationsOnSap = pSap->u.Sap.MaxStationCount;
return Status;
}
NTSTATUS
DlcReset(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure closes immediately a sap and its all link stations or
all saps and all link stations.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
STATUS_SUCCESS
STATUS_PENDING
DLC_STATUS_NO_MEMORY
--*/
{
PDLC_OBJECT pDlcObject;
PDLC_CLOSE_WAIT_INFO pClosingInfo;
NTSTATUS Status;
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
//
// Station id 0 resets the whole DLC
//
if (pDlcParms->Async.Ccb.u.dlc.usStationId == 0) {
PDLC_CLOSE_WAIT_INFO pClosingInfo;
pClosingInfo = ALLOCATE_PACKET_DLC_PKT(pFileContext->hPacketPool);
if (pClosingInfo == NULL) {
Status = DLC_STATUS_NO_MEMORY;
} else {
CloseAllStations(pFileContext,
pIrp,
DLC_COMMAND_COMPLETION,
NULL,
pDlcParms,
pClosingInfo
);
Status = STATUS_PENDING;
}
} else {
BOOLEAN allClosed;
//
// We have a specific sap station
//
Status = GetSapStation(pFileContext,
pDlcParms->Async.Ccb.u.dlc.usStationId,
&pDlcObject
);
if (Status != STATUS_SUCCESS) {
return Status;
}
//
// Allocate the close/reset command completion info,
// the station count is the number of link stations and
// the sap station itself
//
pClosingInfo = ALLOCATE_PACKET_DLC_PKT(pFileContext->hPacketPool);
if (pClosingInfo == NULL) {
return DLC_STATUS_NO_MEMORY;
}
pClosingInfo->pIrp = pIrp;
pClosingInfo->Event = DLC_COMMAND_COMPLETION;
pClosingInfo->CancelStatus = DLC_STATUS_CANCELLED_BY_USER;
pClosingInfo->CloseCounter = 1; // keep command alive over sync path
(USHORT)(pDlcObject->u.Sap.LinkStationCount + 1);
CloseAnyStation(pDlcObject, pClosingInfo, FALSE);
//
// RLF 05/09/93
//
// PC/3270 (DOS program) is hanging forever when we try to quit.
// It is doing this because we returned STATUS_PENDING to a DLC.RESET,
// even though the reset completed; the DOS program spins forever on
// the CCB.uchDlcStatus field, waiting for it to go non-0xFF, which it
// will never do.
//
// If we determine that the station has been reset (all links closed)
// then return success, else pending
//
allClosed = DecrementCloseCounters(pFileContext, pClosingInfo);
//
// RLF 07/21/92 Always return PENDING. Can't complete before return?
//
Status = allClosed ? STATUS_SUCCESS : STATUS_PENDING;
}
return Status;
}
NTSTATUS
DirSetExceptionFlags(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure sets the exception flags for the current adapter context.
Arguments:
pIrp - current io request packet
pFileContext - DLC adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength
Return Value:
NTSTATUS:
STATUS_SUCCESS
--*/
{
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
pFileContext->AdapterCheckFlag = pDlcParms->DirSetExceptionFlags.ulAdapterCheckFlag;
pFileContext->NetworkStatusFlag = pDlcParms->DirSetExceptionFlags.ulNetworkStatusFlag;
pFileContext->PcErrorFlag = pDlcParms->DirSetExceptionFlags.ulPcErrorFlag;
pFileContext->SystemActionFlag = pDlcParms->DirSetExceptionFlags.ulSystemActionFlag;
return STATUS_SUCCESS;
}
VOID
CompleteAsyncCommand(
IN PDLC_FILE_CONTEXT pFileContext,
IN UINT Status,
IN PIRP pIrp,
IN PVOID pUserCcbPointer,
IN BOOLEAN InCancel
)
/*++
Routine Description:
Procedure completes an asynchronous DLC command.
It also copies the optional output parameters to user parameter
table, if there is the second output buffer.
Arguments:
pFileContext - DLC driver client context.
Status - status of the complete command.
pIrp - the completed I/O request packet.
pUserCcbPointer - the next CCB address to which the command will be linked.
InCancel - TRUE if called on Irp cancel path
Return Value:
None.
--*/
{
PNT_DLC_PARMS pDlcParms;
UNREFERENCED_PARAMETER(pFileContext);
ASSUME_IRQL(DISPATCH_LEVEL);
DIAG_FUNCTION("CompleteAsyncCommand");
pDlcParms = (PNT_DLC_PARMS)pIrp->AssociatedIrp.SystemBuffer;
//
// We first map the 32-bit DLC driver status code to 8- bit API status
//
if (Status == STATUS_SUCCESS) {
pDlcParms->Async.Ccb.uchDlcStatus = (UCHAR)STATUS_SUCCESS;
} else if (Status >= DLC_STATUS_ERROR_BASE && Status < DLC_STATUS_MAX_ERROR) {
//
// We can map the normal DLC error codes directly to the 8-bit
// DLC API error codes.
//
pDlcParms->Async.Ccb.uchDlcStatus = (UCHAR)(Status - DLC_STATUS_ERROR_BASE);
} else {
//
// we have an unknown NT error status => we will return it in the CCB
//
pDlcParms->Async.Ccb.uchDlcStatus = (UCHAR)(DLC_STATUS_NT_ERROR_STATUS & 0xff);
}
pDlcParms->Async.Ccb.pCcbAddress = pUserCcbPointer;
//
// We always return success status to the I/O system. The actual status is
// copied to the CCB (= the dafault output buffer)
//
LEAVE_DLC(pFileContext);
RELEASE_DRIVER_LOCK();
DlcCompleteIoRequest(pIrp, InCancel);
ACQUIRE_DRIVER_LOCK();
ENTER_DLC(pFileContext);
DereferenceFileContext(pFileContext);
}
NTSTATUS
GetLinkStation(
IN PDLC_FILE_CONTEXT pFileContext,
IN USHORT StationId,
OUT PDLC_OBJECT *ppLinkStation
)
/*++
Routine Description:
Procedure checks and returns link station
Arguments:
pFileContext - DLC driver client context
StationId - DLC station id (ssnn, where ss = sap and nn = link station id
ppDlcObject - the returned link station
Return Value:
NTSTATUS:
STATUS_SUCCESS
DLC_STATUS_INVALID_SAP_VALUE
DLC_STATUS_INVALID_STATION_ID
--*/
{
if ((StationId & 0xff) == 0 || (StationId & 0xff00) == 0) {
return DLC_STATUS_INVALID_STATION_ID;
}
return GetStation(pFileContext, StationId, ppLinkStation);
}
NTSTATUS
GetSapStation(
IN PDLC_FILE_CONTEXT pFileContext,
IN USHORT StationId,
OUT PDLC_OBJECT *ppStation
)
/*++
Routine Description:
Procedure checks and returns link station
Arguments:
pFileContext - DLC driver client context
StationId - DLC station id (ssnn, where ss = sap and nn = link station id
ppDlcObject - the returned link station
Return Value:
NTSTATUS:
STATUS_SUCCESS
DLC_STATUS_INVALID_SAP_VALUE
DLC_STATUS_INVALID_STATION_ID
--*/
{
UINT SapId = StationId >> 9;
if (SapId >= MAX_SAP_STATIONS
|| SapId == 0
|| (StationId & GROUP_SAP_BIT)
|| (*ppStation = pFileContext->SapStationTable[SapId]) == NULL
|| (*ppStation)->State != DLC_OBJECT_OPEN) {
return DLC_STATUS_INVALID_SAP_VALUE;
}
return STATUS_SUCCESS;
}
NTSTATUS
GetStation(
IN PDLC_FILE_CONTEXT pFileContext,
IN USHORT StationId,
OUT PDLC_OBJECT *ppStation
)
/*++
Routine Description:
Procedure checks the given station id and returns a pointer to
sap, direct or link station object.
Arguments:
pFileContext - DLC driver client context
StationId - DLC station id (ssnn, where ss = sap and nn = link station id
ppStation - the returned station object
Return Value:
NTSTATUS:
STATUS_SUCCESS
DLC_STATUS_INVALID_STATION_ID
--*/
{
UINT SapId = StationId >> 9;
//
// Check if the sap or direct station exists,
// but check also the link station, if we found a valid sap id.
//
if (SapId >= MAX_SAP_STATIONS
|| (StationId & GROUP_SAP_BIT)
|| (*ppStation = pFileContext->SapStationTable[SapId]) == NULL
|| (*ppStation)->State != DLC_OBJECT_OPEN) {
if (SapId == 0) {
return DLC_STATUS_DIRECT_STATIONS_NOT_AVAILABLE;
} else {
return DLC_STATUS_INVALID_STATION_ID;
}
}
//
// The link station table will never be read, if we have found
// a valid sap or direct station. Link station must exist and
// it must be opened.
//
if (SapId != 0
&& (StationId & 0xff) != 0
&& (*ppStation = pFileContext->LinkStationTable[((StationId & 0xff) - 1)]) == NULL
|| (*ppStation)->State != DLC_OBJECT_OPEN) {
return DLC_STATUS_INVALID_STATION_ID;
}
return STATUS_SUCCESS;
}
NTSTATUS
DlcReadCancel(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
This primitive cancels a READ command, that have the given CCB pointer.
Arguments:
pIrp - current io request packet
pFileContext - DLC process specific adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
DLC_STATUS:
STATUS_SUCCESS
--*/
{
PVOID pCcbAddress = NULL;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
DLC_TRACE('Q');
return AbortCommand(pFileContext,
(USHORT)DLC_IGNORE_STATION_ID,
(USHORT)DLC_STATION_MASK_SPECIFIC,
pDlcParms->DlcCancelCommand.CcbAddress,
&pCcbAddress,
DLC_STATUS_CANCELLED_BY_USER,
TRUE // Suppress completion
);
}
NTSTATUS
DirOpenAdapter(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
This primitive binds the DLC API driver to an adapter context of
the LLC module. The LLC mode may also bind to the given NDIS driver
and open it, if this is the first reference to the driver from DLC.
Arguments:
pIrp - current io request packet
pFileContext - DLC process specific adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
DLC_STATUS:
STATUS_SUCCESS
--*/
{
NTSTATUS Status;
UINT OpenErrorCode;
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
if (pDlcParms->DirOpenAdapter.NtDlcIoctlVersion != NT_DLC_IOCTL_VERSION) {
return DLC_STATUS_INVALID_VERSION;
}
//
// This makes the DirOpenAdapter safe, even if there were two adapter
// opens going on simultaneously
//
//
// RLF 04/22/94
//
// this only protects against 2 threads in the same process performing
// simultaneous opens on the same adapter
//
if (pFileContext->pBindingContext) {
return DLC_STATUS_DUPLICATE_COMMAND;
}
pFileContext->pBindingContext = (PVOID)-1;
//
// if a buffer pool handle was supplied (i.e. the app already created a
// buffer pool or is otherwise sharing one) then reference it for this
// file context
//
if (pDlcParms->DirOpenAdapter.hBufferPoolHandle) {
Status = BufferPoolReference(pDlcParms->DirOpenAdapter.hBufferPoolHandle,
&pFileContext->hBufferPool
);
if (Status == STATUS_SUCCESS) {
pFileContext->BufferPoolReferenceCount = 1;
pFileContext->hExternalBufferPool = pDlcParms->DirOpenAdapter.hBufferPoolHandle;
} else {
//
// Invalid buffer pool handle, hopefully this status
// code indicates correctly, that the buffer pool
// handle is not valid.
//
pFileContext->pBindingContext = NULL;
return DLC_STATUS_INVALID_BUFFER_LENGTH;
}
}
LEAVE_DLC(pFileContext);
//
// XXXXXXX: BringUpDiagnostics are still missing!!!
//
//
// RLF 04/19/93
//
// The string we pass to LlcOpenAdapter is a pointer to a zero terminated
// wide character string, NOT a pointer to a UNICODE_STRING structure. The
// string MUST be in system memory space, i.e. copied across the kernel
// interface by NtDeviceIoControlFile
//
Status = LlcOpenAdapter(&pDlcParms->DirOpenAdapter.Buffer[0],
(PVOID)pFileContext,
LlcCommandCompletion,
LlcReceiveIndication,
LlcEventIndication,
NdisMedium802_5, // Always token-ring!
pDlcParms->DirOpenAdapter.LlcEthernetType,
pDlcParms->DirOpenAdapter.AdapterNumber,
&pFileContext->pBindingContext,
&OpenErrorCode,
&pFileContext->MaxFrameLength,
&pFileContext->ActualNdisMedium
);
//
// make sure LlcOpenAdapter didn't return with lowered IRQL
//
ASSUME_IRQL(DISPATCH_LEVEL);
//
// IBM LAN Tech. Ref. defines the open error code as a 16-bit value, the
// high 8 bits of which are 0. The MAC inclusive-ORs the open error code
// into the NDIS status. Extract it
//
pDlcParms->DirOpenAdapter.Adapter.usOpenErrorCode = (USHORT)(UCHAR)OpenErrorCode;
if (Status != STATUS_SUCCESS) {
ENTER_DLC(pFileContext);
//
// It does not matter, if we have null buffer pool handle!
//
#if DBG
BufferPoolDereference(pFileContext,
(PDLC_BUFFER_POOL*)&pFileContext->hBufferPool
);
#else
BufferPoolDereference((PDLC_BUFFER_POOL*)&pFileContext->hBufferPool);
#endif
//
// set the BINDING_CONTEXT pointer back to NULL - other routines check
// for this value, like CloseAdapterFileContext
//
pFileContext->pBindingContext = NULL;
//
// Probably the adapter was missing or it was installed improperly
//
Status = DLC_STATUS_ADAPTER_NOT_INSTALLED;
} else {
//
// Set the optional timer tick one/two values
// (if they have been set in registry)
//
LlcSetInformation(pFileContext->pBindingContext,
DLC_INFO_CLASS_DLC_TIMERS,
(PLLC_SET_INFO_BUFFER)&(pDlcParms->DirOpenAdapter.LlcTicks),
sizeof(LLC_TICKS)
);
LlcQueryInformation(pFileContext->pBindingContext,
DLC_INFO_CLASS_DIR_ADAPTER,
(PLLC_QUERY_INFO_BUFFER)pDlcParms->DirOpenAdapter.Adapter.auchNodeAddress,
sizeof(LLC_ADAPTER_INFO)
);
ENTER_DLC(pFileContext);
//
// take the missing parameters from the hat
//
pDlcParms->DirOpenAdapter.Adapter.usOpenOptions = 0;
pDlcParms->DirOpenAdapter.Adapter.usMaxFrameSize = (USHORT)(pFileContext->MaxFrameLength + 6);
pDlcParms->DirOpenAdapter.Adapter.usBringUps = 0;
pDlcParms->DirOpenAdapter.Adapter.InitWarnings = 0;
pFileContext->AdapterNumber = pDlcParms->DirOpenAdapter.AdapterNumber;
pFileContext->LinkStationCount = 255;
pFileContext->pSecurityDescriptor = pDlcParms->DirOpenAdapter.pSecurityDescriptor;
//
// Read the most recent cumulative NDIS error counters
// to the file context. DLC error counters will be counted
// from 0 and they may be reset.
//
GetDlcErrorCounters(pFileContext, NULL);
pFileContext->State = DLC_FILE_CONTEXT_OPEN;
}
//
// We may directly return whatever the LLC binding primitive gives us
//
return Status;
}
NTSTATUS
DirCloseAdapter(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
This primitive initializes the close adapter operation.
It first closes all open link and sap stations and
then optionally chains the canceled commands and
receive buffers to a READ command. If no read commands
was found, then the CCBs are linked to the CCB pointer
of this command.
The actual file close should only delete the file
object, except, if the application exits, when it still has
open dlc api handles. In that case the file close routine
will call this procedure to shut down the dlc file context.
Arguments:
pIrp - current io request packet
pFileContext - DLC process specific adapter context
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS
STATUS_PENDING
The adapter is being closed
DLC_STATUS_ADAPTER_CLOSED
The adapter is already closed
NOTE: This is a SYNCHRONOUS return code! And will cause the IRP
to be completed
--*/
{
UNREFERENCED_PARAMETER(pDlcParms);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
ASSUME_IRQL(DISPATCH_LEVEL);
DIAG_FUNCTION("DirCloseAdapter");
DLC_TRACE('J');
if (pFileContext->State != DLC_FILE_CONTEXT_OPEN) {
return DLC_STATUS_ADAPTER_CLOSED;
}
#if LLC_DBG == 2
DbgPrint( "*** Top memory consumption (before adapter close) *** \n" );
PrintMemStatus();
#endif
//
// This disables any further commands (including DirCloseAdapter)
//
pFileContext->State = DLC_FILE_CONTEXT_CLOSE_PENDING;
//
// Remove first all functional, group or multicast addresses
// set in the adapter by the current DLC application process
//
if (pFileContext->pBindingContext) {
LEAVE_DLC(pFileContext);
LlcResetBroadcastAddresses(pFileContext->pBindingContext);
ENTER_DLC(pFileContext);
}
//
// We must use the static closing packet, because the adapter close
// must succeed even if we could not allocate any packets
//
CloseAllStations(pFileContext,
pIrp,
DLC_COMMAND_COMPLETION,
CompleteDirCloseAdapter,
pDlcParms,
&pFileContext->ClosingPacket
);
return STATUS_PENDING;
}
VOID
CompleteDirCloseAdapter(
IN PDLC_FILE_CONTEXT pFileContext,
IN PDLC_CLOSE_WAIT_INFO pClosingInfo,
IN PVOID pCcbLink
)
/*++
Routine Description:
Finishes DIR.CLOSE.ADAPTER command
Arguments:
pFileContext - DLC adapter open context
pClosingInfo - packet structure, that includes all data of this command
pCcbLink - the orginal user mode ccb address on the next CCB, that
will be chained to the completed command.
Return Value:
None
--*/
{
ASSUME_IRQL(DISPATCH_LEVEL);
DLC_TRACE('K');
//
// reference the file context to stop any of the dereferences below, or in
// functions called by this routine destroying it
//
ReferenceFileContext(pFileContext);
//
// Disconnect (or unbind) llc driver from us
//
if (pFileContext->pBindingContext) {
LEAVE_DLC(pFileContext);
LlcDisableAdapter(pFileContext->pBindingContext);
ENTER_DLC(pFileContext);
}
if (IoGetCurrentIrpStackLocation(pClosingInfo->pIrp)->MajorFunction == IRP_MJ_DEVICE_CONTROL) {
CompleteAsyncCommand(pFileContext, STATUS_SUCCESS, pClosingInfo->pIrp, pCcbLink, FALSE);
} else {
//
// This is a normal FILE CLOSE !!! (IRP_MJ_CLEANUP)
//
ASSERT(IoGetCurrentIrpStackLocation(pClosingInfo->pIrp)->MajorFunction == IRP_MJ_CLEANUP);
//
// Dereference for the cleanup. This will allow cleanup to become
// unblocked.
//
DereferenceFileContext(pFileContext);
}
//
// We must delete the buffer pool now, because the dereference
// of the driver object starts the final process exit
// completion, that bug chekcs, if the number of the locked
// pages is non zero.
//
DereferenceBufferPool(pFileContext);
//
// We create two references for file context, when it is created
// the other is decremented here and the other when the synchronous
// part of command completion has been done
//
pFileContext->State = DLC_FILE_CONTEXT_CLOSED;
//
// This should be the last reference of the file context
// (if no IRPs operations are in execution or pending.
//
DereferenceFileContext(pFileContext);
}
NTSTATUS
DlcCompleteCommand(
IN PIRP pIrp,
IN PDLC_FILE_CONTEXT pFileContext,
IN PNT_DLC_PARMS pDlcParms,
IN ULONG InputBufferLength,
IN ULONG OutputBufferLength
)
/*++
Routine Description:
Procedure queues the given CCB to the completion list.
This routine is used to save the synchronous commands from
DLC API DLL to event queue. This must be done whenever a
synchronous command has a non null command completion flag.
Arguments:
pIrp - current io request packet
pFileContext - DLC address object
pDlcParms - the current parameter block
InputBufferLength - the length of input parameters
OutputBufferLength -
Return Value:
NTSTATUS:
STATUS_SUCCESS
--*/
{
UNREFERENCED_PARAMETER(pIrp);
UNREFERENCED_PARAMETER(InputBufferLength);
UNREFERENCED_PARAMETER(OutputBufferLength);
if (pDlcParms->CompleteCommand.CommandCompletionFlag == 0
|| pDlcParms->CompleteCommand.pCcbPointer == NULL) {
//
// This is more likely an internal error!
//
return DLC_STATUS_INTERNAL_ERROR;
}
return MakeDlcEvent(pFileContext,
DLC_COMMAND_COMPLETION,
pDlcParms->CompleteCommand.StationId,
NULL,
pDlcParms->CompleteCommand.pCcbPointer,
pDlcParms->CompleteCommand.CommandCompletionFlag,
FALSE
);
}