windows-nt/Source/XPSP1/NT/net/nwlink/nb/send.c

2889 lines
86 KiB
C
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) 1989-1993 Microsoft Corporation
Module Name:
send.c
Abstract:
This module contains the send routines for the Netbios
module of the ISN transport.
Author:
Adam Barr (adamba) 22-November-1993
Environment:
Kernel mode
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
//
// Work Item structure for work items put on the Kernel Excutive worker threads
//
typedef struct
{
WORK_QUEUE_ITEM Item; // Used by OS to queue these requests
PVOID Context;
} NBI_WORK_ITEM_CONTEXT;
VOID
SendDgram(
PNDIS_PACKET Packet
)
/*++
Routine Description:
This routine sends a datagram from a Worker thread.
Earlier, this code was part of the NbiSendComplete module,
but since we could end up with a stack overflow, this may
now be handled over a worker thread.
Arguments:
WorkItem - The work item that was allocated for this.
Return Value:
None.
--*/
{
PNB_SEND_RESERVED Reserved = (PNB_SEND_RESERVED)(Packet->ProtocolReserved);
NDIS_STATUS Status;
PNETBIOS_CACHE CacheName;
PDEVICE Device = NbiDevice;
NB_CONNECTIONLESS UNALIGNED * Header;
PIPX_LOCAL_TARGET LocalTarget;
ULONG HeaderLength;
ULONG PacketLength;
// send the datagram on the next net.
CTEAssert (!Reserved->u.SR_DG.Cache->Unique);
Reserved->SendInProgress = TRUE;
CacheName = Reserved->u.SR_DG.Cache;
//
// Fill in the IPX header -- the default header has the broadcast
// address on net 0 as the destination IPX address, so we modify
// that for the current netbios cache entry if needed.
//
Header = (NB_CONNECTIONLESS UNALIGNED *) (&Reserved->Header[Device->Bind.IncludedHeaderOffset]);
RtlCopyMemory((PVOID)&Header->IpxHeader, &Device->ConnectionlessHeader, sizeof(IPX_HEADER));
*(UNALIGNED ULONG *)Header->IpxHeader.DestinationNetwork = CacheName->Networks[Reserved->u.SR_DG.CurrentNetwork].Network;
RtlCopyMemory (&Header->IpxHeader.DestinationNode, BroadcastAddress, 6);
LocalTarget = &CacheName->Networks[Reserved->u.SR_DG.CurrentNetwork].LocalTarget;
HeaderLength = sizeof(IPX_HEADER) + sizeof(NB_DATAGRAM);
PacketLength = HeaderLength + (ULONG) REQUEST_INFORMATION(Reserved->u.SR_DG.DatagramRequest);
Header->IpxHeader.PacketLength[0] = (UCHAR)(PacketLength / 256);
Header->IpxHeader.PacketLength[1] = (UCHAR)(PacketLength % 256);
Header->IpxHeader.PacketType = 0x04;
//
// Now fill in the Netbios header.
//
Header->Datagram.ConnectionControlFlag = 0x00;
RtlCopyMemory(
Header->Datagram.SourceName,
Reserved->u.SR_DG.AddressFile->Address->NetbiosAddress.NetbiosName,
16);
if (Reserved->u.SR_DG.RemoteName != (PVOID)-1) {
//
// This is a directed, as opposed to broadcast, datagram.
//
Header->Datagram.DataStreamType = NB_CMD_DATAGRAM;
RtlCopyMemory(
Header->Datagram.DestinationName,
Reserved->u.SR_DG.RemoteName->NetbiosName,
16);
} else {
Header->Datagram.DataStreamType = NB_CMD_BROADCAST_DATAGRAM;
RtlZeroMemory(
Header->Datagram.DestinationName,
16);
}
//
// Now send the frame (IPX will adjust the length of the
// first buffer and the whole frame correctly).
//
if ((Status = (*Device->Bind.SendHandler) (LocalTarget,
Packet,
PacketLength,
HeaderLength)) != STATUS_PENDING) {
NbiSendComplete (Packet, Status);
}
}
VOID
NbiDelayedSendDatagram(
IN PVOID pContextInfo
)
{
NBI_WORK_ITEM_CONTEXT *pContext = (NBI_WORK_ITEM_CONTEXT *) pContextInfo;
PNDIS_PACKET Packet = (PNDIS_PACKET) pContext->Context;
PNB_SEND_RESERVED Reserved = (PNB_SEND_RESERVED)(Packet->ProtocolReserved);
Reserved->CurrentSendIteration = 0;
SendDgram (Packet);
NbiFreeMemory (pContextInfo, sizeof(NBI_WORK_ITEM_CONTEXT), MEMORY_WORK_ITEM,
"Free delayed DgramSend work item");
}
VOID
NbiSendComplete(
IN PNDIS_PACKET Packet,
IN NDIS_STATUS Status
)
/*++
Routine Description:
This routine handles a send completion call from IPX.
Arguments:
Packet - The packet which has been completed.
Status - The status of the send.
Return Value:
None.
--*/
{
PDEVICE Device = NbiDevice;
PADDRESS Address;
PADDRESS_FILE AddressFile;
PCONNECTION Connection;
PREQUEST DatagramRequest;
PREQUEST SendRequest, TmpRequest;
PNDIS_BUFFER CurBuffer, TmpBuffer;
PNETBIOS_CACHE CacheName;
PNDIS_BUFFER SecondBuffer = NULL;
PVOID SecondBufferMemory = NULL;
UINT SecondBufferLength;
ULONG oldvalue;
PNB_SEND_RESERVED Reserved = (PNB_SEND_RESERVED)(Packet->ProtocolReserved);
CTELockHandle CancelLH;
#if defined(_PNP_POWER)
CTELockHandle LockHandle;
#endif _PNP_POWER
//
// We jump back here if we re-call send from inside this
// function and it doesn't pend (to avoid stack overflow).
//
++Device->Statistics.PacketsSent;
switch (Reserved->Type) {
case SEND_TYPE_SESSION_DATA:
//
// This was a send on a session. This references the
// IRP.
//
NB_DEBUG2 (SEND, ("Complete NDIS packet %lx\n", Reserved));
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
Connection = Reserved->u.SR_CO.Connection;
SendRequest = Reserved->u.SR_CO.Request;
if (!Reserved->u.SR_CO.NoNdisBuffer) {
CurBuffer = NDIS_BUFFER_LINKAGE (NDIS_BUFFER_LINKAGE(Reserved->HeaderBuffer));
while (CurBuffer) {
TmpBuffer = NDIS_BUFFER_LINKAGE (CurBuffer);
NdisFreeBuffer (CurBuffer);
CurBuffer = TmpBuffer;
}
}
//
// If NoNdisBuffer is TRUE, then we could set
// Connection->SendBufferInUse to FALSE here. The
// problem is that a new send might be in progress
// by the time this completes and it may have
// used the user buffer, then if we need to
// retransmit that packet we would use the buffer
// twice. We instead rely on the fact that whenever
// we make a new send active we set SendBufferInUse
// to FALSE. The net effect is that the user's buffer
// can be used the first time a send is packetize
// but not on resends.
//
NDIS_BUFFER_LINKAGE (NDIS_BUFFER_LINKAGE(Reserved->HeaderBuffer)) = NULL;
NdisRecalculatePacketCounts (Packet);
#if DBG
if (REQUEST_REFCOUNT(SendRequest) > 100) {
DbgPrint ("Request %lx (%lx) has high refcount\n",
Connection, SendRequest);
DbgBreakPoint();
}
#endif
#if defined(__PNP)
NB_GET_LOCK( &Connection->Lock, &LockHandle );
oldvalue = REQUEST_REFCOUNT(SendRequest)--;
if ( DEVICE_NETWORK_PATH_NOT_FOUND == Status ) {
Connection->LocalTarget = Reserved->LocalTarget;
}
NB_FREE_LOCK( &Connection->Lock, LockHandle );
#else
oldvalue = NB_ADD_ULONG(
&REQUEST_REFCOUNT (SendRequest),
(ULONG)-1,
&Connection->Lock);
#endif __PNP
if (oldvalue == 1) {
//
// If the refcount on this request is now zero then
// we already got the ack for it, which means
// that the ack-processing code has unlinked the
// request from Connection->SendQueue. So we
// can just run the queue of connections here
// and complete them.
//
// We dereference the connection for all but one
// of the requests, we hang on to that until a bit
// later so everything stays around.
//
while (TRUE) {
TmpRequest = REQUEST_SINGLE_LINKAGE (SendRequest);
NB_DEBUG2 (SEND, ("Completing request %lx from send complete\n", SendRequest));
REQUEST_STATUS (SendRequest) = STATUS_SUCCESS;
NB_GET_CANCEL_LOCK( &CancelLH );
IoSetCancelRoutine (SendRequest, (PDRIVER_CANCEL)NULL);
NB_FREE_CANCEL_LOCK( CancelLH );
NbiCompleteRequest (SendRequest);
NbiFreeRequest (Device, SendRequest);
++Connection->ConnectionInfo.TransmittedTsdus;
SendRequest = TmpRequest;
if (SendRequest == NULL) {
break;
}
NbiDereferenceConnection (Connection, CREF_SEND);
}
}
if (Reserved->OwnedByConnection) {
Connection->SendPacketInUse = FALSE;
if (Connection->OnWaitPacketQueue) {
//
// This will put the connection on the packetize
// queue if appropriate.
//
NbiCheckForWaitPacket (Connection);
}
} else {
NbiPushSendPacket(Reserved);
}
if (oldvalue == 1) {
NbiDereferenceConnection (Connection, CREF_SEND);
}
break;
case SEND_TYPE_NAME_FRAME:
//
// The frame is an add name/delete name; put it back in
// the pool and deref the address.
//
CTEAssert (Reserved->SendInProgress);
Address = Reserved->u.SR_NF.Address;
Reserved->SendInProgress = FALSE;
NbiPushSendPacket (Reserved);
if (Address) {
NbiDereferenceAddress (Address, AREF_NAME_FRAME);
} else {
NbiDereferenceDevice (Device, DREF_NAME_FRAME);
}
break;
case SEND_TYPE_SESSION_INIT:
//
// This is a session initialize or session init ack; free
// the second buffer, put the packet back in the pool and
// deref the device.
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
NdisUnchainBufferAtBack (Packet, &SecondBuffer);
if (SecondBuffer)
{
NdisQueryBufferSafe (SecondBuffer, &SecondBufferMemory, &SecondBufferLength, HighPagePriority);
CTEAssert (SecondBufferLength == sizeof(NB_SESSION_INIT));
if (SecondBufferMemory)
{
NbiFreeMemory (SecondBufferMemory, sizeof(NB_SESSION_INIT), MEMORY_CONNECTION,
"Session Initialize");
}
NdisFreeBuffer(SecondBuffer);
}
NbiPushSendPacket (Reserved);
NbiDereferenceDevice (Device, DREF_SESSION_INIT);
break;
case SEND_TYPE_SESSION_NO_DATA:
//
// This is a frame which was sent on a connection but
// has no data (ack, session end, session end ack).
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
Connection = Reserved->u.SR_CO.Connection;
if (Reserved->OwnedByConnection) {
CTEAssert (Connection != NULL);
Connection->SendPacketInUse = FALSE;
if (Connection->OnWaitPacketQueue) {
//
// This will put the connection on the packetize
// queue if appropriate.
//
NbiCheckForWaitPacket (Connection);
}
} else {
NbiPushSendPacket(Reserved);
}
if (Connection != NULL) {
NbiDereferenceConnection (Connection, CREF_FRAME);
} else {
NbiDereferenceDevice (Device, DREF_FRAME);
}
break;
case SEND_TYPE_FIND_NAME:
//
// The frame is a find name; just set SendInProgress to
// FALSE and FindNameTimeout will clean it up.
//
#if defined(_PNP_POWER)
NB_GET_LOCK( &Device->Lock, &LockHandle);
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
//
// We keep track of when it finds a net that isn't
// a down wan line so that we can tell when datagram
// sends should fail (otherwise we succeed them, so
// the browser won't think this is a down wan line).
//
if ( STATUS_SUCCESS == Status ) {
NB_SET_SR_FN_SENT_ON_UP_LINE (Reserved, TRUE);
} else {
NB_DEBUG( CACHE, ("Send complete of find name with failure %lx\n",Status ));
}
NB_FREE_LOCK(&Device->Lock, LockHandle);
#else
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
#endif _PNP_POWER
break;
case SEND_TYPE_DATAGRAM:
//
// If there are any more networks to send this on then
// do so, otherwise put it back in the pool and complete
// the request.
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
if ((Reserved->u.SR_DG.Cache == NULL) ||
(++Reserved->u.SR_DG.CurrentNetwork >=
Reserved->u.SR_DG.Cache->NetworksUsed)) {
AddressFile = Reserved->u.SR_DG.AddressFile;
DatagramRequest = Reserved->u.SR_DG.DatagramRequest;
NB_DEBUG2 (DATAGRAM, ("Completing datagram %lx on %lx\n", DatagramRequest, AddressFile));
//
// Remove any user buffers chained on this packet.
//
NdisReinitializePacket (Packet);
NDIS_BUFFER_LINKAGE (NDIS_BUFFER_LINKAGE(Reserved->HeaderBuffer)) = NULL;
NdisChainBufferAtFront (Packet, Reserved->HeaderBuffer);
//
// Complete the request.
//
REQUEST_STATUS(DatagramRequest) = Status;
NbiCompleteRequest(DatagramRequest);
NbiFreeRequest (Device, DatagramRequest);
CacheName = Reserved->u.SR_DG.Cache;
NbiPushSendPacket (Reserved);
//
// Since we are no longer referencing the cache
// name, see if we should delete it (this will
// happen if the cache entry was aged out while
// the datagram was being processed).
//
if (CacheName != NULL) {
oldvalue = NB_ADD_ULONG(
&CacheName->ReferenceCount,
(ULONG)-1,
&Device->Lock);
if (oldvalue == 1) {
NB_DEBUG2 (CACHE, ("Free aged cache entry %lx\n", CacheName));
NbiFreeMemory(
CacheName,
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
MEMORY_CACHE,
"Free old cache");
}
}
NbiDereferenceAddressFile (AddressFile, AFREF_SEND_DGRAM);
} else {
NBI_WORK_ITEM_CONTEXT *WorkItem;
if ((++Reserved->CurrentSendIteration >= MAX_SEND_ITERATIONS) &&
(WorkItem = (NBI_WORK_ITEM_CONTEXT *) NbiAllocateMemory (sizeof(NBI_WORK_ITEM_CONTEXT),
MEMORY_WORK_ITEM,
"Delayed DgramSend work item")))
{
WorkItem->Context = (PVOID) Packet;
ExInitializeWorkItem (&WorkItem->Item, NbiDelayedSendDatagram, (PVOID)WorkItem);
ExQueueWorkItem(&WorkItem->Item, DelayedWorkQueue);
}
else
{
SendDgram (Packet);
}
}
break;
case SEND_TYPE_STATUS_QUERY:
//
// This is an adapter status query, which is a simple
// packet.
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
NbiPushSendPacket (Reserved);
NbiDereferenceDevice (Device, DREF_STATUS_FRAME);
break;
case SEND_TYPE_STATUS_RESPONSE:
//
// This is an adapter status response, we have to free the
// second buffer.
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
NdisUnchainBufferAtBack (Packet, &SecondBuffer);
if (SecondBuffer)
{
NdisQueryBufferSafe (SecondBuffer, &SecondBufferMemory, &SecondBufferLength, HighPagePriority);
if (SecondBufferMemory)
{
NbiFreeMemory (SecondBufferMemory, Reserved->u.SR_AS.ActualBufferLength, MEMORY_STATUS,
"Adapter Status");
}
NdisFreeBuffer(SecondBuffer);
}
NbiPushSendPacket (Reserved);
NbiDereferenceDevice (Device, DREF_STATUS_RESPONSE);
break;
#ifdef RSRC_TIMEOUT_DBG
case SEND_TYPE_DEATH_PACKET:
//
// This is a session initialize or session init ack; free
// the second buffer, put the packet back in the pool and
// deref the device.
//
CTEAssert (Reserved->SendInProgress);
Reserved->SendInProgress = FALSE;
DbgPrint("********Death packet send completed status %lx\n",Status);
DbgBreakPoint();
break;
#endif //RSRC_TIMEOUT_DBG
default:
CTEAssert (FALSE);
break;
}
} /* NbiSendComplete */
#if 0
ULONG NbiLoudSendQueue = 1;
#endif
VOID
NbiAssignSequenceAndSend(
IN PCONNECTION Connection,
IN PNDIS_PACKET Packet
IN NB_LOCK_HANDLE_PARAM(LockHandle)
)
/*++
Routine Description:
This routine is used to ensure that receive sequence numbers on
packets are numbered correctly. It is called in place of the lower-level
send handler; after assigning the receive sequence number it locks out
other sends until the NdisSend call has returned (not necessarily completed),
insuring that the packets with increasing receive sequence numbers
are queue in the right order by the MAC.
NOTE: THIS ROUTINE IS CALLED WITH THE CONNECTION LOCK HELD, AND
RETURNS WITH IT RELEASED.
Arguments:
Connection - The connection the send is on.
Packet - The packet to send.
LockHandle - The handle with which Connection->Lock was acquired.
Return Value:
None.
--*/
{
NDIS_STATUS NdisStatus;
PNB_SEND_RESERVED Reserved;
PLIST_ENTRY p;
NB_CONNECTION UNALIGNED * Header;
PDEVICE Device = NbiDevice;
BOOLEAN NdisSendReference;
ULONG result;
Reserved = (PNB_SEND_RESERVED)(Packet->ProtocolReserved);
CTEAssert (Connection->State == CONNECTION_STATE_ACTIVE);
//
// If there is a send in progress, then queue this packet
// and return.
//
if (Connection->NdisSendsInProgress > 0) {
NB_DEBUG2 (SEND, ("Queueing send packet %lx on %lx\n", Reserved, Connection));
InsertTailList (&Connection->NdisSendQueue, &Reserved->WaitLinkage);
++Connection->NdisSendsInProgress;
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
return;
}
//
// No send in progress. Set the flag to true, and fill in the
// receive sequence fields in the packet.
//
Connection->NdisSendsInProgress = 1;
NdisSendReference = FALSE;
Connection->NdisSendReference = &NdisSendReference;
while (TRUE) {
Header = (NB_CONNECTION UNALIGNED *)
(&Reserved->Header[Device->Bind.IncludedHeaderOffset]);
Header->Session.ReceiveSequence = Connection->ReceiveSequence;
if (Connection->NewNetbios) {
Header->Session.ReceiveSequenceMax = Connection->LocalRcvSequenceMax;
} else {
Header->Session.BytesReceived = (USHORT)Connection->CurrentReceive.MessageOffset;
}
//
// Since we are acking as much as we know, we can clear
// this flag. The connection will eventually get removed
// from the queue by the long timeout.
//
Connection->DataAckPending = FALSE;
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
NdisAdjustBufferLength(NB_GET_NBHDR_BUFF(Packet), sizeof(NB_CONNECTION));
NdisStatus = (*Device->Bind.SendHandler)(
&Connection->LocalTarget,
Packet,
Reserved->u.SR_CO.PacketLength,
sizeof(NB_CONNECTION));
if (NdisStatus != NDIS_STATUS_PENDING) {
NbiSendComplete(
Packet,
NdisStatus);
}
//
// Take the ref count down, which may allow others
// to come through.
//
result = NB_ADD_ULONG(
&Connection->NdisSendsInProgress,
(ULONG)-1,
&Connection->Lock);
//
// We have now sent a packet, see if any queued up while we
// were doing it. If the count was zero after removing ours,
// then anything else queued is being processed, so we can
// exit. If the connection was stopped while we were sending,
// a special reference was added which we remove (NbiStopConnection
// sets NdisSendReference to TRUE, using the pointer saved
// in Connection->NdisSendReference).
//
if (result == 1) {
if (NdisSendReference) {
NB_DEBUG2 (SEND, ("Remove CREF_NDIS_SEND from %lx\n", Connection));
NbiDereferenceConnection (Connection, CREF_NDIS_SEND);
}
return;
}
NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
p = RemoveHeadList(&Connection->NdisSendQueue);
//
// If the refcount was not zero, then nobody else should
// have taken packets off since they would have been
// blocked by us. So, the queue should not be empty.
//
ASSERT (p != &Connection->NdisSendQueue);
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
Packet = CONTAINING_RECORD (Reserved, NDIS_PACKET, ProtocolReserved[0]);
} // while loop
//
// We should never reach here.
//
CTEAssert (FALSE);
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
} /* NbiAssignSequenceAndSend */
NTSTATUS
NbiTdiSend(
IN PDEVICE Device,
IN PREQUEST Request
)
/*++
Routine Description:
This routine does a send on an active connection.
Arguments:
Device - The netbios device.
Request - The request describing the send.
Return Value:
NTSTATUS - status of operation.
--*/
{
PCONNECTION Connection;
PTDI_REQUEST_KERNEL_SEND Parameters;
NB_DEFINE_SYNC_CONTEXT (SyncContext)
NB_DEFINE_LOCK_HANDLE (LockHandle)
CTELockHandle CancelLH;
//
// Check that the file type is valid
//
if (REQUEST_OPEN_TYPE(Request) != (PVOID)TDI_CONNECTION_FILE)
{
CTEAssert(FALSE);
return (STATUS_INVALID_ADDRESS_COMPONENT);
}
//
// First make sure the connection is valid.
//
Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(Request);
if (Connection->Type == NB_CONNECTION_SIGNATURE) {
NB_GET_CANCEL_LOCK( &CancelLH );
NB_BEGIN_SYNC (&SyncContext);
NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
//
// Make sure the connection is in a good state.
//
if (Connection->State == CONNECTION_STATE_ACTIVE) {
//
// If the connection is idle then send it now, otherwise
// queue it.
//
if (!Request->Cancel) {
Parameters = (PTDI_REQUEST_KERNEL_SEND)REQUEST_PARAMETERS(Request);
//
// For old netbios, don't allow sends greater than 64K-1.
//
if ((Connection->NewNetbios) ||
(Parameters->SendLength <= 0xffff)) {
IoSetCancelRoutine (Request, NbiCancelSend);
NB_SYNC_SWAP_IRQL( CancelLH, LockHandle );
NB_FREE_CANCEL_LOCK( CancelLH );
REQUEST_INFORMATION (Request) = Parameters->SendLength; // assume it succeeds.
REQUEST_REFCOUNT (Request) = 1; // refcount starts at 1.
NbiReferenceConnectionSync (Connection, CREF_SEND);
//
// NOTE: The connection send queue is managed such
// that the current send being packetized is not on
// the queue. For multiple-request messages, the
// first one is not on the queue, but its linkage
// field points to the next request in the message
// (which will be on the head of the queue).
//
if ((Parameters->SendFlags & TDI_SEND_PARTIAL) == 0) {
//
// This is a final send.
//
if (Connection->SubState == CONNECTION_SUBSTATE_A_IDLE) {
NB_DEBUG2 (SEND, ("Send %lx, connection %lx idle\n", Request, Connection));
Connection->CurrentSend.Request = Request;
Connection->CurrentSend.MessageOffset = 0;
Connection->CurrentSend.Buffer = REQUEST_NDIS_BUFFER (Request);
Connection->CurrentSend.BufferOffset = 0;
Connection->SendBufferInUse = FALSE;
Connection->UnAckedSend = Connection->CurrentSend;
Connection->FirstMessageRequest = Request;
#ifdef RSRC_TIMEOUT_DBG
KeQuerySystemTime(&Connection->FirstMessageRequestTime);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Connection->FirstMessageRequestTime.QuadPart;
#endif //RSRC_TIMEOUT_DBG
Connection->LastMessageRequest = Request;
Connection->CurrentMessageLength = Parameters->SendLength;
//
// This frees the connection lock.
//
NbiPacketizeSend(
Connection
NB_LOCK_HANDLE_ARG(LockHandle)
);
} else if (Connection->SubState == CONNECTION_SUBSTATE_A_W_EOR) {
//
// We have been collecting partial sends waiting
// for a final one, which we have now received,
// so start packetizing.
//
// We chain it on the back of the send queue,
// in addition if this is the second request in the
// message, we have to link the first request (which
// is not on the queue) to this one.
//
//
NB_DEBUG2 (SEND, ("Send %lx, connection %lx got eor\n", Request, Connection));
Connection->LastMessageRequest = Request;
Connection->CurrentMessageLength += Parameters->SendLength;
if (Connection->SendQueue.Head == NULL) {
REQUEST_SINGLE_LINKAGE(Connection->FirstMessageRequest) = Request;
}
REQUEST_SINGLE_LINKAGE(Request) = NULL;
REQUEST_LIST_INSERT_TAIL(&Connection->SendQueue, Request);
Connection->UnAckedSend = Connection->CurrentSend;
#ifdef RSRC_TIMEOUT_DBG
{
LARGE_INTEGER Time;
KeQuerySystemTime(&Time);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Time.QuadPart;
}
#endif //RSRC_TIMEOUT_DBG
//
// This frees the connection lock.
//
NbiPacketizeSend(
Connection
NB_LOCK_HANDLE_ARG(LockHandle)
);
} else {
//
// The state is PACKETIZE, W_ACK, or W_PACKET.
//
NB_DEBUG2 (SEND, ("Send %lx, connection %lx busy\n", Request, Connection));
REQUEST_SINGLE_LINKAGE(Request) = NULL;
REQUEST_LIST_INSERT_TAIL(&Connection->SendQueue, Request);
#ifdef RSRC_TIMEOUT_DBG
{
LARGE_INTEGER Time;
KeQuerySystemTime(&Time);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Time.QuadPart;
}
#endif //RSRC_TIMEOUT_DBG
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} else {
//
// This is a partial send. We queue them up without
// packetizing until we get a final (this is because
// we have to put a correct Connection->CurrentMessageLength
// in the frames.
//
if (Connection->SubState == CONNECTION_SUBSTATE_A_IDLE) {
//
// Start collecting partial sends. NOTE: Partial sends
// are always inserted in the send queue
//
Connection->CurrentSend.Request = Request;
Connection->CurrentSend.MessageOffset = 0;
Connection->CurrentSend.Buffer = REQUEST_NDIS_BUFFER (Request);
Connection->CurrentSend.BufferOffset = 0;
Connection->SendBufferInUse = FALSE;
Connection->FirstMessageRequest = Request;
#ifdef RSRC_TIMEOUT_DBG
KeQuerySystemTime(&Connection->FirstMessageRequestTime);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Connection->FirstMessageRequestTime.QuadPart;
#endif //RSRC_TIMEOUT_DBG
Connection->CurrentMessageLength = Parameters->SendLength;
Connection->SubState = CONNECTION_SUBSTATE_A_W_EOR;
} else if (Connection->SubState == CONNECTION_SUBSTATE_A_W_EOR) {
//
// We have got another partial send to add to our
// list. We chain it on the back of the send queue,
// in addition if this is the second request in the
// message, we have to link the first request (which
// is not on the queue) to this one.
//
Connection->LastMessageRequest = Request;
Connection->CurrentMessageLength += Parameters->SendLength;
if (Connection->SendQueue.Head == NULL) {
REQUEST_SINGLE_LINKAGE(Connection->FirstMessageRequest) = Request;
}
REQUEST_SINGLE_LINKAGE(Request) = NULL;
REQUEST_LIST_INSERT_TAIL(&Connection->SendQueue, Request);
#ifdef RSRC_TIMEOUT_DBG
{
LARGE_INTEGER Time;
KeQuerySystemTime(&Time);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Time.QuadPart;
}
#endif //RSRC_TIMEOUT_DBG
} else {
REQUEST_SINGLE_LINKAGE(Request) = NULL;
REQUEST_LIST_INSERT_TAIL(&Connection->SendQueue, Request);
#ifdef RSRC_TIMEOUT_DBG
{
LARGE_INTEGER Time;
KeQuerySystemTime(&Time);
(((LARGE_INTEGER UNALIGNED *)&(IoGetCurrentIrpStackLocation(Request))->Parameters.Others.Argument3))->QuadPart =
Time.QuadPart;
}
#endif //RSRC_TIMEOUT_DBG
}
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
NB_END_SYNC (&SyncContext);
return STATUS_PENDING;
} else {
NB_DEBUG2 (SEND, ("Send %lx, too long for connection %lx (%d)\n", Request, Connection, Parameters->SendLength));
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
NB_END_SYNC (&SyncContext);
NB_FREE_CANCEL_LOCK( CancelLH );
return STATUS_INVALID_PARAMETER;
}
} else {
NB_DEBUG2 (SEND, ("Send %lx, connection %lx cancelled\n", Request, Connection));
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
NB_END_SYNC (&SyncContext);
NB_FREE_CANCEL_LOCK( CancelLH );
return STATUS_CANCELLED;
}
} else {
NB_DEBUG (SEND, ("Send connection %lx state is %d\n", Connection, Connection->State));
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
NB_END_SYNC (&SyncContext);
NB_FREE_CANCEL_LOCK( CancelLH );
return STATUS_INVALID_CONNECTION;
}
} else {
NB_DEBUG (SEND, ("Send connection %lx has bad signature\n", Connection));
return STATUS_INVALID_CONNECTION;
}
} /* NbiTdiSend */
VOID
NbiPacketizeSend(
IN PCONNECTION Connection
IN NB_LOCK_HANDLE_PARAM(LockHandle)
)
/*++
Routine Description:
This routine does a send on an active connection.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT RELEASED.
Arguments:
Connection - The connection.
LockHandle - The handle used to acquire the lock.
Return Value:
None.
--*/
{
PREQUEST Request;
PNDIS_PACKET Packet;
PNDIS_BUFFER BufferChain;
PNB_SEND_RESERVED Reserved;
PDEVICE Device = NbiDevice;
NB_CONNECTION UNALIGNED * Header;
ULONG PacketLength;
ULONG PacketSize;
ULONG DesiredLength;
ULONG ActualLength;
NTSTATUS Status;
PSINGLE_LIST_ENTRY s;
USHORT ThisSendSequence;
USHORT ThisOffset;
BOOLEAN ExitAfterSend;
UCHAR ConnectionControlFlag;
CTELockHandle DeviceLockHandle;
//
// We jump back here if we are talking new Netbios and it
// is OK to packetize another send.
//
SendAnotherPacket:
//
// If we decide to packetize another send after this, we
// change ExitAfterSend to FALSE and SubState to PACKETIZE.
// Right now we don't change SubState in case it is W_PACKET.
//
ExitAfterSend = TRUE;
CTEAssert (Connection->CurrentSend.Request != NULL);
if (Connection->NewNetbios) {
//
// Check that we have send window, both that advertised
// by the remote and our own locally-decided window which
// may be smaller.
//
if (((USHORT)(Connection->CurrentSend.SendSequence-1) == Connection->RemoteRcvSequenceMax) ||
(((USHORT)(Connection->CurrentSend.SendSequence - Connection->UnAckedSend.SendSequence)) >= Connection->SendWindowSize)) {
//
// Keep track of whether we are waiting because of his window
// or because of our local window. If it is because of our local
// window then we may want to adjust it after this window
// is acked.
//
if ((USHORT)(Connection->CurrentSend.SendSequence-1) != Connection->RemoteRcvSequenceMax) {
Connection->SubState = CONNECTION_SUBSTATE_A_W_ACK;
NB_DEBUG2 (SEND, ("Connection %lx local shut down at %lx, %lx\n", Connection, Connection->CurrentSend.SendSequence, Connection->UnAckedSend.SendSequence));
} else {
Connection->SubState = CONNECTION_SUBSTATE_A_REMOTE_W;
NB_DEBUG2 (SEND, ("Connection %lx remote shut down at %lx\n", Connection, Connection->CurrentSend.SendSequence));
}
//
// Start the timer so we will keep bugging him about
// this. What if he doesn't get a receive down
// quickly -- but this is better than losing his ack
// and then dying. We won't really back off our timer
// because we will keep getting acks, and resetting it.
//
NbiStartRetransmit (Connection);
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
return;
}
}
Request = Connection->CurrentSend.Request;
//
// If we are in this routine then we know that
// we are coming out of IDLE, W_ACK, or W_PACKET
// and we still have the lock held. We also know
// that there is a send request in progress. If
// an ack for none or part of the last packet was
// received, then our send pointers have been
// adjusted to reflect that.
//
//
// First get a packet for the current send.
//
if (!Connection->SendPacketInUse) {
Connection->SendPacketInUse = TRUE;
Packet = PACKET(&Connection->SendPacket);
Reserved = (PNB_SEND_RESERVED)(Packet->ProtocolReserved);
} else {
s = ExInterlockedPopEntrySList(
&Device->SendPacketList,
&NbiGlobalPoolInterlock);
if (s == NULL) {
//
// This function tries to allocate another packet pool.
//
s = NbiPopSendPacket(Device, FALSE);
if (s == NULL) {
//
// It is possible to come in here and already be in
// W_PACKET state -- this is because we may packetize
// when in that state, and rather than always be
// checking that we weren't in W_PACKET, we go
// ahead and check again here.
//
if (Connection->SubState != CONNECTION_SUBSTATE_A_W_PACKET) {
Connection->SubState = CONNECTION_SUBSTATE_A_W_PACKET;
NB_GET_LOCK (&Device->Lock, &DeviceLockHandle);
if (!Connection->OnWaitPacketQueue) {
NbiReferenceConnectionLock (Connection, CREF_W_PACKET);
Connection->OnWaitPacketQueue = TRUE;
InsertTailList(
&Device->WaitPacketConnections,
&Connection->WaitPacketLinkage
);
// NB_INSERT_TAIL_LIST(
// &Device->WaitPacketConnections,
// &Connection->WaitPacketLinkage,
// &Device->Lock);
}
NB_FREE_LOCK (&Device->Lock, DeviceLockHandle);
}
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
return;
}
}
Reserved = CONTAINING_RECORD (s, NB_SEND_RESERVED, PoolLinkage);
Packet = CONTAINING_RECORD (Reserved, NDIS_PACKET, ProtocolReserved[0]);
}
//
// Set this now, we will change it later if needed.
//
Connection->SubState = CONNECTION_SUBSTATE_A_W_ACK;
//
// Save these since they go in this next packet.
//
ThisSendSequence = Connection->CurrentSend.SendSequence;
ThisOffset = (USHORT)Connection->CurrentSend.MessageOffset;
//
// Now see if we need to copy the buffer chain.
//
PacketSize = Connection->MaximumPacketSize;
if (Connection->CurrentSend.MessageOffset + PacketSize >= Connection->CurrentMessageLength) {
PacketSize = Connection->CurrentMessageLength - Connection->CurrentSend.MessageOffset;
if ((Connection->CurrentSend.MessageOffset == 0) &&
(!Connection->SendBufferInUse)) {
//
// If the entire send remaining fits in one packet,
// and this is also the first packet in the send,
// then the entire send fits in one packet and
// we don't need to build a duplicate buffer chain.
//
BufferChain = Connection->CurrentSend.Buffer;
Reserved->u.SR_CO.NoNdisBuffer = TRUE;
Connection->CurrentSend.Buffer = NULL;
Connection->CurrentSend.BufferOffset = 0;
Connection->CurrentSend.MessageOffset = Connection->CurrentMessageLength;
Connection->CurrentSend.Request = NULL;
++Connection->CurrentSend.SendSequence;
Connection->SendBufferInUse = TRUE;
if (Connection->NewNetbios) {
if ((ThisSendSequence == Connection->RemoteRcvSequenceMax) ||
((((PTDI_REQUEST_KERNEL_SEND)REQUEST_PARAMETERS(Request))->SendFlags) &
TDI_SEND_NO_RESPONSE_EXPECTED)) { // optimize this check
ConnectionControlFlag = NB_CONTROL_EOM | NB_CONTROL_SEND_ACK;
} else {
ConnectionControlFlag = NB_CONTROL_EOM;
}
Connection->PiggybackAckTimeout = FALSE;
} else {
ConnectionControlFlag = NB_CONTROL_SEND_ACK;
}
if (BufferChain != NULL) {
NB_DEBUG2 (SEND, ("Send packet %lx on %lx (%d/%d), user buffer\n",
Reserved, Connection,
Connection->CurrentSend.SendSequence,
Connection->CurrentSend.MessageOffset));
NdisChainBufferAtBack (Packet, BufferChain);
} else {
NB_DEBUG2 (SEND, ("Send packet %lx on %lx (%d/%d), no buffer\n",
Reserved, Connection,
Connection->CurrentSend.SendSequence,
Connection->CurrentSend.MessageOffset));
}
goto GotBufferChain;
}
}
//
// We need to build a partial buffer chain. In the case
// where the current request is a partial one, we may
// build this from the ndis buffer chains of several
// requests.
//
if (PacketSize > 0) {
DesiredLength = PacketSize;
NB_DEBUG2 (SEND, ("Send packet %lx on %lx (%d/%d), allocate buffer\n",
Reserved, Connection,
Connection->CurrentSend.SendSequence,
Connection->CurrentSend.MessageOffset));
while (TRUE) {
Status = NbiBuildBufferChainFromBufferChain (
Device->NdisBufferPoolHandle,
Connection->CurrentSend.Buffer,
Connection->CurrentSend.BufferOffset,
DesiredLength,
&BufferChain,
&Connection->CurrentSend.Buffer,
&Connection->CurrentSend.BufferOffset,
&ActualLength);
if (Status != STATUS_SUCCESS) {
PNDIS_BUFFER CurBuffer, TmpBuffer;
NB_DEBUG2 (SEND, ("Allocate buffer chain failed for packet %lx\n", Reserved));
//
// We could not allocate resources for this send.
// We'll put the connection on the packetize
// queue and hope we get more resources later.
//
NbiReferenceConnectionSync (Connection, CREF_PACKETIZE);
CTEAssert (!Connection->OnPacketizeQueue);
Connection->OnPacketizeQueue = TRUE;
//
// Connection->CurrentSend can stay where it is.
//
NB_INSERT_TAIL_LIST(
&Device->PacketizeConnections,
&Connection->PacketizeLinkage,
&Device->Lock);
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
//
// Free any buffers we have allocated on previous calls
// to BuildBufferChain inside this same while(TRUE) loop,
// then free the packet.
//
CurBuffer = NDIS_BUFFER_LINKAGE (NDIS_BUFFER_LINKAGE(Reserved->HeaderBuffer));
while (CurBuffer) {
TmpBuffer = NDIS_BUFFER_LINKAGE (CurBuffer);
NdisFreeBuffer (CurBuffer);
CurBuffer = TmpBuffer;
}
NDIS_BUFFER_LINKAGE (NDIS_BUFFER_LINKAGE(Reserved->HeaderBuffer)) = NULL;
NdisRecalculatePacketCounts (Packet);
if (Reserved->OwnedByConnection) {
Connection->SendPacketInUse = FALSE;
} else {
NbiPushSendPacket(Reserved);
}
return;
}
NdisChainBufferAtBack (Packet, BufferChain);
Connection->CurrentSend.MessageOffset += ActualLength;
DesiredLength -= ActualLength;
if (DesiredLength == 0) {
//
// We have gotten enough data for our packet.
//
if (Connection->CurrentSend.MessageOffset == Connection->CurrentMessageLength) {
Connection->CurrentSend.Request = NULL;
}
break;
}
//
// We ran out of buffer chain on this send, which means
// that we must have another one behind it (since we
// don't start packetizing partial sends until all of
// them are queued).
//
Request = REQUEST_SINGLE_LINKAGE(Request);
if (Request == NULL) {
KeBugCheck (NDIS_INTERNAL_ERROR);
}
Connection->CurrentSend.Request = Request;
Connection->CurrentSend.Buffer = REQUEST_NDIS_BUFFER (Request);
Connection->CurrentSend.BufferOffset = 0;
}
} else {
//
// This is a zero-length send (in general we will go
// through the code before the if that uses the user's
// buffer, but not on a resend).
//
Connection->CurrentSend.Buffer = NULL;
Connection->CurrentSend.BufferOffset = 0;
CTEAssert (Connection->CurrentSend.MessageOffset == Connection->CurrentMessageLength);
Connection->CurrentSend.Request = NULL;
NB_DEBUG2 (SEND, ("Send packet %lx on %lx (%d/%d), no alloc buf\n",
Reserved, Connection,
Connection->CurrentSend.SendSequence,
Connection->CurrentSend.MessageOffset));
}
Reserved->u.SR_CO.NoNdisBuffer = FALSE;
if (Connection->NewNetbios) {
++Connection->CurrentSend.SendSequence;
if (Connection->CurrentSend.MessageOffset == Connection->CurrentMessageLength) {
if (((USHORT)(Connection->CurrentSend.SendSequence - Connection->UnAckedSend.SendSequence)) >= Connection->SendWindowSize) {
ConnectionControlFlag = NB_CONTROL_EOM | NB_CONTROL_SEND_ACK;
} else if ((ThisSendSequence == Connection->RemoteRcvSequenceMax) ||
((((PTDI_REQUEST_KERNEL_SEND)REQUEST_PARAMETERS(Request))->SendFlags) &
TDI_SEND_NO_RESPONSE_EXPECTED)) { // optimize this check
ConnectionControlFlag = NB_CONTROL_EOM | NB_CONTROL_SEND_ACK;
} else {
ConnectionControlFlag = NB_CONTROL_EOM;
}
Connection->PiggybackAckTimeout = FALSE;
} else if (((USHORT)(Connection->CurrentSend.SendSequence - Connection->UnAckedSend.SendSequence)) >= Connection->SendWindowSize) {
ConnectionControlFlag = NB_CONTROL_SEND_ACK;
} else if (ThisSendSequence == Connection->RemoteRcvSequenceMax) {
ConnectionControlFlag = NB_CONTROL_SEND_ACK;
} else {
ConnectionControlFlag = 0;
ExitAfterSend = FALSE;
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
}
} else {
ConnectionControlFlag = NB_CONTROL_SEND_ACK;
if (Connection->CurrentSend.MessageOffset == Connection->CurrentMessageLength) {
++Connection->CurrentSend.SendSequence;
}
}
GotBufferChain:
//
// We have a packet and a buffer chain, there are
// no other resources required for a send so we can
// fill in the header and go.
//
CTEAssert (Reserved->SendInProgress == FALSE);
Reserved->SendInProgress = TRUE;
Reserved->Type = SEND_TYPE_SESSION_DATA;
Reserved->u.SR_CO.Connection = Connection;
Reserved->u.SR_CO.Request = Connection->FirstMessageRequest;
PacketLength = PacketSize + sizeof(NB_CONNECTION);
Reserved->u.SR_CO.PacketLength = PacketLength;
Header = (NB_CONNECTION UNALIGNED *)
(&Reserved->Header[Device->Bind.IncludedHeaderOffset]);
RtlCopyMemory((PVOID)&Header->IpxHeader, &Connection->RemoteHeader, sizeof(IPX_HEADER));
Header->IpxHeader.PacketLength[0] = (UCHAR)(PacketLength / 256);
Header->IpxHeader.PacketLength[1] = (UCHAR)(PacketLength % 256);
Header->IpxHeader.PacketType = 0x04;
//
// Now fill in the Netbios header. Put this in
// a contiguous buffer in the connection ?
//
Header->Session.ConnectionControlFlag = ConnectionControlFlag;
Header->Session.DataStreamType = NB_CMD_SESSION_DATA;
Header->Session.SourceConnectionId = Connection->LocalConnectionId;
Header->Session.DestConnectionId = Connection->RemoteConnectionId;
Header->Session.SendSequence = ThisSendSequence;
Header->Session.TotalDataLength = (USHORT)Connection->CurrentMessageLength;
Header->Session.Offset = ThisOffset;
Header->Session.DataLength = (USHORT)PacketSize;
#if 0
//
// These are set by NbiAssignSequenceAndSend.
//
Header->Session.ReceiveSequence = Connection->ReceiveSequence;
Header->Session.BytesReceived = (USHORT)Connection->CurrentReceive.MessageOffset;
#endif
//
// Reference the request to account for this send.
//
#if DBG
if (REQUEST_REFCOUNT(Request) > 100) {
DbgPrint ("Request %lx (%lx) has high refcount\n",
Connection, Request);
DbgBreakPoint();
}
#endif
++REQUEST_REFCOUNT (Request);
++Device->TempFramesSent;
Device->TempFrameBytesSent += PacketSize;
//
// Start the timer.
//
NbiStartRetransmit (Connection);
//
// This frees the lock. IPX will adjust the length of
// the first buffer correctly.
//
NbiAssignSequenceAndSend(
Connection,
Packet
NB_LOCK_HANDLE_ARG(LockHandle));
if (!ExitAfterSend) {
//
// Did we need to reference the connection until we
// get the lock back??
//
NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
if ((Connection->State == CONNECTION_STATE_ACTIVE) &&
(Connection->SubState == CONNECTION_SUBSTATE_A_PACKETIZE)) {
//
// Jump back to the beginning of the function to
// repacketize.
goto SendAnotherPacket;
} else {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
}
} /* NbiPacketizeSend */
VOID
NbiAdjustSendWindow(
IN PCONNECTION Connection
)
/*++
Routine Description:
This routine adjusts a connection's send window if needed. It is
assumed that we just got an ack for a full send window.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT HELD.
Arguments:
Connection - The connection.
Return Value:
None.
--*/
{
if (Connection->RetransmitThisWindow) {
//
// Move it down. Check if this keeps happening.
//
if (Connection->SendWindowSize > 2) {
--Connection->SendWindowSize;
NB_DEBUG2 (SEND_WINDOW, ("Lower window to %d on %lx (%lx)\n", Connection->SendWindowSize, Connection, Connection->CurrentSend.SendSequence));
}
if (Connection->SendWindowIncrease) {
//
// We just increased the window.
//
++Connection->IncreaseWindowFailures;
NB_DEBUG2 (SEND_WINDOW, ("%d consecutive increase failues on %lx (%lx)\n", Connection->IncreaseWindowFailures, Connection, Connection->CurrentSend.SendSequence));
if (Connection->IncreaseWindowFailures >= 2) {
if (Connection->MaxSendWindowSize > 2) {
//
// Lock ourselves at a smaller window.
//
Connection->MaxSendWindowSize = Connection->SendWindowSize;
NB_DEBUG2 (SEND_WINDOW, ("Lock send window at %d on %lx (%lx)\n", Connection->MaxSendWindowSize, Connection, Connection->CurrentSend.SendSequence));
}
Connection->IncreaseWindowFailures = 0;
}
Connection->SendWindowIncrease = FALSE;
}
} else {
//
// Increase it if allowed, and make a note
// in case this increase causes problems in
// the next window.
//
if (Connection->SendWindowSize < Connection->MaxSendWindowSize) {
++Connection->SendWindowSize;
NB_DEBUG2 (SEND_WINDOW, ("Raise window to %d on %lx (%lx)\n", Connection->SendWindowSize, Connection, Connection->CurrentSend.SendSequence));
Connection->SendWindowIncrease = TRUE;
} else {
if (Connection->SendWindowIncrease) {
//
// We just increased it and nothing failed,
// which is good.
//
Connection->SendWindowIncrease = FALSE;
Connection->IncreaseWindowFailures = 0;
NB_DEBUG2 (SEND_WINDOW, ("Raised window OK on %lx (%lx)\n", Connection, Connection->CurrentSend.SendSequence));
}
}
}
//
// This controls when we'll check this again.
//
Connection->SendWindowSequenceLimit += Connection->SendWindowSize;
} /* NbiAdjustSendWindow */
VOID
NbiReframeConnection(
IN PCONNECTION Connection,
IN USHORT ReceiveSequence,
IN USHORT BytesReceived,
IN BOOLEAN Resend
IN NB_LOCK_HANDLE_PARAM(LockHandle)
)
/*++
Routine Description:
This routine is called when we have gotten an ack
for some data. It completes any sends that have
been acked, and if needed modifies the current send
pointer and queues the connection for repacketizing.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT RELEASED.
Arguments:
Connection - The connection.
ReceiveSequence - The receive sequence from the remote.
BytesReceived - The number of bytes received in this message.
Resend - If it is OK to resend based on this packet.
LockHandle - The handle with which Connection->Lock was acquired.
Return Value:
None.
--*/
{
PREQUEST Request, TmpRequest;
PREQUEST RequestToComplete;
PDEVICE Device = NbiDevice;
CTELockHandle CancelLH;
//
// We should change to stop the timer
// only if we go idle, since otherwise we still
// want it running, or will restart it when we
// packetize.
//
//
// See how much is acked here.
//
if ((Connection->CurrentSend.MessageOffset == Connection->CurrentMessageLength) &&
(ReceiveSequence == (USHORT)(Connection->CurrentSend.SendSequence)) &&
(Connection->FirstMessageRequest != NULL)) {
// Special check for 0 length send which was not accepted by the remote.
// In this case it will pass the above 3 conditions yet, nothing
// is acked. BUG#10395
if (!Connection->CurrentSend.MessageOffset && Connection->CurrentSend.SendSequence == Connection->UnAckedSend.SendSequence ) {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
return;
}
//
// This acks the entire message.
//
NB_DEBUG2 (SEND, ("Got ack for entire message on %lx (%d)\n", Connection, Connection->CurrentSend.SendSequence));
NbiStopRetransmit (Connection);
Connection->CurrentSend.MessageOffset = 0; // Needed?
Connection->UnAckedSend.MessageOffset = 0;
//
// We don't adjust the send window since we likely stopped
// packetizing before we hit it.
//
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
if (Connection->NewNetbios) {
Connection->RemoteRcvSequenceMax = BytesReceived; // really RcvSeqMac
//
// See if we need to adjust our send window.
//
if (((SHORT)(Connection->CurrentSend.SendSequence - Connection->SendWindowSequenceLimit)) >= 0) {
NbiAdjustSendWindow (Connection);
} else {
//
// Advance this, we won't get meaningful results until we
// send a full window in one message.
//
Connection->SendWindowSequenceLimit = Connection->CurrentSend.SendSequence + Connection->SendWindowSize;
}
}
Connection->RetransmitThisWindow = FALSE;
Request = Connection->FirstMessageRequest;
//
// We dequeue these requests from the connection's
// send queue.
//
if (Connection->FirstMessageRequest == Connection->LastMessageRequest) {
REQUEST_SINGLE_LINKAGE (Request) = NULL;
} else {
Connection->SendQueue.Head = REQUEST_SINGLE_LINKAGE (Connection->LastMessageRequest);
REQUEST_SINGLE_LINKAGE (Connection->LastMessageRequest) = NULL;
}
#if DBG
if (REQUEST_REFCOUNT(Request) > 100) {
DbgPrint ("Request %lx (%lx) has high refcount\n",
Connection, Request);
DbgBreakPoint();
}
#endif
if (--REQUEST_REFCOUNT(Request) == 0) {
RequestToComplete = Request;
} else {
//
// There are still sends pending, this will get
// completed when the last send completes. Since
// we have already unlinked the request from the
// connection's send queue we can do this without
// any locks.
//
RequestToComplete = NULL;
}
//
// Now see if there is a send to activate.
//
NbiRestartConnection (Connection);
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
//
// Now complete any requests we need to.
//
while (RequestToComplete != NULL) {
TmpRequest = REQUEST_SINGLE_LINKAGE (RequestToComplete);
REQUEST_STATUS (RequestToComplete) = STATUS_SUCCESS;
NB_GET_CANCEL_LOCK( &CancelLH );
IoSetCancelRoutine (RequestToComplete, (PDRIVER_CANCEL)NULL);
NB_FREE_CANCEL_LOCK( CancelLH );
NbiCompleteRequest (RequestToComplete);
NbiFreeRequest (Device, RequestToComplete);
++Connection->ConnectionInfo.TransmittedTsdus;
RequestToComplete = TmpRequest;
NbiDereferenceConnection (Connection, CREF_SEND);
}
} else if ((ReceiveSequence == Connection->CurrentSend.SendSequence) &&
(Connection->NewNetbios || (BytesReceived == Connection->CurrentSend.MessageOffset)) &&
(Connection->CurrentSend.Request != NULL)) {
//
// This acks whatever we sent last time, and we are
// not done packetizing this send, so we can repacketize.
// With SendSequence changing as it does now,
// don't need the CurrentSend.Request check???
//
NB_DEBUG2 (SEND, ("Got full ack on %lx (%d)\n", Connection, Connection->CurrentSend.SendSequence));
NbiStopRetransmit (Connection);
if (Connection->NewNetbios) {
//
// If we are waiting for a window, and this does not open it
// anymore, then we don't reset our timers/retries.
//
if (Connection->SubState == CONNECTION_SUBSTATE_A_REMOTE_W) {
if (Connection->RemoteRcvSequenceMax != BytesReceived) {
Connection->RemoteRcvSequenceMax = BytesReceived; // really RcvSeqMac
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
}
//
// Advance this, we won't get meaningful results until we
// send a full window in one message.
//
Connection->SendWindowSequenceLimit = Connection->CurrentSend.SendSequence + Connection->SendWindowSize;
} else {
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
Connection->RemoteRcvSequenceMax = BytesReceived; // really RcvSeqMac
if (((SHORT)(Connection->CurrentSend.SendSequence - Connection->SendWindowSequenceLimit)) >= 0) {
NbiAdjustSendWindow (Connection);
} else {
//
// Advance this, we won't get meaningful results until we
// send a full window in one message.
//
Connection->SendWindowSequenceLimit = Connection->CurrentSend.SendSequence + Connection->SendWindowSize;
}
}
} else {
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
}
Connection->RetransmitThisWindow = FALSE;
Connection->UnAckedSend = Connection->CurrentSend;
if (Connection->SubState != CONNECTION_SUBSTATE_A_PACKETIZE) {
//
// We may be on if this ack is duplicated.
//
NbiReferenceConnectionSync (Connection, CREF_PACKETIZE);
CTEAssert(!Connection->OnPacketizeQueue);
Connection->OnPacketizeQueue = TRUE;
NB_INSERT_TAIL_LIST(
&Device->PacketizeConnections,
&Connection->PacketizeLinkage,
&Device->Lock);
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
}
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
} else if( Connection->FirstMessageRequest ) {
//
// This acked part of the current send. If the
// remote is requesting a resend then we advance
// the current send location by the amount
// acked and resend from there. If he does
// not want a resend, just ignore this.
//
// We repacketize immediately because we have
// backed up the pointer, and this would
// cause us to ignore an ack for the amount
// sent. Since we don't release the lock
// until we have packetized, the current
// pointer will be advanced past there.
//
// If he is acking more than we sent, we
// ignore this -- the remote is confused and there
// is nothing much we can do.
//
if (Resend) {
if (Connection->NewNetbios &&
(((Connection->UnAckedSend.SendSequence < Connection->CurrentSend.SendSequence) &&
(ReceiveSequence >= Connection->UnAckedSend.SendSequence) &&
(ReceiveSequence < Connection->CurrentSend.SendSequence)) ||
((Connection->UnAckedSend.SendSequence > Connection->CurrentSend.SendSequence) &&
((ReceiveSequence >= Connection->UnAckedSend.SendSequence) ||
(ReceiveSequence < Connection->CurrentSend.SendSequence))))) {
BOOLEAN SomethingAcked = (BOOLEAN)
(ReceiveSequence != Connection->UnAckedSend.SendSequence);
//
// New netbios and the receive sequence is valid.
//
NbiStopRetransmit (Connection);
//
// Advance our unacked pointer by the amount
// acked in this response.
//
NbiAdvanceUnAckedBySequence(
Connection,
ReceiveSequence);
Connection->RetransmitThisWindow = TRUE;
++Connection->ConnectionInfo.TransmissionErrors;
++Device->Statistics.DataFramesResent;
ADD_TO_LARGE_INTEGER(
&Device->Statistics.DataFrameBytesResent,
Connection->CurrentSend.MessageOffset - Connection->UnAckedSend.MessageOffset);
//
// Packetize from that point on.
//
Connection->CurrentSend = Connection->UnAckedSend;
//
// If anything was acked, then reset the retry count.
//
if (SomethingAcked) {
//
// See if we need to adjust our send window.
//
if (((SHORT)(Connection->UnAckedSend.SendSequence - Connection->SendWindowSequenceLimit)) >= 0) {
NbiAdjustSendWindow (Connection);
}
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
}
Connection->RemoteRcvSequenceMax = BytesReceived; // really RcvSeqMac
//
// Now packetize. This will set the state to
// something meaningful and release the lock.
//
if (Connection->SubState != CONNECTION_SUBSTATE_A_PACKETIZE) {
NbiPacketizeSend(
Connection
NB_LOCK_HANDLE_ARG(LockHandle)
);
} else {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} else if (!Connection->NewNetbios &&
((ReceiveSequence == Connection->UnAckedSend.SendSequence) &&
(BytesReceived <= Connection->CurrentSend.MessageOffset))) {
ULONG BytesAcked =
BytesReceived - Connection->UnAckedSend.MessageOffset;
//
// Old netbios.
//
NbiStopRetransmit (Connection);
//
// Advance our unacked pointer by the amount
// acked in this response.
//
NbiAdvanceUnAckedByBytes(
Connection,
BytesAcked);
++Connection->ConnectionInfo.TransmissionErrors;
++Device->Statistics.DataFramesResent;
ADD_TO_LARGE_INTEGER(
&Device->Statistics.DataFrameBytesResent,
Connection->CurrentSend.MessageOffset - Connection->UnAckedSend.MessageOffset);
//
// Packetize from that point on.
//
Connection->CurrentSend = Connection->UnAckedSend;
//
// If anything was acked, reset the retry count
//
if ( BytesAcked ) {
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
}
//
// Now packetize. This will set the state to
// something meaningful and release the lock.
//
if (Connection->SubState != CONNECTION_SUBSTATE_A_PACKETIZE) {
NbiPacketizeSend(
Connection
NB_LOCK_HANDLE_ARG(LockHandle)
);
} else {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} else {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} else {
if (Connection->NewNetbios &&
(((Connection->UnAckedSend.SendSequence < Connection->CurrentSend.SendSequence) &&
(ReceiveSequence >= Connection->UnAckedSend.SendSequence) &&
(ReceiveSequence < Connection->CurrentSend.SendSequence)) ||
((Connection->UnAckedSend.SendSequence > Connection->CurrentSend.SendSequence) &&
((ReceiveSequence >= Connection->UnAckedSend.SendSequence) ||
(ReceiveSequence < Connection->CurrentSend.SendSequence))))) {
BOOLEAN SomethingAcked = (BOOLEAN)
(ReceiveSequence != Connection->UnAckedSend.SendSequence);
//
// New netbios and the receive sequence is valid. We advance
// the back of our send window, but we don't repacketize.
//
//
// Advance our unacked pointer by the amount
// acked in this response.
//
NbiAdvanceUnAckedBySequence(
Connection,
ReceiveSequence);
Connection->RemoteRcvSequenceMax = BytesReceived; // really RcvSeqMac
//
// If anything was acked, then reset the retry count.
//
if (SomethingAcked) {
//
// See if we need to adjust our send window.
//
if (((SHORT)(Connection->UnAckedSend.SendSequence - Connection->SendWindowSequenceLimit)) >= 0) {
NbiAdjustSendWindow (Connection);
}
Connection->Retries = NbiDevice->KeepAliveCount;
Connection->CurrentRetransmitTimeout = Connection->BaseRetransmitTimeout;
//
// Now packetize. This will set the state to
// something meaningful and release the lock.
//
if ((Connection->CurrentSend.Request != NULL) &&
(Connection->SubState != CONNECTION_SUBSTATE_A_PACKETIZE)) {
NbiReferenceConnectionSync (Connection, CREF_PACKETIZE);
CTEAssert(!Connection->OnPacketizeQueue);
Connection->OnPacketizeQueue = TRUE;
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
NB_INSERT_TAIL_LIST(
&Device->PacketizeConnections,
&Connection->PacketizeLinkage,
&Device->Lock);
}
}
}
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} else {
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle);
}
} /* NbiReframeConnection */
VOID
NbiRestartConnection(
IN PCONNECTION Connection
)
/*++
Routine Description:
This routine is called when have gotten an ack for
a full message, or received a response to a watchdog
probe, and need to check if the connection should
start packetizing.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT HELD.
Arguments:
Connection - The connection.
Return Value:
None.
--*/
{
PREQUEST Request, TmpRequest;
ULONG TempCount;
PTDI_REQUEST_KERNEL_SEND Parameters;
PDEVICE Device = NbiDevice;
//
// See if there is a send to activate.
//
if (Connection->SendQueue.Head != NULL) {
//
// Take the first send off the queue and make
// it current.
//
Request = Connection->SendQueue.Head;
Connection->SendQueue.Head = REQUEST_SINGLE_LINKAGE (Request);
//
// Cache the information about being EOM
// in a more easily accessible location?
//
Parameters = (PTDI_REQUEST_KERNEL_SEND)REQUEST_PARAMETERS(Request);
if ((Parameters->SendFlags & TDI_SEND_PARTIAL) == 0) {
//
// This is a one-request message.
//
Connection->CurrentSend.Request = Request;
Connection->CurrentSend.MessageOffset = 0;
Connection->CurrentSend.Buffer = REQUEST_NDIS_BUFFER (Request);
Connection->CurrentSend.BufferOffset = 0;
Connection->SendBufferInUse = FALSE;
Connection->UnAckedSend = Connection->CurrentSend;
Connection->FirstMessageRequest = Request;
#ifdef RSRC_TIMEOUT_DBG
KeQuerySystemTime(&Connection->FirstMessageRequestTime);
#endif //RSRC_TIMEOUT_DBG
Connection->LastMessageRequest = Request;
Connection->CurrentMessageLength = Parameters->SendLength;
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
NbiReferenceConnectionSync (Connection, CREF_PACKETIZE);
CTEAssert (!Connection->OnPacketizeQueue);
Connection->OnPacketizeQueue = TRUE;
NB_INSERT_TAIL_LIST(
&Device->PacketizeConnections,
&Connection->PacketizeLinkage,
&Device->Lock);
} else {
//
// This is a multiple-request message. We scan
// to see if we have the end of message received
// yet.
//
TempCount = Parameters->SendLength;
TmpRequest = Request;
Request = REQUEST_SINGLE_LINKAGE(Request);
while (Request != NULL) {
TempCount += Parameters->SendLength;
Parameters = (PTDI_REQUEST_KERNEL_SEND)REQUEST_PARAMETERS(Request);
if ((Parameters->SendFlags & TDI_SEND_PARTIAL) == 0) {
Connection->CurrentSend.Request = TmpRequest;
Connection->CurrentSend.MessageOffset = 0;
Connection->CurrentSend.Buffer = REQUEST_NDIS_BUFFER (TmpRequest);
Connection->CurrentSend.BufferOffset = 0;
Connection->SendBufferInUse = FALSE;
Connection->UnAckedSend = Connection->CurrentSend;
Connection->FirstMessageRequest = TmpRequest;
Connection->LastMessageRequest = Request;
#ifdef RSRC_TIMEOUT_DBG
KeQuerySystemTime(&Connection->FirstMessageRequestTime);
#endif //RSRC_TIMEOUT_DBG
Connection->CurrentMessageLength = TempCount;
Connection->SubState = CONNECTION_SUBSTATE_A_PACKETIZE;
NbiReferenceConnectionSync (Connection, CREF_PACKETIZE);
CTEAssert (!Connection->OnPacketizeQueue);
Connection->OnPacketizeQueue = TRUE;
NB_INSERT_TAIL_LIST(
&Device->PacketizeConnections,
&Connection->PacketizeLinkage,
&Device->Lock);
break;
}
Request = REQUEST_SINGLE_LINKAGE(Request);
}
if (Request == NULL) {
Connection->SubState = CONNECTION_SUBSTATE_A_W_EOR;
}
}
} else {
Connection->FirstMessageRequest = NULL;
Connection->SubState = CONNECTION_SUBSTATE_A_IDLE;
NbiStartWatchdog (Connection);
}
} /* NbiRestartConnection */
VOID
NbiAdvanceUnAckedByBytes(
IN PCONNECTION Connection,
IN ULONG BytesAcked
)
/*++
Routine Description:
This routine advances the Connection->UnAckedSend
send pointer by the specified number of bytes. It
assumes that there are enough send requests to
handle the number specified.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT HELD.
Arguments:
Connection - The connection.
BytesAcked - The number of bytes acked.
Return Value:
NTSTATUS - status of operation.
--*/
{
ULONG CurSendBufferLength;
ULONG BytesLeft = BytesAcked;
ULONG TempBytes;
while (BytesLeft > 0) {
NdisQueryBufferSafe (Connection->UnAckedSend.Buffer, NULL, &CurSendBufferLength, HighPagePriority);
//
// See if bytes acked ends within the current buffer.
//
if (Connection->UnAckedSend.BufferOffset + BytesLeft <
CurSendBufferLength) {
Connection->UnAckedSend.BufferOffset += BytesLeft;
Connection->UnAckedSend.MessageOffset += BytesLeft;
break;
} else {
TempBytes = CurSendBufferLength - Connection->UnAckedSend.BufferOffset;
BytesLeft -= TempBytes;
Connection->UnAckedSend.MessageOffset += TempBytes;
//
// No, so advance the buffer.
//
Connection->UnAckedSend.BufferOffset = 0;
Connection->UnAckedSend.Buffer =
NDIS_BUFFER_LINKAGE (Connection->UnAckedSend.Buffer);
//
// Is there a next buffer in this request?
//
if (Connection->UnAckedSend.Buffer == NULL) {
//
// No, so advance the request unless we are done.
//
if (BytesLeft == 0) {
return;
}
Connection->UnAckedSend.Request =
REQUEST_SINGLE_LINKAGE(Connection->UnAckedSend.Request);
if (Connection->UnAckedSend.Request == NULL) {
KeBugCheck (NDIS_INTERNAL_ERROR);
}
Connection->UnAckedSend.Buffer =
REQUEST_NDIS_BUFFER (Connection->UnAckedSend.Request);
}
}
}
} /* NbiAdvanceUnAckedByBytes */
VOID
NbiAdvanceUnAckedBySequence(
IN PCONNECTION Connection,
IN USHORT ReceiveSequence
)
/*++
Routine Description:
This routine advances the Connection->UnAckedSend
send pointer so that the next packet to send will be
the correct one for ReceiveSequence. UnAckedSend
must point to a known valid combination. It
assumes that there are enough send requests to
handle the sequence specified.
NOTE: THIS FUNCTION IS CALLED WITH CONNECTION->LOCK HELD
AND RETURNS WITH IT HELD.
Arguments:
Connection - The connection.
Return Value:
NTSTATUS - status of operation.
--*/
{
USHORT PacketsAcked;
//
// Fix this to account for partial sends, where
// we might not have used the max. for all packets.
//
PacketsAcked = ReceiveSequence - Connection->UnAckedSend.SendSequence;
NbiAdvanceUnAckedByBytes(
Connection,
PacketsAcked * Connection->MaximumPacketSize);
Connection->UnAckedSend.SendSequence += PacketsAcked;
} /* NbiAdvanceUnAckedBySequence */
VOID
NbiCancelSend(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
This routine is called by the I/O system to cancel a send
The request is found on the connection's send queue.
NOTE: This routine is called with the CancelSpinLock held and
is responsible for releasing it.
Arguments:
DeviceObject - Pointer to the device object for this driver.
Irp - Pointer to the request packet representing the I/O request.
Return Value:
none.
--*/
{
PCONNECTION Connection;
PREQUEST Request = (PREQUEST)Irp;
NB_DEFINE_LOCK_HANDLE (LockHandle)
NB_DEFINE_SYNC_CONTEXT (SyncContext)
CTEAssert ((REQUEST_MAJOR_FUNCTION(Request) == IRP_MJ_INTERNAL_DEVICE_CONTROL) &&
(REQUEST_MINOR_FUNCTION(Request) == TDI_SEND));
CTEAssert (REQUEST_OPEN_TYPE(Request) == (PVOID)TDI_CONNECTION_FILE);
Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(Request);
//
// Just stop the connection, that will tear down any
// sends.
//
// Do we care about cancelling non-active
// sends without stopping the connection??
//
NbiReferenceConnectionSync (Connection, CREF_CANCEL);
IoReleaseCancelSpinLock (Irp->CancelIrql);
NB_BEGIN_SYNC (&SyncContext);
NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle);
//
// This frees the lock, cancels any sends, etc.
//
NbiStopConnection(
Connection,
STATUS_CANCELLED
NB_LOCK_HANDLE_ARG (LockHandle));
NbiDereferenceConnection (Connection, CREF_CANCEL);
NB_END_SYNC (&SyncContext);
} /* NbiCancelSend */
NTSTATUS
NbiBuildBufferChainFromBufferChain (
IN NDIS_HANDLE BufferPoolHandle,
IN PNDIS_BUFFER CurrentSourceBuffer,
IN ULONG CurrentByteOffset,
IN ULONG DesiredLength,
OUT PNDIS_BUFFER *DestinationBuffer,
OUT PNDIS_BUFFER *NewSourceBuffer,
OUT ULONG *NewByteOffset,
OUT ULONG *ActualLength
)
/*++
Routine Description:
This routine is called to build an NDIS_BUFFER chain from a source
NDIS_BUFFER chain and offset into it. We assume we don't know the
length of the source Mdl chain, and we must allocate the NDIS_BUFFERs
for the destination chain, which we do from the NDIS buffer pool.
If the system runs out of memory while we are building the destination
NDIS_BUFFER chain, we completely clean up the built chain and return with
NewCurrentMdl and NewByteOffset set to the current values of CurrentMdl
and ByteOffset.
Environment:
Arguments:
BufferPoolHandle - The buffer pool to allocate buffers from.
CurrentSourceBuffer - Points to the start of the NDIS_BUFFER chain
from which to draw the packet.
CurrentByteOffset - Offset within this NDIS_BUFFER to start the packet at.
DesiredLength - The number of bytes to insert into the packet.
DestinationBuffer - returned pointer to the NDIS_BUFFER chain describing
the packet.
NewSourceBuffer - returned pointer to the NDIS_BUFFER that would
be used for the next byte of packet. NULL if the source NDIS_BUFFER
chain was exhausted.
NewByteOffset - returned offset into the NewSourceBuffer for the next byte
of packet. NULL if the source NDIS_BUFFER chain was exhausted.
ActualLength - The actual length of the data copied.
Return Value:
STATUS_SUCCESS if the build of the returned NDIS_BUFFER chain succeeded
and was the correct length.
STATUS_INSUFFICIENT_RESOURCES if we ran out of NDIS_BUFFERs while
building the destination chain.
--*/
{
ULONG AvailableBytes;
ULONG CurrentByteCount;
ULONG BytesCopied;
PNDIS_BUFFER OldNdisBuffer;
PNDIS_BUFFER NewNdisBuffer;
NDIS_STATUS NdisStatus;
OldNdisBuffer = CurrentSourceBuffer;
NdisQueryBufferSafe (OldNdisBuffer, NULL, &CurrentByteCount, HighPagePriority);
AvailableBytes = CurrentByteCount - CurrentByteOffset;
if (AvailableBytes > DesiredLength) {
AvailableBytes = DesiredLength;
}
//
// Build the first NDIS_BUFFER, which could conceivably be the only one...
//
NdisCopyBuffer(
&NdisStatus,
&NewNdisBuffer,
BufferPoolHandle,
OldNdisBuffer,
CurrentByteOffset,
AvailableBytes);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
*NewSourceBuffer = CurrentSourceBuffer;
*NewByteOffset = CurrentByteOffset;
return STATUS_INSUFFICIENT_RESOURCES;
}
*DestinationBuffer = NewNdisBuffer;
BytesCopied = AvailableBytes;
//
// Was the first NDIS_BUFFER enough data.
//
if (BytesCopied == DesiredLength) {
if (CurrentByteOffset + AvailableBytes == CurrentByteCount) {
*NewSourceBuffer = CurrentSourceBuffer->Next;
*NewByteOffset = 0;
} else {
*NewSourceBuffer = CurrentSourceBuffer;
*NewByteOffset = CurrentByteOffset + AvailableBytes;
}
*ActualLength = BytesCopied;
return STATUS_SUCCESS;
}
if (CurrentSourceBuffer->Next == NULL) {
*NewSourceBuffer = NULL;
*NewByteOffset = 0;
*ActualLength = BytesCopied;
return STATUS_SUCCESS;
}
//
// Need more data, so follow the in Mdl chain to create a packet.
//
OldNdisBuffer = OldNdisBuffer->Next;
NdisQueryBufferSafe (OldNdisBuffer, NULL, &CurrentByteCount, HighPagePriority);
while (OldNdisBuffer != NULL) {
AvailableBytes = DesiredLength - BytesCopied;
if (AvailableBytes > CurrentByteCount) {
AvailableBytes = CurrentByteCount;
}
NdisCopyBuffer(
&NdisStatus,
&(NDIS_BUFFER_LINKAGE(NewNdisBuffer)),
BufferPoolHandle,
OldNdisBuffer,
0,
AvailableBytes);
if (NdisStatus != NDIS_STATUS_SUCCESS) {
//
// ran out of resources. put back what we've used in this call and
// return the error.
//
while (*DestinationBuffer != NULL) {
NewNdisBuffer = NDIS_BUFFER_LINKAGE(*DestinationBuffer);
NdisFreeBuffer (*DestinationBuffer);
*DestinationBuffer = NewNdisBuffer;
}
*NewByteOffset = CurrentByteOffset;
*NewSourceBuffer = CurrentSourceBuffer;
return STATUS_INSUFFICIENT_RESOURCES;
}
NewNdisBuffer = NDIS_BUFFER_LINKAGE(NewNdisBuffer);
BytesCopied += AvailableBytes;
if (BytesCopied == DesiredLength) {
if (AvailableBytes == CurrentByteCount) {
*NewSourceBuffer = OldNdisBuffer->Next;
*NewByteOffset = 0;
} else {
*NewSourceBuffer = OldNdisBuffer;
*NewByteOffset = AvailableBytes;
}
*ActualLength = BytesCopied;
return STATUS_SUCCESS;
}
OldNdisBuffer = OldNdisBuffer->Next;
NdisQueryBufferSafe (OldNdisBuffer, NULL, &CurrentByteCount, HighPagePriority);
}
//
// We ran out of source buffer chain.
//
*NewSourceBuffer = NULL;
*NewByteOffset = 0;
*ActualLength = BytesCopied;
return STATUS_SUCCESS;
} /* NbiBuildBufferChainFromBufferChain */