2020-09-30 16:53:55 +02:00

3100 lines
81 KiB
C
Raw Blame History

This file contains invisible Unicode characters

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

/*++
Copyright (c) 1993 Microsoft Corporation
Module Name:
Write.c
Abstract:
This module implements support for NtWriteFile for the
NetWare redirector called by the dispatch driver.
Author:
Colin Watson [ColinW] 07-Apr-1993
Revision History:
--*/
#include "Procs.h"
#include <stdlib.h>
//
// The local debug trace level
//
#define Dbg (DEBUG_TRACE_WRITE)
//
// The header overhead in the first packet of a burst write.
//
#define BURST_WRITE_HEADER_SIZE \
( sizeof( NCP_BURST_WRITE_REQUEST ) - sizeof( NCP_BURST_HEADER ) )
//
// Local procedure prototypes
//
NTSTATUS
NwCommonWrite (
IN PIRP_CONTEXT IrpContext
);
NTSTATUS
WriteNcp(
PIRP_CONTEXT IrpContext,
LARGE_INTEGER ByteOffset,
ULONG BufferLength,
PVOID WriteBuffer,
PMDL WriteMdl
);
NTSTATUS
QueryEofForWriteCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
);
NTSTATUS
WriteNcpCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
);
NTSTATUS
BurstWrite(
PIRP_CONTEXT IrpContext,
LARGE_INTEGER ByteOffset,
ULONG BufferLength,
PVOID WriteBuffer,
PMDL WriteMdl
);
NTSTATUS
SendWriteBurst(
PIRP_CONTEXT IrpContext,
ULONG Offset,
USHORT Length,
BOOLEAN EndOfBurst,
BOOLEAN Retransmission
);
VOID
BuildBurstWriteFirstReq(
PIRP_CONTEXT IrpContext,
PVOID Buffer,
ULONG DataSize,
PMDL BurstMdl,
UCHAR Flags,
ULONG Handle,
ULONG FileOffset
);
VOID
BuildBurstWriteNextReq(
PIRP_CONTEXT IrpContext,
PVOID Buffer,
ULONG DataSize,
UCHAR BurstFlags,
ULONG BurstOffset,
PMDL BurstHeaderMdl,
PMDL BurstDataMdl
);
NTSTATUS
BurstWriteCompletionSend(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN PVOID Context
);
NTSTATUS
BurstWriteCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
);
VOID
BurstWriteTimeout(
PIRP_CONTEXT IrpContext
);
NTSTATUS
BurstWriteReconnect(
PIRP_CONTEXT IrpContext
);
NTSTATUS
NwCommonFlushBuffers (
IN PIRP_CONTEXT IrpContext
);
NTSTATUS
FlushBuffersCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
);
NTSTATUS
SendSecondaryPacket(
PIRP_CONTEXT IrpContext,
PIRP Irp
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, NwFsdWrite )
#pragma alloc_text( PAGE, NwCommonWrite )
#pragma alloc_text( PAGE, DoWrite )
#pragma alloc_text( PAGE, WriteNcp )
#pragma alloc_text( PAGE, BurstWrite )
#pragma alloc_text( PAGE, SendWriteBurst )
#pragma alloc_text( PAGE, ResubmitBurstWrite )
#pragma alloc_text( PAGE, NwFsdFlushBuffers )
#pragma alloc_text( PAGE, NwCommonFlushBuffers )
#pragma alloc_text( PAGE, BuildBurstWriteFirstReq )
#pragma alloc_text( PAGE, BuildBurstWriteNextReq )
#ifndef QFE_BUILD
#pragma alloc_text( PAGE1, WriteNcpCallback )
#pragma alloc_text( PAGE1, BurstWriteCompletionSend )
#pragma alloc_text( PAGE1, BurstWriteCallback )
#pragma alloc_text( PAGE1, BurstWriteTimeout )
#pragma alloc_text( PAGE1, FlushBuffersCallback )
#pragma alloc_text( PAGE1, SendSecondaryPacket )
#pragma alloc_text( PAGE1, BurstWriteReconnect )
#endif
#endif
#if 0 // Not pageable
// see ifndef QFE_BUILD above
#endif
NTSTATUS
NwFsdWrite(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
This routine is the FSD routine that handles NtWriteFile.
Arguments:
NwfsDeviceObject - Supplies the device object for the write function.
Irp - Supplies the IRP to process.
Return Value:
NTSTATUS - The result status.
--*/
{
PIRP_CONTEXT pIrpContext = NULL;
NTSTATUS status;
BOOLEAN TopLevel;
PAGED_CODE();
DebugTrace(+1, Dbg, "NwFsdWrite\n", 0);
//
// Call the common write routine.
//
FsRtlEnterFileSystem();
TopLevel = NwIsIrpTopLevel( Irp );
try {
pIrpContext = AllocateIrpContext( Irp );
status = NwCommonWrite( pIrpContext );
} except(NwExceptionFilter( Irp, GetExceptionInformation() )) {
if ( pIrpContext == NULL ) {
//
// If we couldn't allocate an irp context, just complete
// irp without any fanfare.
//
status = STATUS_INSUFFICIENT_RESOURCES;
Irp->IoStatus.Status = status;
Irp->IoStatus.Information = 0;
IoCompleteRequest ( Irp, IO_NETWORK_INCREMENT );
} else {
//
// We had some trouble trying to perform the requested
// operation, so we'll abort the I/O request with
// the error Status that we get back from the
// execption code
//
status = NwProcessException( pIrpContext, GetExceptionCode() );
}
}
if ( pIrpContext ) {
if ( status != STATUS_PENDING ) {
NwDequeueIrpContext( pIrpContext, FALSE );
}
NwCompleteRequest( pIrpContext, status );
}
if ( TopLevel ) {
NwSetTopLevelIrp( NULL );
}
FsRtlExitFileSystem();
//
// Return to the caller.
//
DebugTrace(-1, Dbg, "NwFsdWrite -> %08lx\n", status );
Stats.WriteOperations++;
return status;
}
NTSTATUS
NwCommonWrite (
IN PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine does the common code for NtWriteFile.
Arguments:
IrpContext - Supplies the request being processed.
Return Value:
NTSTATUS - The return status for the operation
--*/
{
NTSTATUS status;
PIRP Irp;
PIO_STACK_LOCATION irpSp;
NODE_TYPE_CODE nodeTypeCode;
PICB icb;
PFCB fcb;
PNONPAGED_FCB pNpFcb;
PVOID fsContext;
BOOLEAN WroteToCache;
LARGE_INTEGER ByteOffset;
LARGE_INTEGER PreviousByteOffset;
ULONG BufferLength;
PULONG pFileSize;
// ULONG FileLength;
PAGED_CODE();
//
// Get the current stack location
//
Irp = IrpContext->pOriginalIrp;
irpSp = IoGetCurrentIrpStackLocation( Irp );
DebugTrace(+1, Dbg, "CommonWrite...\n", 0);
DebugTrace( 0, Dbg, "Irp = %08lx\n", (ULONG_PTR)Irp);
//
// Decode the file object to figure out who we are. If the result
// is not the root DCB then its an illegal parameter.
//
nodeTypeCode = NwDecodeFileObject( irpSp->FileObject,
&fsContext,
(PVOID *)&icb );
fcb = (PFCB)icb->SuperType.Fcb;
if (((nodeTypeCode != NW_NTC_ICB) &&
(nodeTypeCode != NW_NTC_ICB_SCB)) ||
(!icb->HasRemoteHandle) ) {
DebugTrace(0, Dbg, "Not a file\n", 0);
status = STATUS_INVALID_PARAMETER;
DebugTrace(-1, Dbg, "CommonWrite -> %08lx\n", status );
return status;
}
//
// Make sure that this ICB is still active.
//
NwVerifyIcbSpecial( icb );
if ( fcb->NodeTypeCode == NW_NTC_FCB ) {
IrpContext->pScb = fcb->Scb;
IrpContext->pNpScb = IrpContext->pScb->pNpScb;
IrpContext->Icb = icb;
pFileSize = &icb->NpFcb->Header.FileSize.LowPart;
} else if ( fcb->NodeTypeCode == NW_NTC_SCB ) {
IrpContext->pScb = icb->SuperType.Scb;
IrpContext->pNpScb = IrpContext->pScb->pNpScb;
IrpContext->Icb = icb;
fcb = NULL;
pFileSize = &icb->FileSize;
} else {
DebugTrace(0, Dbg, "Not a file or a server\n", 0);
status = STATUS_INVALID_PARAMETER;
DebugTrace(-1, Dbg, "CommonWrite -> %08lx\n", status );
return status;
}
ByteOffset = irpSp->Parameters.Write.ByteOffset;
BufferLength = irpSp->Parameters.Write.Length;
//
// Can't handle large byte offset, but write to EOF is okay.
//
if ( ByteOffset.HighPart != 0 ) {
if ( ByteOffset.HighPart != 0xFFFFFFFF ||
ByteOffset.LowPart != 0xFFFFFFFF ) {
return( STATUS_INVALID_PARAMETER );
}
}
if (FlagOn(irpSp->FileObject->Flags, FO_SYNCHRONOUS_IO) &&
!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
PreviousByteOffset.QuadPart = irpSp->FileObject->CurrentByteOffset.QuadPart;
irpSp->FileObject->CurrentByteOffset.QuadPart = ByteOffset.QuadPart;
}
//
// Paging I/O is not allowed to extend the file
//
if ((FlagOn(Irp->Flags, IRP_PAGING_IO)) &&
(ByteOffset.LowPart + BufferLength > *pFileSize )) {
NwAppendToQueueAndWait( IrpContext );
if ( ByteOffset.LowPart + BufferLength <= *pFileSize ) {
//
// Someone else extended the file. Do nothing.
//
// continue;
} else if ( ByteOffset.LowPart > *pFileSize ) {
//
// Whole write is off the end of the buffer
//
NwDequeueIrpContext( IrpContext, FALSE );
Irp->IoStatus.Information = 0;
return( STATUS_SUCCESS );
} else {
//
// Truncate request to size of file
//
BufferLength = *pFileSize - ByteOffset.LowPart;
}
NwDequeueIrpContext( IrpContext, FALSE );
}
//
// Special case 0 length write.
//
if ( BufferLength == 0 ) {
Irp->IoStatus.Information = 0;
return( STATUS_SUCCESS );
}
//
// Remember the original MDL, so that we can restore it when we are done.
//
IrpContext->pOriginalMdlAddress = Irp->MdlAddress;
//
// Attempt to write this data to our private cache
//
if ( fcb != NULL && Irp->UserBuffer != NULL ) {
WroteToCache = CacheWrite(
IrpContext,
fcb->NonPagedFcb,
ByteOffset.LowPart,
BufferLength,
Irp->UserBuffer );
if ( WroteToCache ) {
Irp->IoStatus.Information = BufferLength;
//
// Update the current byte offset in the file if it is a
// synchronous file (and this is not paging I/O).
//
if (FlagOn(irpSp->FileObject->Flags, FO_SYNCHRONOUS_IO) &&
!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
irpSp->FileObject->CurrentByteOffset.QuadPart += BufferLength;
}
//
// Record write offset and size to discover a sequential write pattern.
//
fcb->LastReadOffset = irpSp->Parameters.Write.ByteOffset.LowPart;
fcb->LastReadSize = irpSp->Parameters.Write.Length;
//
// If the file was extended, record the new file size.
//
if ( fcb->LastReadOffset + fcb->LastReadSize >
fcb->NonPagedFcb->Header.FileSize.LowPart ) {
fcb->NonPagedFcb->Header.FileSize.LowPart =
fcb->LastReadOffset + fcb->LastReadSize;
}
DebugTrace(-1, Dbg, "NwCommonWrite -> %08lx\n", STATUS_SUCCESS );
return( STATUS_SUCCESS );
}
}
status = DoWrite(
IrpContext,
ByteOffset,
BufferLength,
Irp->UserBuffer,
IrpContext->pOriginalMdlAddress );
if ( NT_SUCCESS( status ) ) {
//
// We actually wrote something out to the wire. If there was a read
// cache and this write overlapped it, invalidate the read cache data
// so that we get good data on future reads.
//
if ( fcb != NULL ) {
pNpFcb = fcb->NonPagedFcb;
if ( ( pNpFcb->CacheBuffer != NULL ) &&
( pNpFcb->CacheSize != 0 ) &&
( pNpFcb->CacheType == ReadAhead ) ) {
//
// Two cases: (1) offset is less than cache offset
// (2) offset is inside cached region
//
if ( ByteOffset.LowPart < pNpFcb->CacheFileOffset ) {
//
// Did we run into the read cache?
//
if ( BufferLength >
(pNpFcb->CacheFileOffset - ByteOffset.LowPart) ) {
DebugTrace( 0, Dbg, "Invalidated read cache for %08lx.\n", pNpFcb );
pNpFcb->CacheDataSize = 0;
}
} else {
//
// Did we write over any of the cached region.
//
if ( ByteOffset.LowPart <= ( pNpFcb->CacheFileOffset + pNpFcb->CacheDataSize ) ) {
DebugTrace( 0, Dbg, "Invalidated read cache for %08lx.\n", pNpFcb );
pNpFcb->CacheDataSize = 0;
}
}
}
}
Irp->IoStatus.Information = IrpContext->Specific.Write.WriteOffset;
//
// Update the current byte offset in the file if it is a
// synchronous file (and this is not paging I/O).
//
if (FlagOn(irpSp->FileObject->Flags, FO_SYNCHRONOUS_IO) &&
!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
irpSp->FileObject->CurrentByteOffset.QuadPart += BufferLength;
}
NwAppendToQueueAndWait( IrpContext );
if (ByteOffset.LowPart + BufferLength > *pFileSize ) {
*pFileSize = ByteOffset.LowPart + BufferLength;
}
} else {
//
// The request failed, don't move the file pointer.
//
if (FlagOn(irpSp->FileObject->Flags, FO_SYNCHRONOUS_IO) &&
!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
irpSp->FileObject->CurrentByteOffset.QuadPart = PreviousByteOffset.QuadPart;
}
}
DebugTrace(-1, Dbg, "CommonWrite -> %08lx\n", status);
return status;
}
NTSTATUS
DoWrite(
PIRP_CONTEXT IrpContext,
LARGE_INTEGER ByteOffset,
ULONG BufferLength,
PVOID WriteBuffer,
PMDL WriteMdl OPTIONAL
)
/*++
Routine Description:
This routine does a write to the network via the most efficient
available protocol.
Arguments:
IrpContext - A pointer to IRP context information for this request.
ByteOffset - The file offset to write.
BufferLength - The number of bytes to write.
WriteBuffer - A pointer to the source buffer.
WriteMdl = An optional MDL for the write buffer.
Return Value:
Status of transfer.
--*/
{
NTSTATUS status;
PAGED_CODE();
if ( IrpContext->pNpScb->SendBurstModeEnabled &&
BufferLength > IrpContext->pNpScb->BufferSize ) {
status = BurstWrite( IrpContext, ByteOffset, BufferLength, WriteBuffer, WriteMdl );
} else {
status = WriteNcp( IrpContext, ByteOffset, BufferLength, WriteBuffer, WriteMdl );
}
//
// Reset IrpContext parameters
//
IrpContext->TxMdl->Next = NULL;
IrpContext->CompletionSendRoutine = NULL;
IrpContext->TimeoutRoutine = NULL;
IrpContext->Flags &= ~(IRP_FLAG_RETRY_SEND | IRP_FLAG_BURST_REQUEST | IRP_FLAG_BURST_PACKET |
IRP_FLAG_BURST_WRITE | IRP_FLAG_NOT_SYSTEM_PACKET );
IrpContext->pTdiStruct = NULL;
IrpContext->pOriginalIrp->MdlAddress = IrpContext->pOriginalMdlAddress;
IrpContext->pOriginalIrp->AssociatedIrp.SystemBuffer = IrpContext->pOriginalSystemBuffer;
return( status );
}
NTSTATUS
WriteNcp(
PIRP_CONTEXT IrpContext,
LARGE_INTEGER ByteOffset,
ULONG BufferLength,
PVOID WriteBuffer,
PMDL WriteMdl
)
/*++
Routine Description:
This routine exchanges a series of write NCPs with the server.
Arguments:
IrpContext - A pointer to IRP context information for this request.
Icb - Supplies the file specific information.
Return Value:
Status of transfer.
--*/
{
PICB Icb;
PIRP irp;
PIO_STACK_LOCATION irpSp;
ULONG Length; // Size we will send to the server
ULONG FileLength;
PSCB pScb;
NTSTATUS status = STATUS_UNSUCCESSFUL;
PMDL DataMdl;
BOOLEAN Done;
PAGED_CODE();
Icb = IrpContext->Icb;
irp = IrpContext->pOriginalIrp;
irpSp = IoGetCurrentIrpStackLocation( irp );
DebugTrace(+1, Dbg, "WriteNcp...\n", 0);
DebugTrace( 0, Dbg, "irp = %08lx\n", (ULONG_PTR)irp);
DebugTrace( 0, Dbg, "WriteLen= %ld\n", BufferLength);
DebugTrace( 0, Dbg, "HOffset = %lx\n", ByteOffset.HighPart);
DebugTrace( 0, Dbg, "LOffset = %lx\n", ByteOffset.LowPart);
if (Icb->SuperType.Fcb->NodeTypeCode == NW_NTC_FCB) {
pScb = Icb->SuperType.Fcb->Scb;
DebugTrace( 0, Dbg, "File = %wZ\n", &Icb->SuperType.Fcb->FullFileName);
} else {
//
// Write to a queue
//
pScb = Icb->SuperType.Scb;
}
ASSERT (pScb->NodeTypeCode == NW_NTC_SCB);
if ( ByteOffset.HighPart == 0xFFFFFFFF &&
ByteOffset.LowPart == FILE_WRITE_TO_END_OF_FILE ) {
//
// Ensure that you are at the head of the queue
//
NwAppendToQueueAndWait( IrpContext );
//
// Write relative to end of file. Find the end of file.
//
status = ExchangeWithWait(
IrpContext,
SynchronousResponseCallback,
"F-r",
NCP_GET_FILE_SIZE,
&Icb->Handle, sizeof( Icb->Handle ) );
if ( NT_SUCCESS( status ) ) {
status = ParseResponse(
IrpContext,
IrpContext->rsp,
IrpContext->ResponseLength,
"Nd",
&FileLength );
if ( !NT_SUCCESS( status ) ) {
return status;
}
}
IrpContext->Specific.Write.FileOffset = FileLength;
}
Length = MIN( (ULONG)IrpContext->pNpScb->BufferSize, BufferLength );
DebugTrace( 0, Dbg, "Length = %ld\n", Length);
//
// The server will not accept writes that cross 4k boundaries in the file
//
if ((IrpContext->pNpScb->PageAlign) &&
(DIFFERENT_PAGES( ByteOffset.LowPart, Length ))) {
Length = 4096 -
((ULONG)ByteOffset.LowPart & (4096-1));
}
IrpContext->Specific.Write.Buffer = WriteBuffer;
IrpContext->Specific.Write.WriteOffset = 0;
IrpContext->Specific.Write.RemainingLength = BufferLength;
IrpContext->Specific.Write.LastWriteLength = Length;
IrpContext->Specific.Write.FileOffset = ByteOffset.LowPart;
IrpContext->Specific.Write.PartialMdl = NULL;
Done = FALSE;
while ( !Done ) {
//
// Setup to do at most 64K of i/o asynchronously, or buffer length.
//
IrpContext->Specific.Write.BurstLength =
MIN( 64 * 1024, IrpContext->Specific.Write.RemainingLength );
IrpContext->Specific.Write.BurstOffset = 0;
//
// Try to allocate an MDL for this i/o.
//
DataMdl = ALLOCATE_MDL(
(PCHAR)IrpContext->Specific.Write.Buffer +
IrpContext->Specific.Write.WriteOffset,
IrpContext->Specific.Write.BurstLength,
FALSE, // Secondary Buffer
FALSE, // Charge Quota
NULL);
if ( DataMdl == NULL ) {
if ( IrpContext->Specific.Write.PartialMdl != NULL ) {
FREE_MDL( IrpContext->Specific.Write.PartialMdl );
}
DebugTrace(-1, Dbg, "WriteNcp -> %X\n", STATUS_INSUFFICIENT_RESOURCES );
return STATUS_INSUFFICIENT_RESOURCES;
}
IrpContext->Specific.Write.FullMdl = DataMdl;
//
// If there is no MDL for this write probe the data MDL to
// lock it's pages down. Otherwise, use the data MDL as
// a partial MDL.
//
if ( WriteMdl == NULL ) {
//
// The Probe may cause us to page in some data. If the data is from
// the same server we are writing to then we had better not be at
// the front of the queue otherwise it will wait indefinitely behind us.
// Its a good idea to Dequeue ourselves after each burst anyway because
// its a quick operation and it alow smaller requests to overtake a very
// large series of bursts.
//
NwDequeueIrpContext( IrpContext, FALSE );
try {
MmProbeAndLockPages( DataMdl, irp->RequestorMode, IoReadAccess);
} except (EXCEPTION_EXECUTE_HANDLER) {
FREE_MDL( DataMdl );
DebugTrace(-1, Dbg, "WriteNcp -> %X\n", GetExceptionCode() );
return GetExceptionCode();
}
} else {
IoBuildPartialMdl(
WriteMdl,
DataMdl,
(PCHAR)IrpContext->Specific.Write.Buffer,
IrpContext->Specific.Write.BurstLength );
}
//
// Allocate a partial Mdl for the worst possible case of alignment
//
IrpContext->Specific.Write.PartialMdl =
ALLOCATE_MDL( 0 , IrpContext->pNpScb->BufferSize + PAGE_SIZE-1, FALSE, FALSE, NULL);
if ( IrpContext->Specific.Write.PartialMdl == NULL ) {
if ( WriteMdl == NULL ) {
MmUnlockPages( DataMdl );
}
FREE_MDL( DataMdl );
DebugTrace(-1, Dbg, "WriteNcp -> %X\n", STATUS_INSUFFICIENT_RESOURCES );
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Build a partial MDL for this write NCP.
//
IoBuildPartialMdl(
DataMdl,
IrpContext->Specific.Write.PartialMdl,
MmGetMdlVirtualAddress( DataMdl ),
Length );
if ( IrpContext->Specific.Write.BurstLength ==
IrpContext->Specific.Write.RemainingLength ) {
Done = TRUE;
}
//
// Ensure that you are at the head of the queue
//
NwAppendToQueueAndWait( IrpContext );
//
// Send the request.
//
status = ExchangeWithWait(
IrpContext,
WriteNcpCallback,
"F-rdwf",
NCP_WRITE_FILE,
&Icb->Handle, sizeof( Icb->Handle ),
IrpContext->Specific.Write.FileOffset,
Length,
IrpContext->Specific.Write.PartialMdl );
Stats.WriteNcps+=2;
FREE_MDL( IrpContext->Specific.Write.PartialMdl );
//
// Unlock locked pages, and free our MDL.
//
if ( WriteMdl == NULL ) {
MmUnlockPages( DataMdl );
}
FREE_MDL( DataMdl );
//
// If we had a failure, we need to terminate this loop.
// The only status that is set is the Specific->Write
// status. We can not trust what comes back from the
// ExchangeWithWait by design.
//
if ( !NT_SUCCESS( IrpContext->Specific.Write.Status ) ) {
Done = TRUE;
}
//
// Reset the packet length since we may have less than
// a packet to send.
//
Length = MIN( (ULONG)IrpContext->pNpScb->BufferSize,
IrpContext->Specific.Write.RemainingLength );
IrpContext->Specific.Write.LastWriteLength = Length;
}
status = IrpContext->Specific.Write.Status;
DebugTrace(-1, Dbg, "WriteNcp -> %08lx\n", status );
return status;
}
NTSTATUS
WriteNcpCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
)
/*++
Routine Description:
This routine receives the response from a user NCP.
Arguments:
Return Value:
VOID
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
ULONG Length;
ULONG LastLength;
DebugTrace(0, Dbg, "WriteNcpCallback...\n", 0);
if ( BytesAvailable == 0) {
//
// No response from server. Status is in pIrpContext->
// ResponseParameters.Error
//
IrpContext->Specific.Write.Status = STATUS_REMOTE_NOT_LISTENING;
NwSetIrpContextEvent( IrpContext );
return STATUS_REMOTE_NOT_LISTENING;
}
LastLength = IrpContext->Specific.Write.LastWriteLength;
Status = ParseResponse( IrpContext, Response, BytesAvailable, "N" );
if ( NT_SUCCESS(Status) ) {
// If the last write worked then move the pointers appropriately
IrpContext->Specific.Write.RemainingLength -= LastLength;
IrpContext->Specific.Write.BurstLength -= LastLength;
IrpContext->Specific.Write.WriteOffset += LastLength;
IrpContext->Specific.Write.FileOffset += LastLength;
IrpContext->Specific.Write.BurstOffset += LastLength;
// If this is a print job, remember that we actually wrote data
if ( IrpContext->Icb->IsPrintJob ) {
IrpContext->Icb->ActuallyPrinted = TRUE;
}
} else {
//
// Abandon this request
//
IrpContext->Specific.Write.Status = Status;
NwSetIrpContextEvent( IrpContext );
DebugTrace( 0, Dbg, "WriteNcpCallback -> %08lx\n", Status );
return Status;
}
if ( IrpContext->Specific.Write.BurstLength != 0 ) {
// Write the next packet.
DebugTrace( 0, Dbg, "RemainingLength = %ld\n", IrpContext->Specific.Write.RemainingLength);
DebugTrace( 0, Dbg, "FileOffset = %ld\n", IrpContext->Specific.Write.FileOffset);
DebugTrace( 0, Dbg, "WriteOffset = %ld\n", IrpContext->Specific.Write.WriteOffset);
DebugTrace( 0, Dbg, "BurstOffset = %ld\n", IrpContext->Specific.Write.BurstOffset);
Length = MIN( (ULONG)IrpContext->pNpScb->BufferSize,
IrpContext->Specific.Write.BurstLength );
//
// The server will not accept writes that cross 4k boundaries
// in the file.
//
if ((IrpContext->pNpScb->PageAlign) &&
(DIFFERENT_PAGES( IrpContext->Specific.Write.FileOffset, Length ))) {
Length = 4096 -
((ULONG)IrpContext->Specific.Write.FileOffset & (4096-1));
}
IrpContext->Specific.Write.LastWriteLength = Length;
DebugTrace( 0, Dbg, "Length = %ld\n", Length);
MmPrepareMdlForReuse( IrpContext->Specific.Write.PartialMdl );
IoBuildPartialMdl(
IrpContext->Specific.Write.FullMdl,
IrpContext->Specific.Write.PartialMdl,
(PUCHAR)MmGetMdlVirtualAddress( IrpContext->Specific.Write.FullMdl ) +
IrpContext->Specific.Write.BurstOffset,
Length );
//
// Send the request.
//
BuildRequestPacket(
IrpContext,
WriteNcpCallback,
"F-rdwf",
NCP_WRITE_FILE,
&IrpContext->Icb->Handle, sizeof( IrpContext->Icb->Handle ),
IrpContext->Specific.Write.FileOffset,
Length,
IrpContext->Specific.Write.PartialMdl );
Status = PrepareAndSendPacket( IrpContext );
Stats.WriteNcps+=2;
DebugTrace(-1, Dbg, "WriteNcbCallBack -> %08lx\n", Status );
if ( !NT_SUCCESS(Status) ) {
//
// Abandon this request
//
IrpContext->Specific.Write.Status = Status;
NwSetIrpContextEvent( IrpContext );
DebugTrace( 0, Dbg, "WriteNcpCallback -> %08lx\n", Status );
return Status;
}
} else {
//
// We're done with this request, signal the writing thread.
//
IrpContext->Specific.Write.Status = STATUS_SUCCESS;
NwSetIrpContextEvent( IrpContext );
}
DebugTrace( 0, Dbg, "WriteNcpCallback -> %08lx\n", Status );
return STATUS_SUCCESS;
}
NTSTATUS
BurstWrite(
PIRP_CONTEXT IrpContext,
LARGE_INTEGER ByteOffset,
ULONG BufferLength,
PVOID WriteBuffer,
PMDL WriteMdl
)
/*++
Routine Description:
This routine exchanges a series of burst write NCPs with the server.
Arguments:
IrpContext - A pointer to IRP context information for this request.
Return Value:
Status of the transfer.
--*/
{
PICB Icb;
PIRP irp;
PIO_STACK_LOCATION irpSp;
ULONG Length; // Size we will send to the server
PSCB pScb;
PNONPAGED_SCB pNpScb;
NTSTATUS status = STATUS_UNSUCCESSFUL;
PMDL DataMdl;
BOOLEAN Done;
BOOLEAN MissingData;
ULONG TimeInNwUnits;
ULONG LastLength;
ULONG Result;
UCHAR BurstFlags;
USHORT MissingFragmentCount;
USHORT i;
ULONG FragmentOffset;
USHORT FragmentLength;
Icb = IrpContext->Icb;
pNpScb = IrpContext->pNpScb;
irp = IrpContext->pOriginalIrp;
irpSp = IoGetCurrentIrpStackLocation( irp );
IrpContext->Specific.Write.WriteOffset = 0;
IrpContext->Specific.Write.RemainingLength = BufferLength;
IrpContext->Specific.Write.TotalWriteLength = BufferLength;
IrpContext->Specific.Write.TotalWriteOffset = ByteOffset.LowPart;
DebugTrace(+1, Dbg, "BurstWrite...\n", 0);
DebugTrace( 0, Dbg, "irp = %08lx\n", (ULONG_PTR)irp);
DebugTrace( 0, Dbg, "WriteLen= %ld\n", BufferLength);
DebugTrace( 0, Dbg, "HOffset = %lx\n", ByteOffset.HighPart);
DebugTrace( 0, Dbg, "LOffset = %lx\n", ByteOffset.LowPart);
//
// Renegotiate burst mode, if necessary
//
if ( pNpScb->BurstRenegotiateReqd ) {
pNpScb->BurstRenegotiateReqd = FALSE;
RenegotiateBurstMode( IrpContext, pNpScb );
}
SetFlag( IrpContext->Flags, IRP_FLAG_BURST_WRITE );
if (Icb->SuperType.Fcb->NodeTypeCode == NW_NTC_FCB) {
pScb = Icb->SuperType.Fcb->Scb;
DebugTrace( 0, Dbg, "File = %wZ\n", &Icb->SuperType.Fcb->FullFileName);
} else {
//
// Write to a queue
//
pScb = Icb->SuperType.Scb;
}
ASSERT (pScb->NodeTypeCode == NW_NTC_SCB);
//
// Calculate the length of the burst to send.
//
Length = MIN( (ULONG)pNpScb->MaxSendSize, BufferLength );
DebugTrace( 0, Dbg, "Length = %ld\n", Length);
if ( ByteOffset.HighPart == 0xFFFFFFFF &&
ByteOffset.LowPart == FILE_WRITE_TO_END_OF_FILE ) {
ULONG FileLength;
//
// Ensure that you are at the head of the queue
//
NwAppendToQueueAndWait( IrpContext );
//
// Write relative to end of file. Find the end of file.
//
status = ExchangeWithWait(
IrpContext,
SynchronousResponseCallback,
"F-r",
NCP_GET_FILE_SIZE,
&Icb->Handle, sizeof(Icb->Handle) );
if ( NT_SUCCESS( status ) ) {
status = ParseResponse(
IrpContext,
IrpContext->rsp,
IrpContext->ResponseLength,
"Nd",
&FileLength );
}
if ( !NT_SUCCESS( status ) ) {
return( status );
}
IrpContext->Specific.Write.FileOffset = FileLength;
} else {
IrpContext->Specific.Write.FileOffset = ByteOffset.LowPart;
}
//
// Setup context parameters for burst write.
//
IrpContext->Specific.Write.LastWriteLength = Length;
IrpContext->Destination = pNpScb->RemoteAddress;
IrpContext->Specific.Write.Buffer = WriteBuffer;
//
// Set the timeout to be the time for all te burst packets to be sent plus a round
// trip delay plus a second.
//
TimeInNwUnits = pNpScb->NwSingleBurstPacketTime * ((Length / IrpContext->pNpScb->MaxPacketSize) + 1) +
IrpContext->pNpScb->NwLoopTime;
IrpContext->pNpScb->SendTimeout =
(SHORT)(((TimeInNwUnits / 555) *
(ULONG)WriteTimeoutMultiplier) / 100 + 1) ;
if (IrpContext->pNpScb->SendTimeout < 2)
{
IrpContext->pNpScb->SendTimeout = 2 ;
}
if (IrpContext->pNpScb->SendTimeout > (SHORT)MaxWriteTimeout)
{
IrpContext->pNpScb->SendTimeout = (SHORT)MaxWriteTimeout ;
}
IrpContext->pNpScb->TimeOut = IrpContext->pNpScb->SendTimeout;
//
// tommye - MS bug 2743 changed the RetryCount from 20 to be based off the
// default retry count, nudged up a little.
//
pNpScb->RetryCount = DefaultRetryCount * 2;
DebugTrace( 0, DEBUG_TRACE_LIP, "pNpScb->SendTimeout = %08lx\n", IrpContext->pNpScb->SendTimeout );
Done = FALSE;
do {
DataMdl = ALLOCATE_MDL(
(PCHAR)IrpContext->Specific.Write.Buffer +
IrpContext->Specific.Write.WriteOffset,
Length,
FALSE, // Secondary Buffer
FALSE, // Charge Quota
NULL);
if ( DataMdl == NULL ) {
return ( STATUS_INSUFFICIENT_RESOURCES );
}
//
// If there is no MDL for this write, probe the data MDL to lock it's
// pages down.
//
// Otherwise, use the data MDL as a partial MDL and lock the pages
// accordingly.
//
if ( WriteMdl == NULL ) {
//
// The Probe may cause us to page in some data. If the data is from
// the same server we are writing to then we had better not be at
// the front of the queue otherwise it will wait indefinitely behind us.
// Its a good idea to Dequeue ourselves after each burst anyway because
// its a quick operation and it alow smaller requests to overtake a very
// large series of bursts.
//
NwDequeueIrpContext( IrpContext, FALSE );
try {
MmProbeAndLockPages( DataMdl, irp->RequestorMode, IoReadAccess);
} except (EXCEPTION_EXECUTE_HANDLER) {
FREE_MDL( DataMdl );
return GetExceptionCode();
}
} else {
IoBuildPartialMdl(
WriteMdl,
DataMdl,
(PCHAR)IrpContext->Specific.Write.Buffer +
IrpContext->Specific.Write.WriteOffset,
Length );
}
pNpScb->BurstDataWritten += Length;
if (( SendExtraNcp ) &&
( pNpScb->BurstDataWritten >= 0x0000ffff )) {
ULONG Flags;
//
// VLM client sends an NCP when starting a burst mode request
// if the last request was not a write. It also does this every
// 0xfe00 bytes written
//
// When going to a queue we will use handle 2. This is what the vlm
// client always seems to do.
//
Flags = IrpContext->Flags;
//
// Reset IrpContext parameters
//
IrpContext->TxMdl->Next = NULL;
IrpContext->CompletionSendRoutine = NULL;
IrpContext->TimeoutRoutine = NULL;
IrpContext->Flags &= ~(IRP_FLAG_RETRY_SEND | IRP_FLAG_BURST_REQUEST | IRP_FLAG_BURST_PACKET |
IRP_FLAG_BURST_WRITE | IRP_FLAG_NOT_SYSTEM_PACKET );
IrpContext->pTdiStruct = NULL;
//
// Ensure that you are at the head of the queue
//
NwAppendToQueueAndWait( IrpContext );
ExchangeWithWait (
IrpContext,
SynchronousResponseCallback,
"Sb", // NCP Get Directory Path
NCP_DIR_FUNCTION, NCP_GET_DIRECTORY_PATH,
(Icb->SuperType.Fcb->NodeTypeCode == NW_NTC_FCB)?
Icb->SuperType.Fcb->Vcb->Specific.Disk.Handle : 2 );
pNpScb->BurstDataWritten = Length;
IrpContext->Flags = Flags;
SetFlag( IrpContext->Flags, IRP_FLAG_ON_SCB_QUEUE );
}
IrpContext->TimeoutRoutine = BurstWriteTimeout;
IrpContext->CompletionSendRoutine = BurstWriteCompletionSend;
IrpContext->pTdiStruct = &IrpContext->pNpScb->Burst;
IrpContext->PacketType = NCP_BURST;
IrpContext->pEx = BurstWriteCallback;
IrpContext->Specific.Write.FullMdl = DataMdl;
MmGetSystemAddressForMdlSafe( DataMdl, NormalPagePriority );
//
// Allocate a partial Mdl for the worst possible case of alignment
//
IrpContext->Specific.Write.PartialMdl =
ALLOCATE_MDL( 0, IrpContext->pNpScb->MaxPacketSize + PAGE_SIZE - 1, FALSE, FALSE, NULL);
if ( IrpContext->Specific.Write.PartialMdl == NULL ) {
if ( WriteMdl == NULL ) {
MmUnlockPages( DataMdl );
}
FREE_MDL( DataMdl );
return STATUS_INSUFFICIENT_RESOURCES;
}
//
// Get to the front of the SCB queue, if we are not already there.
// Note that can't append this IrpContext to the SCB until after
// the probe and lock, since the probe and lock may cause a paging
// read on this SCB.
//
NwAppendToQueueAndWait( IrpContext );
status = SendWriteBurst(
IrpContext,
BURST_WRITE_HEADER_SIZE,
(USHORT)Length,
TRUE,
FALSE );
MissingData = TRUE;
while ( MissingData ) {
KeWaitForSingleObject( &IrpContext->Event, Executive, KernelMode, FALSE, NULL );
MmPrepareMdlForReuse( IrpContext->Specific.Write.PartialMdl );
if ( BooleanFlagOn( IrpContext->Flags, IRP_FLAG_RETRY_SEND ) ) {
//
// This burst has timed out, simply resend the burst.
//
NwProcessSendBurstFailure( pNpScb, 1 );
status = SendWriteBurst(
IrpContext,
BURST_WRITE_HEADER_SIZE,
(USHORT)Length,
TRUE,
TRUE );
continue;
}
if ( !NT_SUCCESS( IrpContext->Specific.Write.Status ) ) {
status = IrpContext->Specific.Write.Status;
Done = TRUE;
goto EndOfLoop;
} else {
status = ParseResponse(
IrpContext,
IrpContext->rsp,
IrpContext->ResponseLength,
"B_d",
&BurstFlags,
8,
&Result );
}
if ( BurstFlags & BURST_FLAG_SYSTEM_PACKET ) {
//
// The server dropped at least one packet.
//
MissingData = TRUE;
DebugTrace( 0, Dbg, "Received system packet\n", 0 );
//
// This is a missing fragment request.
//
status = ParseResponse(
IrpContext,
IrpContext->rsp,
IrpContext->ResponseLength,
"G_w",
34,
&MissingFragmentCount );
ASSERT( NT_SUCCESS( status ) );
ASSERT( MissingFragmentCount != 0 );
NwProcessSendBurstFailure( pNpScb, MissingFragmentCount );
DebugTrace( 0, Dbg, "Received request for %d missing fragment\n", MissingFragmentCount );
ClearFlag( IrpContext->Flags, IRP_FLAG_RETRY_SEND );
//
// Walk the missing fragment list and send the missing fragments.
//
for ( i = 0; i < MissingFragmentCount && NT_SUCCESS( status ); i++ ) {
status = ParseResponse(
IrpContext,
IrpContext->rsp,
IrpContext->ResponseLength,
"G_dw",
34 + 2 + 6 * i,
&FragmentOffset,
&FragmentLength
);
ASSERT( NT_SUCCESS( status ) );
if ( FragmentOffset < Length + BURST_WRITE_HEADER_SIZE &&
FragmentOffset + FragmentLength <=
Length + BURST_WRITE_HEADER_SIZE ) {
//
// Send a burst with the missing data. Do no set the
// end of burst bit until we have sent the last
// missing fragment packet.
//
status = SendWriteBurst(
IrpContext,
FragmentOffset,
FragmentLength,
(BOOLEAN)( i == (MissingFragmentCount - 1)),
FALSE );
} else {
//
// Received a bogus missing fragment request.
// Ignore the remainder of the request.
//
status = STATUS_INVALID_NETWORK_RESPONSE;
Done = TRUE;
goto EndOfLoop;
}
}
Stats.PacketBurstWriteTimeouts++;
} else {
NwProcessSendBurstSuccess( pNpScb );
MissingData = FALSE;
//
// This is not a system packets, check the response.
//
if ( Result == 0 ) {
//
// If the last write worked then move the pointers appropriately
//
LastLength = IrpContext->Specific.Write.LastWriteLength;
IrpContext->Specific.Write.RemainingLength -= LastLength;
IrpContext->Specific.Write.WriteOffset += LastLength;
IrpContext->Specific.Write.FileOffset += LastLength;
//
// If this is a print job, remember that we actually wrote data
//
if ( IrpContext->Icb->IsPrintJob ) {
IrpContext->Icb->ActuallyPrinted = TRUE;
}
} else {
//
// Abandon this request
//
Done = TRUE;
}
//
// Do we need to send another burst to satisfy the write IRP?
//
if ( IrpContext->Specific.Write.RemainingLength != 0 ) {
//
// Write the next packet.
//
DebugTrace( 0, Dbg, "RemainingLength = %ld\n", IrpContext->Specific.Write.RemainingLength);
DebugTrace( 0, Dbg, "FileOffset = %ld\n", IrpContext->Specific.Write.FileOffset);
DebugTrace( 0, Dbg, "WriteOffset = %ld\n", IrpContext->Specific.Write.WriteOffset);
Length = MIN( (ULONG)IrpContext->pNpScb->MaxSendSize,
IrpContext->Specific.Write.RemainingLength );
IrpContext->Specific.Write.LastWriteLength = Length;
} else {
Done = TRUE;
}
} // else ( not a system packet )
} // while ( missing data )
//
// Update the burst request number now.
//
if ( status != STATUS_REMOTE_NOT_LISTENING ) {
IrpContext->pNpScb->BurstRequestNo++;
}
//
// If we need to reconnect, do it now.
//
if ( BooleanFlagOn( IrpContext->Flags, IRP_FLAG_RECONNECT_ATTEMPT ) ) {
BurstWriteReconnect( IrpContext );
}
//
// Dequeue this Irp context in preparation for the next run
// through the loop.
//
EndOfLoop:
ASSERT( status != STATUS_PENDING );
FREE_MDL( IrpContext->Specific.Write.PartialMdl );
//
// Unlock locked pages, and free our MDL.
//
if ( WriteMdl == NULL ) {
MmUnlockPages( DataMdl );
}
FREE_MDL( DataMdl );
} while ( !Done );
DebugTrace(-1, Dbg, "BurstWrite -> %08lx\n", status );
return status;
}
#ifdef NWDBG
int DropWritePackets;
#endif
NTSTATUS
BurstWriteCompletionSend(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN PVOID Context
)
/*++
Routine Description:
This routine handles completion of a burst write send. If the sending
thread is waiting for send completion notification, it signals the
IrpContext Event.
Note that this routine can be called from SendWriteBurst (i.e. not
at DPC level), if an allocation fails.
Arguments:
DeviceObject - unused.
Irp - Supplies Irp that the transport has finished processing.
Context - Supplies the IrpContext associated with the Irp.
Return Value:
The STATUS_MORE_PROCESSING_REQUIRED so that the IO system stops
processing Irp stack locations at this point.
--*/
{
PIRP_CONTEXT pIrpContext = (PIRP_CONTEXT) Context;
INTERLOCKED_RESULT Result;
KIRQL OldIrql;
NTSTATUS Status;
//
// Avoid completing the Irp because the Mdl etc. do not contain
// their original values.
//
DebugTrace( +1, Dbg, "BurstWriteCompletionSend\n", 0);
DebugTrace( +0, Dbg, "Irp %X\n", Irp);
DebugTrace( +0, Dbg, "pIrpC %X\n", pIrpContext);
if ( Irp != NULL ) {
DebugTrace( 0, Dbg, "Burst Write Send = %08lx\n", Irp->IoStatus.Status );
Status = Irp->IoStatus.Status;
} else {
Status = STATUS_SUCCESS;
}
//
// If this was a secondary IRP, free it now.
//
if ( pIrpContext->NodeTypeCode == NW_NTC_MINI_IRP_CONTEXT ) {
PMINI_IRP_CONTEXT MiniIrpContext;
MiniIrpContext = (PMINI_IRP_CONTEXT)pIrpContext;
ASSERT( MiniIrpContext->Mdl2->Next == NULL );
pIrpContext = MiniIrpContext->IrpContext;
FreeMiniIrpContext( MiniIrpContext );
}
//
// Nothing to do unless the last send has completed.
//
Result = InterlockedDecrement(
&pIrpContext->Specific.Write.PacketCount );
if ( Result ) {
DebugTrace( 0, Dbg, "Packets to go = %d\n", pIrpContext->Specific.Write.PacketCount );
if (Status == STATUS_BAD_NETWORK_PATH) {
//
// IPX has ripped for the destination but failed to find the net. Minimise the
// difference between this case and sending a normal burst by completing the
// transmission as soon as possible.
//
pIrpContext->pNpScb->NwSendDelay = 0;
}
return STATUS_MORE_PROCESSING_REQUIRED;
}
KeAcquireSpinLock( &pIrpContext->pNpScb->NpScbSpinLock, &OldIrql );
ASSERT( pIrpContext->pNpScb->Sending == TRUE );
pIrpContext->pNpScb->Sending = FALSE;
//
// Signal to the writing thread that the send has completed, if it
// is waiting.
//
if ( BooleanFlagOn( pIrpContext->Flags, IRP_FLAG_SIGNAL_EVENT ) ) {
ClearFlag( pIrpContext->Flags, IRP_FLAG_SIGNAL_EVENT );
NwSetIrpContextEvent( pIrpContext );
}
//
// If we processed a receive while waiting for send
// completion call the receive handler routine now.
//
if ( pIrpContext->pNpScb->Received ) {
pIrpContext->pNpScb->Receiving = FALSE;
pIrpContext->pNpScb->Received = FALSE;
KeReleaseSpinLock( &pIrpContext->pNpScb->NpScbSpinLock, OldIrql );
pIrpContext->pEx(
pIrpContext,
pIrpContext->ResponseLength,
pIrpContext->rsp );
} else {
if ((Status == STATUS_BAD_NETWORK_PATH) &&
(pIrpContext->pNpScb->Receiving == FALSE)) {
//
// Usually means a ras connection has gone down during the burst.
// Go through the timeout logic now because the ras timeouts take
// a long time and unless we re rip things won't get better.
//
pIrpContext->Specific.Write.Status = STATUS_REMOTE_NOT_LISTENING;
ClearFlag( pIrpContext->Flags, IRP_FLAG_RETRY_SEND );
NwSetIrpContextEvent( pIrpContext );
}
KeReleaseSpinLock( &pIrpContext->pNpScb->NpScbSpinLock, OldIrql );
}
DebugTrace( -1, Dbg, "BurstWriteCompletionSend -> STATUS_MORE_PROCESSING_REQUIRED\n", 0);
return STATUS_MORE_PROCESSING_REQUIRED;
UNREFERENCED_PARAMETER( DeviceObject );
}
NTSTATUS
BurstWriteCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
)
/*++
Routine Description:
This routine receives the response a burst write.
Arguments:
IrpContext - A pointer to the context information for this IRP.
BytesAvailable - Actual number of bytes in the received message.
Response - Points to the receive buffer.
Return Value:
VOID
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
DebugTrace(0, Dbg, "BurstWriteCallback...\n", 0);
if ( BytesAvailable == 0) {
//
// No response from server. Status is in pIrpContext->Write.Status
// Clear the retry send bit so we don't keep retrying.
//
IrpContext->Specific.Write.Status = STATUS_REMOTE_NOT_LISTENING;
ClearFlag( IrpContext->Flags, IRP_FLAG_RETRY_SEND );
NwSetIrpContextEvent( IrpContext );
DebugTrace(-1, Dbg, "BurstWriteCallback -> %X\n", STATUS_REMOTE_NOT_LISTENING );
return STATUS_REMOTE_NOT_LISTENING;
}
IrpContext->Specific.Write.Status = STATUS_SUCCESS;
ASSERT( BytesAvailable < MAX_RECV_DATA );
++Stats.PacketBurstWriteNcps;
//
// Clear the retry send bit, since we have a response.
//
ClearFlag( IrpContext->Flags, IRP_FLAG_RETRY_SEND );
//
// Copy the burst write response, and signal the users thread
// to continue.
//
TdiCopyLookaheadData(
IrpContext->rsp,
Response,
BytesAvailable < MAX_RECV_DATA ? BytesAvailable : MAX_RECV_DATA,
0
);
IrpContext->ResponseLength = BytesAvailable;
NwSetIrpContextEvent( IrpContext );
return STATUS_SUCCESS;
}
NTSTATUS
SendWriteBurst(
PIRP_CONTEXT IrpContext,
ULONG BurstOffset,
USHORT Length,
BOOLEAN EndOfBurst,
BOOLEAN Retransmission
)
/*++
Routine Description:
This routine does the actual work of sending a series of burst write
NCPs to the server.
Arguments:
IrpContext - A pointer to IRP context information for this request.
BurstOffset - The offset in the burst to start sending. If BurstOffset
equals BURST_WRITE_HEADER_SIZE, start from the beginning of the burst.
Length - The length of the burst.
EndOfBurst - If TRUE set the end of burst bit when sending the last
frame. Otherwise there is more (discontiguous) data to come in
the current burst.
Retransmission - If TRUE, this is a burst write timeout retransmission.
Send the first packet only.
Return Value:
Status of transfer.
--*/
{
UCHAR BurstFlags;
NTSTATUS Status;
BOOLEAN MoreData;
PIRP SendIrp;
PMINI_IRP_CONTEXT MiniIrpContext;
PAGED_CODE();
DebugTrace( +1, Dbg, "SendWriteBurst...\n", 0);
DebugTrace( 0, Dbg, "Data offset = %d\n", BurstOffset );
DebugTrace( 0, Dbg, "Data length = %d\n", Length );
DebugTrace( 0, Dbg, "End of burst = %d\n", EndOfBurst );
//
// Send the request.
//
SetFlag( IrpContext->Flags, IRP_FLAG_BURST_REQUEST | IRP_FLAG_BURST_PACKET );
//
// Set the burst flags
//
IrpContext->Specific.Write.BurstLength =
MIN( IrpContext->pNpScb->MaxPacketSize, Length );
//
// Set the end-of-burst bit (and enable receiving the response), if this
// is the last packet we expect to send.
//
if ( ( !EndOfBurst || IrpContext->Specific.Write.BurstLength < Length )
&& !Retransmission ) {
IrpContext->pNpScb->OkToReceive = FALSE;
SetFlag( IrpContext->Flags, IRP_FLAG_NOT_OK_TO_RECEIVE );
BurstFlags = 0;
} else {
DebugTrace( 0, Dbg, "Last packet in the burst\n", 0);
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_OK_TO_RECEIVE );
BurstFlags = BURST_FLAG_END_OF_BURST;
}
if ( !EndOfBurst ) {
SetFlag( IrpContext->Flags, IRP_FLAG_SIGNAL_EVENT );
}
//
// Build the partial MDL for the first packet in the burst.
//
IoBuildPartialMdl(
IrpContext->Specific.Write.FullMdl,
IrpContext->Specific.Write.PartialMdl,
(PUCHAR)MmGetMdlVirtualAddress( IrpContext->Specific.Write.FullMdl ) +
BurstOffset - BURST_WRITE_HEADER_SIZE,
IrpContext->Specific.Write.BurstLength );
//
// Set the burst flags
//
if ( BurstOffset == BURST_WRITE_HEADER_SIZE ) {
SetFlag( IrpContext->Flags, IRP_FLAG_BURST_REQUEST | IRP_FLAG_BURST_PACKET );
}
if ( ( IrpContext->Specific.Write.BurstLength < Length ) &&
!Retransmission ) {
MoreData = TRUE;
} else {
MoreData = FALSE;
}
if ( BurstOffset == BURST_WRITE_HEADER_SIZE ) {
BuildBurstWriteFirstReq(
IrpContext,
IrpContext->req,
Length,
IrpContext->Specific.Write.PartialMdl,
BurstFlags,
*(ULONG UNALIGNED *)(&IrpContext->Icb->Handle[2]),
IrpContext->Specific.Write.FileOffset );
} else {
BuildBurstWriteNextReq(
IrpContext,
IrpContext->req,
IrpContext->Specific.Write.LastWriteLength + BURST_WRITE_HEADER_SIZE,
BurstFlags,
BurstOffset,
IrpContext->TxMdl,
IrpContext->Specific.Write.PartialMdl
);
}
if ( !Retransmission ) {
IrpContext->Specific.Write.PacketCount =
( Length + IrpContext->pNpScb->MaxPacketSize - 1 ) /
IrpContext->pNpScb->MaxPacketSize;
} else {
IrpContext->Specific.Write.PacketCount = 1;
}
DebugTrace( 0, Dbg, "Packet count = %d\n", IrpContext->Specific.Write.PacketCount );
DebugTrace( 0, DEBUG_TRACE_LIP, "Send delay = %d\n", IrpContext->pNpScb->NwSendDelay );
//
// Use the original IRP context to format the first packet.
//
++Stats.PacketBurstWriteNcps;
PreparePacket( IrpContext, IrpContext->pOriginalIrp, IrpContext->TxMdl );
Status = SendPacket( IrpContext, IrpContext->pNpScb );
while ( MoreData ) {
if ( IrpContext->pNpScb->NwSendDelay > 0 ) {
//
// Introduce a send delay between packets.
//
KeDelayExecutionThread(
KernelMode,
FALSE,
&IrpContext->pNpScb->NtSendDelay );
}
MiniIrpContext = AllocateMiniIrpContext( IrpContext );
DebugTrace( 0, Dbg, "Allocated mini IrpContext = %X\n", MiniIrpContext );
//
// Calculate the total number of bytes to send during this burst. Do this before
// checking to see if MiniIrpContext is NULL so that we skip the packet rather
// than sitting in a tight loop.
//
BurstOffset += IrpContext->Specific.Write.BurstLength;
//
// Do we need to send another burst write packet?
//
Length -= (USHORT)IrpContext->Specific.Write.BurstLength;
ASSERT ( Length > 0 );
IrpContext->Specific.Write.BurstLength =
MIN( IrpContext->pNpScb->MaxPacketSize, (ULONG)Length );
DebugTrace( +0, Dbg, "More data, sending %d bytes\n", IrpContext->Specific.Write.BurstLength );
//
// If we can't allocate a mini irp context to send the packet,
// just skip it and wait for the server to ask a retranmit. At
// this point performance isn't exactly stellar, so don't worry
// about having to wait for a timeout.
//
if ( MiniIrpContext == NULL ) {
InterlockedDecrement(
&IrpContext->Specific.Write.PacketCount );
if ( Length == IrpContext->Specific.Write.BurstLength ) {
MoreData = FALSE;
break;
}
continue;
}
#ifdef NWDBG
//
// If DropWritePackets is enabled, simulate missing packets, by
// occasionally dropping 500 bytes of data.
//
if ( DropWritePackets != 0 ) {
if ( ( rand() % DropWritePackets ) == 0 &&
Length != IrpContext->Specific.Write.BurstLength ) {
FreeMiniIrpContext( MiniIrpContext );
InterlockedDecrement(
&IrpContext->Specific.Write.PacketCount );
continue;
}
}
#endif
//
// Build the MDL for the data to send.
//
IoBuildPartialMdl(
IrpContext->Specific.Write.FullMdl,
MiniIrpContext->Mdl2,
(PUCHAR)MmGetMdlVirtualAddress( IrpContext->Specific.Write.FullMdl ) +
BurstOffset - BURST_WRITE_HEADER_SIZE,
IrpContext->Specific.Write.BurstLength );
//
// Set the burst flags
//
if ( !EndOfBurst || IrpContext->Specific.Write.BurstLength < Length ) {
IrpContext->pNpScb->OkToReceive = FALSE;
SetFlag( IrpContext->Flags, IRP_FLAG_NOT_OK_TO_RECEIVE );
BurstFlags = 0;
} else {
DebugTrace( 0, Dbg, "Last packet in the burst\n", 0);
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_OK_TO_RECEIVE );
BurstFlags = BURST_FLAG_END_OF_BURST;
}
if ( IrpContext->Specific.Write.BurstLength == Length ) {
MoreData = FALSE;
}
BuildBurstWriteNextReq(
IrpContext,
MiniIrpContext->Mdl1->MappedSystemVa,
IrpContext->Specific.Write.LastWriteLength +
BURST_WRITE_HEADER_SIZE,
BurstFlags,
BurstOffset,
MiniIrpContext->Mdl1,
MiniIrpContext->Mdl2
);
++Stats.PacketBurstWriteNcps;
SendIrp = MiniIrpContext->Irp;
PreparePacket( IrpContext, SendIrp, MiniIrpContext->Mdl1 );
IoSetCompletionRoutine( SendIrp, BurstWriteCompletionSend, MiniIrpContext, TRUE, TRUE, TRUE);
ASSERT( MiniIrpContext->Mdl2->Next == NULL );
Status = SendSecondaryPacket( IrpContext, SendIrp );
}
//
// If this is not the end-of-burst, wait for send completion here,
// since the caller is about to send more data.
//
if ( !EndOfBurst ) {
KeWaitForSingleObject( &IrpContext->Event, Executive, KernelMode, FALSE, NULL );
}
DebugTrace( -1, Dbg, "SendWriteBurst -> %X\n", Status );
return( Status );
}
VOID
BurstWriteTimeout(
PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine handles a burst write timeout.
Arguments:
IrpContext - A pointer to IRP context information for this request.
Return Value:
None
--*/
{
NTSTATUS Status = STATUS_UNSUCCESSFUL;
PIRP Irp;
DebugTrace(0, Dbg, "BurstWriteTimeout\n", 0 );
Irp = IrpContext->pOriginalIrp;
//
// Set the RetrySend flag, so that we know to retransmit the request.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RETRY_SEND );
//
// Signal the write thread to wakeup and resend the burst.
//
NwSetIrpContextEvent( IrpContext );
Stats.PacketBurstWriteTimeouts++;
return;
}
NTSTATUS
ResubmitBurstWrite(
PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine resubmits a burst write over a new burst connection.
Arguments:
IrpContext - A pointer to IRP context information for this request.
Return Value:
None
--*/
{
PNONPAGED_SCB pNpScb = IrpContext->pNpScb;
PAGED_CODE();
//
// Remember that we need to establish a new burst connection.
//
SetFlag( IrpContext->Flags, IRP_FLAG_RECONNECT_ATTEMPT );
//
// Set the packet size down the largest packet we can use, that
// is guaranteed to be routable.
//
pNpScb->MaxPacketSize = DEFAULT_PACKET_SIZE;
//
// Crank the delay times down so we give the new connection a chance.
//
pNpScb->NwGoodSendDelay = pNpScb->NwBadSendDelay = pNpScb->NwSendDelay = MinSendDelay;
pNpScb->NwGoodReceiveDelay = pNpScb->NwBadReceiveDelay = pNpScb->NwReceiveDelay = MinReceiveDelay;
pNpScb->SendBurstSuccessCount = 0;
pNpScb->ReceiveBurstSuccessCount = 0;
pNpScb->NtSendDelay.QuadPart = MinSendDelay;
//
// Signal the write thread to wakeup and resend the burst.
//
NwSetIrpContextEvent( IrpContext );
return( STATUS_PENDING );
}
NTSTATUS
BurstWriteReconnect(
PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine allocates a new IRP context and renegotiates burst mode.
Arguments:
IrpContext - A pointer to IRP context information for this request.
Return Value:
None
--*/
{
PIRP_CONTEXT pNewIrpContext;
PNONPAGED_SCB pNpScb = IrpContext->pNpScb;
BOOLEAN LIPNegotiated ;
PAGED_CODE();
//
// Attempt to allocate an extra IRP context.
//
if ( !NwAllocateExtraIrpContext( &pNewIrpContext, pNpScb ) ) {
return STATUS_INSUFFICIENT_RESOURCES;
}
pNewIrpContext->Specific.Create.UserUid = IrpContext->Specific.Create.UserUid;
SetFlag( pNewIrpContext->Flags, IRP_FLAG_RECONNECT_ATTEMPT );
pNewIrpContext->pNpScb = pNpScb;
//
// Insert this new IrpContext to the head of
// the SCB queue for processing. We can get away with this
// because we own the IRP context currently at the front of
// the queue.
//
ExInterlockedInsertHeadList(
&pNpScb->Requests,
&pNewIrpContext->NextRequest,
&pNpScb->NpScbSpinLock );
SetFlag( pNewIrpContext->Flags, IRP_FLAG_ON_SCB_QUEUE );
//
// Renegotiate the burst connection, this will automatically re-sync
// the burst connection.
//
NegotiateBurstMode( pNewIrpContext, pNpScb, &LIPNegotiated );
//
// Reset the sequence numbers.
//
pNpScb->BurstSequenceNo = 0;
pNpScb->BurstRequestNo = 0;
//
// Dequeue and free the bonus IRP context.
//
ExInterlockedRemoveHeadList(
&pNpScb->Requests,
&pNpScb->NpScbSpinLock );
ClearFlag( pNewIrpContext->Flags, IRP_FLAG_ON_SCB_QUEUE );
NwFreeExtraIrpContext( pNewIrpContext );
ClearFlag( IrpContext->Flags, IRP_FLAG_RECONNECT_ATTEMPT );
return( STATUS_SUCCESS );
}
NTSTATUS
NwFsdFlushBuffers(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
This routine is the FSD routine that handles NtFlushBuffersFile.
Arguments:
DeviceObject - Supplies the device object for the write function.
Irp - Supplies the IRP to process.
Return Value:
NTSTATUS - The result status.
--*/
{
PIRP_CONTEXT pIrpContext = NULL;
NTSTATUS status;
BOOLEAN TopLevel;
PAGED_CODE();
DebugTrace(+1, Dbg, "NwFsdFlushBuffers\n", 0);
//
// Call the common write routine.
//
FsRtlEnterFileSystem();
TopLevel = NwIsIrpTopLevel( Irp );
try {
pIrpContext = AllocateIrpContext( Irp );
status = NwCommonFlushBuffers( pIrpContext );
} except(NwExceptionFilter( Irp, GetExceptionInformation() )) {
if ( pIrpContext == NULL ) {
//
// If we couldn't allocate an irp context, just complete
// irp without any fanfare.
//
status = STATUS_INSUFFICIENT_RESOURCES;
Irp->IoStatus.Status = status;
Irp->IoStatus.Information = 0;
IoCompleteRequest ( Irp, IO_NETWORK_INCREMENT );
} else {
//
// We had some trouble trying to perform the requested
// operation, so we'll abort the I/O request with
// the error Status that we get back from the
// execption code
//
status = NwProcessException( pIrpContext, GetExceptionCode() );
}
}
if ( pIrpContext ) {
NwCompleteRequest( pIrpContext, status );
}
if ( TopLevel ) {
NwSetTopLevelIrp( NULL );
}
FsRtlExitFileSystem();
//
// Return to the caller.
//
DebugTrace(-1, Dbg, "NwFsdFlushBuffers -> %08lx\n", status );
return status;
}
NTSTATUS
NwCommonFlushBuffers (
IN PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine requests all dirty cache buffers to be flushed for a
given file.
Arguments:
IrpContext - Supplies the request being processed.
Return Value:
The status of the operation.
--*/
{
PIRP Irp;
PIO_STACK_LOCATION IrpSp;
NTSTATUS Status;
PFCB Fcb;
PICB Icb;
NODE_TYPE_CODE NodeTypeCode;
PVOID FsContext;
PAGED_CODE();
DebugTrace(0, Dbg, "NwCommonFlushBuffers...\n", 0);
//
// Get the current stack location
//
Irp = IrpContext->pOriginalIrp;
IrpSp = IoGetCurrentIrpStackLocation( Irp );
DebugTrace( 0, Dbg, "Irp = %08lx\n", (ULONG_PTR)Irp);
//
// Decode the file object to figure out who we are. If the result
// is not the a file then its an illegal parameter.
//
if (( NodeTypeCode = NwDecodeFileObject( IrpSp->FileObject,
&FsContext,
(PVOID *)&Icb )) != NW_NTC_ICB) {
DebugTrace(0, Dbg, "Not a file\n", 0);
Status = STATUS_INVALID_PARAMETER;
DebugTrace(-1, Dbg, "NwCommonFlushBuffers -> %08lx\n", Status );
return Status;
}
//
// Make sure that this ICB is still active.
//
NwVerifyIcbSpecial( Icb );
Fcb = (PFCB)Icb->SuperType.Fcb;
NodeTypeCode = Fcb->NodeTypeCode;
if ( NodeTypeCode != NW_NTC_FCB ) {
DebugTrace(0, Dbg, "Not a file\n", 0);
Status = STATUS_INVALID_PARAMETER;
DebugTrace(-1, Dbg, "CommonFlushBuffers -> %08lx\n", Status );
return Status;
}
//
// Set up the IRP context to do an exchange
//
IrpContext->pScb = Fcb->Scb;
IrpContext->pNpScb = IrpContext->pScb->pNpScb;
IrpContext->Icb = Icb;
//
// Send any user data to the server. Note we must not be on the
// queue when we do this.
//
MmFlushImageSection(&Icb->NpFcb->SegmentObject, MmFlushForWrite);
//
// Flush our dirty data.
//
Status = AcquireFcbAndFlushCache( IrpContext, Fcb->NonPagedFcb );
if ( !NT_SUCCESS( Status )) {
return( Status );
}
//
// Send a flush NCP
//
Status = Exchange (
IrpContext,
FlushBuffersCallback,
"F-r",
NCP_FLUSH_FILE,
&Icb->Handle, sizeof( Icb->Handle ) );
return( Status );
}
NTSTATUS
FlushBuffersCallback (
IN PIRP_CONTEXT IrpContext,
IN ULONG BytesAvailable,
IN PUCHAR Response
)
/*++
Routine Description:
This routine receives the flush file size response and completes the
flush IRP.
Arguments:
Return Value:
VOID
--*/
{
NTSTATUS Status;
DebugTrace(0, Dbg, "FlushBuffersCallback...\n", 0);
if ( BytesAvailable == 0) {
//
// We're done with this request. Dequeue the IRP context from
// SCB and complete the request.
//
NwDequeueIrpContext( IrpContext, FALSE );
NwCompleteRequest( IrpContext, STATUS_REMOTE_NOT_LISTENING );
//
// No response from server. Status is in pIrpContext->
// ResponseParameters.Error
//
DebugTrace( 0, Dbg, "Timeout\n", 0);
return STATUS_REMOTE_NOT_LISTENING;
}
//
// Get the data from the response.
//
Status = ParseResponse(
IrpContext,
Response,
BytesAvailable,
"N" );
//
// We're done with this request. Dequeue the IRP context from
// SCB and complete the request.
//
NwDequeueIrpContext( IrpContext, FALSE );
NwCompleteRequest( IrpContext, Status );
return Status;
}
VOID
BuildBurstWriteFirstReq(
PIRP_CONTEXT IrpContext,
PVOID Buffer,
ULONG DataSize,
PMDL BurstMdl,
UCHAR Flags,
ULONG Handle,
ULONG FileOffset
)
{
PNCP_BURST_WRITE_REQUEST BurstWrite;
PNONPAGED_SCB pNpScb;
ULONG RealDataLength;
USHORT RealBurstLength;
PAGED_CODE();
BurstWrite = (PNCP_BURST_WRITE_REQUEST)Buffer;
pNpScb = IrpContext->pNpScb;
RealDataLength = DataSize + sizeof( *BurstWrite ) - sizeof( NCP_BURST_HEADER );
RealBurstLength = (USHORT)MdlLength( BurstMdl ) + sizeof( *BurstWrite ) - sizeof( NCP_BURST_HEADER );
BurstWrite->BurstHeader.Command = PEP_COMMAND_BURST;
BurstWrite->BurstHeader.Flags = Flags;
BurstWrite->BurstHeader.StreamType = 0x02;
BurstWrite->BurstHeader.SourceConnection = pNpScb->SourceConnectionId;
BurstWrite->BurstHeader.DestinationConnection = pNpScb->DestinationConnectionId;
if ( !BooleanFlagOn( IrpContext->Flags, IRP_FLAG_RETRY_SEND ) ) {
//
// Use the same delay on all retransmissions of the burst. Save
// the current time.
//
pNpScb->CurrentBurstDelay = pNpScb->NwSendDelay;
//
// Send system packet next retransmission.
//
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
} else {
//
// This is a retransmission. Alternate between sending a system
// packet and the first write.
//
if ( !BooleanFlagOn( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET ) ) {
SetFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
BurstWrite->BurstHeader.Flags = BURST_FLAG_SYSTEM_PACKET;
LongByteSwap( BurstWrite->BurstHeader.SendDelayTime, pNpScb->CurrentBurstDelay );
BurstWrite->BurstHeader.DataSize = 0;
BurstWrite->BurstHeader.BurstOffset = 0;
BurstWrite->BurstHeader.BurstLength = 0;
BurstWrite->BurstHeader.MissingFragmentCount = 0;
IrpContext->TxMdl->ByteCount = sizeof( NCP_BURST_HEADER );
IrpContext->TxMdl->Next = NULL;
return;
}
//
// Send system packet next retransmission.
//
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
}
LongByteSwap( BurstWrite->BurstHeader.SendDelayTime, pNpScb->CurrentBurstDelay );
LongByteSwap( BurstWrite->BurstHeader.DataSize, RealDataLength );
BurstWrite->BurstHeader.BurstOffset = 0;
ShortByteSwap( BurstWrite->BurstHeader.BurstLength, RealBurstLength );
BurstWrite->BurstHeader.MissingFragmentCount = 0;
BurstWrite->Function = BURST_REQUEST_WRITE;
BurstWrite->Handle = Handle;
LongByteSwap( BurstWrite->TotalWriteOffset, IrpContext->Specific.Write.TotalWriteOffset );
LongByteSwap( BurstWrite->TotalWriteLength, IrpContext->Specific.Write.TotalWriteLength );
LongByteSwap( BurstWrite->Offset, FileOffset );
LongByteSwap( BurstWrite->Length, DataSize );
IrpContext->TxMdl->ByteCount = sizeof( *BurstWrite );
IrpContext->TxMdl->Next = BurstMdl;
return;
}
VOID
BuildBurstWriteNextReq(
PIRP_CONTEXT IrpContext,
PVOID Buffer,
ULONG DataSize,
UCHAR BurstFlags,
ULONG BurstOffset,
PMDL BurstHeaderMdl,
PMDL BurstDataMdl
)
{
PNCP_BURST_HEADER BurstHeader;
PNONPAGED_SCB pNpScb;
USHORT BurstLength;
PAGED_CODE();
BurstHeader = (PNCP_BURST_HEADER)Buffer;
pNpScb = IrpContext->pNpScb;
BurstLength = (USHORT)MdlLength( BurstDataMdl );
BurstHeader->Command = PEP_COMMAND_BURST;
BurstHeader->Flags = BurstFlags;
BurstHeader->StreamType = 0x02;
BurstHeader->SourceConnection = pNpScb->SourceConnectionId;
BurstHeader->DestinationConnection = pNpScb->DestinationConnectionId;
LongByteSwap( BurstHeader->SendDelayTime, pNpScb->CurrentBurstDelay );
if ( BooleanFlagOn( IrpContext->Flags, IRP_FLAG_RETRY_SEND ) ) {
//
// This is a retransmission. Alternate between sending a system
// packet and the first write.
//
if ( !BooleanFlagOn( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET ) ) {
SetFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
BurstHeader->Flags = BURST_FLAG_SYSTEM_PACKET;
LongByteSwap( BurstHeader->SendDelayTime, pNpScb->CurrentBurstDelay );
BurstHeader->DataSize = 0;
BurstHeader->BurstOffset = 0;
BurstHeader->BurstLength = 0;
BurstHeader->MissingFragmentCount = 0;
IrpContext->TxMdl->ByteCount = sizeof( NCP_BURST_HEADER );
IrpContext->TxMdl->Next = NULL;
return;
}
//
// Send system packet next retransmission.
//
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
} else {
//
// Send system packet next retransmission.
//
ClearFlag( IrpContext->Flags, IRP_FLAG_NOT_SYSTEM_PACKET );
}
LongByteSwap( BurstHeader->DataSize, DataSize );
LongByteSwap( BurstHeader->BurstOffset, BurstOffset );
ShortByteSwap( BurstHeader->BurstLength, BurstLength );
BurstHeader->MissingFragmentCount = 0;
BurstHeaderMdl->ByteCount = sizeof( *BurstHeader );
BurstHeaderMdl->Next = BurstDataMdl;
return;
}
NTSTATUS
SendSecondaryPacket(
PIRP_CONTEXT IrpContext,
PIRP Irp
)
/*++
Routine Description:
This routine submits a TDI send request to the tranport layer.
Arguments:
IrpContext - A pointer to IRP context information for the request
being processed.
Irp - The IRP for the packet to send.
Return Value:
None.
--*/
{
PNONPAGED_SCB pNpScb;
NTSTATUS Status;
PNCP_BURST_HEADER BurstHeader;
pNpScb = IrpContext->pNpScb;
DebugTrace( 0, Dbg, "SendSecondaryPacket\n", 0 );
BurstHeader = (PNCP_BURST_HEADER)( MmGetMdlVirtualAddress( Irp->MdlAddress ) );
if ( !BooleanFlagOn( IrpContext->Flags, IRP_FLAG_NOT_OK_TO_RECEIVE ) ) {
pNpScb->OkToReceive = TRUE;
}
LongByteSwap( BurstHeader->PacketSequenceNo, pNpScb->BurstSequenceNo );
pNpScb->BurstSequenceNo++;
ShortByteSwap( BurstHeader->BurstSequenceNo, pNpScb->BurstRequestNo );
ShortByteSwap( BurstHeader->AckSequenceNo, pNpScb->BurstRequestNo );
DebugTrace( +0, Dbg, "Irp %X\n", Irp );
DebugTrace( +0, Dbg, "pIrpC %X\n", IrpContext);
#if NWDBG
dumpMdl( Dbg, IrpContext->TxMdl);
#endif
Stats.BytesTransmitted.QuadPart += MdlLength( Irp->MdlAddress );
Stats.NcpsTransmitted.QuadPart += 1;
Status = IoCallDriver( pNpScb->Server.pDeviceObject, Irp );
DebugTrace( -1, Dbg, " %X\n", Status );
if ( !NT_SUCCESS( Status ) ) {
Error( EVENT_NWRDR_NETWORK_ERROR, Status, NULL, 0, 0 );
}
return Status;
}
#if NWFASTIO
BOOLEAN
NwFastWrite (
IN PFILE_OBJECT FileObject,
IN PLARGE_INTEGER FileOffset,
IN ULONG Length,
IN BOOLEAN Wait,
IN ULONG LockKey,
OUT PVOID Buffer,
OUT PIO_STATUS_BLOCK IoStatus,
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
This routine does a fast cached read bypassing the usual file system
entry routine (i.e., without the Irp). It is used to do a copy read
of a cached file object. For a complete description of the arguments
see CcCopyRead.
Arguments:
FileObject - Pointer to the file object being read.
FileOffset - Byte offset in file for desired data.
Length - Length of desired data in bytes.
Wait - FALSE if caller may not block, TRUE otherwise
Buffer - Pointer to output buffer to which data should be copied.
IoStatus - Pointer to standard I/O status block to receive the status
for the transfer.
Return Value:
FALSE - if Wait was supplied as FALSE and the data was not delivered, or
if there is an I/O error.
TRUE - if the data is being delivered
--*/
{
NODE_TYPE_CODE nodeTypeCode;
PICB icb;
PFCB fcb;
PVOID fsContext;
ULONG offset;
BOOLEAN wroteToCache;
try {
FsRtlEnterFileSystem();
DebugTrace(+1, Dbg, "NwFastWrite...\n", 0);
//
// Special case a read of zero length
//
if (Length == 0) {
//
// A zero length transfer was requested.
//
IoStatus->Status = STATUS_SUCCESS;
IoStatus->Information = 0;
DebugTrace(+1, Dbg, "NwFastWrite -> TRUE\n", 0);
return TRUE;
}
//
// Decode the file object to figure out who we are. If the result
// is not FCB then its an illegal parameter.
//
if ((nodeTypeCode = NwDecodeFileObject( FileObject,
&fsContext,
(PVOID *)&icb )) != NW_NTC_ICB) {
DebugTrace(0, Dbg, "Not a file\n", 0);
DebugTrace(-1, Dbg, "NwFastWrite -> FALSE\n", 0);
return FALSE;
}
fcb = (PFCB)icb->SuperType.Fcb;
nodeTypeCode = fcb->NodeTypeCode;
offset = FileOffset->LowPart;
IoStatus->Status = STATUS_SUCCESS;
IoStatus->Information = Length;
wroteToCache = CacheWrite(
NULL,
fcb->NonPagedFcb,
offset,
Length,
Buffer );
DebugTrace(-1, Dbg, "NwFastWrite -> %s\n", wroteToCache ? "TRUE" : "FALSE" );
if ( wroteToCache ) {
//
// If the file was extended, record the new file size.
//
if ( ( offset + Length ) > fcb->NonPagedFcb->Header.FileSize.LowPart ) {
fcb->NonPagedFcb->Header.FileSize.LowPart = ( offset + Length );
}
}
#ifndef NT1057
//
// Update the file object if we succeeded. We know that this
// is synchronous and not paging io because it's coming in through
// the cache.
//
if ( wroteToCache ) {
FileObject->CurrentByteOffset.QuadPart = FileOffset->QuadPart + Length;
}
#endif
return( wroteToCache );
} finally {
FsRtlExitFileSystem();
}
}
#endif