windows-nt/Source/XPSP1/NT/drivers/parallel/parclass/devobj.c

1411 lines
47 KiB
C
Raw Permalink Normal View History

2020-09-26 03:20:57 -05:00
//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1998 - 1999
//
// File: devobj.c
//
//--------------------------------------------------------------------------
// this file contains functions to create, initialize, manage, and destroy ParClass device objects
#include "pch.h"
extern WCHAR ParInt2Wchar[];
VOID
ParMakeClassNameFromPortLptName(
IN PUNICODE_STRING PortSymbolicLinkName,
OUT PUNICODE_STRING ClassName
)
/*
Get LPTx name from ParPort device and use this name to construct the \Device\Parallely name.
y = x-1 (i.e., LPT3 => \Device\Parallel2)
*/
{
NTSTATUS status;
PDEVICE_OBJECT portDeviceObject;
PFILE_OBJECT portDeviceFileObject;
PWSTR portName;
ULONG portNumber;
LONG count;
UNICODE_STRING str;
// Get a pointer to the ParPort device
status = IoGetDeviceObjectPointer(PortSymbolicLinkName, STANDARD_RIGHTS_ALL,
&portDeviceFileObject, &portDeviceObject);
if( !NT_SUCCESS(status) ) {
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - unable to get handle to parport device\n"));
return;
}
// Get LPTx portName from ParPort device
portName = ParGetPortLptName( portDeviceObject );
// Done with handle
ObDereferenceObject( portDeviceFileObject );
// Did we get a portName?
if( 0 == portName ) {
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - unable to get portName from parport device\n"));
return;
}
// Verify that the portname looks like LPTx where x is a number
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - portName = <%S>\n", portName));
if( portName[0] != L'L' || portName[1] != L'P' || portName[2] != L'T' ) {
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - name prefix doesn't look like LPT\n"));
return;
}
// prefix is LPT, check for integer suffix with value > 0
RtlInitUnicodeString( &str, (PWSTR)&portName[3] );
status = RtlUnicodeStringToInteger( &str, 10, &portNumber );
if( !NT_SUCCESS( status ) ) {
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - name suffix doesn't look like an integer\n"));
return;
}
if( portNumber == 0 ) {
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - name suffix == 0 - FAIL - Invalid value\n"));
return;
}
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - LPT name suffix= %d\n", portNumber));
// Build \Device\Parallely name from LPTx name
ParMakeClassNameFromNumber( portNumber-1, ClassName );
ParDump2(PARPNP1, ("devobj::ParMakeClassNameFromPortLptName - portName=<%S> className=<%wZ>\n",
portName, ClassName));
}
// return TRUE if given a pointer to a ParClass PODO, FALSE otherwise
BOOLEAN
ParIsPodo(PDEVICE_OBJECT DevObj) {
PDEVICE_EXTENSION devExt = DevObj->DeviceExtension;
if( !devExt->IsPdo ) {
// this is an FDO
return FALSE;
}
// still here? - this is either a PODO or a PDO
if( devExt->DeviceIdString[0] != 0 ) {
// this device object has a device ID string - It's a PDO
return FALSE;
}
// still here? - this is either a PODO, or a PDO marked "hardware gone" that
// is simply waiting for PnP to send it a REMOVE.
if( devExt->DeviceStateFlags & PAR_DEVICE_HARDWARE_GONE) {
// this is a PDO marked "hardware gone"
return FALSE;
}
// still here? - this is a PODO
return TRUE;
}
#if USE_TEMP_FIX_FOR_MULTIPLE_INTERFACE_ARRIVAL
//
// Temp fix for PnP calling us multiple times for the same interface arrival
//
//
// Return Value: Did we already process an interface arrival for this interface?
//
BOOLEAN
ParIsDuplicateInterfaceArrival(
IN PDEVICE_INTERFACE_CHANGE_NOTIFICATION NotificationStruct,
IN PDEVICE_OBJECT Fdo
)
{
PUNICODE_STRING portSymbolicLinkName = NotificationStruct->SymbolicLinkName;
PDEVICE_EXTENSION fdoExt = Fdo->DeviceExtension;
PDEVICE_OBJECT curDevObj;
PDEVICE_EXTENSION curDevExt;
PAGED_CODE();
ParDump2(PARPNP1, ("Enter ParIsDuplicateInterfaceArrival()\n"));
//
// Find the ParClass PODOs and compare the PortSymbolicLinkName in the
// device extension with the one in the interface arrival notification.
//
curDevObj = fdoExt->ParClassPdo;
while( curDevObj ) {
curDevExt = curDevObj->DeviceExtension;
if( ParIsPodo(curDevObj) ) {
ParDump2(PARPNP1, ("DevObj= %x IS PODO\n",curDevObj) );
ParDump2(PARPNP1, ("Comparing port symbolic link names\n"));
if( RtlEqualUnicodeString(portSymbolicLinkName,&curDevExt->PortSymbolicLinkName, FALSE) ) {
ParDump2(PARPNP1, ("MATCH! - Second Arrival of this Interface\n"));
return TRUE;
} else {
ParDump2(PARPNP1, ("NO match on port symbolic link name for interface arrival - keep searching\n"));
}
} else {
ParDump2(PARPNP1, ("DevObj= %x NOT a PODO\n",curDevObj) );
}
curDevObj = curDevExt->Next;
}
return FALSE;
}
#endif
NTSTATUS
ParPnpNotifyInterfaceChange(
IN PDEVICE_INTERFACE_CHANGE_NOTIFICATION NotificationStruct,
IN PDEVICE_OBJECT Fdo
)
/*++dvdf
Routine Description:
This routine is the PnP "interface change notification" callback routine.
This gets called on a ParPort triggered device interface arrival or removal.
- Interface arrival corresponds to a ParPort device being STARTed
- Interface removal corresponds to a ParPort device being REMOVEd
On arrival:
- Create the LPTx PODO (PlainOldDeviceObject) for legacy/raw port access.
- Query for an EOC (End-Of-Chain) PnP Device, create PDO if device found.
- Enumerate all 1284.3 DC (Daisy Chain) devices attached to the port.
This callback is a NO-OP for interface removal because all ParClass created
P[O]DOs register for PnP EventCategoryTargetDeviceChange callbacks and
use that callback to clean up when their associated ParPort device goes
away.
Arguments:
NotificationStruct - Structure defining the change.
Fdo - pointer to ParClass FDO
(supplied as the "context" when we
registered for this callback)
Return Value:
STATUS_SUCCESS - always, even if something goes wrong
--*/
{
PUNICODE_STRING portSymbolicLinkName = NotificationStruct->SymbolicLinkName;
PDEVICE_OBJECT legacyPodo;
PDEVICE_EXTENSION legacyExt;
// PDEVICE_OBJECT endOfChainPdo;
PDEVICE_OBJECT portDeviceObject;
PFILE_OBJECT portDeviceFileObject;
NTSTATUS status;
BOOLEAN foundNewDevice = FALSE;
// UCHAR dot3DeviceCount;
PAGED_CODE();
//
// Verify that interface class is a ParPort device interface.
//
// Any other InterfaceClassGuid is an error, but let it go since
// it is not fatal to the machine.
//
if( !IsEqualGUID( (LPGUID)&(NotificationStruct->InterfaceClassGuid),
(LPGUID)&GUID_PARALLEL_DEVICE) ) {
ParDump2(PARERRORS, ("ParPnpNotifyInterfaceChange() - ERROR - Bad InterfaceClassGuid\n") );
return STATUS_SUCCESS;
}
//
// This callback is a NO-OP for interface removal.
//
// All ParClass DO's that depend on this interface register for target
// device change PnP notification on the ParPort device associated
// with this interface.
//
// Thus, all such ParClass DO's that depend on this interface will
// have already received target device change notification callbacks
// that the ParPort device associated with this interface is being
// removed prior to the arrival of this interface removal
// notification callback.
//
if(!IsEqualGUID( (LPGUID)&(NotificationStruct->Event),
(LPGUID)&GUID_DEVICE_INTERFACE_ARRIVAL )) {
ParDump2(PARPNP1, ("Interface Removal Notification\n") );
return STATUS_SUCCESS;
}
//
// A ParPort device has STARTed and we are the bus driver for the port.
// Continue...
//
#if USE_TEMP_FIX_FOR_MULTIPLE_INTERFACE_ARRIVAL
//
// Temp fix for PnP calling us multiple times for the same interface arrival
//
if( ParIsDuplicateInterfaceArrival(NotificationStruct, Fdo) ) {
ParDump2(PARERRORS, ("Duplicate Interface Arrival Notification - Returning/Ignoring\n") );
return STATUS_SUCCESS;
}
#endif
//
// Get a pointer to and create a FILE against the ParPort device
//
status = IoGetDeviceObjectPointer(portSymbolicLinkName, STANDARD_RIGHTS_ALL,
&portDeviceFileObject, &portDeviceObject);
if( !NT_SUCCESS(status) ) {
ParDump2(PARERRORS, ("ParPnpNotifyInterfaceChange() - ERROR - unable to get device object port to ParPort device\n") );
return STATUS_SUCCESS;
}
//
// Acquire the ParPort device
//
status = ParAcquirePort(portDeviceObject, NULL);
if( !NT_SUCCESS(status) ) {
ParDump2(PARERRORS, ("ParPnpNotifyInterfaceChange() - ERROR - unable to acquire port\n") );
ObDereferenceObject(portDeviceFileObject);
return STATUS_SUCCESS;
}
//
// Create the Legacy LPTx PODO (PlainOldDeviceObject) for raw port access.
//
legacyPodo = ParCreateLegacyPodo(Fdo, portSymbolicLinkName);
if( !legacyPodo ) {
// If we can't create the legacyPodo, then nothing following will
// succeed, so bail out.
ParDump2(PARERRORS, ("ParPnpNotifyInterfaceChange() - ERROR - unable to create legacyPodo\n") );
ParReleasePort(portDeviceObject);
ObDereferenceObject(portDeviceFileObject);
return STATUS_SUCCESS;
}
//
// SUCCESS - Legacy PODO created
//
legacyExt = legacyPodo->DeviceExtension;
ParDump2(PARPNP1, ("devobj::ParPnpNotifyInterfaceChange - CREATED legacyPODO - <%wZ> <%wZ>\n",
&legacyExt->ClassName, &legacyExt->SymbolicLinkName) );
//
// Legacy PODO - add to list of ParClass created device objects
//
ParAddDevObjToFdoList(legacyPodo);
//
// release port and close our FILE to Parport device (our PODO and PDOs
// have their own FILEs open against parport)
//
ParReleasePort(portDeviceObject);
ObDereferenceObject(portDeviceFileObject);
//
// Tell PnP that we might have some new children
//
{
PDEVICE_EXTENSION fdoExt = Fdo->DeviceExtension;
ParDump2(PARPNP1, ("devobj::ParPnpNotifyInterfaceChange - calling IoInvalidateDeviceRelations(,BusRelations)\n") );
IoInvalidateDeviceRelations(fdoExt->PhysicalDeviceObject, BusRelations);
}
return STATUS_SUCCESS;
}
PDEVICE_OBJECT
ParCreateLegacyPodo(
IN PDEVICE_OBJECT Fdo,
IN PUNICODE_STRING PortSymbolicLinkName
)
/*++dvdf
Routine Description:
This routine creates the LPTx PODO (PlainOldDeviceObject) that is
used for legacy/raw port access.
- create a classname of the form "\Device\ParallelN"
- create device object
- initialize device object and extension
- create symbolic link
- register for PnP TargetDeviceChange notification
Arguments:
Fdo - pointer to ParClass FDO
PortSymbolicLinkName - symbolic link name of the ParPort device
Return Value:
PDEVICE_OBJECT - on success
NULL - otherwise
--*/
{
NTSTATUS status;
UNICODE_STRING className = {0,0,0};
PDEVICE_OBJECT legacyPodo;
PDEVICE_EXTENSION legacyExt;
PDRIVER_OBJECT driverObject = Fdo->DriverObject;
#define PAR_CLASSNAME_OFFSET 8
PAGED_CODE();
//
// Legacy PODO - try to build a \Device\Parallely classname based on the LPTx name
// retrieved from the ParPort device
//
ParMakeClassNameFromPortLptName(PortSymbolicLinkName, &className);
if( !className.Buffer ) {
//
// We failed to construct a ClassName from the Port's
// LPTx name - just make up a name
//
// Use an offset so that the name we make up doesn't collide
// with other ports
//
ParMakeClassNameFromNumber( PAR_CLASSNAME_OFFSET + g_NumPorts++, &className );
if( !className.Buffer ) {
// unable to create class name, bail out
return NULL;
}
}
//
// Legacy PODO - create device object
//
status = ParCreateDevice(driverObject, sizeof(DEVICE_EXTENSION), &className,
FILE_DEVICE_PARALLEL_PORT, 0, TRUE, &legacyPodo);
if( !NT_SUCCESS( status ) ) {
//
// We failed to create a device, if failure was due to a name collision, then
// make up a new classname and try again
//
if( status == STATUS_OBJECT_NAME_COLLISION || status == STATUS_OBJECT_NAME_EXISTS ) {
// name collision - make up a new classname and try again
RtlFreeUnicodeString( &className );
ParMakeClassNameFromNumber( PAR_CLASSNAME_OFFSET + g_NumPorts++, &className );
if( !className.Buffer ) {
// unable to create class name, bail out
return NULL;
}
status = ParCreateDevice(driverObject, sizeof(DEVICE_EXTENSION), &className,
FILE_DEVICE_PARALLEL_PORT, 0, TRUE, &legacyPodo);
}
}
if( !NT_SUCCESS( status ) ) {
// unable to create device object, bail out
RtlFreeUnicodeString(&className);
ParLogError(driverObject, NULL, PhysicalZero, PhysicalZero,
0, 0, 0, 9, STATUS_SUCCESS, PAR_INSUFFICIENT_RESOURCES);
return NULL;
}
legacyExt = legacyPodo->DeviceExtension;
//
// Legacy PODO - initialize device object and extension
//
ParInitCommonDOPre(legacyPodo, Fdo, &className);
status = ParInitLegacyPodo(legacyPodo, PortSymbolicLinkName);
if( !NT_SUCCESS( status ) ) {
// initialization failed, clean up and bail out
ParAcquireListMutexAndKillDeviceObject(Fdo, legacyPodo);
return NULL;
}
//
// Note that we are a PODO in our extension
//
legacyExt->DeviceType = PAR_DEVTYPE_PODO;
ParInitCommonDOPost(legacyPodo);
//
// Legacy PODO - create symbolic link
//
if( legacyExt->SymbolicLinkName.Buffer ) {
status = IoCreateUnprotectedSymbolicLink(&legacyExt->SymbolicLinkName,
&legacyExt->ClassName);
if ( NT_SUCCESS(status) ) {
// note this in our extension for later cleanup
legacyExt->CreatedSymbolicLink = TRUE;
// Write symbolic link map info to the registry.
status = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
(PWSTR)L"PARALLEL PORTS",
legacyExt->ClassName.Buffer,
REG_SZ,
legacyExt->SymbolicLinkName.Buffer,
legacyExt->SymbolicLinkName.Length +
sizeof(WCHAR));
if (!NT_SUCCESS(status)) {
// unable to write map info to registry - continue anyway
ParLogError(legacyPodo->DriverObject, legacyPodo,
legacyExt->OriginalController, PhysicalZero,
0, 0, 0, 6, status, PAR_NO_DEVICE_MAP_CREATED);
}
} else {
// unable to create the symbolic link.
legacyExt->CreatedSymbolicLink = FALSE;
RtlFreeUnicodeString(&legacyExt->SymbolicLinkName);
ParLogError(legacyPodo->DriverObject, legacyPodo,
legacyExt->OriginalController, PhysicalZero,
0, 0, 0, 5, status, PAR_NO_SYMLINK_CREATED);
}
} else {
// extension does not contain a symbolic link name for us to use
legacyExt->CreatedSymbolicLink = FALSE;
}
if( FALSE == legacyExt->CreatedSymbolicLink ) {
// Couldn't create a symbolic Link
// clean up and bail out
ParAcquireListMutexAndKillDeviceObject(Fdo, legacyPodo);
return NULL;
}
//
// Legacy PODO - register for PnP TargetDeviceChange notification
//
status = IoRegisterPlugPlayNotification(EventCategoryTargetDeviceChange,
0,
(PVOID)legacyExt->PortDeviceFileObject,
driverObject,
(PDRIVER_NOTIFICATION_CALLBACK_ROUTINE) ParPnpNotifyTargetDeviceChange,
(PVOID)legacyPodo,
&legacyExt->NotificationHandle);
if( !NT_SUCCESS(status) ) {
// PnP registration for TargetDeviceChange notification failed,
// clean up and bail out
ParAcquireListMutexAndKillDeviceObject(Fdo, legacyPodo);
return NULL;
}
//
// Register for WMI
//
status = ParWmiPdoInitWmi( legacyPodo );
if( !NT_SUCCESS( status ) ) {
// WMI registration failed, clean up and bail out
ParAcquireListMutexAndKillDeviceObject(Fdo, legacyPodo);
return NULL;
} else {
//
// Note in our extension that we have registered for WMI
// so we can clean up later
//
legacyExt->PodoRegForWMI = TRUE;
}
return legacyPodo;
}
VOID
ParInitCommonDOPre(
IN PDEVICE_OBJECT DevObj,
IN PDEVICE_OBJECT Fdo,
IN PUNICODE_STRING ClassName
)
/*++dvdf - code complete - compiles clean - not tested
Routine Description:
This routine contains common initialization code for ParClass
created PDOs and PODOs that should be called before (Pre) the
device object type specific (PDO/PODO) intialization function
is called.
Arguments:
DevObj - points to the DeviceObject to be initialized
Fdo - points to the ParClass FDO
ClassName - points to the ClassName for the PDO/PODO
Return Value:
None - This function can not fail.
--*/
{
PDEVICE_EXTENSION extension;
PAGED_CODE();
// - we use buffered IO
// - force power dispatch at PASSIVE_LEVEL IRQL so we can page driver
// DevObj->Flags |= (DO_BUFFERED_IO | DO_POWER_PAGABLE);
DevObj->Flags |= DO_BUFFERED_IO; // RMT - should also set POWER_PAGABLE
// initialize extension to all zeros
extension = DevObj->DeviceExtension;
RtlZeroMemory(extension, sizeof(DEVICE_EXTENSION));
// used by debugger extension
extension->ExtensionSignature = PARCLASS_EXTENSION_SIGNATURE;
extension->ExtensionSignatureEnd = PARCLASS_EXTENSION_SIGNATURE;
// initialize synchronization and list mechanisms
ExInitializeFastMutex(&extension->OpenCloseMutex);
InitializeListHead(&extension->WorkQueue);
KeInitializeSemaphore(&extension->RequestSemaphore, 0, MAXLONG);
KeInitializeEvent(&extension->PauseEvent, NotificationEvent, TRUE);
// general info
extension->ClassName = *ClassName; // copy the struct
extension->DeviceObject = DevObj;
extension->EndOfChain = TRUE; // override later if this is a
extension->Ieee1284_3DeviceId = DOT3_END_OF_CHAIN_ID; // 1284.3 Daisy Chain device
extension->IsPdo = TRUE; // really means !FDO
extension->ParClassFdo = Fdo;
extension->BusyDelay = 0;
extension->BusyDelayDetermined = FALSE;
// timing constants
extension->TimerStart = PAR_WRITE_TIMEOUT_VALUE;
extension->IdleTimeout.QuadPart -= 250*10*1000; // 250 ms
extension->AbsoluteOneSecond.QuadPart = 10*1000*1000;
extension->OneSecond.QuadPart = -(extension->AbsoluteOneSecond.QuadPart);
// init IEEE 1284 protocol settings
ParInitializeExtension1284Info( extension );
}
NTSTATUS
ParInitLegacyPodo(
IN PDEVICE_OBJECT LegacyPodo,
IN PUNICODE_STRING PortSymbolicLinkName
)
/*++
Routine Description:
This function performs ParClass DeviceObject and DeviceExtension
initialization specific to ParClass Legacy PODOs (Plain Old Device
Objects). A Legacy PODO represents the "raw" parallel port and is
used by legacy drivers to communicate with parallel port connected
devices.
Precondition:
- ParInitCommonPre(...) must be called with this DeviceObject before
this function is called.
Postcondition:
- ParInitCommonPost(...) must be called with this DeviceObject after
this function is called.
On error, this routine defers cleaning up the LegacyPodo DeviceObject
and any pool allocations to the caller.
Arguments:
LegacyPodo - pointer to the legacy device object to initialize
PortSymbolicLinkName - symbolic link name of ParPort device
Return Value:
STATUS_SUCCESS - on success
STATUS_UNSUCCESSFUL - otherwise
--*/
{
NTSTATUS status = STATUS_SUCCESS;
PDEVICE_EXTENSION legacyExt = LegacyPodo->DeviceExtension;
PWSTR buffer = NULL;
PFILE_OBJECT portDeviceFileObject = NULL;
PDEVICE_OBJECT portDeviceObject = NULL;
PAGED_CODE();
//
// Get a pointer to and create a FILE against the ParPort device
//
// - We need a pointer to the ParPort DeviceObject so that we can send
// it IRPs.
//
// - We need a FILE against the DeviceObject to keep the ParPort
// DeviceObject from "going away" while we are using it.
//
// - Having an open FILE against the ParPort DeviceObject will also prevent
// PnP from doing a resource rebalance on the ParPort DeviceObject and
// changing its resources while we are holding pointers to the ParPort
// device's registers. This works because a ParPort DeviceObject fails
// PnP QUERY_STOP IRPs if anyone has an open FILE against it.
//
status = IoGetDeviceObjectPointer(PortSymbolicLinkName,
STANDARD_RIGHTS_ALL,
&portDeviceFileObject,
&portDeviceObject);
if( !NT_SUCCESS(status) ) {
return STATUS_UNSUCCESSFUL;
}
//
// success - save pointer to ParPort DeviceObject and a copy of the
// REFERENCED pointer to the FILE created against the ParPort
// DeviceObject in our extension
//
legacyExt->PortDeviceObject = portDeviceObject;
legacyExt->PortDeviceFileObject = portDeviceFileObject;
//
// Save a copy of the ParPort SymbolicLinkName in our device extension.
//
buffer = ParCreateWideStringFromUnicodeString(PortSymbolicLinkName);
if( !buffer ) {
// unable to copy PortSymbolicLinkName, bail out
return STATUS_UNSUCCESSFUL;
}
// copy ParPort SymbolicLinkName to our device extension
RtlInitUnicodeString(&legacyExt->PortSymbolicLinkName, buffer);
//
// make sure that IRPs sent to us have enough stack locations so that we
// can forward them to ParPort if needed
//
legacyExt->DeviceObject->StackSize = (CHAR)( legacyExt->PortDeviceObject->StackSize + 1 );
//
// Obtain PARALLEL_PORT_INFORMATION and PARALLEL_PNP_INFORMATION from
// the ParPort device and save it in our device extension
//
status = ParGetPortInfoFromPortDevice(legacyExt);
if (!NT_SUCCESS(status)) {
ParLogError(LegacyPodo->DriverObject, LegacyPodo, PhysicalZero, PhysicalZero,
0, 0, 0, 4, status, PAR_CANT_FIND_PORT_DRIVER);
return STATUS_UNSUCCESSFUL;
}
if (legacyExt->OriginalController.HighPart == 0 &&
legacyExt->OriginalController.LowPart == (ULONG_PTR) legacyExt->Controller) {
legacyExt->UsePIWriteLoop = FALSE;
} else {
legacyExt->UsePIWriteLoop = TRUE;
}
return STATUS_SUCCESS;
}
VOID
ParInitCommonDOPost(
IN PDEVICE_OBJECT DevObj
)
/*++dvdf - code complete - compiles clean - not tested
Routine Description:
This routine contains common initialization code for ParClass
created PDOs and PODOs that should be called after (Post) the
device object type specific (PDO/PODO) intialization function
is called.
Arguments:
DevObj - points to the DeviceObject to be initialized
Return Value:
None - This function can not fail.
--*/
{
PAGED_CODE();
// Check the registry for parameter overrides
ParCheckParameters(DevObj->DeviceExtension);
// Tell the IO system that we are ready to receive IRPs
DevObj->Flags &= ~DO_DEVICE_INITIALIZING;
}
PDEVICE_OBJECT
ParDetectCreateEndOfChainPdo(
IN PDEVICE_OBJECT LegacyPodo
)
/*++
Routine Description:
Detect if an EndOfChain device is connected. If so, create a PDO for the
device and add the PDO to the list of ParClass children.
Arguments:
LegacyPodo - pointer to the legacy PODO for the port.
Return Value:
Pointer to the new PDO if device found
NULL otherwise
--*/
{
PDEVICE_OBJECT newPdo = ParDetectCreatePdo( LegacyPodo, DOT3_END_OF_CHAIN_ID, FALSE );
if( newPdo ) {
ParAddDevObjToFdoList( newPdo );
}
return newPdo;
}
PDEVICE_OBJECT
ParDetectCreatePdo(
IN PDEVICE_OBJECT LegacyPodo,
IN UCHAR Dot3Id,
IN BOOLEAN bStlDot3Id
)
/*++
Routine Description:
Detect if there is a 1284 device attached.
If a device is detected then create a PDO to represent the device.
Preconditions:
Caller has acquired the ParPort device
Caller has SELECTed the device
Postconditions:
ParPort device is still acquired
Device is still SELECTed
Arguments:
LegacyPodo - points to the Legacy PODO for the port
Dot3Id - 1284.3 daisy chain id
0..3 is daisy chain device
4 is end of chain device
Return Value:
PDEVICE_OBJECT - on success, points to the PDO we create
NULL - otherwise
--*/
{
PDEVICE_EXTENSION legacyExt = LegacyPodo->DeviceExtension;
PDRIVER_OBJECT driverObject = LegacyPodo->DriverObject;
PDEVICE_OBJECT fdo = legacyExt->ParClassFdo;
UNICODE_STRING className = {0,0,NULL};
NTSTATUS status;
PCHAR deviceIdString = NULL;
ULONG deviceIdLength;
PDEVICE_OBJECT newDevObj = NULL;
PDEVICE_EXTENSION newDevExt;
ULONG idTry = 1;
ULONG maxIdTries = 3;
BOOLEAN useModifiedClassName = FALSE;
UNICODE_STRING modifiedClassName;
UNICODE_STRING suffix;
UNICODE_STRING dash;
WCHAR suffixBuffer[10];
ULONG number = 0;
ULONG maxNumber = 256;
ULONG newLength;
PAGED_CODE();
//
// Query for PnP device
//
while( (NULL == deviceIdString) && (idTry <= maxIdTries) ) {
deviceIdString = Par3QueryDeviceId(legacyExt, NULL, 0, &deviceIdLength, FALSE, bStlDot3Id);
if( NULL == deviceIdString ) {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - no 1284 ID on try %d\n", idTry) );
KeStallExecutionProcessor(1);
++idTry;
} else {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - devIdString=<%s> on try %d\n", deviceIdString, idTry) );
}
}
if( !deviceIdString ) {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - no 1284 ID, bail out\n") );
return NULL;
}
//
// Found PnP Device, create a PDO to represent the device
// - create classname
// - create device object
// - initialize device object and extension
// - create symbolic link
// - register for PnP TargetDeviceChange notification
//
//
// Create a class name of the form \Device\ParallelN,
//
ParMakeDotClassNameFromBaseClassName(&legacyExt->ClassName, Dot3Id, &className);
if( !className.Buffer ) {
// unable to construct ClassName
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - unable to construct ClassName for device\n") );
ExFreePool(deviceIdString);
return NULL;
}
//
// create device object
//
status = ParCreateDevice(driverObject, sizeof(DEVICE_EXTENSION), &className,
FILE_DEVICE_PARALLEL_PORT, 0, TRUE, &newDevObj);
///
if( status == STATUS_OBJECT_NAME_COLLISION ) {
//
// old name is still in use, appending a suffix and try again
//
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - ParCreateDevice FAILED due to name Collision on <%wZ> - retry\n", &className));
useModifiedClassName = TRUE;
suffix.Length = 0;
suffix.MaximumLength = sizeof(suffixBuffer);
suffix.Buffer = suffixBuffer;
RtlInitUnicodeString( &dash, (PWSTR)L"-" );
newLength = className.MaximumLength + 5*sizeof(WCHAR); // L"-XXX" suffix
modifiedClassName.Length = 0;
modifiedClassName.MaximumLength = (USHORT)newLength;
modifiedClassName.Buffer = ExAllocatePool(PagedPool, newLength);
if( NULL == modifiedClassName.Buffer ) {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - ParCreateDevice FAILED - no PagedPool avail\n"));
ExFreePool(deviceIdString);
RtlFreeUnicodeString( &className );
return NULL;
}
while( ( status == STATUS_OBJECT_NAME_COLLISION ) && ( number <= maxNumber ) ) {
status = RtlIntegerToUnicodeString(number, 10, &suffix);
if ( !NT_SUCCESS(status) ) {
ExFreePool(deviceIdString);
RtlFreeUnicodeString( &className );
RtlFreeUnicodeString( &modifiedClassName );
return NULL;
}
RtlCopyUnicodeString( &modifiedClassName, &className );
RtlAppendUnicodeStringToString( &modifiedClassName, &dash );
RtlAppendUnicodeStringToString( &modifiedClassName, &suffix );
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - trying ParCreateDevice with className <%wZ>\n", &modifiedClassName));
status = ParCreateDevice(driverObject, sizeof(DEVICE_EXTENSION), &modifiedClassName,
FILE_DEVICE_PARALLEL_PORT, 0, TRUE, &newDevObj);
if( NT_SUCCESS( status ) ) {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - ParCreateDevice returned SUCCESS with className <%wZ>\n", &modifiedClassName));
} else {
++number;
}
}
}
///
if( useModifiedClassName ) {
// copy modifiedClassName to className
RtlFreeUnicodeString( &className );
className = modifiedClassName;
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - copy useModifiedClassName to className - className=<%wZ>\n", &className));
}
if( !NT_SUCCESS(status) ) {
// unable to create device object, bail out
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - unable to create device object "
"className=<%wZ>, bail out - status=%x\n", &className, status) );
ExFreePool(deviceIdString);
RtlFreeUnicodeString(&className);
ParLogError(fdo->DriverObject, NULL, PhysicalZero, PhysicalZero,
0, 0, 0, 9, STATUS_SUCCESS, PAR_INSUFFICIENT_RESOURCES);
return NULL;
} else {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - device created <%wZ>\n", &className));
}
//
// device object created
//
newDevExt = newDevObj->DeviceExtension;
//
// initialize device object and extension
//
ParInitCommonDOPre(newDevObj, fdo, &className);
status = ParInitPdo(newDevObj, (PUCHAR)deviceIdString, deviceIdLength, LegacyPodo, Dot3Id);
if( !NT_SUCCESS( status ) ) {
// initialization failed, clean up and bail out
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - call to ParInitPdo failed, bail out\n") );
ParAcquireListMutexAndKillDeviceObject(fdo, newDevObj);
return NULL;
}
ParInitCommonDOPost(newDevObj);
//
// create symbolic link
//
if( newDevExt->SymbolicLinkName.Buffer ) {
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - ready to create symlink - SymbolicLinkName <%wZ>, ClassName <%wZ>\n",
&newDevExt->SymbolicLinkName, &newDevExt->ClassName) );
ParDump2(PARPNP1, (" - Length=%hd, MaximumLength=%hd\n", newDevExt->ClassName.Length, newDevExt->ClassName.MaximumLength) );
// doug
ASSERT(newDevExt->ClassName.Length < 100);
PAGED_CODE();
status = IoCreateUnprotectedSymbolicLink(&newDevExt->SymbolicLinkName, &newDevExt->ClassName);
if ( NT_SUCCESS(status) ) {
// note this in our extension for later cleanup
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - SymbolicLinkName -> ClassName = <%wZ> -> <%wZ>\n",
&newDevExt->SymbolicLinkName, &newDevExt->ClassName) );
newDevExt->CreatedSymbolicLink = TRUE;
// Write symbolic link map info to the registry.
status = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
(PWSTR)L"PARALLEL PORTS",
newDevExt->ClassName.Buffer,
REG_SZ,
newDevExt->SymbolicLinkName.Buffer,
newDevExt->SymbolicLinkName.Length +
sizeof(WCHAR));
if (!NT_SUCCESS(status)) {
// unable to write map info to registry - continue anyway
ParLogError(newDevObj->DriverObject, newDevObj,
newDevExt->OriginalController, PhysicalZero,
0, 0, 0, 6, status, PAR_NO_DEVICE_MAP_CREATED);
}
} else {
// unable to create the symbolic link.
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - unable to create SymbolicLink - status = %x\n",status));
newDevExt->CreatedSymbolicLink = FALSE;
RtlFreeUnicodeString(&newDevExt->SymbolicLinkName);
ParLogError(newDevObj->DriverObject, newDevObj,
newDevExt->OriginalController, PhysicalZero,
0, 0, 0, 5, status, PAR_NO_SYMLINK_CREATED);
}
} else {
// extension does not contain a symbolic link name for us to use
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - extension does not contain a symbolic link for us to use\n"));
newDevExt->CreatedSymbolicLink = FALSE;
}
// End-Of-Chain PDO - register for PnP TargetDeviceChange notification
status = IoRegisterPlugPlayNotification(EventCategoryTargetDeviceChange,
0,
(PVOID)newDevExt->PortDeviceFileObject,
driverObject,
(PDRIVER_NOTIFICATION_CALLBACK_ROUTINE) ParPnpNotifyTargetDeviceChange,
(PVOID)newDevObj,
&newDevExt->NotificationHandle);
if( !NT_SUCCESS(status) ) {
// PnP registration for TargetDeviceChange notification failed,
// clean up and bail out
ParDump2(PARPNP1, ("devobj::ParDetectCreatePdo - PnP registration failed, killing PDO\n") );
ParAcquireListMutexAndKillDeviceObject(fdo, newDevObj);
return NULL;
}
return newDevObj;
}
NTSTATUS
ParInitPdo(
IN PDEVICE_OBJECT NewPdo,
IN PUCHAR DeviceIdString,
IN ULONG DeviceIdLength,
IN PDEVICE_OBJECT LegacyPodo,
IN UCHAR Dot3Id
)
{
static WCHAR ParInt2Wchar[] = { L'0', L'1', L'2', L'3', L'4', L'5' };
NTSTATUS status;
PDEVICE_EXTENSION legacyExt = LegacyPodo->DeviceExtension;
PDEVICE_EXTENSION newExt = NewPdo->DeviceExtension;
//
// start with identical initialization as done for Legacy PODO
//
status = ParInitLegacyPodo(NewPdo, &legacyExt->PortSymbolicLinkName);
if( !NT_SUCCESS(status) ) {
return status;
}
//
// Fixup extension if we are a daisy chain device or legacy Zip
//
newExt->Ieee1284Flags = legacyExt->Ieee1284Flags;
if( Dot3Id != DOT3_END_OF_CHAIN_ID ) {
newExt->EndOfChain = FALSE;
newExt->Ieee1284_3DeviceId = Dot3Id;
// NewPdo->Flags &= ~DO_POWER_PAGABLE; // RMT - temp clear bit until ppa, disk, partmgr set it
}
//
// Note that we are a PDO in our extension
//
newExt->DeviceType = PAR_DEVTYPE_PDO;
//
// use PODO's SymbolicLinkName + a ".N" suffix as the PDO's SymbolicLinkName
//
{
UNICODE_STRING newSymLinkName;
USHORT index;
USHORT length = (USHORT)( newExt->SymbolicLinkName.Length + ( 3 * sizeof(WCHAR) ) );
ParDump2(PARPNP1, ("devobj::ParInitPdo - old SymLinkName=%wZ\n", &newExt->SymbolicLinkName) );
RtlInitUnicodeString(&newSymLinkName, NULL);
newSymLinkName.Buffer = ExAllocatePool(PagedPool, length);
if( !newSymLinkName.Buffer ) {
return STATUS_UNSUCCESSFUL;
}
newSymLinkName.Length=0;
newSymLinkName.MaximumLength=length;
RtlCopyUnicodeString(&newSymLinkName, &newExt->SymbolicLinkName);
index = (USHORT) ( (newExt->SymbolicLinkName.Length)/sizeof(WCHAR) );
newSymLinkName.Buffer[index+0] = L'.';
newSymLinkName.Buffer[index+1] = ParInt2Wchar[Dot3Id];
newSymLinkName.Buffer[index+2] = L'\0';
newSymLinkName.Length += (2 * sizeof(WCHAR));
RtlFreeUnicodeString(&newExt->SymbolicLinkName);
newExt->SymbolicLinkName = newSymLinkName;
ParDump2(PARPNP1, ("devobj::ParInitPdo - new SymLinkName=%wZ\n", &newExt->SymbolicLinkName) );
ParDump2(PARPNP1, (" - Length=%hd, MaximumLength=%hd\n", newExt->SymbolicLinkName.Length, newExt->SymbolicLinkName.MaximumLength) );
}
// initialize PnP fields of device extension
{
UCHAR RawString[128];
UCHAR DescriptionString[128];
PUCHAR deviceIdString;
deviceIdString = ExAllocatePool(PagedPool, DeviceIdLength + sizeof(UCHAR) );
if( deviceIdString ) {
RtlCopyMemory(deviceIdString, DeviceIdString, DeviceIdLength);
*(deviceIdString+DeviceIdLength) = 0; // NULL terminate
}
RtlZeroMemory( RawString, sizeof(RawString) );
RtlZeroMemory( DescriptionString, sizeof(DescriptionString) );
status = ParPnpGetId(DeviceIdString, BusQueryDeviceID, RawString, DescriptionString);
if (NT_SUCCESS(status)) {
RtlCopyMemory(newExt->DeviceIdString, RawString, strlen((const PCHAR)RawString));
RtlCopyMemory(newExt->DeviceDescription, DescriptionString, strlen((const PCHAR)DescriptionString));
if( deviceIdString ) {
ParDetectDot3DataLink(newExt, deviceIdString);
}
}
if( deviceIdString ) {
ExFreePool(deviceIdString);
}
}
return status;
}
PWSTR
ParGetPortLptName(
IN PDEVICE_OBJECT PortDeviceObject
)
// return 0 on any error
{
NTSTATUS status;
PARALLEL_PNP_INFORMATION pnpInfo;
// Get Parallel Pnp Info from ParPort device, return PortName
status = ParBuildSendInternalIoctl(IOCTL_INTERNAL_GET_PARALLEL_PNP_INFO,
PortDeviceObject, NULL, 0,
&pnpInfo, sizeof(PARALLEL_PNP_INFORMATION), NULL);
if( NT_SUCCESS(status) ) {
return (PWSTR)(pnpInfo.PortName);
} else {
ParDump2(PARERRORS, ("devobj::ParGetPortLptName - FAILED - returning 0\n"));
return 0;
}
}
UCHAR
ParGet1284_3DeviceCount(
IN PDEVICE_OBJECT PortDeviceObject
)
// return 0 on any error
{
NTSTATUS status;
PARALLEL_PNP_INFORMATION pnpInfo;
// Get Parallel Pnp Info from ParPort device, return Dot3 device Id count returned
status = ParBuildSendInternalIoctl(IOCTL_INTERNAL_GET_PARALLEL_PNP_INFO,
PortDeviceObject, NULL, 0,
&pnpInfo, sizeof(PARALLEL_PNP_INFORMATION), NULL);
if( NT_SUCCESS(status) ) {
return (UCHAR)(pnpInfo.Ieee1284_3DeviceCount);
} else {
ParDump2(PARERRORS, ("devobj::ParGet1284_3DeviceCount - FAILED - returning 0\n"));
return 0;
}
}
NTSTATUS
ParBuildSendInternalIoctl(
IN ULONG IoControlCode,
IN PDEVICE_OBJECT TargetDeviceObject,
IN PVOID InputBuffer OPTIONAL,
IN ULONG InputBufferLength,
OUT PVOID OutputBuffer OPTIONAL,
IN ULONG OutputBufferLength,
IN PLARGE_INTEGER RequestedTimeout OPTIONAL
)
/*++dvdf
Routine Description:
This routine builds and sends an Internal IOCTL to the TargetDeviceObject, waits
for the IOCTL to complete, and returns status to the caller.
*** WORKWORK - dvdf 12Dec98: This function does not support Input and Output in the same IOCTL
Arguments:
IoControlCode - the IOCTL to send
TargetDeviceObject - who to send the IOCTL to
InputBuffer - pointer to input buffer, if any
InputBufferLength, - length of input buffer
OutputBuffer - pointer to output buffer, if any
OutputBufferLength, - length of output buffer
Timeout - how long to wait for request to complete, NULL==use driver global AcquirePortTimeout
Return Value:
Status
--*/
{
NTSTATUS status;
PIRP irp;
LARGE_INTEGER timeout;
KEVENT event;
PIO_STACK_LOCATION irpSp;
BOOLEAN needToCopyOutputBuffer = FALSE;
PAGED_CODE();
ParDump2(PARIOCTL1,("devobj::ParBuildSendInternalIoctl: Enter\n"));
//
// Current limitation is that this function does not handle a request with
// both InputBufferLength and OutputBufferLength > 0
//
if( InputBufferLength != 0 && OutputBufferLength != 0 ) {
// ASSERTMSG("ParBuildSendInternalIoctl does not support input and output in the same IOCTL \n", FALSE);
return STATUS_UNSUCCESSFUL;
}
//
// Allocate and initialize IRP
//
irp = IoAllocateIrp( (CCHAR)(TargetDeviceObject->StackSize + 1), FALSE );
if( !irp ) {
return STATUS_INSUFFICIENT_RESOURCES;
}
irpSp = IoGetNextIrpStackLocation( irp );
irpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
irpSp->Parameters.DeviceIoControl.OutputBufferLength = OutputBufferLength;
irpSp->Parameters.DeviceIoControl.InputBufferLength = InputBufferLength;
irpSp->Parameters.DeviceIoControl.IoControlCode = IoControlCode;
if( InputBufferLength != 0 ) {
irp->AssociatedIrp.SystemBuffer = InputBuffer;
} else if( OutputBufferLength != 0 ) {
irp->AssociatedIrp.SystemBuffer = OutputBuffer;
}
//
// Set completion routine and send IRP
//
KeInitializeEvent( &event, NotificationEvent, FALSE );
IoSetCompletionRoutine( irp, ParSynchCompletionRoutine, &event, TRUE, TRUE, TRUE );
status = ParCallDriver(TargetDeviceObject, irp);
if( !NT_SUCCESS(status) ) {
ParDump2(PARIOCTL1,("devobj::ParBuildSendInternalIoctl: ParCallDriver FAILED - status=%x\n",status));
IoFreeIrp( irp );
return status;
}
//
// Set timeout and wait
//
// user specified : default
timeout = (NULL != RequestedTimeout) ? *RequestedTimeout : AcquirePortTimeout;
status = KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, &timeout);
//
// Did we timeout or did the IRP complete?
//
if( status == STATUS_TIMEOUT ) {
// we timed out - cancel the IRP
IoCancelIrp( irp );
KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL);
}
//
// Irp is complete, grab the status and free the irp
//
status = irp->IoStatus.Status;
IoFreeIrp( irp );
return status;
}
NTSTATUS
ParSelect1284_3Device(
IN PDEVICE_OBJECT PortDeviceObject,
IN UCHAR Dot3DeviceId
)
/*++dvdf
Routine Description:
This routine selects a 1284.3 daisy chain device via an
IOCTL_INTERNAL_SELECT_DEVICE sent to the ParPort to which
our device is connected.
Note: Caller must have already Acquired the Port prior to calling this function.
Arguments:
PortDeviceObject - points to the ParPort that the device is connected to.
Dot3DeviceId - IEEE 1284.3 daisy chain id (in the range [0..3]) to be selected.
Return Value:
STATUS_SUCCESS - if the device was selected
!STATUS_SUCCESS - otherwise
--*/
{
PARALLEL_1284_COMMAND par1284Command;
par1284Command.ID = Dot3DeviceId;
par1284Command.Port = 0;
par1284Command.CommandFlags = PAR_HAVE_PORT_KEEP_PORT; // we have already Acquired the port
return ParBuildSendInternalIoctl(IOCTL_INTERNAL_SELECT_DEVICE,
PortDeviceObject,
&par1284Command, sizeof(PARALLEL_1284_COMMAND),
NULL, 0, NULL);
}
NTSTATUS
ParDeselect1284_3Device(
IN PDEVICE_OBJECT PortDeviceObject,
IN UCHAR Dot3DeviceId
)
/*++dvdf
Routine Description:
This routine selects a 1284.3 daisy chain device via an
IOCTL_INTERNAL_SELECT_DEVICE sent to the ParPort to which
our device is connected.
Note: This function does not Release the port so the Caller still has
the port after this function returns.
Arguments:
PortDeviceObject - points to the ParPort that the device is connected to.
Dot3DeviceId - IEEE 1284.3 daisy chain id (in the range [0..3]) to be selected.
Return Value:
STATUS_SUCCESS - if the device was selected
!STATUS_SUCCESS - otherwise
--*/
{
PARALLEL_1284_COMMAND par1284Command;
par1284Command.ID = Dot3DeviceId;
par1284Command.Port = 0;
par1284Command.CommandFlags = PAR_HAVE_PORT_KEEP_PORT; // just Deselect device, don't Release port
return ParBuildSendInternalIoctl(IOCTL_INTERNAL_DESELECT_DEVICE,
PortDeviceObject,
&par1284Command, sizeof(PARALLEL_1284_COMMAND),
NULL, 0, NULL);
}