1048 lines
25 KiB
C
1048 lines
25 KiB
C
/*++
|
|
|
|
Copyright (c) 1997 SCM Microsystems, Inc.
|
|
|
|
Module Name:
|
|
|
|
usbcom.c
|
|
|
|
Abstract:
|
|
|
|
Hardware access functions for USB smartcard reader
|
|
|
|
|
|
Environment:
|
|
|
|
WDM
|
|
|
|
Revision History:
|
|
|
|
PP 01/19/1999 1.01
|
|
PP 12/18/1998 1.00 Initial Version
|
|
|
|
|
|
--*/
|
|
|
|
|
|
#include "common.h"
|
|
#include "stcCmd.h"
|
|
#include "usbcom.h"
|
|
#include "stcusbnt.h"
|
|
|
|
#pragma optimize( "", off )
|
|
|
|
|
|
|
|
NTSTATUS STCtoNT(
|
|
UCHAR ucData[])
|
|
/*++
|
|
|
|
Routine Description:
|
|
Error code translation routine
|
|
|
|
Arguments:
|
|
ucData Error code returned by the STC
|
|
|
|
Return Value:
|
|
Corresponding NT error code
|
|
|
|
--*/
|
|
{
|
|
USHORT usCode = ucData[0]*0x100 +ucData[1];
|
|
NTSTATUS NtStatus;
|
|
|
|
|
|
switch (usCode)
|
|
{
|
|
case 0x9000:
|
|
NtStatus = STATUS_SUCCESS;
|
|
break;
|
|
case 0x5800:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x2000:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x4000:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x64A1:
|
|
NtStatus = STATUS_NO_MEDIA;
|
|
break;
|
|
case 0x64A0:
|
|
NtStatus = STATUS_MEDIA_CHANGED;
|
|
break;
|
|
case 0x6203:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x6300:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x6500:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x6A00:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
case 0x6A80:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
default:
|
|
NtStatus = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
}
|
|
return(NtStatus);
|
|
}
|
|
|
|
|
|
//******************************************************************************
|
|
//
|
|
// UsbSyncCompletionRoutine()
|
|
//
|
|
// Completion routine used by UsbCallUSBD.
|
|
//
|
|
// Signals an Irp completion event and then returns MORE_PROCESSING_REQUIRED
|
|
// to stop further completion of the Irp.
|
|
//
|
|
// If the Irp is one we allocated ourself, DeviceObject is NULL.
|
|
//
|
|
//******************************************************************************
|
|
|
|
NTSTATUS
|
|
UsbSyncCompletionRoutine (
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PVOID Context
|
|
)
|
|
{
|
|
PKEVENT kevent;
|
|
|
|
kevent = (PKEVENT)Context;
|
|
|
|
KeSetEvent(kevent,
|
|
IO_NO_INCREMENT,
|
|
FALSE);
|
|
|
|
return STATUS_MORE_PROCESSING_REQUIRED;
|
|
}
|
|
|
|
//******************************************************************************
|
|
//
|
|
// UsbCallUSBD()
|
|
//
|
|
// Synchronously sends a URB down the device stack. Blocks until the request
|
|
// completes normally or until the request is timed out and cancelled.
|
|
//
|
|
// Must be called at IRQL PASSIVE_LEVEL
|
|
//
|
|
//******************************************************************************
|
|
|
|
NTSTATUS
|
|
UsbCallUSBD (
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PURB Urb
|
|
)
|
|
{
|
|
PDEVICE_EXTENSION deviceExtension;
|
|
KEVENT localevent;
|
|
PIRP irp;
|
|
PIO_STACK_LOCATION nextStack;
|
|
NTSTATUS ntStatus;
|
|
|
|
deviceExtension = DeviceObject->DeviceExtension;
|
|
|
|
// Initialize the event we'll wait on
|
|
//
|
|
KeInitializeEvent(&localevent,
|
|
SynchronizationEvent,
|
|
FALSE);
|
|
|
|
// Allocate the Irp
|
|
//
|
|
irp = IoAllocateIrp(deviceExtension->AttachedPDO->StackSize,
|
|
FALSE);
|
|
|
|
if (irp == NULL)
|
|
{
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
// Set the Irp parameters
|
|
//
|
|
nextStack = IoGetNextIrpStackLocation(irp);
|
|
|
|
nextStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
|
|
|
nextStack->Parameters.DeviceIoControl.IoControlCode =
|
|
IOCTL_INTERNAL_USB_SUBMIT_URB;
|
|
|
|
nextStack->Parameters.Others.Argument1 = Urb;
|
|
|
|
// Set the completion routine, which will signal the event
|
|
//
|
|
IoSetCompletionRoutine(irp,
|
|
UsbSyncCompletionRoutine,
|
|
&localevent,
|
|
TRUE, // InvokeOnSuccess
|
|
TRUE, // InvokeOnError
|
|
TRUE); // InvokeOnCancel
|
|
|
|
// Pass the Irp & Urb down the stack
|
|
//
|
|
ntStatus = IoCallDriver(deviceExtension->AttachedPDO,
|
|
irp);
|
|
|
|
// If the request is pending, block until it completes
|
|
//
|
|
if (ntStatus == STATUS_PENDING)
|
|
{
|
|
LARGE_INTEGER timeout;
|
|
|
|
// We used to wait for 1 second, but that made this timeout longer
|
|
// than the polling period of 500ms. So, if this read failed (e.g.,
|
|
// because of device or USB failure) and timeout, two more worker items
|
|
// would get queued and eventually hundreds of working items would be
|
|
// backed up. By reducing this timeout we have a good chance that this
|
|
// will finish before the next item is queued. 450ms seems a good value.
|
|
//
|
|
timeout.QuadPart = -4500000; // 450ms
|
|
|
|
ntStatus = KeWaitForSingleObject(&localevent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
&timeout);
|
|
|
|
if (ntStatus == STATUS_TIMEOUT)
|
|
{
|
|
ntStatus = STATUS_IO_TIMEOUT;
|
|
|
|
// Cancel the Irp we just sent.
|
|
//
|
|
IoCancelIrp(irp);
|
|
|
|
// And wait until the cancel completes
|
|
//
|
|
KeWaitForSingleObject(&localevent,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
ntStatus = irp->IoStatus.Status;
|
|
}
|
|
}
|
|
|
|
// Done with the Irp, now free it.
|
|
//
|
|
IoFreeIrp(irp);
|
|
|
|
// If the request was not sucessful, delay for 1 second. (Why?)
|
|
//
|
|
if (ntStatus != STATUS_SUCCESS)
|
|
{
|
|
SysDelay(1000);
|
|
}
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbSelectInterfaces(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PUSB_CONFIGURATION_DESCRIPTOR ConfigurationDescriptor)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Initializes an USB reader with (possibly) multiple interfaces;
|
|
This driver only supports one interface (with multiple endpoints).
|
|
|
|
|
|
Arguments:
|
|
DeviceObject - pointer to the device object for this instance of the device.
|
|
|
|
ConfigurationDescriptor - pointer to the USB configuration
|
|
descriptor containing the interface and endpoint
|
|
descriptors.
|
|
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
--*/
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension= DeviceObject->DeviceExtension;
|
|
NTSTATUS NtStatus;
|
|
PURB pUrb = NULL;
|
|
USHORT usSize;
|
|
ULONG ulNumberOfInterfaces, i;
|
|
UCHAR ucNumberOfPipes, ucAlternateSetting, ucMyInterfaceNumber;
|
|
PUSB_INTERFACE_DESCRIPTOR InterfaceDescriptor;
|
|
PUSBD_INTERFACE_INFORMATION InterfaceObject;
|
|
|
|
// This driver only supports one interface, we must parse
|
|
// the configuration descriptor for the interface
|
|
// and remember the pipes.
|
|
//
|
|
|
|
pUrb = USBD_CreateConfigurationRequest(ConfigurationDescriptor, &usSize);
|
|
|
|
if (pUrb)
|
|
{
|
|
//
|
|
// USBD_ParseConfigurationDescriptorEx searches a given configuration
|
|
// descriptor and returns a pointer to an interface that matches the
|
|
// given search criteria. We only support one interface on this device
|
|
//
|
|
InterfaceDescriptor = USBD_ParseConfigurationDescriptorEx(
|
|
ConfigurationDescriptor,
|
|
ConfigurationDescriptor, //search from start of config descriptro
|
|
-1, // interface number not a criteria; we only support one interface
|
|
-1, // not interested in alternate setting here either
|
|
-1, // interface class not a criteria
|
|
-1, // interface subclass not a criteria
|
|
-1); // interface protocol not a criteria
|
|
|
|
ASSERT( InterfaceDescriptor != NULL );
|
|
|
|
InterfaceObject = &pUrb->UrbSelectConfiguration.Interface;
|
|
|
|
for (i = 0; i < InterfaceObject->NumberOfPipes; i++)
|
|
{
|
|
InterfaceObject->Pipes[i].PipeFlags = 0;
|
|
}
|
|
|
|
UsbBuildSelectConfigurationRequest(
|
|
pUrb,
|
|
usSize,
|
|
ConfigurationDescriptor);
|
|
|
|
NtStatus = UsbCallUSBD(DeviceObject, pUrb);
|
|
}
|
|
else
|
|
{
|
|
NtStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
if(NtStatus == STATUS_SUCCESS)
|
|
{
|
|
|
|
// Save the configuration handle for this device
|
|
DeviceExtension->ConfigurationHandle =
|
|
pUrb->UrbSelectConfiguration.ConfigurationHandle;
|
|
|
|
ASSERT(DeviceExtension->Interface == NULL);
|
|
|
|
DeviceExtension->Interface = ExAllocatePool(
|
|
NonPagedPool,
|
|
InterfaceObject->Length
|
|
);
|
|
|
|
if (DeviceExtension->Interface)
|
|
{
|
|
// save a copy of the interface information returned
|
|
RtlCopyMemory(
|
|
DeviceExtension->Interface,
|
|
InterfaceObject,
|
|
InterfaceObject->Length);
|
|
}
|
|
else
|
|
{
|
|
NtStatus = STATUS_NO_MEMORY;
|
|
}
|
|
}
|
|
|
|
if (pUrb)
|
|
{
|
|
ExFreePool(pUrb);
|
|
}
|
|
|
|
return NtStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbConfigureDevice(
|
|
IN PDEVICE_OBJECT DeviceObject)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Initializes a given instance of the device on the USB and
|
|
selects and saves the configuration.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - pointer to the physical device object for this instance of the device.
|
|
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
|
|
--*/
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension= DeviceObject->DeviceExtension;
|
|
NTSTATUS NtStatus;
|
|
PURB pUrb = NULL;
|
|
ULONG ulSize;
|
|
PUSB_CONFIGURATION_DESCRIPTOR ConfigurationDescriptor = NULL;
|
|
|
|
__try {
|
|
|
|
pUrb = ExAllocatePool(
|
|
NonPagedPool,
|
|
sizeof( struct _URB_CONTROL_DESCRIPTOR_REQUEST )
|
|
);
|
|
|
|
if( pUrb == NULL)
|
|
{
|
|
NtStatus = STATUS_NO_MEMORY;
|
|
__leave;
|
|
}
|
|
|
|
// When USB_CONFIGURATION_DESCRIPTOR_TYPE is specified for DescriptorType
|
|
// in a call to UsbBuildGetDescriptorRequest(),
|
|
// all interface, endpoint, class-specific, and vendor-specific descriptors
|
|
// for the configuration also are retrieved.
|
|
// The caller must allocate a buffer large enough to hold all of this
|
|
// information or the data is truncated without error.
|
|
// Therefore the 'siz' set below is just a 'good guess', and we may have to retry
|
|
ulSize = sizeof( USB_CONFIGURATION_DESCRIPTOR ) + 16;
|
|
|
|
// We will break out of this 'retry loop' when UsbBuildGetDescriptorRequest()
|
|
// has a big enough deviceExtension->UsbConfigurationDescriptor buffer not to truncate
|
|
while( 1 )
|
|
{
|
|
ConfigurationDescriptor = ExAllocatePool( NonPagedPool, ulSize );
|
|
|
|
if(ConfigurationDescriptor == NULL)
|
|
{
|
|
NtStatus = STATUS_NO_MEMORY;
|
|
__leave;
|
|
}
|
|
|
|
UsbBuildGetDescriptorRequest(
|
|
pUrb,
|
|
sizeof( struct _URB_CONTROL_DESCRIPTOR_REQUEST ),
|
|
USB_CONFIGURATION_DESCRIPTOR_TYPE,
|
|
0,
|
|
0,
|
|
ConfigurationDescriptor,
|
|
NULL,
|
|
ulSize,
|
|
NULL );
|
|
|
|
NtStatus = UsbCallUSBD( DeviceObject, pUrb );
|
|
|
|
// if we got some data see if it was enough.
|
|
// NOTE: we may get an error in URB because of buffer overrun
|
|
if (pUrb->UrbControlDescriptorRequest.TransferBufferLength == 0 ||
|
|
ConfigurationDescriptor->wTotalLength <= ulSize)
|
|
{
|
|
break;
|
|
}
|
|
|
|
ulSize = ConfigurationDescriptor->wTotalLength;
|
|
ExFreePool(ConfigurationDescriptor);
|
|
ConfigurationDescriptor = NULL;
|
|
}
|
|
|
|
//
|
|
// We have the configuration descriptor for the configuration we want.
|
|
// Now we issue the select configuration command to get
|
|
// the pipes associated with this configuration.
|
|
//
|
|
if(NT_SUCCESS(NtStatus))
|
|
{
|
|
NtStatus = UsbSelectInterfaces(
|
|
DeviceObject,
|
|
ConfigurationDescriptor);
|
|
}
|
|
}
|
|
__finally {
|
|
|
|
if( pUrb )
|
|
{
|
|
ExFreePool( pUrb );
|
|
}
|
|
if( ConfigurationDescriptor )
|
|
{
|
|
ExFreePool( ConfigurationDescriptor );
|
|
}
|
|
}
|
|
|
|
return NtStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbWriteSTCData(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
PUCHAR pucData,
|
|
ULONG ulSize)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
Write data in the STC
|
|
|
|
Arguments:
|
|
ReaderExtension Context of the call
|
|
APDU Buffer to write
|
|
ulAPDULen Length of the buffer to write
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
PUCHAR pucCmd;
|
|
UCHAR ucResponse[3];
|
|
BOOLEAN resend = TRUE;
|
|
LONG Len;
|
|
ULONG Index;
|
|
LONG refLen = (LONG) ulSize;
|
|
ULONG Retries;
|
|
|
|
pucCmd = ExAllocatePool( NonPagedPool, MIN_BUFFER_SIZE);
|
|
|
|
if(pucCmd == NULL)
|
|
{
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ReaderExtension->ulReadBufferLen = 0;
|
|
|
|
// Build the write data command
|
|
Len = refLen;
|
|
Index = 0;
|
|
|
|
while (resend == TRUE)
|
|
{
|
|
if(Len > 62)
|
|
{
|
|
Len = 62;
|
|
resend = TRUE;
|
|
}
|
|
else
|
|
{
|
|
resend = FALSE;
|
|
}
|
|
|
|
*pucCmd = 0xA0;
|
|
*(pucCmd+1) = (UCHAR) Len;
|
|
memcpy( pucCmd + 2, pucData+Index, Len );
|
|
|
|
Retries = USB_WRITE_RETRIES;
|
|
do
|
|
{
|
|
// Send the Write data command
|
|
NTStatus = UsbWrite( ReaderExtension, pucCmd, 2 + Len);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
("%s!UsbWriteSTCData: write error %X \n",
|
|
DRIVER_NAME,
|
|
NTStatus)
|
|
);
|
|
break;
|
|
}
|
|
// Read the response
|
|
NTStatus = UsbRead( ReaderExtension, ucResponse, 3);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
("%s!UsbWriteSTCData: read error %X \n",
|
|
DRIVER_NAME,
|
|
NTStatus)
|
|
);
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
// Test if what we read is really a response to a write
|
|
if(ucResponse[0] != 0xA0)
|
|
{
|
|
NTStatus = STCtoNT(ucResponse);
|
|
}
|
|
}
|
|
} while(( NTStatus != STATUS_SUCCESS ) && --Retries );
|
|
|
|
if( NTStatus != STATUS_SUCCESS )
|
|
break;
|
|
|
|
Index += 62;
|
|
Len = refLen - 62;
|
|
refLen = refLen - 62;
|
|
}
|
|
|
|
ExFreePool( pucCmd );
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbReadSTCData(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
PUCHAR pucData,
|
|
ULONG ulDataLen)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
Read data from the STC
|
|
|
|
Arguments:
|
|
ReaderExtension Context of the call
|
|
ulAPDULen Length of the buffer to write
|
|
pucData Output Buffer
|
|
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
UCHAR ucCmd[1];
|
|
PUCHAR pucResponse;
|
|
int i;
|
|
ULONG ulLenExpected = ulDataLen;
|
|
ULONG Index=0;
|
|
BOOLEAN SendReadCommand = TRUE;
|
|
LARGE_INTEGER Begin;
|
|
LARGE_INTEGER End;
|
|
|
|
pucResponse = ExAllocatePool( NonPagedPool, MIN_BUFFER_SIZE);
|
|
if(pucResponse == NULL)
|
|
{
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
|
|
KeQuerySystemTime( &Begin );
|
|
End = Begin;
|
|
End.QuadPart = End.QuadPart + (LONGLONG)10 * 1000 * ReaderExtension->ReadTimeout;
|
|
|
|
// First let see if we have not already read the data that
|
|
// we need
|
|
if(ReaderExtension->ulReadBufferLen != 0)
|
|
{
|
|
if(ReaderExtension->ulReadBufferLen >= ulLenExpected)
|
|
{
|
|
// all the data that we need are available
|
|
memcpy(pucData,ReaderExtension->ucReadBuffer,ulLenExpected);
|
|
ReaderExtension->ulReadBufferLen = ReaderExtension->ulReadBufferLen - ulLenExpected;
|
|
if(ReaderExtension->ulReadBufferLen != 0)
|
|
{
|
|
memcpy(
|
|
ReaderExtension->ucReadBuffer,
|
|
ReaderExtension->ucReadBuffer+ulLenExpected,
|
|
ReaderExtension->ulReadBufferLen);
|
|
}
|
|
SendReadCommand = FALSE;
|
|
}
|
|
else
|
|
{
|
|
// all the data that we need are not available
|
|
memcpy(pucData,ReaderExtension->ucReadBuffer,ReaderExtension->ulReadBufferLen);
|
|
ulLenExpected = ulLenExpected - ReaderExtension->ulReadBufferLen;
|
|
Index = ReaderExtension->ulReadBufferLen;
|
|
ReaderExtension->ulReadBufferLen = 0;
|
|
SendReadCommand = TRUE;
|
|
}
|
|
}
|
|
while( SendReadCommand == TRUE)
|
|
{
|
|
// Build the Read Register command
|
|
ucCmd[0] = 0xE0;
|
|
|
|
NTStatus = UsbWrite( ReaderExtension, ucCmd, 1);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
("%s!UsbReadSTCData: write error %X \n",
|
|
DRIVER_NAME,
|
|
NTStatus)
|
|
);
|
|
break;
|
|
}
|
|
|
|
NTStatus = UsbRead( ReaderExtension, pucResponse, 64);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
("%s!UsbReadSTCData: read error %X \n",
|
|
DRIVER_NAME,
|
|
NTStatus)
|
|
);
|
|
break;
|
|
}
|
|
// Test if what we read is really a READ DATA frame
|
|
if(*pucResponse != 0xE0)
|
|
{
|
|
if(*pucResponse == 0x64 && *(pucResponse + 1) == 0xA0)
|
|
{
|
|
NTStatus = STATUS_NO_MEDIA;
|
|
}
|
|
else
|
|
{
|
|
NTStatus = STCtoNT(pucResponse);
|
|
}
|
|
break;
|
|
}
|
|
// If there is no data available
|
|
if (*(pucResponse + 1) == 0)
|
|
{
|
|
KeQuerySystemTime( &Begin );
|
|
if(RtlLargeIntegerGreaterThan(End, Begin))
|
|
{
|
|
SendReadCommand = TRUE;
|
|
}
|
|
else
|
|
{
|
|
ReaderExtension->ulReadBufferLen = 0;
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
("%s!UsbReadSTCData: Timeout %X \n",
|
|
DRIVER_NAME,
|
|
STATUS_IO_TIMEOUT));
|
|
NTStatus =STATUS_IO_TIMEOUT;
|
|
break;
|
|
}
|
|
}
|
|
if ((ULONG) *(pucResponse+1) < ulLenExpected)
|
|
{
|
|
memcpy(pucData+Index,pucResponse+2,(ULONG) *(pucResponse+1));
|
|
Index = Index + (ULONG) *(pucResponse+1);
|
|
ulLenExpected = ulLenExpected - (ULONG) *(pucResponse+1);
|
|
SendReadCommand = TRUE;
|
|
}
|
|
else
|
|
{
|
|
SendReadCommand = FALSE;
|
|
memcpy(pucData+Index,pucResponse+2,ulLenExpected);
|
|
|
|
if((ULONG) *(pucResponse+1) > ulLenExpected)
|
|
{
|
|
memcpy(
|
|
ReaderExtension->ucReadBuffer,
|
|
pucResponse+ulLenExpected+2,
|
|
(ULONG) *(pucResponse+1) - ulLenExpected);
|
|
|
|
ReaderExtension->ulReadBufferLen =
|
|
(ULONG) *(pucResponse+1) - ulLenExpected;
|
|
}
|
|
else
|
|
{
|
|
ReaderExtension->ulReadBufferLen = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
ExFreePool( pucResponse );
|
|
return NTStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbWriteSTCRegister(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
UCHAR ucAddress,
|
|
ULONG ulSize,
|
|
PUCHAR pucValue)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
PUCHAR pucCmd;
|
|
UCHAR ucResponse[2];
|
|
|
|
if(ulSize > 16)
|
|
{
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
pucCmd = ExAllocatePool( NonPagedPool, MAX_READ_REGISTER_BUFFER_SIZE);
|
|
if(pucCmd == NULL)
|
|
{
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
ReaderExtension->ulReadBufferLen = 0;
|
|
|
|
// Build the write register command
|
|
*pucCmd = 0x80 | ucAddress;
|
|
*(pucCmd+1) = (UCHAR) ulSize;
|
|
memcpy( pucCmd + 2, pucValue, ulSize );
|
|
|
|
// Send the Write Register command
|
|
NTStatus = UsbWrite( ReaderExtension, pucCmd, 2 + ulSize);
|
|
if (NTStatus == STATUS_SUCCESS)
|
|
{
|
|
// Read the acknowledge
|
|
NTStatus = UsbRead( ReaderExtension, ucResponse, 2);
|
|
if (NTStatus == STATUS_SUCCESS)
|
|
{
|
|
NTStatus = STCtoNT(ucResponse);
|
|
}
|
|
}
|
|
|
|
ExFreePool( pucCmd );
|
|
return NTStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbReadSTCRegister(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
UCHAR ucAddress,
|
|
ULONG ulSize,
|
|
PUCHAR pucValue)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
UCHAR ucCmd[2];
|
|
PUCHAR pucResponse;
|
|
|
|
if(ulSize > 16)
|
|
{
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
pucResponse = ExAllocatePool(
|
|
NonPagedPool,
|
|
MAX_READ_REGISTER_BUFFER_SIZE
|
|
);
|
|
|
|
if(pucResponse == NULL)
|
|
{
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
|
|
// Build the Read Register command
|
|
ucCmd[0] = 0xC0 | ucAddress;
|
|
ucCmd[1] = (UCHAR) ulSize;
|
|
|
|
// Send the Read Register command
|
|
NTStatus = UsbWrite( ReaderExtension, ucCmd, 2);
|
|
if (NTStatus == STATUS_SUCCESS)
|
|
{
|
|
// Read the response from the reader
|
|
NTStatus = UsbRead(
|
|
ReaderExtension,
|
|
pucResponse,
|
|
6
|
|
);
|
|
|
|
if (NTStatus == STATUS_SUCCESS)
|
|
{
|
|
// Test if what we read is really a READ frame
|
|
if(*pucResponse == 0x21)
|
|
{
|
|
if(*(pucResponse + 1) > 16)
|
|
{
|
|
NTStatus = STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
else
|
|
{
|
|
memcpy(
|
|
pucValue,
|
|
pucResponse + 2,
|
|
(ULONG) *(pucResponse + 1)
|
|
);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
NTStatus = STCtoNT(pucResponse);
|
|
}
|
|
}
|
|
}
|
|
|
|
ExFreePool( pucResponse );
|
|
return NTStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbGetFirmwareRevision(
|
|
PREADER_EXTENSION ReaderExtension)
|
|
/*++
|
|
Description:
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
UCHAR ucCmd[1];
|
|
UCHAR ucResponse[4];
|
|
|
|
ucCmd[0] = 0xE1;
|
|
NTStatus = UsbWrite( ReaderExtension, ucCmd, 2 );
|
|
|
|
if( NTStatus == STATUS_SUCCESS )
|
|
{
|
|
ReaderExtension->ReadTimeout = 1000;
|
|
NTStatus = UsbRead( ReaderExtension, ucResponse, 4 );
|
|
|
|
if( NTStatus == STATUS_SUCCESS )
|
|
{
|
|
ReaderExtension->FirmwareMajor = ucResponse[ 2 ];
|
|
ReaderExtension->FirmwareMinor = ucResponse[ 3 ];
|
|
}
|
|
}
|
|
return NTStatus ;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
UsbRead(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
PUCHAR pData,
|
|
ULONG DataLen )
|
|
/*++
|
|
Description:
|
|
Read data on the USB bus
|
|
|
|
Arguments:
|
|
ReaderExtension context of call
|
|
pData ptr to data buffer
|
|
DataLen length of data buffer
|
|
pNBytes number of bytes returned
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
STATUS_BUFFER_TOO_SMALL
|
|
STATUS_UNSUCCESSFUL
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NtStatus = STATUS_SUCCESS;
|
|
PURB pUrb;
|
|
USBD_INTERFACE_INFORMATION* pInterfaceInfo;
|
|
USBD_PIPE_INFORMATION* pPipeInfo;
|
|
PDEVICE_OBJECT DeviceObject = ReaderExtension->DeviceObject;
|
|
PDEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
ULONG ulSize;
|
|
|
|
pInterfaceInfo = DeviceExtension->Interface;
|
|
|
|
ASSERT(pInterfaceInfo != NULL);
|
|
|
|
if (pInterfaceInfo == NULL) {
|
|
|
|
// The device has likely been disconnected during hibernate / stand by
|
|
return STATUS_DEVICE_NOT_CONNECTED;
|
|
}
|
|
|
|
// Read pipe number is 0 on this device
|
|
pPipeInfo = &( pInterfaceInfo->Pipes[ 0 ] );
|
|
|
|
ulSize = sizeof( struct _URB_BULK_OR_INTERRUPT_TRANSFER );
|
|
pUrb = ExAllocatePool( NonPagedPool, ulSize );
|
|
|
|
if(pUrb == NULL)
|
|
{
|
|
return STATUS_NO_MEMORY;
|
|
}
|
|
else
|
|
{
|
|
UsbBuildInterruptOrBulkTransferRequest(
|
|
pUrb,
|
|
(USHORT)ulSize,
|
|
pPipeInfo->PipeHandle,
|
|
pData,
|
|
NULL,
|
|
DataLen,
|
|
USBD_SHORT_TRANSFER_OK,
|
|
NULL
|
|
);
|
|
|
|
NtStatus = UsbCallUSBD( DeviceObject, pUrb );
|
|
ExFreePool( pUrb );
|
|
}
|
|
|
|
return NtStatus;
|
|
}
|
|
|
|
NTSTATUS
|
|
UsbWrite(
|
|
PREADER_EXTENSION ReaderExtension,
|
|
PUCHAR pData,
|
|
ULONG DataLen)
|
|
/*++
|
|
Description:
|
|
Write data on the usb port
|
|
|
|
Arguments:
|
|
ReaderExtension context of call
|
|
pData ptr to data buffer
|
|
DataLen length of data buffer (exclusive LRC!)
|
|
|
|
Return Value:
|
|
return value of
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS NtStatus = STATUS_SUCCESS;
|
|
PURB pUrb;
|
|
USBD_INTERFACE_INFORMATION* pInterfaceInfo;
|
|
USBD_PIPE_INFORMATION* pPipeInfo;
|
|
PDEVICE_OBJECT DeviceObject = ReaderExtension->DeviceObject;
|
|
PDEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
ULONG ulSize;
|
|
|
|
pInterfaceInfo = DeviceExtension->Interface;
|
|
|
|
ASSERT(pInterfaceInfo != NULL);
|
|
|
|
if (pInterfaceInfo == NULL) {
|
|
|
|
// The device has likely been disconnected during hibernate / stand by
|
|
return STATUS_DEVICE_NOT_CONNECTED;
|
|
}
|
|
|
|
// Write pipe number is 1 on this device
|
|
pPipeInfo = &( pInterfaceInfo->Pipes[ 1 ] );
|
|
|
|
ulSize = sizeof( struct _URB_BULK_OR_INTERRUPT_TRANSFER );
|
|
pUrb = ExAllocatePool( NonPagedPool, ulSize );
|
|
if(pUrb == NULL)
|
|
{
|
|
NtStatus = STATUS_NO_MEMORY;
|
|
}
|
|
else
|
|
{
|
|
UsbBuildInterruptOrBulkTransferRequest(
|
|
pUrb,
|
|
(USHORT)ulSize,
|
|
pPipeInfo->PipeHandle,
|
|
pData,
|
|
NULL,
|
|
DataLen,
|
|
USBD_SHORT_TRANSFER_OK,
|
|
NULL );
|
|
|
|
NtStatus = UsbCallUSBD( DeviceObject, pUrb );
|
|
ExFreePool( pUrb );
|
|
}
|
|
return NtStatus;
|
|
}
|