759 lines
20 KiB
C
759 lines
20 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) 1999 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
ocrw.c
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
read/write io code for printing
|
||
|
|
||
|
Environment:
|
||
|
|
||
|
kernel mode only
|
||
|
|
||
|
Notes:
|
||
|
|
||
|
THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
|
||
|
KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR
|
||
|
PURPOSE.
|
||
|
|
||
|
Copyright (c) 1999 Microsoft Corporation. All Rights Reserved.
|
||
|
|
||
|
|
||
|
Revision History:
|
||
|
|
||
|
5-4-96 : created
|
||
|
|
||
|
--*/
|
||
|
|
||
|
#define DRIVER
|
||
|
|
||
|
#include "wdm.h"
|
||
|
#include "stdarg.h"
|
||
|
#include "stdio.h"
|
||
|
|
||
|
#include <usb.h>
|
||
|
#include <usbdrivr.h>
|
||
|
#include "usbdlib.h"
|
||
|
#include "usbprint.h"
|
||
|
|
||
|
|
||
|
|
||
|
//******************************************************************************
|
||
|
//
|
||
|
// USBPRINT_CompletionStop()
|
||
|
//
|
||
|
// IO Completion Routine which just stops further completion of the Irp
|
||
|
//
|
||
|
//******************************************************************************
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_CompletionStop (
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp,
|
||
|
IN PVOID Context
|
||
|
)
|
||
|
{
|
||
|
return STATUS_MORE_PROCESSING_REQUIRED;
|
||
|
}
|
||
|
|
||
|
//******************************************************************************
|
||
|
//
|
||
|
// USBPRINT_GetCurrentFrame()
|
||
|
//
|
||
|
//******************************************************************************
|
||
|
|
||
|
ULONG
|
||
|
USBPRINT_GetCurrentFrame (
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp
|
||
|
)
|
||
|
{
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
PIO_STACK_LOCATION nextStack;
|
||
|
NTSTATUS ntStatus;
|
||
|
struct _URB_GET_CURRENT_FRAME_NUMBER urb;
|
||
|
|
||
|
deviceExtension = DeviceObject->DeviceExtension;
|
||
|
|
||
|
// Initialize the URB
|
||
|
//
|
||
|
urb.Hdr.Function = URB_FUNCTION_GET_CURRENT_FRAME_NUMBER;
|
||
|
urb.Hdr.Length = sizeof(urb);
|
||
|
urb.FrameNumber = (ULONG)-1;
|
||
|
|
||
|
// Set the IRP parameters to pass the URB down the stack
|
||
|
//
|
||
|
nextStack = IoGetNextIrpStackLocation(Irp);
|
||
|
|
||
|
nextStack->Parameters.Others.Argument1 = &urb;
|
||
|
|
||
|
nextStack->Parameters.DeviceIoControl.IoControlCode =
|
||
|
IOCTL_INTERNAL_USB_SUBMIT_URB;
|
||
|
|
||
|
nextStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
||
|
|
||
|
// Since this Irp is borrowed for URB_FUNCTION_GET_CURRENT_FRAME_NUMBER
|
||
|
// before it is passed down later for the real URB request after this
|
||
|
// routine returns, set a completion routine which stop further completion
|
||
|
// of the Irp.
|
||
|
//
|
||
|
IoSetCompletionRoutine(
|
||
|
Irp,
|
||
|
USBPRINT_CompletionStop,
|
||
|
NULL, // Context
|
||
|
TRUE, // InvokeOnSuccess
|
||
|
TRUE, // InvokeOnError
|
||
|
TRUE // InvokeOnCancel
|
||
|
);
|
||
|
|
||
|
// Now pass the Irp down the stack
|
||
|
//
|
||
|
ntStatus = IoCallDriver(
|
||
|
deviceExtension->TopOfStackDeviceObject,
|
||
|
Irp
|
||
|
);
|
||
|
|
||
|
// Don't need to wait for completion because JD guarantees that
|
||
|
// URB_FUNCTION_GET_CURRENT_FRAME_NUMBER will never return STATUS_PENDING
|
||
|
|
||
|
return urb.FrameNumber;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
PURB
|
||
|
USBPRINT_BuildAsyncRequest(
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp,
|
||
|
IN PUSBD_PIPE_INFORMATION PipeHandle,
|
||
|
IN BOOLEAN Read
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - pointer to the device extension for this instance of the
|
||
|
printer
|
||
|
|
||
|
Irp -
|
||
|
|
||
|
PipeHandle -
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
initialized async urb.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
ULONG siz;
|
||
|
ULONG length;
|
||
|
PURB urb = NULL;
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: handle = 0x%x\n", PipeHandle));
|
||
|
|
||
|
if ( Irp->MdlAddress == NULL )
|
||
|
return NULL;
|
||
|
|
||
|
length = MmGetMdlByteCount(Irp->MdlAddress);
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: length = 0x%x\n", length));
|
||
|
|
||
|
siz = sizeof(struct _URB_BULK_OR_INTERRUPT_TRANSFER);
|
||
|
urb = ExAllocatePoolWithTag(NonPagedPool, siz, USBP_TAG);
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: siz = 0x%x urb 0x%x\n", siz, urb));
|
||
|
|
||
|
if (urb) {
|
||
|
RtlZeroMemory(urb, siz);
|
||
|
|
||
|
urb->UrbBulkOrInterruptTransfer.Hdr.Length = (USHORT) siz;
|
||
|
urb->UrbBulkOrInterruptTransfer.Hdr.Function =
|
||
|
URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER;
|
||
|
urb->UrbBulkOrInterruptTransfer.PipeHandle =
|
||
|
PipeHandle->PipeHandle;
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferFlags =
|
||
|
Read ? USBD_TRANSFER_DIRECTION_IN : 0;
|
||
|
|
||
|
// short packet is not treated as an error.
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferFlags |=
|
||
|
USBD_SHORT_TRANSFER_OK;
|
||
|
|
||
|
//
|
||
|
// no linkage for now
|
||
|
//
|
||
|
|
||
|
urb->UrbBulkOrInterruptTransfer.UrbLink = NULL;
|
||
|
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferMDL =
|
||
|
Irp->MdlAddress;
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferLength =
|
||
|
length;
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: Init async urb Length = 0x%x buf = 0x%x, mdlBuff=0x%x\n",
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferLength,
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBuffer,
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferMDL));
|
||
|
}
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: exit USBPRINT_BuildAsyncRequest\n"));
|
||
|
|
||
|
return urb;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_AsyncReadWrite_Complete(
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp,
|
||
|
IN PVOID Context
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - Pointer to the device object for the USBPRINT device.
|
||
|
|
||
|
Irp - Irp completed.
|
||
|
|
||
|
Context - Driver defined context.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
The function value is the final status from the operation.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
||
|
PURB urb;
|
||
|
PUSBPRINT_RW_CONTEXT context = Context;
|
||
|
PDEVICE_OBJECT deviceObject;
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
PUSBPRINT_WORKITEM_CONTEXT pResetWorkItemObj;
|
||
|
LONG ResetPending;
|
||
|
|
||
|
//Always mark irp pending in dispatch routine now
|
||
|
// if (Irp->PendingReturned) {
|
||
|
// IoMarkIrpPending(Irp);
|
||
|
// }
|
||
|
|
||
|
urb = context->Urb;
|
||
|
deviceObject = context->DeviceObject;
|
||
|
deviceExtension=deviceObject->DeviceExtension;
|
||
|
|
||
|
USBPRINT_KdPrint2 (("USBPRINT.SYS: Async Completion: Length %d, Status 0x%08X\n",
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferLength,
|
||
|
urb->UrbHeader.Status));
|
||
|
|
||
|
|
||
|
//ASSERT(urb->UrbHeader.Status==0);
|
||
|
|
||
|
ntStatus=urb->UrbHeader.Status;
|
||
|
|
||
|
//
|
||
|
// set the length based on the TransferBufferLength
|
||
|
// value in the URB
|
||
|
//
|
||
|
Irp->IoStatus.Information =
|
||
|
urb->UrbBulkOrInterruptTransfer.TransferBufferLength;
|
||
|
|
||
|
|
||
|
if((!NT_SUCCESS(ntStatus))&&(ntStatus!=STATUS_CANCELLED)&&(ntStatus!=STATUS_DEVICE_NOT_CONNECTED)&&(ntStatus!=STATUS_DELETE_PENDING))
|
||
|
{ //We've got an error, and it's not "not connected" or "cancelled", we need to reset the connection
|
||
|
ResetPending=InterlockedCompareExchange(&deviceExtension->ResetWorkItemPending,
|
||
|
1,
|
||
|
0); //Check to see if ResetWorkItem is 0, if so, set it to 1, and start a Reset
|
||
|
if(!ResetPending)
|
||
|
{
|
||
|
pResetWorkItemObj=ExAllocatePoolWithTag(NonPagedPool,sizeof(USBPRINT_WORKITEM_CONTEXT),USBP_TAG);
|
||
|
if(pResetWorkItemObj)
|
||
|
{
|
||
|
pResetWorkItemObj->ioWorkItem=IoAllocateWorkItem(DeviceObject);
|
||
|
if(pResetWorkItemObj==NULL)
|
||
|
{
|
||
|
USBPRINT_KdPrint1 (("USBPRINT.SYS: Unable to allocate IoAllocateWorkItem in ReadWrite_Complete\n"));
|
||
|
ExFreePool(pResetWorkItemObj);
|
||
|
pResetWorkItemObj=NULL;
|
||
|
}
|
||
|
} //if ALloc RestWorkItem OK
|
||
|
else
|
||
|
{
|
||
|
USBPRINT_KdPrint1 (("USBPRINT.SYS: Unable to allocate WorkItemObj in ReadWrite_Complete\n"));
|
||
|
}
|
||
|
if(pResetWorkItemObj)
|
||
|
{
|
||
|
pResetWorkItemObj->irp=Irp;
|
||
|
pResetWorkItemObj->deviceObject=DeviceObject;
|
||
|
if(context->IsWrite)
|
||
|
pResetWorkItemObj->pPipeInfo=deviceExtension->pWritePipe;
|
||
|
else
|
||
|
pResetWorkItemObj->pPipeInfo=deviceExtension->pReadPipe;
|
||
|
|
||
|
USBPRINT_IncrementIoCount(deviceObject);
|
||
|
IoQueueWorkItem(pResetWorkItemObj->ioWorkItem,
|
||
|
USBPRINT_ResetWorkItem,
|
||
|
DelayedWorkQueue,
|
||
|
pResetWorkItemObj);
|
||
|
ntStatus=STATUS_MORE_PROCESSING_REQUIRED;
|
||
|
//Leave the IRP pending until the reset is complete. This way we won't get flooded with irp's we're not
|
||
|
//prepaired to deal with. The Reset WorkItem completes the IRP when it's done.
|
||
|
} //end if the allocs went OK
|
||
|
} //end if ! Reset Pending
|
||
|
} //end if we need to reset
|
||
|
|
||
|
USBPRINT_DecrementIoCount(deviceObject); //still +1 on the IO count after this, leaving one for the workitem to decrement
|
||
|
|
||
|
ExFreePool(context);
|
||
|
ExFreePool(urb);
|
||
|
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
NTSTATUS USBPRINT_ResetWorkItem(IN PDEVICE_OBJECT deviceObject, IN PVOID Context)
|
||
|
{
|
||
|
|
||
|
PUSBPRINT_WORKITEM_CONTEXT pResetWorkItemObj;
|
||
|
PDEVICE_EXTENSION DeviceExtension;
|
||
|
NTSTATUS ntStatus;
|
||
|
ULONG portStatus;
|
||
|
PDEVICE_OBJECT devObj;
|
||
|
|
||
|
|
||
|
USBPRINT_KdPrint2(("USBPRINT.SYS: Entering USBPRINT_ResetWorkItem\n"));
|
||
|
pResetWorkItemObj=(PUSBPRINT_WORKITEM_CONTEXT)Context;
|
||
|
DeviceExtension=pResetWorkItemObj->deviceObject->DeviceExtension;
|
||
|
ntStatus=USBPRINT_ResetPipe(pResetWorkItemObj->deviceObject,pResetWorkItemObj->pPipeInfo,FALSE);
|
||
|
IoCompleteRequest(pResetWorkItemObj->irp,IO_NO_INCREMENT);
|
||
|
IoFreeWorkItem(pResetWorkItemObj->ioWorkItem);
|
||
|
|
||
|
// save off work item device object before freeing work item
|
||
|
devObj = pResetWorkItemObj->deviceObject;
|
||
|
|
||
|
ExFreePool(pResetWorkItemObj);
|
||
|
InterlockedExchange(&(DeviceExtension->ResetWorkItemPending),0);
|
||
|
USBPRINT_DecrementIoCount(devObj);
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_Read(
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - pointer to the device object for this instance of a printer.
|
||
|
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
NT status code
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
||
|
PUSBD_PIPE_INFORMATION pipeHandle = NULL;
|
||
|
PFILE_OBJECT fileObject;
|
||
|
PIO_STACK_LOCATION irpStack, nextStack;
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
PURB urb;
|
||
|
PUSBPRINT_RW_CONTEXT context = NULL;
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: /*dd enter USBPRINT_Read\n\n\n\n\n\n"));
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: /*dd **************************************************************************\n"));
|
||
|
|
||
|
USBPRINT_IncrementIoCount(DeviceObject);
|
||
|
|
||
|
deviceExtension = DeviceObject->DeviceExtension;
|
||
|
|
||
|
if (deviceExtension->AcceptingRequests == FALSE) {
|
||
|
ntStatus = STATUS_DELETE_PENDING;
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
|
||
|
IoCompleteRequest (Irp,
|
||
|
IO_NO_INCREMENT
|
||
|
);
|
||
|
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
||
|
fileObject = irpStack->FileObject;
|
||
|
|
||
|
pipeHandle = deviceExtension->pReadPipe;
|
||
|
|
||
|
if (!pipeHandle) {
|
||
|
ntStatus = STATUS_INVALID_HANDLE;
|
||
|
goto USBPRINT_Read_Reject;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// submit the Read request to USB
|
||
|
//
|
||
|
|
||
|
switch (pipeHandle->PipeType) {
|
||
|
case UsbdPipeTypeInterrupt:
|
||
|
case UsbdPipeTypeBulk:
|
||
|
urb = USBPRINT_BuildAsyncRequest(DeviceObject,
|
||
|
Irp,
|
||
|
pipeHandle,
|
||
|
TRUE);
|
||
|
if (urb) {
|
||
|
context = ExAllocatePoolWithTag(NonPagedPool, sizeof(USBPRINT_RW_CONTEXT), USBP_TAG);
|
||
|
|
||
|
if ( !context )
|
||
|
ExFreePool(urb);
|
||
|
}
|
||
|
|
||
|
if (urb && context) {
|
||
|
context->Urb = urb;
|
||
|
context->DeviceObject = DeviceObject;
|
||
|
context->IsWrite=FALSE;
|
||
|
|
||
|
nextStack = IoGetNextIrpStackLocation(Irp);
|
||
|
ASSERT(nextStack != NULL);
|
||
|
ASSERT(DeviceObject->StackSize>1);
|
||
|
|
||
|
nextStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
||
|
nextStack->Parameters.Others.Argument1 = urb;
|
||
|
nextStack->Parameters.DeviceIoControl.IoControlCode =
|
||
|
IOCTL_INTERNAL_USB_SUBMIT_URB;
|
||
|
|
||
|
IoSetCompletionRoutine(Irp,
|
||
|
USBPRINT_AsyncReadWrite_Complete,
|
||
|
context,
|
||
|
TRUE,
|
||
|
TRUE,
|
||
|
TRUE);
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: IRP = 0x%x current = 0x%x next = 0x%x\n",
|
||
|
Irp, irpStack, nextStack));
|
||
|
|
||
|
// start perf timer here if needed
|
||
|
|
||
|
IoMarkIrpPending(Irp);
|
||
|
ntStatus = IoCallDriver(deviceExtension->TopOfStackDeviceObject,
|
||
|
Irp);
|
||
|
ntStatus=STATUS_PENDING;
|
||
|
goto USBPRINT_Read_Done;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
default:
|
||
|
ntStatus = STATUS_INVALID_PARAMETER;
|
||
|
TRAP();
|
||
|
}
|
||
|
|
||
|
USBPRINT_Read_Reject:
|
||
|
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
|
||
|
IoCompleteRequest (Irp,
|
||
|
IO_NO_INCREMENT
|
||
|
);
|
||
|
|
||
|
USBPRINT_Read_Done:
|
||
|
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_Write(
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This function services WRITE requests for this device (probably from the user mode USB port monitor)
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - pointer to the device object for this printer
|
||
|
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
NT status code
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
||
|
PUSBD_PIPE_INFORMATION pipeHandle = NULL;
|
||
|
PFILE_OBJECT fileObject;
|
||
|
PIO_STACK_LOCATION irpStack, nextStack;
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
PURB urb;
|
||
|
PUSBPRINT_RW_CONTEXT context = NULL;
|
||
|
|
||
|
USBPRINT_KdPrint2 (("USBPRINT.SYS: enter USBPRINT_Write (foo)\n"));
|
||
|
|
||
|
USBPRINT_IncrementIoCount(DeviceObject);
|
||
|
|
||
|
deviceExtension = DeviceObject->DeviceExtension;
|
||
|
|
||
|
if (deviceExtension->AcceptingRequests == FALSE)
|
||
|
{
|
||
|
USBPRINT_KdPrint1 (("USBPRINT.SYS: failure because AcceptingRequests=FALSE\n"));
|
||
|
ntStatus = STATUS_DELETE_PENDING;
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
|
||
|
IoCompleteRequest (Irp,IO_NO_INCREMENT);
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
||
|
|
||
|
|
||
|
fileObject = irpStack->FileObject;
|
||
|
|
||
|
// MmProbeAndLockPages(Irp->MdlAddress,
|
||
|
// KernelMode,
|
||
|
// IoReadAccess);
|
||
|
|
||
|
pipeHandle = deviceExtension->pWritePipe;
|
||
|
if (!pipeHandle)
|
||
|
{
|
||
|
USBPRINT_KdPrint1 (("USBPRINT.SYS: failure because pipe is bad\n"));
|
||
|
ntStatus = STATUS_INVALID_HANDLE;
|
||
|
goto USBPRINT_Write_Reject;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// submit the write request to USB
|
||
|
//
|
||
|
|
||
|
switch (pipeHandle->PipeType)
|
||
|
{
|
||
|
case UsbdPipeTypeInterrupt:
|
||
|
case UsbdPipeTypeBulk:
|
||
|
urb = USBPRINT_BuildAsyncRequest(DeviceObject,
|
||
|
Irp,
|
||
|
pipeHandle,
|
||
|
FALSE);
|
||
|
|
||
|
if (urb)
|
||
|
{
|
||
|
context = ExAllocatePoolWithTag(NonPagedPool, sizeof(USBPRINT_RW_CONTEXT), USBP_TAG);
|
||
|
|
||
|
if(!context)
|
||
|
ExFreePool(urb);
|
||
|
|
||
|
}
|
||
|
|
||
|
if (urb && context)
|
||
|
{
|
||
|
context->Urb = urb;
|
||
|
context->DeviceObject = DeviceObject;
|
||
|
context->IsWrite=TRUE;
|
||
|
|
||
|
nextStack = IoGetNextIrpStackLocation(Irp);
|
||
|
ASSERT(nextStack != NULL);
|
||
|
ASSERT(DeviceObject->StackSize>1);
|
||
|
|
||
|
nextStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
||
|
nextStack->Parameters.Others.Argument1 = urb;
|
||
|
nextStack->Parameters.DeviceIoControl.IoControlCode =
|
||
|
IOCTL_INTERNAL_USB_SUBMIT_URB;
|
||
|
|
||
|
IoSetCompletionRoutine(Irp,
|
||
|
USBPRINT_AsyncReadWrite_Complete,
|
||
|
context,
|
||
|
TRUE,
|
||
|
TRUE,
|
||
|
TRUE);
|
||
|
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: IRP = 0x%x current = 0x%x next = 0x%x\n",Irp, irpStack, nextStack));
|
||
|
|
||
|
IoMarkIrpPending(Irp);
|
||
|
ntStatus = IoCallDriver(deviceExtension->TopOfStackDeviceObject,Irp);
|
||
|
ntStatus=STATUS_PENDING;
|
||
|
goto USBPRINT_Write_Done;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
default:
|
||
|
ntStatus = STATUS_INVALID_PARAMETER;
|
||
|
TRAP();
|
||
|
}
|
||
|
|
||
|
USBPRINT_Write_Reject:
|
||
|
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
|
||
|
IoCompleteRequest (Irp,
|
||
|
IO_NO_INCREMENT
|
||
|
);
|
||
|
|
||
|
USBPRINT_Write_Done:
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: Write Done, status= 0x%08X\n",ntStatus));
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_Close(
|
||
|
IN PDEVICE_OBJECT DeviceObject,
|
||
|
IN PIRP Irp
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - pointer to the device object for this printer
|
||
|
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
NT status code
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
NTSTATUS ntStatus;
|
||
|
PFILE_OBJECT fileObject;
|
||
|
PIO_STACK_LOCATION irpStack;
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
PUSBD_PIPE_INFORMATION pipeHandle = NULL;
|
||
|
|
||
|
USBPRINT_KdPrint2 (("USBPRINT.SYS: entering USBPRINT_Close\n"));
|
||
|
|
||
|
USBPRINT_IncrementIoCount(DeviceObject);
|
||
|
|
||
|
deviceExtension = DeviceObject->DeviceExtension;
|
||
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
||
|
fileObject = irpStack->FileObject;
|
||
|
|
||
|
if (fileObject->FsContext)
|
||
|
{
|
||
|
// closing pipe handle
|
||
|
pipeHandle = fileObject->FsContext;
|
||
|
USBPRINT_KdPrint3 (("USBPRINT.SYS: closing pipe %x\n", pipeHandle));
|
||
|
|
||
|
}
|
||
|
deviceExtension->OpenCnt--;
|
||
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
|
||
|
|
||
|
ntStatus = Irp->IoStatus.Status;
|
||
|
|
||
|
IoCompleteRequest (Irp,IO_NO_INCREMENT);
|
||
|
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
|
||
|
if(!deviceExtension->IsChildDevice)
|
||
|
{
|
||
|
USBPRINT_FdoSubmitIdleRequestIrp(deviceExtension);
|
||
|
}
|
||
|
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
NTSTATUS
|
||
|
USBPRINT_Create(IN PDEVICE_OBJECT DeviceObject,IN PIRP Irp)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
//
|
||
|
// Entry point for CreateFile calls
|
||
|
// user mode apps may open "\\.\USBPRINT-x\"
|
||
|
// where x is the "USB virtual printer port"
|
||
|
//
|
||
|
// No more exposing the pipe # to usermode
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
DeviceObject - pointer to the device object for this printer.
|
||
|
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
NT status code
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
NTSTATUS ntStatus = STATUS_SUCCESS;
|
||
|
PFILE_OBJECT fileObject;
|
||
|
PIO_STACK_LOCATION irpStack;
|
||
|
PDEVICE_EXTENSION deviceExtension;
|
||
|
|
||
|
USBPRINT_KdPrint2 (("USBPRINT.SYS: entering USBPRINT_Create\n"));
|
||
|
USBPRINT_IncrementIoCount(DeviceObject);
|
||
|
deviceExtension = DeviceObject->DeviceExtension;
|
||
|
|
||
|
if (deviceExtension->IsChildDevice==TRUE) {
|
||
|
ntStatus = STATUS_NOT_SUPPORTED;
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
IoCompleteRequest (Irp,IO_NO_INCREMENT);
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|
||
|
if (deviceExtension->AcceptingRequests == FALSE) {
|
||
|
ntStatus = STATUS_DELETE_PENDING;
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
IoCompleteRequest (Irp,IO_NO_INCREMENT);
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
USBPRINT_FdoRequestWake(deviceExtension);
|
||
|
|
||
|
irpStack = IoGetCurrentIrpStackLocation (Irp);
|
||
|
fileObject = irpStack->FileObject;
|
||
|
// fscontext is null for device
|
||
|
fileObject->FsContext = NULL;
|
||
|
deviceExtension->OpenCnt++;
|
||
|
ntStatus = STATUS_SUCCESS;
|
||
|
Irp->IoStatus.Status = ntStatus;
|
||
|
Irp->IoStatus.Information = 0;
|
||
|
IoCompleteRequest (Irp,IO_NO_INCREMENT);
|
||
|
USBPRINT_DecrementIoCount(DeviceObject);
|
||
|
USBPRINT_KdPrint2 (("USBPRINT.SYS: exit USBPRINT_Create %x\n", ntStatus));
|
||
|
return ntStatus;
|
||
|
}
|
||
|
|
||
|
|