Leaked source code of windows server 2003
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.
 
 
 
 
 
 

522 lines
17 KiB

/*++
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"
#define MIN_CONFLICT_LIST_SIZE (sizeof(PLUGPLAY_CONTROL_CONFLICT_LIST) - sizeof(PLUGPLAY_CONTROL_CONFLICT_ENTRY) + sizeof(PLUGPLAY_CONTROL_CONFLICT_STRINGS))
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;
KGUARDED_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) \
{ \
PWCHAR end; \
LONG len; \
\
StringCchPrintfExW((PWCHAR)(ub), (ubl) / sizeof(WCHAR), &end, NULL, 0, REGSTR_VALUE_STANDARD_ULONG_FORMAT, (i)); \
len = (LONG)(end - (PWCHAR)(ub)); \
(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) \
{ \
PWCHAR end; \
LONG len; \
\
StringCchPrintfExW((PWCHAR)(ub), (ubl) / sizeof(WCHAR), &end, NULL, 0, REGSTR_KEY_INSTANCE_KEY_FORMAT, (i)); \
len = (LONG)(end - (PWCHAR)(ub)); \
(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]))
//
// Enter critical section and acquire a lock on the registry. Both these
// mechanisms are required to prevent deadlock in the case where an APC
// routine calls this routine after the registry resource has been claimed
// in this case it would wait blocking this thread so the registry would
// never be released -> deadlock. Critical sectioning the registry manipulation
// portion solves this problem
//
#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(); \
}
#define PiIsPnpRegistryLocked(Exclusive) \
((Exclusive) ? ExIsResourceAcquiredExclusiveLite(&PpRegistryDeviceResource) : \
((ExIsResourceAcquiredSharedLite(&PpRegistryDeviceResource) > 0) ? TRUE : FALSE))
//
// 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
);
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;
VOID
PpLogEvent(
IN PUNICODE_STRING InsertionString1,
IN PUNICODE_STRING InsertionString2,
IN NTSTATUS Status,
IN PVOID DumpData,
IN ULONG DumpDataSize
);
NTSTATUS
PpIrpQueryDeviceText(
IN PDEVICE_OBJECT DeviceObject,
IN DEVICE_TEXT_TYPE DeviceTextType,
IN LCID POINTER_ALIGNMENT LocaleId,
OUT PWCHAR *Description
);
#define PpQueryDeviceDescription(dn, desc) PpIrpQueryDeviceText((dn)->PhysicalDeviceObject, DeviceTextDescription, PsDefaultSystemLocaleId, desc)
#define PpQueryDeviceLocationInformation(dn, loc) PpIrpQueryDeviceText((dn)->PhysicalDeviceObject, DeviceTextLocationInformation, PsDefaultSystemLocaleId, loc)
NTSTATUS
PpIrpQueryCapabilities(
IN PDEVICE_OBJECT DeviceObject,
OUT PDEVICE_CAPABILITIES Capabilities
);
NTSTATUS
PpIrpQueryResourceRequirements(
IN PDEVICE_OBJECT DeviceObject,
OUT PIO_RESOURCE_REQUIREMENTS_LIST *Requirements
);
NTSTATUS
PpIrpQueryID(
IN PDEVICE_OBJECT DeviceObject,
IN BUS_QUERY_ID_TYPE IDType,
OUT PWCHAR *ID
);
NTSTATUS
PpQueryID(
IN PDEVICE_NODE DeviceNode,
IN BUS_QUERY_ID_TYPE IDType,
OUT PWCHAR *ID,
OUT PULONG IDLength
);
NTSTATUS
PpQueryDeviceID(
IN PDEVICE_NODE DeviceNode,
OUT PWCHAR *BusID,
OUT PWCHAR *DeviceID
);
#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 PpQuerySerialNumber(dn, id) PpIrpQueryID((dn)->PhysicalDeviceObject, BusQueryDeviceSerialNumber, id)
NTSTATUS
PpIrpQueryBusInformation(
IN PDEVICE_OBJECT DeviceObject,
OUT PPNP_BUS_INFORMATION *BusInfo
);
NTSTATUS
PpQueryBusInformation(
IN PDEVICE_NODE DeviceNode
);
NTSTATUS
PpSaveDeviceCapabilities (
IN PDEVICE_NODE DeviceNode,
IN PDEVICE_CAPABILITIES Capabilities
);
NTSTATUS
PpBusTypeGuidInitialize(
VOID
);
USHORT
PpBusTypeGuidGetIndex(
IN LPGUID BusTypeGuid
);
NTSTATUS
PpBusTypeGuidGet(
IN USHORT Index,
IN OUT LPGUID BusTypeGuid
);
extern BOOLEAN PpDisableFirmwareMapper;
#if defined(_X86_)
NTSTATUS
PnPBiosMapper(
VOID
);
NTSTATUS
PnPBiosGetBiosInfo(
OUT PVOID *BiosInfo,
OUT ULONG *BiosInfoLength
);
VOID
PnPBiosShutdownSystem(
IN ULONG Phase,
IN OUT PVOID *Context
);
NTSTATUS
PnPBiosInitializePnPBios(
VOID
);
#endif
//
// Firmware mapper external declarations.
//
BOOLEAN
PipIsFirmwareMapperDevicePresent(
IN HANDLE KeyHandle
);
VOID
MapperProcessFirmwareTree(
IN BOOLEAN OnlyProcessSerialPorts
);
VOID
MapperConstructRootEnumTree(
IN BOOLEAN CreatePhantomDevices
);
VOID
MapperFreeList(
VOID
);
VOID
MapperPhantomizeDetectedComPorts(
VOID
);
//
// True iff textmode setup.
//
extern BOOLEAN ExpInTextModeSetup;
VOID
PpMarkDeviceStackStartPending(
IN PDEVICE_OBJECT DeviceObject,
IN BOOLEAN Set
);
NTSTATUS
PiControlMakeUserModeCallersCopy(
PVOID *Destination,
PVOID Src,
ULONG Length,
ULONG Alignment,
KPROCESSOR_MODE CallerMode,
BOOLEAN AllocateDestination
);
#if DBG
LONG
PiControlExceptionFilter(
IN PEXCEPTION_POINTERS ExceptionPointers
);
#else
#define PiControlExceptionFilter(a) EXCEPTION_EXECUTE_HANDLER
#endif
#endif // _KERNEL_PNPI_