mirror of
https://github.com/reactos/reactos.git
synced 2024-12-29 10:35:28 +00:00
878 lines
24 KiB
C
878 lines
24 KiB
C
/*
|
|
* PROJECT: ReactOS Named Pipe FileSystem
|
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
|
* FILE: drivers/filesystems/npfs/fsctrl.c
|
|
* PURPOSE: Named Pipe FileSystem I/O Controls
|
|
* PROGRAMMERS: ReactOS Portable Systems Group
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#include "npfs.h"
|
|
|
|
// File ID number for NPFS bugchecking support
|
|
#define NPFS_BUGCHECK_FILE_ID (NPFS_BUGCHECK_FSCTRL)
|
|
|
|
/* GLOBALS ********************************************************************/
|
|
|
|
IO_STATUS_BLOCK NpUserIoStatusBlock;
|
|
|
|
/* FUNCTIONS ******************************************************************/
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpInternalTransceive(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpInternalRead(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN BOOLEAN Overflow,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpInternalWrite(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpQueryClientProcess(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
PIO_STACK_LOCATION IoStackLocation;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PNP_CCB Ccb;
|
|
PNP_CLIENT_PROCESS ClientSession, QueryBuffer;
|
|
ULONG Length;
|
|
PAGED_CODE();
|
|
|
|
/* Get the current stack location */
|
|
IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
/* Decode the file object and check the node type */
|
|
NodeTypeCode = NpDecodeFileObject(IoStackLocation->FileObject, 0, &Ccb, 0);
|
|
if (NodeTypeCode != NPFS_NTC_CCB)
|
|
{
|
|
return STATUS_PIPE_DISCONNECTED;
|
|
}
|
|
|
|
/* Get the length of the query buffer */
|
|
Length = IoStackLocation->Parameters.QueryFile.Length;
|
|
if (Length < 8)
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
QueryBuffer = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
/* Lock the Ccb */
|
|
ExAcquireResourceExclusiveLite(&Ccb->NonPagedCcb->Lock, TRUE);
|
|
|
|
/* Get the CCBs client session and check if it's set */
|
|
ClientSession = Ccb->ClientSession;
|
|
if (ClientSession != NULL)
|
|
{
|
|
/* Copy first 2 fields */
|
|
QueryBuffer->Unknown = ClientSession->Unknown;
|
|
QueryBuffer->Process = ClientSession->Process;
|
|
}
|
|
else
|
|
{
|
|
/* Copy the process from the CCB */
|
|
QueryBuffer->Unknown = NULL;
|
|
QueryBuffer->Process = Ccb->Process;
|
|
}
|
|
|
|
/* Does the caller provide a large enough buffer for the full data? */
|
|
if (Length >= sizeof(NP_CLIENT_PROCESS))
|
|
{
|
|
Irp->IoStatus.Information = sizeof(NP_CLIENT_PROCESS);
|
|
|
|
/* Do we have a ClientSession structure? */
|
|
if (ClientSession != NULL)
|
|
{
|
|
/* Copy length and the data */
|
|
QueryBuffer->DataLength = ClientSession->DataLength;
|
|
RtlCopyMemory(QueryBuffer->Buffer,
|
|
ClientSession->Buffer,
|
|
ClientSession->DataLength);
|
|
|
|
/* NULL terminate the buffer */
|
|
NT_ASSERT(QueryBuffer->DataLength <= 30);
|
|
QueryBuffer->Buffer[QueryBuffer->DataLength / sizeof(WCHAR)] = 0;
|
|
}
|
|
else
|
|
{
|
|
/* No data */
|
|
QueryBuffer->DataLength = 0;
|
|
QueryBuffer->Buffer[0] = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Irp->IoStatus.Information = FIELD_OFFSET(NP_CLIENT_PROCESS, DataLength);
|
|
}
|
|
|
|
/* Unlock the Ccb */
|
|
ExReleaseResourceLite(&Ccb->NonPagedCcb->Lock);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpSetClientProcess(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
PIO_STACK_LOCATION IoStackLocation;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PNP_CCB Ccb;
|
|
ULONG Length;
|
|
PNP_CLIENT_PROCESS InputBuffer, ClientSession, OldClientSession;
|
|
PAGED_CODE();
|
|
|
|
/* Get the current stack location */
|
|
IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
/* Only kernel calls are allowed! */
|
|
if (IoStackLocation->MinorFunction != IRP_MN_KERNEL_CALL)
|
|
{
|
|
return STATUS_ACCESS_DENIED;
|
|
}
|
|
|
|
/* Decode the file object and check the node type */
|
|
NodeTypeCode = NpDecodeFileObject(IoStackLocation->FileObject, 0, &Ccb, 0);
|
|
if (NodeTypeCode != NPFS_NTC_CCB)
|
|
{
|
|
return STATUS_PIPE_DISCONNECTED;
|
|
}
|
|
|
|
/* Get the length of the query buffer and check if it's valid */
|
|
Length = IoStackLocation->Parameters.QueryFile.Length;
|
|
if (Length != sizeof(NP_CLIENT_PROCESS))
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
/* Get the buffer and check if the data Length is valid */
|
|
InputBuffer = Irp->AssociatedIrp.SystemBuffer;
|
|
if (InputBuffer->DataLength > 30)
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
/* Allocate a new structure */
|
|
ClientSession = ExAllocatePoolWithQuotaTag(PagedPool,
|
|
sizeof(NP_CLIENT_PROCESS),
|
|
'iFpN');
|
|
|
|
/* Copy the full input buffer */
|
|
RtlCopyMemory(ClientSession, InputBuffer, sizeof(NP_CLIENT_PROCESS));
|
|
|
|
/* Lock the Ccb */
|
|
ExAcquireResourceExclusiveLite(&Ccb->NonPagedCcb->Lock, TRUE);
|
|
|
|
/* Get the old ClientSession and set the new */
|
|
OldClientSession = Ccb->ClientSession;
|
|
Ccb->ClientSession = ClientSession;
|
|
|
|
/* Copy the process to the CCB */
|
|
Ccb->Process = ClientSession->Process;
|
|
|
|
/* Unlock the Ccb */
|
|
ExReleaseResourceLite(&Ccb->NonPagedCcb->Lock);
|
|
|
|
/* Check if there was already a ClientSession */
|
|
if (OldClientSession != NULL)
|
|
{
|
|
/* Free it */
|
|
ExFreePoolWithTag(OldClientSession, 'iFpN');
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpAssignEvent(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpQueryEvent(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpImpersonate(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
ULONG NamedPipeEnd;
|
|
PNP_CCB Ccb;
|
|
NTSTATUS Status;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PIO_STACK_LOCATION IoStack;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
NodeTypeCode = NpDecodeFileObject(IoStack->FileObject, NULL, &Ccb, &NamedPipeEnd);
|
|
if (NodeTypeCode == NPFS_NTC_CCB && NamedPipeEnd == FILE_PIPE_SERVER_END)
|
|
{
|
|
Status = NpImpersonateClientContext(Ccb);
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_ILLEGAL_FUNCTION;
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpDisconnect(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
ULONG NamedPipeEnd;
|
|
PNP_CCB Ccb;
|
|
NTSTATUS Status;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PIO_STACK_LOCATION IoStack;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
NodeTypeCode = NpDecodeFileObject(IoStack->FileObject, NULL, &Ccb, &NamedPipeEnd);
|
|
if (NodeTypeCode == NPFS_NTC_CCB)
|
|
{
|
|
if (NamedPipeEnd == FILE_PIPE_SERVER_END)
|
|
{
|
|
ExAcquireResourceExclusiveLite(&Ccb->NonPagedCcb->Lock, TRUE);
|
|
|
|
Status = NpSetDisconnectedPipeState(Ccb, List);
|
|
|
|
NpUninitializeSecurity(Ccb);
|
|
|
|
ExReleaseResourceLite(&Ccb->NonPagedCcb->Lock);
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_ILLEGAL_FUNCTION;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_PIPE_DISCONNECTED;
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpListen(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
ULONG NamedPipeEnd;
|
|
PNP_CCB Ccb;
|
|
NTSTATUS Status;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PIO_STACK_LOCATION IoStack;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
NodeTypeCode = NpDecodeFileObject(IoStack->FileObject, NULL, &Ccb, &NamedPipeEnd);
|
|
if (NodeTypeCode == NPFS_NTC_CCB)
|
|
{
|
|
if (NamedPipeEnd == FILE_PIPE_SERVER_END)
|
|
{
|
|
ExAcquireResourceExclusiveLite(&Ccb->NonPagedCcb->Lock, TRUE);
|
|
|
|
Status = NpSetListeningPipeState(Ccb, Irp, List);
|
|
|
|
NpUninitializeSecurity(Ccb);
|
|
|
|
ExReleaseResourceLite(&Ccb->NonPagedCcb->Lock);
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_ILLEGAL_FUNCTION;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_ILLEGAL_FUNCTION;
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpPeek(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
PIO_STACK_LOCATION IoStack;
|
|
NODE_TYPE_CODE Type;
|
|
ULONG OutputLength;
|
|
ULONG NamedPipeEnd;
|
|
PNP_CCB Ccb;
|
|
PFILE_PIPE_PEEK_BUFFER PeekBuffer;
|
|
PNP_DATA_QUEUE DataQueue;
|
|
ULONG_PTR BytesPeeked;
|
|
IO_STATUS_BLOCK IoStatus;
|
|
NTSTATUS Status;
|
|
PNP_DATA_QUEUE_ENTRY DataEntry;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
OutputLength = IoStack->Parameters.FileSystemControl.OutputBufferLength;
|
|
Type = NpDecodeFileObject(IoStack->FileObject, NULL, &Ccb, &NamedPipeEnd);
|
|
|
|
if (!Type)
|
|
{
|
|
return STATUS_PIPE_DISCONNECTED;
|
|
}
|
|
|
|
if ((Type != NPFS_NTC_CCB) &&
|
|
(OutputLength < FIELD_OFFSET(FILE_PIPE_PEEK_BUFFER, Data)))
|
|
{
|
|
return STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
PeekBuffer = Irp->AssociatedIrp.SystemBuffer;
|
|
if (NamedPipeEnd != FILE_PIPE_CLIENT_END)
|
|
{
|
|
if (NamedPipeEnd != FILE_PIPE_SERVER_END)
|
|
{
|
|
NpBugCheck(NamedPipeEnd, 0, 0);
|
|
}
|
|
|
|
DataQueue = &Ccb->DataQueue[FILE_PIPE_INBOUND];
|
|
}
|
|
else
|
|
{
|
|
DataQueue = &Ccb->DataQueue[FILE_PIPE_OUTBOUND];
|
|
}
|
|
|
|
if (Ccb->NamedPipeState != FILE_PIPE_CONNECTED_STATE)
|
|
{
|
|
if (Ccb->NamedPipeState != FILE_PIPE_CLOSING_STATE)
|
|
{
|
|
return STATUS_INVALID_PIPE_STATE;
|
|
}
|
|
|
|
if (DataQueue->QueueState != WriteEntries)
|
|
{
|
|
return STATUS_PIPE_BROKEN;
|
|
}
|
|
}
|
|
|
|
PeekBuffer->NamedPipeState = 0;
|
|
PeekBuffer->ReadDataAvailable = 0;
|
|
PeekBuffer->NumberOfMessages = 0;
|
|
PeekBuffer->MessageLength = 0;
|
|
PeekBuffer->NamedPipeState = Ccb->NamedPipeState;
|
|
BytesPeeked = FIELD_OFFSET(FILE_PIPE_PEEK_BUFFER, Data);
|
|
|
|
if (DataQueue->QueueState == WriteEntries)
|
|
{
|
|
DataEntry = CONTAINING_RECORD(DataQueue->Queue.Flink,
|
|
NP_DATA_QUEUE_ENTRY,
|
|
QueueEntry);
|
|
ASSERT((DataEntry->DataEntryType == Buffered) || (DataEntry->DataEntryType == Unbuffered));
|
|
|
|
PeekBuffer->ReadDataAvailable = DataQueue->BytesInQueue - DataQueue->ByteOffset;
|
|
if (Ccb->Fcb->NamedPipeType == FILE_PIPE_MESSAGE_TYPE)
|
|
{
|
|
PeekBuffer->NumberOfMessages = DataQueue->EntriesInQueue;
|
|
PeekBuffer->MessageLength = DataEntry->DataSize - DataQueue->ByteOffset;
|
|
}
|
|
|
|
if (OutputLength == FIELD_OFFSET(FILE_PIPE_PEEK_BUFFER, Data))
|
|
{
|
|
Status = PeekBuffer->ReadDataAvailable ? STATUS_BUFFER_OVERFLOW : STATUS_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
IoStatus = NpReadDataQueue(DataQueue,
|
|
TRUE,
|
|
FALSE,
|
|
PeekBuffer->Data,
|
|
OutputLength - FIELD_OFFSET(FILE_PIPE_PEEK_BUFFER, Data),
|
|
Ccb->Fcb->NamedPipeType == FILE_PIPE_MESSAGE_TYPE,
|
|
Ccb,
|
|
List);
|
|
Status = IoStatus.Status;
|
|
BytesPeeked += IoStatus.Information;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Status = STATUS_SUCCESS;
|
|
}
|
|
|
|
Irp->IoStatus.Information = BytesPeeked;
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpCompleteTransceiveIrp(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PVOID Context)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
if (Irp->AssociatedIrp.SystemBuffer)
|
|
{
|
|
ExFreePool(Irp->AssociatedIrp.SystemBuffer);
|
|
}
|
|
|
|
IoFreeIrp(Irp);
|
|
return STATUS_MORE_PROCESSING_REQUIRED;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpTransceive(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PLIST_ENTRY List)
|
|
{
|
|
PIO_STACK_LOCATION IoStack;
|
|
PVOID InBuffer, OutBuffer;
|
|
ULONG InLength, OutLength, BytesWritten;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PNP_CCB Ccb;
|
|
ULONG NamedPipeEnd;
|
|
PNP_NONPAGED_CCB NonPagedCcb;
|
|
PNP_DATA_QUEUE ReadQueue, WriteQueue;
|
|
PNP_EVENT_BUFFER EventBuffer;
|
|
NTSTATUS Status;
|
|
PIRP NewIrp;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
InLength = IoStack->Parameters.FileSystemControl.InputBufferLength;
|
|
InBuffer = IoStack->Parameters.FileSystemControl.Type3InputBuffer;
|
|
OutLength = IoStack->Parameters.FileSystemControl.OutputBufferLength;
|
|
OutBuffer = Irp->UserBuffer;
|
|
|
|
if (Irp->RequestorMode == UserMode)
|
|
{
|
|
_SEH2_TRY
|
|
{
|
|
ProbeForRead(InBuffer, InLength, sizeof(CHAR));
|
|
ProbeForWrite(OutBuffer, OutLength, sizeof(CHAR));
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
|
}
|
|
_SEH2_END;
|
|
}
|
|
|
|
NodeTypeCode = NpDecodeFileObject(IoStack->FileObject, NULL, &Ccb, &NamedPipeEnd);
|
|
if (NodeTypeCode != NPFS_NTC_CCB)
|
|
{
|
|
return STATUS_PIPE_DISCONNECTED;
|
|
}
|
|
|
|
NonPagedCcb = Ccb->NonPagedCcb;
|
|
ExAcquireResourceExclusiveLite(&NonPagedCcb->Lock, TRUE);
|
|
|
|
if (Ccb->NamedPipeState != FILE_PIPE_CONNECTED_STATE)
|
|
{
|
|
Status = STATUS_INVALID_PIPE_STATE;
|
|
goto Quickie;
|
|
}
|
|
|
|
if (NamedPipeEnd != FILE_PIPE_CLIENT_END)
|
|
{
|
|
if (NamedPipeEnd != FILE_PIPE_SERVER_END)
|
|
{
|
|
NpBugCheck(NamedPipeEnd, 0, 0);
|
|
}
|
|
ReadQueue = &Ccb->DataQueue[FILE_PIPE_INBOUND];
|
|
WriteQueue = &Ccb->DataQueue[FILE_PIPE_OUTBOUND];
|
|
}
|
|
else
|
|
{
|
|
ReadQueue = &Ccb->DataQueue[FILE_PIPE_OUTBOUND];
|
|
WriteQueue = &Ccb->DataQueue[FILE_PIPE_INBOUND];
|
|
}
|
|
|
|
EventBuffer = NonPagedCcb->EventBuffer[NamedPipeEnd];
|
|
|
|
if (Ccb->Fcb->NamedPipeConfiguration != FILE_PIPE_FULL_DUPLEX ||
|
|
Ccb->ReadMode[NamedPipeEnd] != FILE_PIPE_MESSAGE_MODE)
|
|
{
|
|
Status = STATUS_INVALID_PIPE_STATE;
|
|
goto Quickie;
|
|
}
|
|
|
|
if (ReadQueue->QueueState != Empty)
|
|
{
|
|
Status = STATUS_PIPE_BUSY;
|
|
goto Quickie;
|
|
}
|
|
|
|
Status = NpWriteDataQueue(WriteQueue,
|
|
FILE_PIPE_MESSAGE_MODE,
|
|
InBuffer,
|
|
InLength,
|
|
Ccb->Fcb->NamedPipeType,
|
|
&BytesWritten,
|
|
Ccb,
|
|
NamedPipeEnd,
|
|
Irp->Tail.Overlay.Thread,
|
|
List);
|
|
if (Status == STATUS_MORE_PROCESSING_REQUIRED)
|
|
{
|
|
ASSERT(WriteQueue->QueueState != ReadEntries);
|
|
NewIrp = IoAllocateIrp(DeviceObject->StackSize, TRUE);
|
|
if (!NewIrp)
|
|
{
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Quickie;
|
|
}
|
|
|
|
IoSetCompletionRoutine(Irp, NpCompleteTransceiveIrp, NULL, TRUE, TRUE, TRUE);
|
|
|
|
if (BytesWritten)
|
|
{
|
|
NewIrp->AssociatedIrp.SystemBuffer = ExAllocatePoolWithQuotaTag(PagedPool | POOL_QUOTA_FAIL_INSTEAD_OF_RAISE,
|
|
BytesWritten,
|
|
NPFS_WRITE_BLOCK_TAG);
|
|
if (!NewIrp->AssociatedIrp.SystemBuffer)
|
|
{
|
|
IoFreeIrp(NewIrp);
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Quickie;
|
|
}
|
|
|
|
_SEH2_TRY
|
|
{
|
|
RtlCopyMemory(NewIrp->AssociatedIrp.SystemBuffer,
|
|
(PVOID)((ULONG_PTR)InBuffer + InLength - BytesWritten),
|
|
BytesWritten);
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
Status = _SEH2_GetExceptionCode();
|
|
_SEH2_YIELD(goto Quickie);
|
|
}
|
|
_SEH2_END;
|
|
}
|
|
else
|
|
{
|
|
NewIrp->AssociatedIrp.SystemBuffer = NULL;
|
|
}
|
|
|
|
IoStack = IoGetNextIrpStackLocation(NewIrp);
|
|
IoSetNextIrpStackLocation(NewIrp);
|
|
|
|
NewIrp->Tail.Overlay.Thread = Irp->Tail.Overlay.Thread;
|
|
NewIrp->IoStatus.Information = BytesWritten;
|
|
|
|
IoStack->Parameters.Read.Length = BytesWritten;
|
|
IoStack->MajorFunction = IRP_MJ_WRITE;
|
|
|
|
if (BytesWritten > 0) NewIrp->Flags = IRP_DEALLOCATE_BUFFER | IRP_BUFFERED_IO;
|
|
NewIrp->UserIosb = &NpUserIoStatusBlock;
|
|
|
|
Status = NpAddDataQueueEntry(NamedPipeEnd,
|
|
Ccb,
|
|
WriteQueue,
|
|
WriteEntries,
|
|
Unbuffered,
|
|
BytesWritten,
|
|
NewIrp,
|
|
NULL,
|
|
0);
|
|
if (Status != STATUS_PENDING)
|
|
{
|
|
NewIrp->IoStatus.Status = Status;
|
|
InsertTailList(List, &NewIrp->Tail.Overlay.ListEntry);
|
|
}
|
|
}
|
|
|
|
if (!NT_SUCCESS(Status)) goto Quickie;
|
|
|
|
if (EventBuffer) KeSetEvent(EventBuffer->Event, IO_NO_INCREMENT, FALSE);
|
|
ASSERT(ReadQueue->QueueState == Empty);
|
|
Status = NpAddDataQueueEntry(NamedPipeEnd,
|
|
Ccb,
|
|
ReadQueue,
|
|
ReadEntries,
|
|
Buffered,
|
|
OutLength,
|
|
Irp,
|
|
NULL,
|
|
0);
|
|
if (NT_SUCCESS(Status))
|
|
{
|
|
if (EventBuffer) KeSetEvent(EventBuffer->Event, IO_NO_INCREMENT, FALSE);
|
|
}
|
|
|
|
Quickie:
|
|
ExReleaseResourceLite(&Ccb->NonPagedCcb->Lock);
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpWaitForNamedPipe(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
PIO_STACK_LOCATION IoStack;
|
|
ULONG InLength, NameLength;
|
|
UNICODE_STRING SourceString, Prefix;
|
|
ULONG NamedPipeEnd;
|
|
PNP_CCB Ccb;
|
|
PFILE_PIPE_WAIT_FOR_BUFFER Buffer;
|
|
NTSTATUS Status;
|
|
NODE_TYPE_CODE NodeTypeCode;
|
|
PLIST_ENTRY NextEntry;
|
|
PNP_FCB Fcb;
|
|
PWCHAR OriginalBuffer;
|
|
PAGED_CODE();
|
|
|
|
IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
InLength = IoStack->Parameters.FileSystemControl.InputBufferLength;
|
|
|
|
SourceString.Buffer = NULL;
|
|
|
|
if (NpDecodeFileObject(IoStack->FileObject,
|
|
NULL,
|
|
&Ccb,
|
|
&NamedPipeEnd) != NPFS_NTC_ROOT_DCB)
|
|
{
|
|
Status = STATUS_ILLEGAL_FUNCTION;
|
|
goto Quickie;
|
|
}
|
|
|
|
Buffer = (PFILE_PIPE_WAIT_FOR_BUFFER)Irp->AssociatedIrp.SystemBuffer;
|
|
if (InLength < sizeof(*Buffer))
|
|
{
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto Quickie;
|
|
}
|
|
|
|
NameLength = Buffer->NameLength;
|
|
if ((NameLength > (0xFFFF - sizeof(UNICODE_NULL))) ||
|
|
((NameLength + FIELD_OFFSET(FILE_PIPE_WAIT_FOR_BUFFER, Name)) > InLength))
|
|
{
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
goto Quickie;
|
|
}
|
|
|
|
SourceString.Length = (USHORT)NameLength + sizeof(OBJ_NAME_PATH_SEPARATOR);
|
|
SourceString.Buffer = ExAllocatePoolWithTag(PagedPool,
|
|
SourceString.Length,
|
|
NPFS_WRITE_BLOCK_TAG);
|
|
if (!SourceString.Buffer)
|
|
{
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
goto Quickie;
|
|
}
|
|
|
|
SourceString.Buffer[0] = OBJ_NAME_PATH_SEPARATOR;
|
|
RtlCopyMemory(&SourceString.Buffer[1], Buffer->Name, Buffer->NameLength);
|
|
|
|
Status = STATUS_SUCCESS;
|
|
OriginalBuffer = SourceString.Buffer;
|
|
//Status = NpTranslateAlias(&SourceString);
|
|
if (!NT_SUCCESS(Status)) goto Quickie;
|
|
|
|
Fcb = NpFindPrefix(&SourceString, 1, &Prefix);
|
|
Fcb = (PNP_FCB)((ULONG_PTR)Fcb & ~1);
|
|
|
|
NodeTypeCode = Fcb ? Fcb->NodeType : 0;
|
|
if (NodeTypeCode != NPFS_NTC_FCB)
|
|
{
|
|
Status = STATUS_OBJECT_NAME_NOT_FOUND;
|
|
goto Quickie;
|
|
}
|
|
|
|
for (NextEntry = Fcb->CcbList.Flink;
|
|
NextEntry != &Fcb->CcbList;
|
|
NextEntry = NextEntry->Flink)
|
|
{
|
|
Ccb = CONTAINING_RECORD(NextEntry, NP_CCB, CcbEntry);
|
|
if (Ccb->NamedPipeState == FILE_PIPE_LISTENING_STATE) break;
|
|
}
|
|
|
|
if (NextEntry != &Fcb->CcbList)
|
|
{
|
|
Status = STATUS_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
Status = NpAddWaiter(&NpVcb->WaitQueue,
|
|
Fcb->Timeout,
|
|
Irp,
|
|
OriginalBuffer == SourceString.Buffer ?
|
|
NULL : &SourceString);
|
|
}
|
|
|
|
Quickie:
|
|
if (SourceString.Buffer) ExFreePool(SourceString.Buffer);
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpCommonFileSystemControl(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
ULONG Fsctl;
|
|
BOOLEAN Overflow = FALSE;
|
|
LIST_ENTRY DeferredList;
|
|
NTSTATUS Status;
|
|
PAGED_CODE();
|
|
|
|
InitializeListHead(&DeferredList);
|
|
Fsctl = IoGetCurrentIrpStackLocation(Irp)->Parameters.FileSystemControl.FsControlCode;
|
|
|
|
switch (Fsctl)
|
|
{
|
|
case FSCTL_PIPE_PEEK:
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpPeek(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_INTERNAL_WRITE:
|
|
NpAcquireSharedVcb();
|
|
Status = NpInternalWrite(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_TRANSCEIVE:
|
|
NpAcquireSharedVcb();
|
|
Status = NpTransceive(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_INTERNAL_TRANSCEIVE:
|
|
NpAcquireSharedVcb();
|
|
Status = NpInternalTransceive(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_INTERNAL_READ_OVFLOW:
|
|
Overflow = TRUE;
|
|
// on purpose
|
|
|
|
case FSCTL_PIPE_INTERNAL_READ:
|
|
NpAcquireSharedVcb();
|
|
Status = NpInternalRead(DeviceObject, Irp, Overflow, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_QUERY_CLIENT_PROCESS:
|
|
|
|
NpAcquireSharedVcb();
|
|
Status = NpQueryClientProcess(DeviceObject, Irp);
|
|
break;
|
|
|
|
case FSCTL_PIPE_ASSIGN_EVENT:
|
|
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpAssignEvent(DeviceObject, Irp);
|
|
break;
|
|
|
|
case FSCTL_PIPE_DISCONNECT:
|
|
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpDisconnect(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_LISTEN:
|
|
|
|
NpAcquireSharedVcb();
|
|
Status = NpListen(DeviceObject, Irp, &DeferredList);
|
|
break;
|
|
|
|
case FSCTL_PIPE_QUERY_EVENT:
|
|
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpQueryEvent(DeviceObject, Irp);
|
|
break;
|
|
|
|
case FSCTL_PIPE_WAIT:
|
|
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpWaitForNamedPipe(DeviceObject, Irp);
|
|
break;
|
|
|
|
case FSCTL_PIPE_IMPERSONATE:
|
|
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpImpersonate(DeviceObject, Irp);
|
|
break;
|
|
|
|
case FSCTL_PIPE_SET_CLIENT_PROCESS:
|
|
NpAcquireExclusiveVcb();
|
|
Status = NpSetClientProcess(DeviceObject, Irp);
|
|
break;
|
|
|
|
default:
|
|
return STATUS_NOT_SUPPORTED;
|
|
}
|
|
|
|
NpReleaseVcb();
|
|
NpCompleteDeferredIrps(&DeferredList);
|
|
|
|
return Status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NpFsdFileSystemControl(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
NTSTATUS Status;
|
|
PAGED_CODE();
|
|
|
|
FsRtlEnterFileSystem();
|
|
|
|
Status = NpCommonFileSystemControl(DeviceObject, Irp);
|
|
|
|
FsRtlExitFileSystem();
|
|
|
|
if (Status != STATUS_PENDING)
|
|
{
|
|
Irp->IoStatus.Status = Status;
|
|
IoCompleteRequest(Irp, IO_NAMED_PIPE_INCREMENT);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
/* EOF */
|