mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1668 lines
54 KiB
1668 lines
54 KiB
/*****************************************************************************
|
|
@doc INT EXT
|
|
******************************************************************************
|
|
* $ProjectName: $
|
|
* $ProjectRevision: $
|
|
*-----------------------------------------------------------------------------
|
|
* $Source: z:/pr/cmbp0/sw/cmbp0.ms/rcs/cmbp0wdm.c $
|
|
* $Revision: 1.11 $
|
|
*-----------------------------------------------------------------------------
|
|
* $Author: WFrischauf $
|
|
*-----------------------------------------------------------------------------
|
|
* History: see EOF
|
|
*-----------------------------------------------------------------------------
|
|
*
|
|
* Copyright © 2000 OMNIKEY AG
|
|
******************************************************************************/
|
|
|
|
#include <cmbp0wdm.h>
|
|
|
|
#ifdef NT4
|
|
#include <cmbp0nt4.h>
|
|
#else
|
|
#include <cmbp0pnp.h>
|
|
#endif
|
|
|
|
#include <cmbp0scr.h>
|
|
#include <cmbp0log.h>
|
|
|
|
BOOLEAN DeviceSlot[CMMOB_MAX_DEVICE];
|
|
|
|
// this is a list of our supported data rates
|
|
ULONG SupportedDataRates[] = { 9600, 19200, 38400, 76800, 115200,
|
|
153600, 192000, 307200};
|
|
|
|
// this is a list of our supported clock frequencies
|
|
ULONG SupportedCLKFrequencies[] = { 4000, 8000};
|
|
|
|
|
|
/*****************************************************************************
|
|
DriverEntry:
|
|
entry function of the driver. setup the callbacks for the OS and try to
|
|
initialize a device object for every device in the system
|
|
|
|
Arguments:
|
|
DriverObject context of the driver
|
|
RegistryPath path to the registry entry for the driver
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
STATUS_UNSUCCESSFUL
|
|
******************************************************************************/
|
|
NTSTATUS DriverEntry(
|
|
PDRIVER_OBJECT DriverObject,
|
|
PUNICODE_STRING RegistryPath
|
|
)
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
ULONG ulDevice;
|
|
|
|
|
|
//#if DBG
|
|
// SmartcardSetDebugLevel(DEBUG_ALL);
|
|
//#endif
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DriverEntry: Enter - %s %s\n",DRIVER_NAME,__DATE__,__TIME__));
|
|
|
|
//
|
|
// tell the system our entry points
|
|
//
|
|
DriverObject->MajorFunction[IRP_MJ_CREATE] = CMMOB_CreateClose;
|
|
DriverObject->MajorFunction[IRP_MJ_CLOSE] = CMMOB_CreateClose;
|
|
DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = CMMOB_DeviceIoControl;
|
|
DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = CMMOB_SystemControl;
|
|
DriverObject->MajorFunction[IRP_MJ_CLEANUP] = CMMOB_Cleanup;
|
|
DriverObject->DriverUnload = CMMOB_UnloadDriver;
|
|
|
|
#ifndef NT4
|
|
DriverObject->MajorFunction[IRP_MJ_PNP] = CMMOB_PnPDeviceControl;
|
|
DriverObject->MajorFunction[IRP_MJ_POWER] = CMMOB_PowerDeviceControl;
|
|
DriverObject->DriverExtension->AddDevice = CMMOB_AddDevice;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DriverEntry: PnP Version\n",DRIVER_NAME));
|
|
#else
|
|
//
|
|
// try to initialize all devices
|
|
//
|
|
for (ulDevice = 0; ulDevice < CMMOB_MAX_DEVICE; ulDevice++)
|
|
{
|
|
//
|
|
// create and start all connected devices
|
|
//
|
|
if (CMMOB_CreateAndStartDevice(DriverObject) != STATUS_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// report a failure if no device was found
|
|
//
|
|
if (DriverObject->DeviceObject == NULL)
|
|
{
|
|
NTStatus = STATUS_UNSUCCESSFUL;
|
|
|
|
SmartcardLogError(DriverObject,
|
|
CMMOB_NO_DEVICE_FOUND,
|
|
NULL,
|
|
0);
|
|
}
|
|
|
|
#endif
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DriverEntry: Exit %x\n",DRIVER_NAME,NTStatus));
|
|
|
|
return NTStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
Trys to read the reader name from the registry
|
|
|
|
Arguments:
|
|
DriverObject context of call
|
|
SmartcardExtension ptr to smartcard extension
|
|
|
|
Return Value:
|
|
none
|
|
|
|
******************************************************************************/
|
|
VOID CMMOB_SetVendorAndIfdName(
|
|
IN PDEVICE_OBJECT PhysicalDeviceObject,
|
|
IN PSMARTCARD_EXTENSION SmartcardExtension
|
|
)
|
|
{
|
|
|
|
RTL_QUERY_REGISTRY_TABLE parameters[3];
|
|
UNICODE_STRING vendorNameU;
|
|
ANSI_STRING vendorNameA;
|
|
UNICODE_STRING ifdTypeU;
|
|
ANSI_STRING ifdTypeA;
|
|
HANDLE regKey = NULL;
|
|
|
|
RtlZeroMemory (parameters, sizeof(parameters));
|
|
RtlZeroMemory (&vendorNameU, sizeof(vendorNameU));
|
|
RtlZeroMemory (&vendorNameA, sizeof(vendorNameA));
|
|
RtlZeroMemory (&ifdTypeU, sizeof(ifdTypeU));
|
|
RtlZeroMemory (&ifdTypeA, sizeof(ifdTypeA));
|
|
|
|
try
|
|
{
|
|
//
|
|
// try to read the reader name from the registry
|
|
// if that does not work, we will use the default
|
|
// (hardcoded) name
|
|
//
|
|
if (IoOpenDeviceRegistryKey(PhysicalDeviceObject,
|
|
PLUGPLAY_REGKEY_DEVICE,
|
|
KEY_READ,
|
|
®Key) != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!SetVendorAndIfdName: IoOpenDeviceRegistryKey failed\n",DRIVER_NAME));
|
|
leave;
|
|
}
|
|
|
|
parameters[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
|
|
parameters[0].Name = L"VendorName";
|
|
parameters[0].EntryContext = &vendorNameU;
|
|
parameters[0].DefaultType = REG_SZ;
|
|
parameters[0].DefaultData = &vendorNameU;
|
|
parameters[0].DefaultLength = 0;
|
|
|
|
parameters[1].Flags = RTL_QUERY_REGISTRY_DIRECT;
|
|
parameters[1].Name = L"IfdType";
|
|
parameters[1].EntryContext = &ifdTypeU;
|
|
parameters[1].DefaultType = REG_SZ;
|
|
parameters[1].DefaultData = &ifdTypeU;
|
|
parameters[1].DefaultLength = 0;
|
|
|
|
if (RtlQueryRegistryValues(RTL_REGISTRY_HANDLE,
|
|
(PWSTR) regKey,
|
|
parameters,
|
|
NULL,
|
|
NULL) != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!SetVendorAndIfdName: RtlQueryRegistryValues failed\n",DRIVER_NAME));
|
|
leave;
|
|
}
|
|
|
|
if (RtlUnicodeStringToAnsiString(&vendorNameA,&vendorNameU,TRUE) != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!SetVendorAndIfdName: RtlUnicodeStringToAnsiString failed\n",DRIVER_NAME));
|
|
leave;
|
|
}
|
|
|
|
if (RtlUnicodeStringToAnsiString(&ifdTypeA,&ifdTypeU,TRUE) != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!SetVendorAndIfdName: RtlUnicodeStringToAnsiString failed\n",DRIVER_NAME));
|
|
leave;
|
|
}
|
|
|
|
if (vendorNameA.Length == 0 ||
|
|
vendorNameA.Length > MAXIMUM_ATTR_STRING_LENGTH ||
|
|
ifdTypeA.Length == 0 ||
|
|
ifdTypeA.Length > MAXIMUM_ATTR_STRING_LENGTH)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!SetVendorAndIfdName: vendor name or ifdtype not found or to long\n",DRIVER_NAME));
|
|
leave;
|
|
}
|
|
|
|
RtlCopyMemory(SmartcardExtension->VendorAttr.VendorName.Buffer,
|
|
vendorNameA.Buffer,
|
|
vendorNameA.Length);
|
|
SmartcardExtension->VendorAttr.VendorName.Length = vendorNameA.Length;
|
|
|
|
RtlCopyMemory(SmartcardExtension->VendorAttr.IfdType.Buffer,
|
|
ifdTypeA.Buffer,
|
|
ifdTypeA.Length);
|
|
SmartcardExtension->VendorAttr.IfdType.Length = ifdTypeA.Length;
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!SetVendorAndIfdName: overwritting vendor name and ifdtype\n",DRIVER_NAME));
|
|
|
|
}
|
|
|
|
finally
|
|
{
|
|
if (vendorNameU.Buffer != NULL)
|
|
{
|
|
RtlFreeUnicodeString(&vendorNameU);
|
|
}
|
|
if (vendorNameA.Buffer != NULL)
|
|
{
|
|
RtlFreeAnsiString(&vendorNameA);
|
|
}
|
|
if (ifdTypeU.Buffer != NULL)
|
|
{
|
|
RtlFreeUnicodeString(&ifdTypeU);
|
|
}
|
|
if (ifdTypeA.Buffer != NULL)
|
|
{
|
|
RtlFreeAnsiString(&ifdTypeA);
|
|
}
|
|
if (regKey != NULL)
|
|
{
|
|
ZwClose (regKey);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
creates a new device object for the driver, allocates & initializes all
|
|
neccessary structures (i.e. SmartcardExtension & ReaderExtension).
|
|
|
|
Arguments:
|
|
DriverObject context of call
|
|
DeviceObject ptr to the created device object
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
STATUS_INSUFFICIENT_RESOURCES
|
|
NTStatus returned by smclib.sys
|
|
******************************************************************************/
|
|
NTSTATUS CMMOB_CreateDevice(
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
IN PDEVICE_OBJECT PhysicalDeviceObject,
|
|
OUT PDEVICE_OBJECT *DeviceObject
|
|
)
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
NTSTATUS RegStatus;
|
|
RTL_QUERY_REGISTRY_TABLE ParamTable[2];
|
|
UNICODE_STRING RegistryPath;
|
|
UNICODE_STRING RegistryValue;
|
|
WCHAR szRegValue[256];
|
|
UNICODE_STRING DeviceName;
|
|
UNICODE_STRING Tmp;
|
|
WCHAR Buffer[64];
|
|
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!CreateDevice: Enter\n",DRIVER_NAME ));
|
|
|
|
try
|
|
{
|
|
ULONG ulDeviceInstance;
|
|
PDEVICE_EXTENSION DeviceExtension;
|
|
PREADER_EXTENSION ReaderExtension;
|
|
PSMARTCARD_EXTENSION SmartcardExtension;
|
|
|
|
*DeviceObject = NULL;
|
|
|
|
for (ulDeviceInstance = 0; ulDeviceInstance < CMMOB_MAX_DEVICE; ulDeviceInstance++)
|
|
{
|
|
if (DeviceSlot[ulDeviceInstance] == FALSE)
|
|
{
|
|
DeviceSlot[ulDeviceInstance] = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ulDeviceInstance == CMMOB_MAX_DEVICE)
|
|
{
|
|
NTStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
leave;
|
|
}
|
|
|
|
//
|
|
// construct the device name
|
|
//
|
|
DeviceName.Buffer = Buffer;
|
|
DeviceName.MaximumLength = sizeof(Buffer);
|
|
DeviceName.Length = 0;
|
|
RtlInitUnicodeString(&Tmp,CARDMAN_MOBILE_DEVICE_NAME);
|
|
RtlCopyUnicodeString(&DeviceName,&Tmp);
|
|
DeviceName.Buffer[(DeviceName.Length)/sizeof(WCHAR)-1] = L'0' + (WCHAR)ulDeviceInstance;
|
|
|
|
//
|
|
// Create the device object
|
|
//
|
|
NTStatus = IoCreateDevice(DriverObject,
|
|
sizeof(DEVICE_EXTENSION),
|
|
&DeviceName,
|
|
FILE_DEVICE_SMARTCARD,
|
|
0,
|
|
TRUE,
|
|
DeviceObject);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DriverObject,
|
|
CMMOB_INSUFFICIENT_RESOURCES,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
|
|
//
|
|
// tell the OS that we supposed to do buffered io
|
|
//
|
|
(*DeviceObject)->Flags |= DO_BUFFERED_IO;
|
|
|
|
#ifndef NT4
|
|
// this is necessary, that power routine is called at IRQL_PASSIVE
|
|
(*DeviceObject)->Flags |= DO_POWER_PAGABLE;
|
|
// tells the IO Manager initialization is done
|
|
(*DeviceObject)->Flags &= ~DO_DEVICE_INITIALIZING;
|
|
#endif
|
|
//
|
|
// set up the device extension.
|
|
//
|
|
DeviceExtension = (*DeviceObject)->DeviceExtension;
|
|
RtlZeroMemory( DeviceExtension, sizeof( DEVICE_EXTENSION ));
|
|
|
|
SmartcardExtension = &DeviceExtension->SmartcardExtension;
|
|
|
|
// used for synchronise access to lIoCount
|
|
KeInitializeSpinLock(&DeviceExtension->SpinLockIoCount);
|
|
|
|
// Used for stop / start notification
|
|
KeInitializeEvent(&DeviceExtension->ReaderStarted,
|
|
NotificationEvent,
|
|
FALSE);
|
|
|
|
// Used for update thread notification after hibernation
|
|
KeInitializeEvent(&DeviceExtension->CanRunUpdateThread,
|
|
NotificationEvent,
|
|
TRUE);
|
|
|
|
//
|
|
// allocate the reader extension
|
|
//
|
|
ReaderExtension = ExAllocatePool(NonPagedPool,sizeof( READER_EXTENSION ));
|
|
if (ReaderExtension == NULL)
|
|
{
|
|
SmartcardLogError(DriverObject,
|
|
CMMOB_INSUFFICIENT_RESOURCES,
|
|
NULL,
|
|
0);
|
|
NTStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
leave;
|
|
}
|
|
|
|
RtlZeroMemory( ReaderExtension, sizeof( READER_EXTENSION ));
|
|
|
|
SmartcardExtension->ReaderExtension = ReaderExtension;
|
|
|
|
// ----------------------------------------------
|
|
// initialize mutex
|
|
// ----------------------------------------------
|
|
KeInitializeMutex(&SmartcardExtension->ReaderExtension->CardManIOMutex,0L);
|
|
|
|
//
|
|
// enter correct version of the lib
|
|
//
|
|
SmartcardExtension->Version = SMCLIB_VERSION;
|
|
|
|
//
|
|
// setup smartcard extension - callback's
|
|
//
|
|
|
|
SmartcardExtension->ReaderFunction[RDF_CARD_POWER] = CMMOB_CardPower;
|
|
SmartcardExtension->ReaderFunction[RDF_TRANSMIT] = CMMOB_Transmit;
|
|
SmartcardExtension->ReaderFunction[RDF_SET_PROTOCOL] = CMMOB_SetProtocol;
|
|
SmartcardExtension->ReaderFunction[RDF_CARD_TRACKING] = CMMOB_CardTracking;
|
|
SmartcardExtension->ReaderFunction[RDF_IOCTL_VENDOR] = CMMOB_IoCtlVendor;
|
|
|
|
|
|
//
|
|
// setup smartcard extension - vendor attribute
|
|
//
|
|
|
|
// default values
|
|
RtlCopyMemory(SmartcardExtension->VendorAttr.VendorName.Buffer,
|
|
CMMOB_VENDOR_NAME,sizeof(CMMOB_VENDOR_NAME));
|
|
SmartcardExtension->VendorAttr.VendorName.Length = sizeof(CMMOB_VENDOR_NAME);
|
|
|
|
RtlCopyMemory(SmartcardExtension->VendorAttr.IfdType.Buffer,
|
|
CMMOB_PRODUCT_NAME,sizeof(CMMOB_PRODUCT_NAME));
|
|
SmartcardExtension->VendorAttr.IfdType.Length = sizeof(CMMOB_PRODUCT_NAME);
|
|
|
|
|
|
// try to overwrite with registry values
|
|
CMMOB_SetVendorAndIfdName(PhysicalDeviceObject, SmartcardExtension);
|
|
|
|
|
|
SmartcardExtension->VendorAttr.UnitNo = ulDeviceInstance;
|
|
SmartcardExtension->VendorAttr.IfdVersion.VersionMajor = CMMOB_MAJOR_VERSION;
|
|
SmartcardExtension->VendorAttr.IfdVersion.VersionMinor = CMMOB_MINOR_VERSION;
|
|
SmartcardExtension->VendorAttr.IfdVersion.BuildNumber = CMMOB_BUILD_NUMBER;
|
|
SmartcardExtension->VendorAttr.IfdSerialNo.Length = 0;
|
|
|
|
|
|
//
|
|
// setup smartcard extension - reader capabilities
|
|
//
|
|
SmartcardExtension->ReaderCapabilities.SupportedProtocols = SCARD_PROTOCOL_T0 | SCARD_PROTOCOL_T1;
|
|
SmartcardExtension->ReaderCapabilities.ReaderType = SCARD_READER_TYPE_PCMCIA;
|
|
SmartcardExtension->ReaderCapabilities.MechProperties = 0;
|
|
SmartcardExtension->ReaderCapabilities.Channel = 0;
|
|
|
|
// set supported frequencies
|
|
SmartcardExtension->ReaderCapabilities.CLKFrequency.Default = 4000; //not used if CLKFrequenciesSupported is supplied
|
|
SmartcardExtension->ReaderCapabilities.CLKFrequency.Max = 8000; //not used if CLKFrequenciesSupported is supplied
|
|
SmartcardExtension->ReaderCapabilities.CLKFrequenciesSupported.Entries =
|
|
sizeof(SupportedCLKFrequencies) / sizeof(SupportedCLKFrequencies[0]);
|
|
SmartcardExtension->ReaderCapabilities.CLKFrequenciesSupported.List =
|
|
SupportedCLKFrequencies;
|
|
|
|
// set supported baud rates
|
|
SmartcardExtension->ReaderCapabilities.DataRate.Default = 9600; //not used if DataRatesSupported is supplied
|
|
SmartcardExtension->ReaderCapabilities.DataRate.Max = 307200; //not used if DataRatesSupported is supplied
|
|
SmartcardExtension->ReaderCapabilities.DataRatesSupported.Entries =
|
|
sizeof(SupportedDataRates) / sizeof(SupportedDataRates[0]);
|
|
SmartcardExtension->ReaderCapabilities.DataRatesSupported.List =
|
|
SupportedDataRates;
|
|
|
|
// maximum buffer size
|
|
SmartcardExtension->ReaderCapabilities.MaxIFSD = 254;
|
|
|
|
//
|
|
// Current state of the reader
|
|
//
|
|
SmartcardExtension->ReaderCapabilities.CurrentState = SCARD_UNKNOWN;
|
|
SmartcardExtension->ReaderExtension->ulOldCardState = UNKNOWN;
|
|
SmartcardExtension->ReaderExtension->ulNewCardState = UNKNOWN;
|
|
|
|
SmartcardExtension->ReaderExtension->ulFWVersion = 100;
|
|
|
|
//
|
|
// Initialization of buffers
|
|
//
|
|
SmartcardExtension->SmartcardRequest.BufferSize = MIN_BUFFER_SIZE;
|
|
SmartcardExtension->SmartcardReply.BufferSize = MIN_BUFFER_SIZE;
|
|
|
|
NTStatus = SmartcardInitialize(SmartcardExtension);
|
|
|
|
SmartcardExtension->ReaderExtension->ReaderPowerState = PowerReaderWorking;
|
|
SmartcardExtension->ReaderExtension->CardParameters.bStopBits=2;
|
|
SmartcardExtension->ReaderExtension->CardParameters.fSynchronousCard=FALSE;
|
|
SmartcardExtension->ReaderExtension->CardParameters.fInversRevers=FALSE;
|
|
SmartcardExtension->ReaderExtension->CardParameters.bClockFrequency=4;
|
|
SmartcardExtension->ReaderExtension->CardParameters.fT0Mode=FALSE;
|
|
SmartcardExtension->ReaderExtension->CardParameters.fT0Write=FALSE;
|
|
SmartcardExtension->ReaderExtension->fReadCIS = FALSE;
|
|
SmartcardExtension->ReaderExtension->bPreviousFlags1 = 0;
|
|
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DriverObject,
|
|
CMMOB_INSUFFICIENT_RESOURCES,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
|
|
//
|
|
// tell the lib our device object & create symbolic link
|
|
//
|
|
SmartcardExtension->OsData->DeviceObject = *DeviceObject;
|
|
|
|
|
|
#ifndef NT4
|
|
// W2000 is till now the only OS which supports WDM version 1.10
|
|
// So check this to determine if we have an Plug&Play able resource manager
|
|
// Otherwise it is Windows 98 / ME
|
|
if (IoIsWdmVersionAvailable (1,10))
|
|
{
|
|
DeviceExtension->OSVersion = OS_Windows2000;
|
|
}
|
|
else
|
|
{
|
|
// Read HKLM\Software\Microsoft\Windows\CurrentVersion\VersionNumber
|
|
// to determine the version of Windows (95, 98, 98SE and ME)
|
|
|
|
RegistryValue.Length=0;
|
|
RegistryValue.MaximumLength=sizeof(szRegValue);
|
|
RegistryValue.Buffer=szRegValue;
|
|
|
|
RtlZeroMemory(ParamTable,sizeof(ParamTable));
|
|
|
|
ParamTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
|
|
ParamTable[0].Name = L"VersionNumber";
|
|
ParamTable[0].EntryContext = &RegistryValue;
|
|
ParamTable[0].DefaultType = REG_SZ;
|
|
ParamTable[0].DefaultData = &RegistryValue;
|
|
ParamTable[0].DefaultLength = RegistryValue.MaximumLength;
|
|
|
|
RtlInitUnicodeString(&RegistryPath,
|
|
L"\\Registry\\Machine\\Software\\Microsoft\\Windows\\CurrentVersion");
|
|
|
|
|
|
RegStatus = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE,
|
|
RegistryPath.Buffer,
|
|
ParamTable,
|
|
NULL,
|
|
NULL);
|
|
|
|
if (RegStatus == STATUS_SUCCESS)
|
|
{
|
|
ULONG ulMajorVersion;
|
|
ULONG ulMinorVersion;
|
|
ULONG ulBuild;
|
|
ULONG ulStrLength;
|
|
ULONG ulStrPos;
|
|
ULONG ulStrPosPrevious;
|
|
|
|
ulStrLength=RegistryValue.Length;
|
|
ulStrPos=0;
|
|
|
|
ulStrPosPrevious=ulStrPos;
|
|
while (szRegValue[ulStrPos] != L'.' &&
|
|
szRegValue[ulStrPos] != L'\0' &&
|
|
ulStrPos != ulStrLength/sizeof(WCHAR))
|
|
{
|
|
ulStrPos ++;
|
|
}
|
|
szRegValue[ulStrPos] = L'\0';
|
|
RegistryValue.Length= (USHORT) ((ulStrPos-ulStrPosPrevious)*sizeof(WCHAR));
|
|
RtlUnicodeStringToInteger(&RegistryValue,10,&ulMajorVersion);
|
|
|
|
ulStrPos++;
|
|
if (ulStrPos < ulStrLength/sizeof(WCHAR))
|
|
{
|
|
ulStrPosPrevious=ulStrPos;
|
|
while (szRegValue[ulStrPos] != L'.' &&
|
|
szRegValue[ulStrPos] != L'\0' &&
|
|
ulStrPos != ulStrLength)
|
|
{
|
|
ulStrPos ++;
|
|
}
|
|
szRegValue[ulStrPos] = L'\0';
|
|
RegistryValue.Length= (USHORT) ((ulStrPos-ulStrPosPrevious)*sizeof(WCHAR));
|
|
RegistryValue.Buffer=&szRegValue[ulStrPosPrevious];
|
|
RtlUnicodeStringToInteger(&RegistryValue,10,&ulMinorVersion);
|
|
|
|
ulStrPos++;
|
|
if (ulStrPos < ulStrLength/sizeof(WCHAR))
|
|
{
|
|
ulStrPosPrevious=ulStrPos;
|
|
while (szRegValue[ulStrPos] != L'.' &&
|
|
szRegValue[ulStrPos] != L'\0' &&
|
|
ulStrPos != ulStrLength)
|
|
{
|
|
ulStrPos ++;
|
|
}
|
|
szRegValue[ulStrPos] = L'\0';
|
|
RegistryValue.Length= (USHORT) ((ulStrPos-ulStrPosPrevious)*sizeof(WCHAR));
|
|
RegistryValue.Buffer=&szRegValue[ulStrPosPrevious];
|
|
RtlUnicodeStringToInteger(&RegistryValue,10,&ulBuild);
|
|
}
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: MS Windows Version %li.%li.%li\n",DRIVER_NAME,
|
|
ulMajorVersion,ulMinorVersion,ulBuild));
|
|
|
|
if (ulMajorVersion > 4 ||
|
|
(ulMajorVersion == 4 &&
|
|
ulMinorVersion > 10 ))
|
|
{
|
|
DeviceExtension->OSVersion = OS_WindowsME;
|
|
}
|
|
else if (ulMajorVersion == 4 &&
|
|
ulMinorVersion == 10 )
|
|
{
|
|
if (ulBuild < 2222)
|
|
{
|
|
DeviceExtension->OSVersion = OS_Windows98;
|
|
}
|
|
else
|
|
{
|
|
DeviceExtension->OSVersion = OS_Windows98SE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DeviceExtension->OSVersion = OS_Windows95;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
// assume it's Windows 98 -> no standby mode
|
|
DeviceExtension->OSVersion = OS_Windows98;
|
|
}
|
|
|
|
|
|
}
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: Operating system is %s\n",DRIVER_NAME,
|
|
szOSVersion[DeviceExtension->OSVersion]));
|
|
|
|
|
|
if (DeviceExtension->OSVersion == OS_Windows2000)
|
|
{
|
|
if (DeviceExtension->PnPDeviceName.Buffer == NULL)
|
|
{
|
|
// ----------------------------------------------
|
|
// register our new device
|
|
// ----------------------------------------------
|
|
NTStatus = IoRegisterDeviceInterface(PhysicalDeviceObject,
|
|
&SmartCardReaderGuid,
|
|
NULL,
|
|
&DeviceExtension->PnPDeviceName);
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: PnPDeviceName.Buffer = %lx\n",DRIVER_NAME,
|
|
DeviceExtension->PnPDeviceName.Buffer));
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: PnPDeviceName.BufferLength = %lx\n",DRIVER_NAME,
|
|
DeviceExtension->PnPDeviceName.Length));
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: IoRegisterDeviceInterface returned=%lx\n",DRIVER_NAME,NTStatus));
|
|
}
|
|
else
|
|
{
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: Interface already exists\n",DRIVER_NAME));
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// ----------------------------------------------
|
|
// create symbolic link
|
|
// ----------------------------------------------
|
|
|
|
NTStatus = SmartcardCreateLink(&DeviceExtension->LinkDeviceName,&DeviceName);
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!CreateDevice: SmartcardCreateLink returned=%lx\n",DRIVER_NAME,NTStatus));
|
|
}
|
|
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DriverObject,
|
|
CMMOB_INSUFFICIENT_RESOURCES,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
|
|
}
|
|
|
|
finally
|
|
{
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
CMMOB_UnloadDevice(*DeviceObject);
|
|
*DeviceObject = NULL;
|
|
}
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!CreateDevice: Exit %x\n",DRIVER_NAME,NTStatus ));
|
|
return NTStatus;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
get the actual configuration from the passed FullResourceDescriptor
|
|
and initializes the reader hardware
|
|
|
|
Note:
|
|
for an NT 4.00 build the resources must be translated by the HAL
|
|
|
|
Arguments:
|
|
DeviceObject context of call
|
|
FullResourceDescriptor actual configuration of the reader
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
NTStatus returned from the HAL (NT 4.00 only )
|
|
NTStatus returned by LowLevel routines
|
|
******************************************************************************/
|
|
NTSTATUS CMMOB_StartDevice(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PCM_FULL_RESOURCE_DESCRIPTOR FullResourceDescriptor
|
|
)
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialDescriptor;
|
|
PDEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
PSMARTCARD_EXTENSION SmartcardExtension = &DeviceExtension->SmartcardExtension;
|
|
PREADER_EXTENSION ReaderExtension = SmartcardExtension->ReaderExtension;
|
|
ULONG ulCount;
|
|
ULONG ulCISIndex;
|
|
UCHAR bTupleCode[2];
|
|
UCHAR bFirmware[2];
|
|
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartDevice: Enter\n",DRIVER_NAME));
|
|
|
|
//
|
|
// Get the number of resources we need
|
|
//
|
|
ulCount = FullResourceDescriptor->PartialResourceList.Count;
|
|
|
|
PartialDescriptor = FullResourceDescriptor->PartialResourceList.PartialDescriptors;
|
|
|
|
//
|
|
// parse all partial descriptors
|
|
//
|
|
while (ulCount--)
|
|
{
|
|
switch (PartialDescriptor->Type)
|
|
{
|
|
#ifdef MEMORYCARD
|
|
case CmResourceTypeMemory:
|
|
{
|
|
//
|
|
// 0 - memory, 1 - IO
|
|
//
|
|
#ifdef NT4
|
|
ULONG AddressSpace = 0;
|
|
BOOLEAN fTranslated;
|
|
PHYSICAL_ADDRESS PhysicalAddress;
|
|
#endif
|
|
|
|
|
|
//
|
|
// Get Mem-length
|
|
//
|
|
ReaderExtension->ulMemWindow = PartialDescriptor->u.Memory.Length;
|
|
ASSERT(PartialDescriptor->u.Memory.Length >= 4096);
|
|
|
|
//
|
|
// Get Mem-base
|
|
//
|
|
#ifndef NT4
|
|
ReaderExtension->pMemBase = MmMapIoSpace (PartialDescriptor->u.Memory.Start,
|
|
PartialDescriptor->u.Memory.Length,
|
|
FALSE);
|
|
DeviceExtension->fUnMapMem = TRUE;
|
|
#else
|
|
//
|
|
// let the hal translate the address
|
|
//
|
|
fTranslated = HalTranslateBusAddress(FullResourceDescriptor->InterfaceType,
|
|
FullResourceDescriptor->BusNumber,
|
|
PartialDescriptor->u.Memory.Start,
|
|
&AddressSpace,
|
|
&PhysicalAddress);
|
|
|
|
if (fTranslated == TRUE && AddressSpace == 0)
|
|
{
|
|
ReaderExtension->pMemBase = MmMapIoSpace(PhysicalAddress,
|
|
PartialDescriptor->u.Memory.Length,
|
|
FALSE);
|
|
DeviceExtension->fUnMapMem = TRUE;
|
|
}
|
|
else
|
|
{
|
|
ReaderExtension->pMemBase = (PVOID) PhysicalAddress.LowPart;
|
|
}
|
|
#endif
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartDevice: MemBase = %lxh\n",DRIVER_NAME,ReaderExtension->pMemBase));
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
#ifdef IOCARD
|
|
case CmResourceTypePort:
|
|
{
|
|
#ifdef NT4
|
|
//
|
|
// 0 - memory, 1 - IO
|
|
//
|
|
ULONG AddressSpace = 1;
|
|
BOOLEAN fTranslated;
|
|
PHYSICAL_ADDRESS PhysicalAddress;
|
|
#endif
|
|
|
|
//
|
|
// Get IO-length
|
|
//
|
|
ReaderExtension->ulIoWindow = PartialDescriptor->u.Port.Length;
|
|
ASSERT(PartialDescriptor->u.Port.Length >= 8);
|
|
|
|
//
|
|
// Get IO-base
|
|
//
|
|
#ifndef NT4
|
|
|
|
#ifndef _WIN64
|
|
ReaderExtension->pIoBase = (PVOID) PartialDescriptor->u.Port.Start.LowPart;
|
|
#else
|
|
ReaderExtension->pIoBase = (PVOID) PartialDescriptor->u.Port.Start.QuadPart;
|
|
#endif
|
|
|
|
#else
|
|
//
|
|
// let the hal translate the address
|
|
//
|
|
fTranslated = HalTranslateBusAddress(FullResourceDescriptor->InterfaceType,
|
|
FullResourceDescriptor->BusNumber,
|
|
PartialDescriptor->u.Port.Start,
|
|
&AddressSpace,
|
|
&PhysicalAddress);
|
|
|
|
if (fTranslated == TRUE && AddressSpace == 0)
|
|
{
|
|
ReaderExtension->pIoBase = MmMapIoSpace(PhysicalAddress,
|
|
PartialDescriptor->u.Port.Length,
|
|
FALSE);
|
|
DeviceExtension->fUnMapMem = TRUE;
|
|
}
|
|
else
|
|
{
|
|
ReaderExtension->pIoBase = (PVOID) PhysicalAddress.LowPart;
|
|
}
|
|
#endif
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartDevice: IoBase = %lxh\n",DRIVER_NAME,ReaderExtension->pIoBase));
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
|
|
default:
|
|
break;
|
|
}
|
|
PartialDescriptor++;
|
|
}
|
|
|
|
try
|
|
{
|
|
|
|
//
|
|
// Base initialized ?
|
|
//
|
|
#ifdef MEMORYCARD
|
|
if (ReaderExtension->pMemBase == NULL)
|
|
#endif
|
|
#ifdef IOCARD
|
|
if (ReaderExtension->pIoBase == NULL)
|
|
#endif
|
|
{
|
|
|
|
#ifndef NT4
|
|
//
|
|
// under NT 4.0 the failure of this fct for the second reader
|
|
// means there is only one device
|
|
//
|
|
SmartcardLogError(DeviceObject,
|
|
CMMOB_ERROR_MEM_PORT,
|
|
NULL,
|
|
0);
|
|
#endif
|
|
NTStatus = STATUS_INSUFFICIENT_RESOURCES;
|
|
leave;
|
|
}
|
|
|
|
// initialize base addresses
|
|
#ifdef MEMORYCARD
|
|
ReaderExtension->pbCISBase= (PUCHAR) ReaderExtension->pMemBase;
|
|
ReaderExtension->pbRegsBase= ReaderExtension->pbCISBase + 0x400;
|
|
ReaderExtension->pbDataBase= ReaderExtension->pbCISBase + 0x800;
|
|
#endif
|
|
#ifdef IOCARD
|
|
ReaderExtension->pbRegsBase= (PUCHAR) ReaderExtension->pIoBase;
|
|
#endif
|
|
|
|
NTStatus=CMMOB_ResetReader (ReaderExtension);
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!DEBUG_DRIVER: ResetReader retval = %x\n",DRIVER_NAME, NTStatus));
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DeviceObject,
|
|
CMMOB_CANT_INITIALIZE_READER,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
|
|
#ifdef IOCARD
|
|
//
|
|
// read firmware version from CIS
|
|
//
|
|
ReaderExtension->fReadCIS=TRUE;
|
|
ReaderExtension->fTActive=TRUE;
|
|
NTStatus=CMMOB_SetFlags1 (ReaderExtension);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DeviceObject,
|
|
CMMOB_CANT_INITIALIZE_READER,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
|
|
ulCISIndex = 0;
|
|
do
|
|
{
|
|
NTStatus=CMMOB_ReadBuffer(ReaderExtension, ulCISIndex, 2, bTupleCode);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
leave;
|
|
}
|
|
if (bTupleCode[0] == 0x15)
|
|
{
|
|
// this is the version tuple
|
|
// read firmware version
|
|
NTStatus=CMMOB_ReadBuffer(ReaderExtension, ulCISIndex+2, 2, bFirmware);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
leave;
|
|
}
|
|
SmartcardExtension->ReaderExtension->ulFWVersion = 100*(ULONG)bFirmware[0]+bFirmware[1];
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartDevice: Firmware version = %li\n",
|
|
DRIVER_NAME, SmartcardExtension->ReaderExtension->ulFWVersion));
|
|
}
|
|
ulCISIndex += bTupleCode[1] + 2;
|
|
}
|
|
while (bTupleCode[1] != 0 &&
|
|
bTupleCode[0] != 0x15 &&
|
|
bTupleCode[0] != 0xFF &&
|
|
ulCISIndex < CMMOB_MAX_CIS_SIZE);
|
|
|
|
ReaderExtension->fReadCIS=FALSE;
|
|
ReaderExtension->fTActive=FALSE;
|
|
NTStatus=CMMOB_SetFlags1 (ReaderExtension);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
SmartcardLogError(DeviceObject,
|
|
CMMOB_CANT_INITIALIZE_READER,
|
|
NULL,
|
|
0);
|
|
leave;
|
|
}
|
|
#endif
|
|
|
|
|
|
//
|
|
// start update thread
|
|
//
|
|
NTStatus = CMMOB_StartCardTracking(DeviceObject);
|
|
|
|
// signal that the reader has been started (again)
|
|
KeSetEvent(&DeviceExtension->ReaderStarted, 0, FALSE);
|
|
|
|
#ifndef NT4
|
|
if (DeviceExtension->OSVersion == OS_Windows2000)
|
|
{
|
|
NTStatus = IoSetDeviceInterfaceState(&DeviceExtension->PnPDeviceName,TRUE);
|
|
}
|
|
#endif
|
|
|
|
}
|
|
finally
|
|
{
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
CMMOB_StopDevice(DeviceObject);
|
|
}
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartDevice: Exit %x\n",DRIVER_NAME,NTStatus));
|
|
return NTStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
Unmap the IO port
|
|
|
|
Arguments:
|
|
DeviceObject context of call
|
|
|
|
Return Value:
|
|
void
|
|
******************************************************************************/
|
|
VOID
|
|
CMMOB_StopDevice(
|
|
PDEVICE_OBJECT DeviceObject
|
|
)
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension;
|
|
|
|
if (DeviceObject == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!StopDevice: Enter\n",DRIVER_NAME ));
|
|
|
|
DeviceExtension = DeviceObject->DeviceExtension;
|
|
KeClearEvent(&DeviceExtension->ReaderStarted);
|
|
|
|
//
|
|
// stop update thread
|
|
//
|
|
CMMOB_StopCardTracking(DeviceObject);
|
|
|
|
// power down the card for saftey reasons
|
|
if (DeviceExtension->SmartcardExtension.ReaderExtension->ulOldCardState == POWERED)
|
|
{
|
|
// we have to wait for the mutex before
|
|
KeWaitForSingleObject(&DeviceExtension->SmartcardExtension.ReaderExtension->CardManIOMutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
CMMOB_PowerOffCard(&DeviceExtension->SmartcardExtension);
|
|
KeReleaseMutex(&DeviceExtension->SmartcardExtension.ReaderExtension->CardManIOMutex,
|
|
FALSE);
|
|
}
|
|
|
|
|
|
//
|
|
// unmap ports
|
|
//
|
|
if (DeviceExtension->fUnMapMem)
|
|
{
|
|
#ifdef MEMORYCARD
|
|
MmUnmapIoSpace(DeviceExtension->SmartcardExtension.ReaderExtension->pMemBase,
|
|
DeviceExtension->SmartcardExtension.ReaderExtension->ulMemWindow);
|
|
#endif
|
|
#ifdef IOCARD
|
|
MmUnmapIoSpace(DeviceExtension->SmartcardExtension.ReaderExtension->pIoBase,
|
|
DeviceExtension->SmartcardExtension.ReaderExtension->ulIoWindow);
|
|
#endif
|
|
|
|
DeviceExtension->fUnMapMem = FALSE;
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!StopDevice: Exit\n",DRIVER_NAME ));
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
close connections to smclib.sys and the pcmcia driver, delete symbolic
|
|
link and mark the slot as unused.
|
|
|
|
|
|
Arguments:
|
|
DeviceObject device to unload
|
|
|
|
Return Value:
|
|
void
|
|
******************************************************************************/
|
|
VOID CMMOB_UnloadDevice(
|
|
PDEVICE_OBJECT DeviceObject
|
|
)
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension;
|
|
|
|
if (DeviceObject == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!UnloadDevice: Enter\n",DRIVER_NAME ));
|
|
|
|
DeviceExtension = DeviceObject->DeviceExtension;
|
|
|
|
//
|
|
// It is possible that someone yanks out the pcmcia card when the
|
|
// resource manager still has a connection to the device.
|
|
// We need this flag to know that the device has been removed.
|
|
//
|
|
DeviceExtension->fDeviceRemoved = TRUE;
|
|
|
|
ASSERT(DeviceExtension->SmartcardExtension.VendorAttr.UnitNo < CMMOB_MAX_DEVICE);
|
|
|
|
//
|
|
// Mark this slot as available
|
|
//
|
|
DeviceSlot[DeviceExtension->SmartcardExtension.VendorAttr.UnitNo] = FALSE;
|
|
|
|
//
|
|
// report to the lib that the device will be unloaded
|
|
//
|
|
if (DeviceExtension->SmartcardExtension.OsData != NULL)
|
|
{
|
|
//
|
|
// finish pending tracking requests
|
|
//
|
|
CMMOB_CompleteCardTracking (&DeviceExtension->SmartcardExtension);
|
|
}
|
|
|
|
#ifdef NT4
|
|
// for WDM driver the remove lock is acquired in the PnP routine
|
|
// so in case of NT4 we have to acquire it her
|
|
SmartcardAcquireRemoveLock(&DeviceExtension->SmartcardExtension);
|
|
#endif
|
|
|
|
// Wait until we can safely unload the device
|
|
SmartcardReleaseRemoveLockAndWait(&DeviceExtension->SmartcardExtension);
|
|
|
|
SmartcardExit(&DeviceExtension->SmartcardExtension);
|
|
|
|
#ifndef NT4
|
|
if (DeviceExtension->OSVersion == OS_Windows2000)
|
|
{
|
|
//
|
|
// disable interface
|
|
//
|
|
if (DeviceExtension->PnPDeviceName.Buffer != NULL)
|
|
{
|
|
// disable our device so no one can open it
|
|
IoSetDeviceInterfaceState(&DeviceExtension->PnPDeviceName,FALSE);
|
|
RtlFreeUnicodeString(&DeviceExtension->PnPDeviceName);
|
|
DeviceExtension->PnPDeviceName.Buffer = NULL;
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
//
|
|
// Delete the symbolic link of the smart card reader
|
|
//
|
|
if (DeviceExtension->LinkDeviceName.Buffer != NULL)
|
|
{
|
|
NTSTATUS NTStatus;
|
|
|
|
NTStatus = IoDeleteSymbolicLink(&DeviceExtension->LinkDeviceName);
|
|
//
|
|
// we continue even if an error occurs
|
|
//
|
|
ASSERT(NTStatus == STATUS_SUCCESS);
|
|
|
|
RtlFreeUnicodeString(&DeviceExtension->LinkDeviceName);
|
|
DeviceExtension->LinkDeviceName.Buffer = NULL;
|
|
}
|
|
}
|
|
|
|
if (DeviceExtension->SmartcardExtension.ReaderExtension != NULL)
|
|
{
|
|
ExFreePool(DeviceExtension->SmartcardExtension.ReaderExtension);
|
|
DeviceExtension->SmartcardExtension.ReaderExtension = NULL;
|
|
}
|
|
|
|
//
|
|
// Detach from the pcmcia driver
|
|
// Under NT 4.0 we did not attach to the pcmcia driver
|
|
//
|
|
if (DeviceExtension->AttachedDeviceObject)
|
|
{
|
|
IoDetachDevice(DeviceExtension->AttachedDeviceObject);
|
|
DeviceExtension->AttachedDeviceObject = NULL;
|
|
}
|
|
|
|
//
|
|
// delete the device object
|
|
//
|
|
IoDeleteDevice(DeviceObject);
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!UnloadDevice: Exit\n",DRIVER_NAME ));
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
CMMOB_UnloadDriver:
|
|
unloads all devices for a given driver object
|
|
|
|
Arguments:
|
|
DriverObject context of driver
|
|
|
|
Return Value:
|
|
void
|
|
******************************************************************************/
|
|
VOID CMMOB_UnloadDriver(
|
|
PDRIVER_OBJECT DriverObject
|
|
)
|
|
{
|
|
PCM_FULL_RESOURCE_DESCRIPTOR FullResourceDescriptor;
|
|
ULONG ulSizeOfResources;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!UnloadDriver: Enter\n",DRIVER_NAME ));
|
|
|
|
#ifdef NT4
|
|
if (DriverObject->DeviceObject != NULL)
|
|
{
|
|
|
|
ulSizeOfResources = sizeof(CM_FULL_RESOURCE_DESCRIPTOR) +
|
|
sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR);
|
|
|
|
//
|
|
// allocate memory for the resource descriptor
|
|
//
|
|
FullResourceDescriptor = ExAllocatePool(PagedPool,ulSizeOfResources);
|
|
if (FullResourceDescriptor == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
RtlZeroMemory(FullResourceDescriptor, ulSizeOfResources);
|
|
|
|
//
|
|
// unload all device objects of that driver object
|
|
//
|
|
while (DriverObject->DeviceObject)
|
|
{
|
|
CMMOB_StopDevice(DriverObject->DeviceObject);
|
|
CMMOB_ReportResources(DriverObject, FullResourceDescriptor);
|
|
CMMOB_UnloadDevice(DriverObject->DeviceObject);
|
|
}
|
|
|
|
//
|
|
// free local resources
|
|
//
|
|
ExFreePool(FullResourceDescriptor);
|
|
}
|
|
#endif
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
( "%s!UnloadDriver: Exit\n",DRIVER_NAME ));
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
CMMOB_CreateClose:
|
|
allowes only one open process a time
|
|
|
|
Arguments:
|
|
DeviceObject context of device
|
|
Irp context of call
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
STATUS_DEVICE_BUSY
|
|
******************************************************************************/
|
|
NTSTATUS CMMOB_CreateClose(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
NTSTATUS NTStatus = STATUS_SUCCESS;
|
|
PDEVICE_EXTENSION DeviceExtension;
|
|
PSMARTCARD_EXTENSION SmartcardExtension;
|
|
PIO_STACK_LOCATION IrpStack;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!CreateClose: Enter ",DRIVER_NAME));
|
|
|
|
DeviceExtension = DeviceObject->DeviceExtension;
|
|
SmartcardExtension = &DeviceExtension->SmartcardExtension;
|
|
IrpStack = IoGetCurrentIrpStackLocation( Irp );
|
|
|
|
//
|
|
// dispatch major function
|
|
//
|
|
switch (IrpStack->MajorFunction)
|
|
{
|
|
case IRP_MJ_CREATE:
|
|
SmartcardDebug(DEBUG_IOCTL,("%s!CreateClose: IRP_MJ_CREATE\n",DRIVER_NAME));
|
|
if (DeviceExtension->fRemovePending)
|
|
{
|
|
NTStatus = STATUS_DEVICE_BUSY;
|
|
}
|
|
else
|
|
{
|
|
if (InterlockedIncrement(&DeviceExtension->lOpenCount) > 1)
|
|
{
|
|
InterlockedDecrement(&DeviceExtension->lOpenCount);
|
|
NTStatus = STATUS_ACCESS_DENIED;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case IRP_MJ_CLOSE:
|
|
SmartcardDebug(DEBUG_IOCTL,("%s!CreateClose: IRP_MJ_CLOSE\n",DRIVER_NAME));
|
|
if (InterlockedDecrement(&DeviceExtension->lOpenCount) < 0)
|
|
{
|
|
InterlockedIncrement(&DeviceExtension->lOpenCount);
|
|
}
|
|
break;
|
|
|
|
|
|
default:
|
|
//
|
|
// unrecognized command
|
|
//
|
|
SmartcardDebug(DEBUG_IOCTL,("unexpected IRP\n"));
|
|
NTStatus = STATUS_INVALID_DEVICE_REQUEST;
|
|
break;
|
|
}
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
Irp->IoStatus.Status = NTStatus;
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!CreateClose: Exit %x\n",DRIVER_NAME,NTStatus));
|
|
|
|
return NTStatus;
|
|
}
|
|
|
|
NTSTATUS CMMOB_SystemControl(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
DeviceExtension = DeviceObject->DeviceExtension;
|
|
|
|
IoSkipCurrentIrpStackLocation(Irp);
|
|
status = IoCallDriver(DeviceExtension->AttachedDeviceObject, Irp);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
CMMOB_DeviceIoControl:
|
|
all IRP's requiring IO are queued to the StartIo routine, other requests
|
|
are served immediately
|
|
|
|
Arguments:
|
|
DeviceObject context of device
|
|
Irp context of call
|
|
|
|
Return Value:
|
|
STATUS_SUCCESS
|
|
STATUS_PENDING
|
|
******************************************************************************/
|
|
NTSTATUS CMMOB_DeviceIoControl(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
{
|
|
NTSTATUS NTStatus;
|
|
PDEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
KIRQL irql;
|
|
PIO_STACK_LOCATION irpSL;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DeviceIoControl: Enter\n",DRIVER_NAME));
|
|
|
|
irpSL = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
|
|
#if DBG
|
|
switch (irpSL->Parameters.DeviceIoControl.IoControlCode)
|
|
{
|
|
case IOCTL_SMARTCARD_EJECT:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_EJECT"));
|
|
break;
|
|
case IOCTL_SMARTCARD_GET_ATTRIBUTE:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_GET_ATTRIBUTE"));
|
|
break;
|
|
case IOCTL_SMARTCARD_GET_LAST_ERROR:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_GET_LAST_ERROR"));
|
|
break;
|
|
case IOCTL_SMARTCARD_GET_STATE:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_GET_STATE"));
|
|
break;
|
|
case IOCTL_SMARTCARD_IS_ABSENT:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_IS_ABSENT"));
|
|
break;
|
|
case IOCTL_SMARTCARD_IS_PRESENT:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_IS_PRESENT"));
|
|
break;
|
|
case IOCTL_SMARTCARD_POWER:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_POWER"));
|
|
break;
|
|
case IOCTL_SMARTCARD_SET_ATTRIBUTE:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_SET_ATTRIBUTE"));
|
|
break;
|
|
case IOCTL_SMARTCARD_SET_PROTOCOL:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_SET_PROTOCOL"));
|
|
break;
|
|
case IOCTL_SMARTCARD_SWALLOW:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_SWALLOW"));
|
|
break;
|
|
case IOCTL_SMARTCARD_TRANSMIT:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "IOCTL_SMARTCARD_TRANSMIT"));
|
|
break;
|
|
default:
|
|
SmartcardDebug(DEBUG_IOCTL,
|
|
("%s!DeviceIoControl: %s\n", DRIVER_NAME, "Vendor specific or unexpected IOCTL"));
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
KeAcquireSpinLock(&DeviceExtension->SpinLockIoCount, &irql);
|
|
if (DeviceExtension->lIoCount == 0)
|
|
{
|
|
KeReleaseSpinLock(&DeviceExtension->SpinLockIoCount, irql);
|
|
|
|
NTStatus = KeWaitForSingleObject(&DeviceExtension->ReaderStarted,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
ASSERT(NTStatus == STATUS_SUCCESS);
|
|
|
|
KeAcquireSpinLock(&DeviceExtension->SpinLockIoCount, &irql);
|
|
}
|
|
ASSERT(DeviceExtension->lIoCount >= 0);
|
|
DeviceExtension->lIoCount++;
|
|
KeReleaseSpinLock(&DeviceExtension->SpinLockIoCount, irql);
|
|
|
|
// Can't accept a new io request if:
|
|
if (DeviceExtension->fDeviceRemoved == TRUE ||
|
|
//flag set when processing IRP_MN_REMOVE_DEVICE
|
|
DeviceExtension->fRemovePending == TRUE)
|
|
// flag set when driver has answered success to IRP_MN_QUERY_REMOVE_DEVICE
|
|
{
|
|
NTStatus = STATUS_DELETE_PENDING;
|
|
Irp->IoStatus.Status = NTStatus;
|
|
Irp->IoStatus.Information = 0;
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
KeAcquireSpinLock(&DeviceExtension->SpinLockIoCount, &irql);
|
|
DeviceExtension->lIoCount--;
|
|
ASSERT(DeviceExtension->lIoCount >= 0);
|
|
KeReleaseSpinLock(&DeviceExtension->SpinLockIoCount, irql);
|
|
|
|
return NTStatus;
|
|
}
|
|
|
|
NTStatus = SmartcardAcquireRemoveLock(&DeviceExtension->SmartcardExtension);
|
|
if (NTStatus != STATUS_SUCCESS)
|
|
{
|
|
// the device has been removed. Fail the call
|
|
NTStatus = STATUS_DELETE_PENDING;
|
|
Irp->IoStatus.Status = NTStatus;
|
|
Irp->IoStatus.Information = 0;
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
KeAcquireSpinLock(&DeviceExtension->SpinLockIoCount, &irql);
|
|
DeviceExtension->lIoCount--;
|
|
ASSERT(DeviceExtension->lIoCount >= 0);
|
|
KeReleaseSpinLock(&DeviceExtension->SpinLockIoCount, irql);
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DeviceIoControl: Exit %x\n",DRIVER_NAME,NTStatus));
|
|
|
|
return NTStatus;
|
|
}
|
|
|
|
KeWaitForSingleObject(&DeviceExtension->SmartcardExtension.ReaderExtension->CardManIOMutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
// get current card state
|
|
NTStatus = CMMOB_UpdateCurrentState(&DeviceExtension->SmartcardExtension);
|
|
|
|
NTStatus = SmartcardDeviceControl(&DeviceExtension->SmartcardExtension,Irp);
|
|
|
|
KeReleaseMutex(&DeviceExtension->SmartcardExtension.ReaderExtension->CardManIOMutex,
|
|
FALSE);
|
|
|
|
SmartcardReleaseRemoveLock(&DeviceExtension->SmartcardExtension);
|
|
|
|
KeAcquireSpinLock(&DeviceExtension->SpinLockIoCount, &irql);
|
|
DeviceExtension->lIoCount--;
|
|
ASSERT(DeviceExtension->lIoCount >= 0);
|
|
KeReleaseSpinLock(&DeviceExtension->SpinLockIoCount, irql);
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!DeviceIoControl: Exit %x\n",DRIVER_NAME,NTStatus));
|
|
|
|
return NTStatus;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This routine is called by the I/O system when the calling thread terminates
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Pointer to device object for this miniport
|
|
Irp - IRP involved.
|
|
|
|
Return Value:
|
|
|
|
STATUS_CANCELLED
|
|
******************************************************************************/
|
|
NTSTATUS CMMOB_Cleanup(
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp
|
|
)
|
|
{
|
|
PDEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
PSMARTCARD_EXTENSION SmartcardExtension = &DeviceExtension->SmartcardExtension;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!Cleanup: Enter\n",DRIVER_NAME));
|
|
|
|
if (SmartcardExtension->OsData->NotificationIrp != NULL &&
|
|
// test if there is a pending IRP at all
|
|
SmartcardExtension->ReaderExtension != NULL &&
|
|
// if the device has been removed ReaderExtension == NULL
|
|
DeviceExtension->lOpenCount == 1 )
|
|
// complete card tracking only if this is the the last close call
|
|
// otherwise the card tracking of the resource manager is canceled
|
|
{
|
|
//
|
|
// We need to complete the notification irp
|
|
//
|
|
CMMOB_CompleteCardTracking(SmartcardExtension);
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!Cleanup: Completing IRP %lx\n",DRIVER_NAME,Irp));
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
|
|
IoCompleteRequest(Irp,IO_NO_INCREMENT);
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!Cleanup: Exit\n",DRIVER_NAME));
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
SysDelay:
|
|
performs a required delay. The usage of KeStallExecutionProcessor is
|
|
very nasty, but it happends only if SysDelay is called in the context of
|
|
our DPC routine (which is only called if a card change was detected).
|
|
|
|
For 'normal' IO we have Irql < DISPATCH_LEVEL, so if the reader is polled
|
|
while waiting for response we will not block the entire system
|
|
|
|
Arguments:
|
|
Timeout delay in milli seconds
|
|
|
|
Return Value:
|
|
void
|
|
******************************************************************************/
|
|
VOID SysDelay(
|
|
ULONG Timeout
|
|
)
|
|
{
|
|
LARGE_INTEGER SysTimeout;
|
|
|
|
if (KeGetCurrentIrql() >= DISPATCH_LEVEL)
|
|
{
|
|
ULONG Cnt = 20 * Timeout;
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s! Waiting at IRQL >= DISPATCH_LEVEL %l\n",DRIVER_NAME,Timeout));
|
|
while (Cnt--)
|
|
{
|
|
//
|
|
// KeStallExecutionProcessor: counted in us
|
|
//
|
|
KeStallExecutionProcessor( 50 );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SysTimeout = RtlConvertLongToLargeInteger(Timeout * -10000L);
|
|
//
|
|
// KeDelayExecutionThread: counted in 100 ns
|
|
//
|
|
KeDelayExecutionThread( KernelMode, FALSE, &SysTimeout );
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* History:
|
|
* $Log: cmbp0wdm.c $
|
|
* Revision 1.11 2001/01/22 08:12:22 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.9 2000/09/25 14:24:33 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.8 2000/08/24 09:05:14 TBruendl
|
|
* No comment given
|
|
*
|
|
* Revision 1.7 2000/08/16 16:52:17 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.6 2000/08/09 12:46:01 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.5 2000/07/27 13:53:06 WFrischauf
|
|
* No comment given
|
|
*
|
|
*
|
|
******************************************************************************/
|
|
|
|
|