|
|
/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
pnpi.h
Abstract:
This module contains the internal structure definitions and APIs used by the kernel-mode Plug and Play manager.
Author:
Lonny McMichael (lonnym) 02/08/1995
Revision History:
--*/
#ifndef _KERNEL_PNPI_
#define _KERNEL_PNPI_
#include <wdmguid.h>
#include "regstrp.h"
typedef struct _DEVICE_NODE DEVICE_NODE, *PDEVICE_NODE; //
// Extract DeviceNode from DeviceObject.
//
#define PP_DO_TO_DN(DO) \
((PDEVICE_NODE)((DO)? (DO)->DeviceObjectExtension->DeviceNode : NULL))
//
// Macros to save useful information into memory dumps.
//
#define PP_SAVE_DEVNODE_TO_TRIAGE_DUMP(dn) { \
if((dn)) { \ IoAddTriageDumpDataBlock(dn, sizeof(DEVICE_NODE)); \ if ((dn)->InstancePath.Length != 0) { \ IoAddTriageDumpDataBlock(&(dn)->InstancePath.Length, sizeof((dn)->InstancePath.Length)); \ IoAddTriageDumpDataBlock((dn)->InstancePath.Buffer, (dn)->InstancePath.Length); \ } \ if ((dn)->ServiceName.Length != 0) { \ IoAddTriageDumpDataBlock(&(dn)->ServiceName.Length, sizeof((dn)->ServiceName.Length)); \ IoAddTriageDumpDataBlock((dn)->ServiceName.Buffer, (dn)->ServiceName.Length); \ } \ if ((dn)->Parent && (dn)->Parent->ServiceName.Length != 0) { \ IoAddTriageDumpDataBlock(&(dn)->Parent->ServiceName.Length, sizeof((dn)->Parent->ServiceName.Length)); \ IoAddTriageDumpDataBlock((dn)->Parent->ServiceName.Buffer, (dn)->Parent->ServiceName.Length); \ } \ } \ }
#define PP_SAVE_DRIVEROBJECT_TO_TRIAGE_DUMP(drvo) { \
if(drvo) { \ IoAddTriageDumpDataBlock(drvo, (drvo)->Size); \ if((drvo)->DriverName.Length != 0) { \ IoAddTriageDumpDataBlock(&(drvo)->DriverName.Length, sizeof((drvo)->DriverName.Length)); \ IoAddTriageDumpDataBlock((drvo)->DriverName.Buffer, (drvo)->DriverName.Length); \ } \ } \ }
#define PP_SAVE_DEVICEOBJECT_TO_TRIAGE_DUMP(do) { \
if((do)) { \ IoAddTriageDumpDataBlock(do, (do)->Size); \ PP_SAVE_DRIVEROBJECT_TO_TRIAGE_DUMP((do)->DriverObject); \ PP_SAVE_DEVNODE_TO_TRIAGE_DUMP(PP_DO_TO_DN(do)); \ } \ }
#define GUID_STRING_LEN 39
#define MAX_DEVICE_ID_LEN 200 // size in chars
#define MAX_SERVICE_NAME_LEN 256 // in characters
//
// PNP_EVENT_LIST
//
// This is the head of the master device event list for both user-mode and
// kernel-mode.
//
typedef struct _PNP_DEVICE_EVENT_LIST { NTSTATUS Status; KMUTEX EventQueueMutex; FAST_MUTEX Lock; LIST_ENTRY List; } PNP_DEVICE_EVENT_LIST, *PPNP_DEVICE_EVENT_LIST;
//
// PNP_DEVICE_EVENT_ENTRY
//
// One of these structures is allocated for each dynamic device event and
// is removed after the event has been posted to all waiting recipients.
// The notify block contains a pointer to this list.
//
typedef struct _PNP_DEVICE_EVENT_ENTRY { LIST_ENTRY ListEntry; ULONG Argument; PKEVENT CallerEvent; PDEVICE_CHANGE_COMPLETE_CALLBACK Callback; PVOID Context; PPNP_VETO_TYPE VetoType; PUNICODE_STRING VetoName; PLUGPLAY_EVENT_BLOCK Data; } PNP_DEVICE_EVENT_ENTRY, *PPNP_DEVICE_EVENT_ENTRY;
//
// Defines the enum type to distinguish between REMOVE device
// and EJECT device.
//
typedef enum _PLUGPLAY_DEVICE_DELETE_TYPE { QueryRemoveDevice, CancelRemoveDevice, RemoveDevice, SurpriseRemoveDevice, EjectDevice, RemoveFailedDevice, RemoveUnstartedFailedDevice, MaxDeviceDeleteType } PLUGPLAY_DEVICE_DELETE_TYPE, *PPLUGPLAY_DEVICE_DELETE_TYPE;
//++
//
// VOID
// PiWstrToUnicodeString(
// OUT PUNICODE_STRING u,
// IN PCWSTR p
// )
//
//--
#define PiWstrToUnicodeString(u, p) { \
if (p) { \ (u)->Length = ((u)->MaximumLength = sizeof((p))) - sizeof(WCHAR); \ } else { \ (u)->Length = (u)->MaximumLength = 0; \ } \ (u)->Buffer = (p); \ }
//++
//
// VOID
// PiUlongToUnicodeString(
// OUT PUNICODE_STRING u,
// IN OUT PWCHAR ub,
// IN ULONG ubl,
// IN ULONG i
// )
//
//--
#define PiUlongToUnicodeString(u, ub, ubl, i) \
{ \ LONG len; \ \ len = _snwprintf((PWCHAR)(ub), (ubl) / sizeof(WCHAR), REGSTR_VALUE_STANDARD_ULONG_FORMAT, (i)); \ (u)->MaximumLength = (USHORT)(ubl); \ (u)->Length = (len == -1) ? (USHORT)(ubl) : (USHORT)len * sizeof(WCHAR); \ (u)->Buffer = (PWSTR)(ub); \ }
//++
//
// VOID
// PiUlongToInstanceKeyUnicodeString(
// OUT PUNICODE_STRING u,
// IN OUT PWCHAR ub,
// IN ULONG ubl,
// IN ULONG i
// )
//
//--
#define PiUlongToInstanceKeyUnicodeString(u, ub, ubl, i) \
{ \ LONG len; \ \ len = _snwprintf((PWCHAR)(ub), (ubl) / sizeof(WCHAR), REGSTR_KEY_INSTANCE_KEY_FORMAT, (i)); \ (u)->MaximumLength = (USHORT)(ubl); \ (u)->Length = (len == -1) ? (USHORT)(ubl) : (USHORT)len * sizeof(WCHAR); \ (u)->Buffer = (PWSTR)(ub); \ }
//
// The following macros convert between a Count of Wide Characters (CWC) and a Count
// of Bytes (CB).
//
#define CWC_TO_CB(c) ((c) * sizeof(WCHAR))
#define CB_TO_CWC(c) ((c) / sizeof(WCHAR))
//
// Macro to determine the number of elements in a statically
// initialized array.
//
#define ELEMENT_COUNT(x) (sizeof(x)/sizeof((x)[0]))
//
// Block write access to Pnp portion of registry.
//
#define PiLockPnpRegistry(Exclusive) { \
KeEnterCriticalRegion(); \ if (Exclusive) { \ ExAcquireResourceExclusiveLite( \ &PpRegistryDeviceResource, \ TRUE); \ } else { \ ExAcquireResourceSharedLite( \ &PpRegistryDeviceResource, \ TRUE); \ } \ } //
// Unblock write access to Pnp portion of registry.
//
#define PiUnlockPnpRegistry() { \
ExReleaseResourceLite(&PpRegistryDeviceResource); \ KeLeaveCriticalRegion(); \ } //
// Function to complete an event asynchronously.
//
VOID PpCompleteDeviceEvent( IN OUT PPNP_DEVICE_EVENT_ENTRY DeviceEvent, IN NTSTATUS FinalStatus );
//
// Global PnP Manager initialization data.
//
extern PVOID PiScratchBuffer;
//
// Private Entry Points
//
BOOLEAN PiRegSzToString( IN PWCHAR RegSzData, IN ULONG RegSzLength, OUT PULONG StringLength OPTIONAL, OUT PWSTR *CopiedString OPTIONAL );
VOID PiUserResponse( IN ULONG Response, IN PNP_VETO_TYPE VetoType, IN LPWSTR VetoName, IN ULONG VetoNameLength );
NTSTATUS PiDeviceRegistration( IN PUNICODE_STRING DeviceInstancePath, IN BOOLEAN Add, IN PUNICODE_STRING ServiceKeyName OPTIONAL );
BOOLEAN PiCompareGuid( CONST GUID *Guid1, CONST GUID *Guid2 );
VOID PnPBiosShutdownSystem ( IN ULONG Phase, IN OUT PVOID *Context );
NTSTATUS PnPBiosInitializePnPBios ( VOID );
NTSTATUS PiGetDeviceRegistryProperty( IN PDEVICE_OBJECT DeviceObject, IN ULONG ValueType, IN PWSTR ValueName, IN PWSTR KeyName, OUT PVOID Buffer, IN OUT PULONG BufferLength );
VOID PpInitializeDeviceReferenceTable( VOID );
PVOID NTAPI PiAllocateGenericTableEntry ( PRTL_GENERIC_TABLE Table, CLONG ByteSize );
VOID NTAPI PiFreeGenericTableEntry ( PRTL_GENERIC_TABLE Table, PVOID Buffer );
VOID PpRemoveDeviceActionRequests( IN PDEVICE_OBJECT DeviceObject );
typedef struct _SYSTEM_HIVE_LIMITS { ULONG Low; ULONG High; } SYSTEM_HIVE_LIMITS, *PSYSTEM_HIVE_LIMITS;
VOID PpSystemHiveLimitCallback( PSYSTEM_HIVE_LIMITS HiveLimits, ULONG Level );
extern SYSTEM_HIVE_LIMITS PpSystemHiveLimits; extern BOOLEAN PpSystemHiveTooLarge;
extern BOOLEAN PpCallerInitializesRequestTable;
NTSTATUS PpBusTypeGuidInitialize( VOID );
USHORT PpBusTypeGuidGetIndex( IN LPGUID BusTypeGuid );
NTSTATUS PpBusTypeGuidGet( IN USHORT Index, IN OUT LPGUID BusTypeGuid );
VOID PiMarkDeviceStackStartPending( IN PDEVICE_OBJECT DeviceObject, IN BOOLEAN Set );
NTSTATUS PpIrpQueryResourceRequirements( IN PDEVICE_OBJECT DeviceObject, OUT PIO_RESOURCE_REQUIREMENTS_LIST *Requirements );
NTSTATUS PpQueryDeviceID( IN PDEVICE_NODE DeviceNode, OUT PWCHAR *BusID, OUT PWCHAR *DeviceID );
NTSTATUS PpQueryID( IN PDEVICE_NODE DeviceNode, IN BUS_QUERY_ID_TYPE IDType, OUT PWCHAR *ID, OUT PULONG IDLength );
NTSTATUS PpIrpQueryID( IN PDEVICE_OBJECT DeviceObject, IN BUS_QUERY_ID_TYPE IDType, OUT PWCHAR *ID );
#define PpQueryInstanceID(dn, id, l) PpQueryID(dn, BusQueryInstanceID, id, l)
#define PpQueryHardwareIDs(dn, id, l) PpQueryID(dn, BusQueryHardwareIDs, id, l)
#define PpQueryCompatibleIDs(dn, id, l) PpQueryID(dn, BusQueryCompatibleIDs, id, l)
#define IopQueryDeviceSerialNumber(do, id) PpIrpQueryID(do, BusQueryDeviceSerialNumber, id)
#endif // _KERNEL_PNPI_
|