windows-nt/Source/XPSP1/NT/printscan/print/drivers/kernelmode/usbprint/ocrw.c
2020-09-26 16:20:57 +08:00

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;
}