2006-01-11 22:13:02 +00:00
|
|
|
|
/*
|
2008-08-18 13:30:17 +00:00
|
|
|
|
* PROJECT: ReactOS Kernel
|
|
|
|
|
* COPYRIGHT: GPL - See COPYING in the top level directory
|
|
|
|
|
* FILE: ntoskrnl/io/pnpmgr/pnpnotify.c
|
2005-01-26 13:58:37 +00:00
|
|
|
|
* PURPOSE: Plug & Play notification functions
|
|
|
|
|
* PROGRAMMERS: Filip Navara (xnavara@volny.cz)
|
2006-01-11 22:13:02 +00:00
|
|
|
|
* Herv<EFBFBD> Poussineau (hpoussin@reactos.org)
|
2010-06-23 19:37:22 +00:00
|
|
|
|
* Pierre Schweitzer
|
Plug & Play manager improvments + few other things
ntoskrnl/ntoskrnl.edf:
Added exports for IoGetDmaAdapter, IoIsWdmVersionAvailable,
KefAcquireSpinLockAtDpcLevel, KefReleaseSpinLockFromDpcLevel,
ExFreePoolWithTag.
ntoskrnl/io/driver.c:
Implementation of NtUnloadDriver.
ntoskrnl/io/device.c:
Added prepending "\\SystemRoot\" and displaying "PnP Loading xxx..." message
in IopInitializeDeviceNodeService.
ntoskrnl/io/pnpmgr.c:
Split into ntoskrnl/io/pnpmgr.c, ntoskrnl/io/deviface.c,
ntoskrnl/io/pnpnotify.c, ntoskrnl/io/pnpmgr/pnpdma.c,
ntoskrnl/io/pnpmgr.c, ntoskrnl/io/pnpmgr/pnproot.c,
ntoskrnl/io/remlock.c, ntoskrnl/io/pnpmgr/pnpreport.c.
Fixed registry handling in IopActionInterrogateDeviceStack and
IopActionConfigureChildServices.
Partial implementation of IoGetDeviceProperty.
ntoskrnl/io/pnpdma.c:
Implementation of IoGetDmaAdapter.
ntoskrnl/io/wdm.c:
New file. Contains implementation of IoIsWdmVersionAvailable.
ntoskrnl/ke/spinlock.c:
Added KeAcquireSpinLockAtDpcLevel and KefReleaseSpinLockFromDpcLevel.
ntoskrnl/mm/pool.c:
Partial implementation of ExFreePoolWithTag.
svn path=/trunk/; revision=6136
2003-09-25 15:54:43 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* INCLUDES ******************************************************************/
|
|
|
|
|
|
2004-08-15 16:39:12 +00:00
|
|
|
|
#include <ntoskrnl.h>
|
2007-08-04 08:49:47 +00:00
|
|
|
|
#define NDEBUG
|
|
|
|
|
#include <debug.h>
|
Plug & Play manager improvments + few other things
ntoskrnl/ntoskrnl.edf:
Added exports for IoGetDmaAdapter, IoIsWdmVersionAvailable,
KefAcquireSpinLockAtDpcLevel, KefReleaseSpinLockFromDpcLevel,
ExFreePoolWithTag.
ntoskrnl/io/driver.c:
Implementation of NtUnloadDriver.
ntoskrnl/io/device.c:
Added prepending "\\SystemRoot\" and displaying "PnP Loading xxx..." message
in IopInitializeDeviceNodeService.
ntoskrnl/io/pnpmgr.c:
Split into ntoskrnl/io/pnpmgr.c, ntoskrnl/io/deviface.c,
ntoskrnl/io/pnpnotify.c, ntoskrnl/io/pnpmgr/pnpdma.c,
ntoskrnl/io/pnpmgr.c, ntoskrnl/io/pnpmgr/pnproot.c,
ntoskrnl/io/remlock.c, ntoskrnl/io/pnpmgr/pnpreport.c.
Fixed registry handling in IopActionInterrogateDeviceStack and
IopActionConfigureChildServices.
Partial implementation of IoGetDeviceProperty.
ntoskrnl/io/pnpdma.c:
Implementation of IoGetDmaAdapter.
ntoskrnl/io/wdm.c:
New file. Contains implementation of IoIsWdmVersionAvailable.
ntoskrnl/ke/spinlock.c:
Added KeAcquireSpinLockAtDpcLevel and KefReleaseSpinLockFromDpcLevel.
ntoskrnl/mm/pool.c:
Partial implementation of ExFreePoolWithTag.
svn path=/trunk/; revision=6136
2003-09-25 15:54:43 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
/* TYPES *******************************************************************/
|
|
|
|
|
|
|
|
|
|
typedef struct _PNP_NOTIFY_ENTRY
|
|
|
|
|
{
|
2008-08-18 13:30:17 +00:00
|
|
|
|
LIST_ENTRY PnpNotifyList;
|
|
|
|
|
IO_NOTIFICATION_EVENT_CATEGORY EventCategory;
|
|
|
|
|
PVOID Context;
|
|
|
|
|
UNICODE_STRING Guid;
|
|
|
|
|
PFILE_OBJECT FileObject;
|
|
|
|
|
PDRIVER_NOTIFICATION_CALLBACK_ROUTINE PnpNotificationProc;
|
2005-04-30 09:39:09 +00:00
|
|
|
|
} PNP_NOTIFY_ENTRY, *PPNP_NOTIFY_ENTRY;
|
|
|
|
|
|
2006-10-17 05:14:57 +00:00
|
|
|
|
KGUARDED_MUTEX PnpNotifyListLock;
|
|
|
|
|
LIST_ENTRY PnpNotifyListHead;
|
2005-04-30 09:39:09 +00:00
|
|
|
|
|
Plug & Play manager improvments + few other things
ntoskrnl/ntoskrnl.edf:
Added exports for IoGetDmaAdapter, IoIsWdmVersionAvailable,
KefAcquireSpinLockAtDpcLevel, KefReleaseSpinLockFromDpcLevel,
ExFreePoolWithTag.
ntoskrnl/io/driver.c:
Implementation of NtUnloadDriver.
ntoskrnl/io/device.c:
Added prepending "\\SystemRoot\" and displaying "PnP Loading xxx..." message
in IopInitializeDeviceNodeService.
ntoskrnl/io/pnpmgr.c:
Split into ntoskrnl/io/pnpmgr.c, ntoskrnl/io/deviface.c,
ntoskrnl/io/pnpnotify.c, ntoskrnl/io/pnpmgr/pnpdma.c,
ntoskrnl/io/pnpmgr.c, ntoskrnl/io/pnpmgr/pnproot.c,
ntoskrnl/io/remlock.c, ntoskrnl/io/pnpmgr/pnpreport.c.
Fixed registry handling in IopActionInterrogateDeviceStack and
IopActionConfigureChildServices.
Partial implementation of IoGetDeviceProperty.
ntoskrnl/io/pnpdma.c:
Implementation of IoGetDmaAdapter.
ntoskrnl/io/wdm.c:
New file. Contains implementation of IoIsWdmVersionAvailable.
ntoskrnl/ke/spinlock.c:
Added KeAcquireSpinLockAtDpcLevel and KefReleaseSpinLockFromDpcLevel.
ntoskrnl/mm/pool.c:
Partial implementation of ExFreePoolWithTag.
svn path=/trunk/; revision=6136
2003-09-25 15:54:43 +00:00
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
VOID
|
|
|
|
|
IopNotifyPlugPlayNotification(
|
|
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
|
IN IO_NOTIFICATION_EVENT_CATEGORY EventCategory,
|
2006-01-11 22:13:02 +00:00
|
|
|
|
IN LPCGUID Event,
|
2005-04-30 09:39:09 +00:00
|
|
|
|
IN PVOID EventCategoryData1,
|
|
|
|
|
IN PVOID EventCategoryData2)
|
|
|
|
|
{
|
|
|
|
|
PPNP_NOTIFY_ENTRY ChangeEntry;
|
2006-09-12 21:18:22 +00:00
|
|
|
|
PLIST_ENTRY ListEntry;
|
2005-04-30 09:39:09 +00:00
|
|
|
|
PVOID NotificationStructure;
|
|
|
|
|
BOOLEAN CallCurrentEntry;
|
2010-04-29 22:35:49 +00:00
|
|
|
|
UNICODE_STRING GuidString;
|
|
|
|
|
NTSTATUS Status;
|
2010-06-23 19:37:22 +00:00
|
|
|
|
PDEVICE_OBJECT EntryDeviceObject = NULL;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
ASSERT(DeviceObject);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
KeAcquireGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
if (IsListEmpty(&PnpNotifyListHead))
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
switch (EventCategory)
|
|
|
|
|
{
|
|
|
|
|
case EventCategoryDeviceInterfaceChange:
|
|
|
|
|
{
|
|
|
|
|
PDEVICE_INTERFACE_CHANGE_NOTIFICATION NotificationInfos;
|
|
|
|
|
NotificationStructure = NotificationInfos = ExAllocatePoolWithTag(
|
|
|
|
|
PagedPool,
|
|
|
|
|
sizeof(DEVICE_INTERFACE_CHANGE_NOTIFICATION),
|
|
|
|
|
TAG_PNP_NOTIFY);
|
2009-09-02 13:02:30 +00:00
|
|
|
|
if (!NotificationInfos)
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2005-04-30 09:39:09 +00:00
|
|
|
|
NotificationInfos->Version = 1;
|
|
|
|
|
NotificationInfos->Size = sizeof(DEVICE_INTERFACE_CHANGE_NOTIFICATION);
|
|
|
|
|
RtlCopyMemory(&NotificationInfos->Event, Event, sizeof(GUID));
|
|
|
|
|
RtlCopyMemory(&NotificationInfos->InterfaceClassGuid, EventCategoryData1, sizeof(GUID));
|
|
|
|
|
NotificationInfos->SymbolicLinkName = (PUNICODE_STRING)EventCategoryData2;
|
2010-04-29 22:35:49 +00:00
|
|
|
|
Status = RtlStringFromGUID(&NotificationInfos->InterfaceClassGuid, &GuidString);
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
ExFreePool(NotificationStructure);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2005-04-30 09:39:09 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryHardwareProfileChange:
|
|
|
|
|
{
|
|
|
|
|
PHWPROFILE_CHANGE_NOTIFICATION NotificationInfos;
|
|
|
|
|
NotificationStructure = NotificationInfos = ExAllocatePoolWithTag(
|
|
|
|
|
PagedPool,
|
|
|
|
|
sizeof(HWPROFILE_CHANGE_NOTIFICATION),
|
|
|
|
|
TAG_PNP_NOTIFY);
|
2009-09-02 13:02:30 +00:00
|
|
|
|
if (!NotificationInfos)
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2005-04-30 09:39:09 +00:00
|
|
|
|
NotificationInfos->Version = 1;
|
|
|
|
|
NotificationInfos->Size = sizeof(HWPROFILE_CHANGE_NOTIFICATION);
|
|
|
|
|
RtlCopyMemory(&NotificationInfos->Event, Event, sizeof(GUID));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryTargetDeviceChange:
|
|
|
|
|
{
|
2010-06-23 19:37:22 +00:00
|
|
|
|
if (Event != &GUID_PNP_CUSTOM_NOTIFICATION)
|
2009-09-02 13:02:30 +00:00
|
|
|
|
{
|
2010-06-23 19:37:22 +00:00
|
|
|
|
PTARGET_DEVICE_REMOVAL_NOTIFICATION NotificationInfos;
|
|
|
|
|
NotificationStructure = NotificationInfos = ExAllocatePoolWithTag(
|
|
|
|
|
PagedPool,
|
|
|
|
|
sizeof(TARGET_DEVICE_REMOVAL_NOTIFICATION),
|
|
|
|
|
TAG_PNP_NOTIFY);
|
|
|
|
|
if (!NotificationInfos)
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
NotificationInfos->Version = 1;
|
|
|
|
|
NotificationInfos->Size = sizeof(TARGET_DEVICE_REMOVAL_NOTIFICATION);
|
|
|
|
|
RtlCopyMemory(&NotificationInfos->Event, Event, sizeof(GUID));
|
|
|
|
|
NotificationInfos->FileObject = (PFILE_OBJECT)EventCategoryData1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
PTARGET_DEVICE_CUSTOM_NOTIFICATION NotificationInfos;
|
|
|
|
|
NotificationStructure = NotificationInfos = ExAllocatePoolWithTag(
|
|
|
|
|
PagedPool,
|
|
|
|
|
sizeof(TARGET_DEVICE_CUSTOM_NOTIFICATION),
|
|
|
|
|
TAG_PNP_NOTIFY);
|
|
|
|
|
if (!NotificationInfos)
|
|
|
|
|
{
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
RtlCopyMemory(NotificationInfos, EventCategoryData1, sizeof(TARGET_DEVICE_CUSTOM_NOTIFICATION));
|
2009-09-02 13:02:30 +00:00
|
|
|
|
}
|
2005-04-30 09:39:09 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("IopNotifyPlugPlayNotification(): unknown EventCategory 0x%x UNIMPLEMENTED\n", EventCategory);
|
2008-11-01 00:36:24 +00:00
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
2005-04-30 09:39:09 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
/* Loop through procedures registred in PnpNotifyListHead
|
|
|
|
|
* list to find those that meet some criteria.
|
|
|
|
|
*/
|
2006-09-12 21:18:22 +00:00
|
|
|
|
ListEntry = PnpNotifyListHead.Flink;
|
|
|
|
|
while (ListEntry != &PnpNotifyListHead)
|
2005-04-30 09:39:09 +00:00
|
|
|
|
{
|
2006-09-12 21:18:22 +00:00
|
|
|
|
ChangeEntry = CONTAINING_RECORD(ListEntry, PNP_NOTIFY_ENTRY, PnpNotifyList);
|
2005-04-30 09:39:09 +00:00
|
|
|
|
CallCurrentEntry = FALSE;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2010-04-29 22:35:49 +00:00
|
|
|
|
if (ChangeEntry->EventCategory != EventCategory)
|
|
|
|
|
{
|
|
|
|
|
ListEntry = ListEntry->Flink;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
switch (EventCategory)
|
|
|
|
|
{
|
|
|
|
|
case EventCategoryDeviceInterfaceChange:
|
|
|
|
|
{
|
2010-04-29 22:35:49 +00:00
|
|
|
|
if (RtlCompareUnicodeString(&ChangeEntry->Guid, &GuidString, FALSE) == 0)
|
2005-04-30 09:39:09 +00:00
|
|
|
|
{
|
|
|
|
|
CallCurrentEntry = TRUE;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryHardwareProfileChange:
|
|
|
|
|
{
|
|
|
|
|
CallCurrentEntry = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryTargetDeviceChange:
|
|
|
|
|
{
|
2010-06-23 19:37:22 +00:00
|
|
|
|
if (Event != &GUID_PNP_CUSTOM_NOTIFICATION)
|
|
|
|
|
{
|
|
|
|
|
if (ChangeEntry->FileObject == (PFILE_OBJECT)EventCategoryData1)
|
|
|
|
|
CallCurrentEntry = TRUE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Status = IoGetRelatedTargetDevice(ChangeEntry->FileObject, &EntryDeviceObject);
|
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
if (DeviceObject == EntryDeviceObject)
|
|
|
|
|
{
|
|
|
|
|
((PTARGET_DEVICE_CUSTOM_NOTIFICATION)NotificationStructure)->FileObject = ChangeEntry->FileObject;
|
|
|
|
|
CallCurrentEntry = TRUE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-04-30 09:39:09 +00:00
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("IopNotifyPlugPlayNotification(): unknown EventCategory 0x%x UNIMPLEMENTED\n", EventCategory);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2006-09-12 21:18:22 +00:00
|
|
|
|
/* Move to the next element now, as callback may unregister itself */
|
|
|
|
|
ListEntry = ListEntry->Flink;
|
|
|
|
|
/* FIXME: If ListEntry was the last element and that callback registers
|
|
|
|
|
* new notifications, those won't be checked... */
|
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
if (CallCurrentEntry)
|
|
|
|
|
{
|
|
|
|
|
/* Call entry into new allocated memory */
|
|
|
|
|
DPRINT("IopNotifyPlugPlayNotification(): found suitable callback %p\n",
|
|
|
|
|
ChangeEntry);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2006-09-12 21:18:22 +00:00
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
2005-04-30 09:39:09 +00:00
|
|
|
|
(ChangeEntry->PnpNotificationProc)(
|
|
|
|
|
NotificationStructure,
|
|
|
|
|
ChangeEntry->Context);
|
2006-09-12 21:18:22 +00:00
|
|
|
|
KeAcquireGuardedMutex(&PnpNotifyListLock);
|
2005-04-30 09:39:09 +00:00
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
|
2005-04-30 09:39:09 +00:00
|
|
|
|
}
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
ExFreePoolWithTag(NotificationStructure, TAG_PNP_NOTIFY);
|
2010-04-29 22:35:49 +00:00
|
|
|
|
if (EventCategory == EventCategoryDeviceInterfaceChange)
|
|
|
|
|
RtlFreeUnicodeString(&GuidString);
|
2005-04-30 09:39:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-08-18 13:30:17 +00:00
|
|
|
|
/* PUBLIC FUNCTIONS **********************************************************/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* @unimplemented
|
|
|
|
|
*/
|
|
|
|
|
ULONG
|
|
|
|
|
NTAPI
|
|
|
|
|
IoPnPDeliverServicePowerNotification(ULONG VetoedPowerOperation OPTIONAL,
|
|
|
|
|
ULONG PowerNotification,
|
|
|
|
|
ULONG Unknown OPTIONAL,
|
|
|
|
|
BOOLEAN Synchronous)
|
|
|
|
|
{
|
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* @implemented
|
|
|
|
|
*/
|
|
|
|
|
NTSTATUS
|
|
|
|
|
NTAPI
|
|
|
|
|
IoRegisterPlugPlayNotification(IN IO_NOTIFICATION_EVENT_CATEGORY EventCategory,
|
|
|
|
|
IN ULONG EventCategoryFlags,
|
|
|
|
|
IN PVOID EventCategoryData OPTIONAL,
|
|
|
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
|
|
|
IN PDRIVER_NOTIFICATION_CALLBACK_ROUTINE CallbackRoutine,
|
|
|
|
|
IN PVOID Context,
|
|
|
|
|
OUT PVOID *NotificationEntry)
|
|
|
|
|
{
|
|
|
|
|
PPNP_NOTIFY_ENTRY Entry;
|
|
|
|
|
PWSTR SymbolicLinkList;
|
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
|
|
DPRINT("__FUNCTION__(EventCategory 0x%x, EventCategoryFlags 0x%lx, DriverObject %p) called.\n",
|
|
|
|
|
EventCategory,
|
|
|
|
|
EventCategoryFlags,
|
|
|
|
|
DriverObject);
|
|
|
|
|
|
|
|
|
|
ObReferenceObject(DriverObject);
|
|
|
|
|
|
|
|
|
|
/* Try to allocate entry for notification before sending any notification */
|
|
|
|
|
Entry = ExAllocatePoolWithTag(NonPagedPool,
|
|
|
|
|
sizeof(PNP_NOTIFY_ENTRY),
|
|
|
|
|
TAG_PNP_NOTIFY);
|
|
|
|
|
|
|
|
|
|
if (!Entry)
|
|
|
|
|
{
|
|
|
|
|
DPRINT("ExAllocatePool() failed\n");
|
|
|
|
|
ObDereferenceObject(DriverObject);
|
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (EventCategory == EventCategoryDeviceInterfaceChange &&
|
|
|
|
|
EventCategoryFlags & PNPNOTIFY_DEVICE_INTERFACE_INCLUDE_EXISTING_INTERFACES)
|
|
|
|
|
{
|
|
|
|
|
DEVICE_INTERFACE_CHANGE_NOTIFICATION NotificationInfos;
|
|
|
|
|
UNICODE_STRING SymbolicLinkU;
|
|
|
|
|
PWSTR SymbolicLink;
|
|
|
|
|
|
|
|
|
|
Status = IoGetDeviceInterfaces((LPGUID)EventCategoryData,
|
|
|
|
|
NULL, /* PhysicalDeviceObject OPTIONAL */
|
|
|
|
|
0, /* Flags */
|
|
|
|
|
&SymbolicLinkList);
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
DPRINT("IoGetDeviceInterfaces() failed with status 0x%08lx\n",
|
|
|
|
|
Status);
|
|
|
|
|
ExFreePoolWithTag(Entry, TAG_PNP_NOTIFY);
|
|
|
|
|
ObDereferenceObject(DriverObject);
|
|
|
|
|
return Status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Enumerate SymbolicLinkList */
|
|
|
|
|
NotificationInfos.Version = 1;
|
|
|
|
|
NotificationInfos.Size = sizeof(DEVICE_INTERFACE_CHANGE_NOTIFICATION);
|
|
|
|
|
RtlCopyMemory(&NotificationInfos.Event,
|
|
|
|
|
&GUID_DEVICE_INTERFACE_ARRIVAL,
|
|
|
|
|
sizeof(GUID));
|
|
|
|
|
RtlCopyMemory(&NotificationInfos.InterfaceClassGuid,
|
|
|
|
|
EventCategoryData,
|
|
|
|
|
sizeof(GUID));
|
|
|
|
|
NotificationInfos.SymbolicLinkName = &SymbolicLinkU;
|
|
|
|
|
|
|
|
|
|
for (SymbolicLink = SymbolicLinkList;
|
|
|
|
|
*SymbolicLink;
|
|
|
|
|
SymbolicLink += wcslen(SymbolicLink) + 1)
|
|
|
|
|
{
|
|
|
|
|
RtlInitUnicodeString(&SymbolicLinkU, SymbolicLink);
|
|
|
|
|
DPRINT("Calling callback routine for %S\n", SymbolicLink);
|
|
|
|
|
(*CallbackRoutine)(&NotificationInfos, Context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ExFreePool(SymbolicLinkList);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Entry->PnpNotificationProc = CallbackRoutine;
|
|
|
|
|
Entry->EventCategory = EventCategory;
|
|
|
|
|
Entry->Context = Context;
|
|
|
|
|
switch (EventCategory)
|
|
|
|
|
{
|
|
|
|
|
case EventCategoryDeviceInterfaceChange:
|
|
|
|
|
{
|
|
|
|
|
Status = RtlStringFromGUID(EventCategoryData, &Entry->Guid);
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
|
{
|
|
|
|
|
ExFreePoolWithTag(Entry, TAG_PNP_NOTIFY);
|
|
|
|
|
ObDereferenceObject(DriverObject);
|
|
|
|
|
return Status;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryHardwareProfileChange:
|
|
|
|
|
{
|
|
|
|
|
/* nothing to do */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case EventCategoryTargetDeviceChange:
|
|
|
|
|
{
|
|
|
|
|
Entry->FileObject = (PFILE_OBJECT)EventCategoryData;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
{
|
|
|
|
|
DPRINT1("__FUNCTION__(): unknown EventCategory 0x%x UNIMPLEMENTED\n",
|
|
|
|
|
EventCategory);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
KeAcquireGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
InsertHeadList(&PnpNotifyListHead,
|
|
|
|
|
&Entry->PnpNotifyList);
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
|
|
|
|
|
DPRINT("IoRegisterPlugPlayNotification() returns NotificationEntry %p\n",
|
|
|
|
|
Entry);
|
|
|
|
|
|
|
|
|
|
*NotificationEntry = Entry;
|
|
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* @implemented
|
|
|
|
|
*/
|
|
|
|
|
NTSTATUS
|
|
|
|
|
NTAPI
|
|
|
|
|
IoUnregisterPlugPlayNotification(IN PVOID NotificationEntry)
|
|
|
|
|
{
|
|
|
|
|
PPNP_NOTIFY_ENTRY Entry;
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
|
|
Entry = (PPNP_NOTIFY_ENTRY)NotificationEntry;
|
|
|
|
|
DPRINT("__FUNCTION__(NotificationEntry %p) called\n", Entry);
|
|
|
|
|
|
|
|
|
|
KeAcquireGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
RtlFreeUnicodeString(&Entry->Guid);
|
|
|
|
|
RemoveEntryList(&Entry->PnpNotifyList);
|
|
|
|
|
KeReleaseGuardedMutex(&PnpNotifyListLock);
|
|
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
|
}
|