xref: /reactos/hal/halx86/acpi/halpnpdd.c (revision 77e6348f)
1 /*
2  * PROJECT:         ReactOS HAL
3  * LICENSE:         BSD - See COPYING.ARM in the top level directory
4  * FILE:            hal/halx86/acpi/halpnpdd.c
5  * PURPOSE:         HAL Plug and Play Device Driver
6  * PROGRAMMERS:     ReactOS Portable Systems Group
7  */
8 
9 /* INCLUDES *******************************************************************/
10 
11 #include <hal.h>
12 
13 #include <initguid.h>
14 #include <wdmguid.h>
15 
16 #define NDEBUG
17 #include <debug.h>
18 
19 typedef enum _EXTENSION_TYPE
20 {
21     PdoExtensionType = 0xC0,
22     FdoExtensionType
23 } EXTENSION_TYPE;
24 
25 typedef enum _PDO_TYPE
26 {
27     AcpiPdo = 0x80,
28     WdPdo
29 } PDO_TYPE;
30 
31 typedef struct _FDO_EXTENSION
32 {
33     EXTENSION_TYPE ExtensionType;
34     struct _PDO_EXTENSION* ChildPdoList;
35     PDEVICE_OBJECT PhysicalDeviceObject;
36     PDEVICE_OBJECT FunctionalDeviceObject;
37     PDEVICE_OBJECT AttachedDeviceObject;
38 } FDO_EXTENSION, *PFDO_EXTENSION;
39 
40 typedef struct _PDO_EXTENSION
41 {
42     EXTENSION_TYPE ExtensionType;
43     struct _PDO_EXTENSION* Next;
44     PDEVICE_OBJECT PhysicalDeviceObject;
45     PFDO_EXTENSION ParentFdoExtension;
46     PDO_TYPE PdoType;
47     PDESCRIPTION_HEADER WdTable;
48     LONG InterfaceReferenceCount;
49 } PDO_EXTENSION, *PPDO_EXTENSION;
50 
51 /* GLOBALS ********************************************************************/
52 
53 PDRIVER_OBJECT HalpDriverObject;
54 
55 /* PRIVATE FUNCTIONS **********************************************************/
56 
57 NTSTATUS
58 NTAPI
59 HalpAddDevice(IN PDRIVER_OBJECT DriverObject,
60               IN PDEVICE_OBJECT TargetDevice)
61 {
62     NTSTATUS Status;
63     PFDO_EXTENSION FdoExtension;
64     PPDO_EXTENSION PdoExtension;
65     PDEVICE_OBJECT DeviceObject, AttachedDevice;
66     PDEVICE_OBJECT PdoDeviceObject;
67     PDESCRIPTION_HEADER Wdrt;
68 
69     DPRINT("HAL: PnP Driver ADD!\n");
70 
71     /* Create the FDO */
72     Status = IoCreateDevice(DriverObject,
73                             sizeof(FDO_EXTENSION),
74                             NULL,
75                             FILE_DEVICE_BUS_EXTENDER,
76                             0,
77                             FALSE,
78                             &DeviceObject);
79     if (!NT_SUCCESS(Status))
80     {
81         /* Should not happen */
82         DbgBreakPoint();
83         return Status;
84     }
85 
86     /* Setup the FDO extension */
87     FdoExtension = DeviceObject->DeviceExtension;
88     FdoExtension->ExtensionType = FdoExtensionType;
89     FdoExtension->PhysicalDeviceObject = TargetDevice;
90     FdoExtension->FunctionalDeviceObject = DeviceObject;
91     FdoExtension->ChildPdoList = NULL;
92 
93     /* FDO is done initializing */
94     DeviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
95 
96     /* Attach to the physical device object (the bus) */
97     AttachedDevice = IoAttachDeviceToDeviceStack(DeviceObject, TargetDevice);
98     if (!AttachedDevice)
99     {
100         /* Failed, undo everything */
101         IoDeleteDevice(DeviceObject);
102         return STATUS_NO_SUCH_DEVICE;
103     }
104 
105     /* Save the attachment */
106     FdoExtension->AttachedDeviceObject = AttachedDevice;
107 
108     /* Create the PDO */
109     Status = IoCreateDevice(DriverObject,
110                             sizeof(PDO_EXTENSION),
111                             NULL,
112                             FILE_DEVICE_BUS_EXTENDER,
113                             FILE_AUTOGENERATED_DEVICE_NAME,
114                             FALSE,
115                             &PdoDeviceObject);
116     if (!NT_SUCCESS(Status))
117     {
118         /* Fail */
119         DPRINT1("HAL: Could not create ACPI device object status=0x%08x\n", Status);
120         return Status;
121     }
122 
123     /* Setup the PDO device extension */
124     PdoExtension = PdoDeviceObject->DeviceExtension;
125     PdoExtension->ExtensionType = PdoExtensionType;
126     PdoExtension->PhysicalDeviceObject = PdoDeviceObject;
127     PdoExtension->ParentFdoExtension = FdoExtension;
128     PdoExtension->PdoType = AcpiPdo;
129 
130     /* Add the PDO to the head of the list */
131     PdoExtension->Next = FdoExtension->ChildPdoList;
132     FdoExtension->ChildPdoList = PdoExtension;
133 
134     /* Initialization is finished */
135     PdoDeviceObject->Flags &= ~DO_DEVICE_INITIALIZING;
136 
137     /* Find the ACPI watchdog table */
138     Wdrt = HalAcpiGetTable(0, 'TRDW');
139     if (Wdrt)
140     {
141         /* FIXME: TODO */
142         DPRINT1("You have an ACPI Watchdog. That's great! You should be proud ;-)\n");
143     }
144 
145     /* Return status */
146     DPRINT("Device added %lx\n", Status);
147     return Status;
148 }
149 
150 NTSTATUS
151 NTAPI
152 HalpQueryInterface(IN PDEVICE_OBJECT DeviceObject,
153                    IN CONST GUID* InterfaceType,
154                    IN USHORT Version,
155                    IN PVOID InterfaceSpecificData,
156                    IN ULONG InterfaceBufferSize,
157                    IN PINTERFACE Interface,
158                    OUT PULONG Length)
159 {
160     if (IsEqualIID(InterfaceType, &GUID_ACPI_REGS_INTERFACE_STANDARD))
161     {
162         DPRINT1("HalpQueryInterface(GUID_ACPI_REGS_INTERFACE_STANDARD) is UNIMPLEMENTED\n");
163     }
164     else if (IsEqualIID(InterfaceType, &GUID_ACPI_PORT_RANGES_INTERFACE_STANDARD))
165     {
166         DPRINT1("HalpQueryInterface(GUID_ACPI_PORT_RANGES_INTERFACE_STANDARD) is UNIMPLEMENTED\n");
167     }
168     else
169     {
170         DPRINT1("HalpQueryInterface({%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}) is UNIMPLEMENTED\n",
171                 InterfaceType->Data1, InterfaceType->Data2, InterfaceType->Data3,
172                 InterfaceType->Data4[0], InterfaceType->Data4[1],
173                 InterfaceType->Data4[2], InterfaceType->Data4[3],
174                 InterfaceType->Data4[4], InterfaceType->Data4[5],
175                 InterfaceType->Data4[6], InterfaceType->Data4[7]);
176     }
177     return STATUS_NOT_SUPPORTED;
178 }
179 
180 NTSTATUS
181 NTAPI
182 HalpQueryDeviceRelations(IN PDEVICE_OBJECT DeviceObject,
183                          IN DEVICE_RELATION_TYPE RelationType,
184                          OUT PDEVICE_RELATIONS* DeviceRelations)
185 {
186     EXTENSION_TYPE ExtensionType;
187     PPDO_EXTENSION PdoExtension;
188     PFDO_EXTENSION FdoExtension;
189     PDEVICE_RELATIONS PdoRelations, FdoRelations;
190     PDEVICE_OBJECT* ObjectEntry;
191     ULONG i = 0, PdoCount = 0;
192 
193     /* Get FDO device extension and PDO count */
194     FdoExtension = DeviceObject->DeviceExtension;
195     ExtensionType = FdoExtension->ExtensionType;
196 
197     /* What do they want? */
198     if (RelationType == BusRelations)
199     {
200         /* This better be an FDO */
201         if (ExtensionType == FdoExtensionType)
202         {
203             /* Count how many PDOs we have */
204             PdoExtension = FdoExtension->ChildPdoList;
205             while (PdoExtension)
206             {
207                 /* Next one */
208                 PdoExtension = PdoExtension->Next;
209                 PdoCount++;
210             }
211 
212             /* Add the PDOs that already exist in the device relations */
213             if (*DeviceRelations)
214             {
215                 PdoCount += (*DeviceRelations)->Count;
216             }
217 
218             /* Allocate our structure */
219             FdoRelations = ExAllocatePoolWithTag(PagedPool,
220                                                  FIELD_OFFSET(DEVICE_RELATIONS,
221                                                               Objects) +
222                                                  sizeof(PDEVICE_OBJECT) * PdoCount,
223                                                  TAG_HAL);
224             if (!FdoRelations) return STATUS_INSUFFICIENT_RESOURCES;
225 
226             /* Save our count */
227             FdoRelations->Count = PdoCount;
228 
229             /* Query existing relations */
230             ObjectEntry = FdoRelations->Objects;
231             if (*DeviceRelations)
232             {
233                 /* Check if there were any */
234                 if ((*DeviceRelations)->Count)
235                 {
236                     /* Loop them all */
237                     do
238                     {
239                         /* Copy into our structure */
240                         *ObjectEntry++ = (*DeviceRelations)->Objects[i];
241                     }
242                     while (++i < (*DeviceRelations)->Count);
243                 }
244 
245                 /* Free existing structure */
246                 ExFreePool(*DeviceRelations);
247             }
248 
249             /* Now check if we have a PDO list */
250             PdoExtension = FdoExtension->ChildPdoList;
251             if (PdoExtension)
252             {
253                 /* Loop the PDOs */
254                 do
255                 {
256                     /* Save our own PDO and reference it */
257                     *ObjectEntry++ = PdoExtension->PhysicalDeviceObject;
258                     ObReferenceObject(PdoExtension->PhysicalDeviceObject);
259 
260                     /* Go to our next PDO */
261                     PdoExtension = PdoExtension->Next;
262                 }
263                 while (PdoExtension);
264             }
265 
266             /* Return the new structure */
267             *DeviceRelations = FdoRelations;
268             return STATUS_SUCCESS;
269         }
270     }
271     else
272     {
273         /* The only other thing we support is a target relation for the PDO */
274         if ((RelationType == TargetDeviceRelation) &&
275             (ExtensionType == PdoExtensionType))
276         {
277             /* Only one entry */
278             PdoRelations = ExAllocatePoolWithTag(PagedPool,
279                                                  sizeof(DEVICE_RELATIONS),
280                                                  TAG_HAL);
281             if (!PdoRelations) return STATUS_INSUFFICIENT_RESOURCES;
282 
283             /* Fill it out and reference us */
284             PdoRelations->Count = 1;
285             PdoRelations->Objects[0] = DeviceObject;
286             ObReferenceObject(DeviceObject);
287 
288             /* Return it */
289             *DeviceRelations = PdoRelations;
290             return STATUS_SUCCESS;
291         }
292     }
293 
294     /* We don't support anything else */
295     return STATUS_NOT_SUPPORTED;
296 }
297 
298 NTSTATUS
299 NTAPI
300 HalpQueryCapabilities(IN PDEVICE_OBJECT DeviceObject,
301                       OUT PDEVICE_CAPABILITIES Capabilities)
302 {
303     //PPDO_EXTENSION PdoExtension;
304     NTSTATUS Status;
305     PAGED_CODE();
306 
307     /* Get the extension and check for valid version */
308     //PdoExtension = DeviceObject->DeviceExtension;
309     ASSERT(Capabilities->Version == 1);
310     if (Capabilities->Version == 1)
311     {
312         /* Can't lock or eject us */
313         Capabilities->LockSupported = FALSE;
314         Capabilities->EjectSupported = FALSE;
315 
316         /* Can't remove or dock us */
317         Capabilities->Removable = FALSE;
318         Capabilities->DockDevice = FALSE;
319 
320         /* Can't access us raw */
321         Capabilities->RawDeviceOK = FALSE;
322 
323         /* We have a unique ID, and don't bother the user */
324         Capabilities->UniqueID = TRUE;
325         Capabilities->SilentInstall = TRUE;
326 
327         /* Fill out the adress */
328         Capabilities->Address = InterfaceTypeUndefined;
329         Capabilities->UINumber = InterfaceTypeUndefined;
330 
331         /* Fill out latencies */
332         Capabilities->D1Latency = 0;
333         Capabilities->D2Latency = 0;
334         Capabilities->D3Latency = 0;
335 
336         /* Fill out supported device states */
337         Capabilities->DeviceState[PowerSystemWorking] = PowerDeviceD0;
338         Capabilities->DeviceState[PowerSystemHibernate] = PowerDeviceD3;
339         Capabilities->DeviceState[PowerSystemShutdown] = PowerDeviceD3;
340         Capabilities->DeviceState[PowerSystemSleeping3] = PowerDeviceD3;
341 
342         /* Done */
343         Status = STATUS_SUCCESS;
344     }
345     else
346     {
347         /* Fail */
348         Status = STATUS_NOT_SUPPORTED;
349     }
350 
351     /* Return status */
352     return Status;
353 }
354 
355 NTSTATUS
356 NTAPI
357 HalpQueryResources(IN PDEVICE_OBJECT DeviceObject,
358                    OUT PCM_RESOURCE_LIST *Resources)
359 {
360     PPDO_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
361     NTSTATUS Status;
362     PCM_RESOURCE_LIST ResourceList;
363     PIO_RESOURCE_REQUIREMENTS_LIST RequirementsList;
364     PIO_RESOURCE_DESCRIPTOR Descriptor;
365     PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialDesc;
366     ULONG i;
367     PAGED_CODE();
368 
369     /* Only the ACPI PDO has requirements */
370     if (DeviceExtension->PdoType == AcpiPdo)
371     {
372         /* Query ACPI requirements */
373         Status = HalpQueryAcpiResourceRequirements(&RequirementsList);
374         if (!NT_SUCCESS(Status)) return Status;
375 
376         ASSERT(RequirementsList->AlternativeLists == 1);
377 
378         /* Allocate the resourcel ist */
379         ResourceList = ExAllocatePoolWithTag(PagedPool,
380                                              sizeof(CM_RESOURCE_LIST),
381                                              TAG_HAL);
382         if (!ResourceList )
383         {
384             /* Fail, no memory */
385             Status = STATUS_INSUFFICIENT_RESOURCES;
386             ExFreePoolWithTag(RequirementsList, TAG_HAL);
387             return Status;
388         }
389 
390         /* Initialize it */
391         RtlZeroMemory(ResourceList, sizeof(CM_RESOURCE_LIST));
392         ResourceList->Count = 1;
393 
394         /* Setup the list fields */
395         ResourceList->List[0].BusNumber = -1;
396         ResourceList->List[0].InterfaceType = PNPBus;
397         ResourceList->List[0].PartialResourceList.Version = 1;
398         ResourceList->List[0].PartialResourceList.Revision = 1;
399         ResourceList->List[0].PartialResourceList.Count = 0;
400 
401         /* Setup the first descriptor */
402         PartialDesc = ResourceList->List[0].PartialResourceList.PartialDescriptors;
403 
404         /* Find the requirement descriptor for the SCI */
405         for (i = 0; i < RequirementsList->List[0].Count; i++)
406         {
407             /* Get this descriptor */
408             Descriptor = &RequirementsList->List[0].Descriptors[i];
409             if (Descriptor->Type == CmResourceTypeInterrupt)
410             {
411                 /* Copy requirements descriptor into resource descriptor */
412                 PartialDesc->Type = CmResourceTypeInterrupt;
413                 PartialDesc->ShareDisposition = Descriptor->ShareDisposition;
414                 PartialDesc->Flags = Descriptor->Flags;
415                 ASSERT(Descriptor->u.Interrupt.MinimumVector ==
416                        Descriptor->u.Interrupt.MaximumVector);
417                 PartialDesc->u.Interrupt.Vector = Descriptor->u.Interrupt.MinimumVector;
418                 PartialDesc->u.Interrupt.Level = Descriptor->u.Interrupt.MinimumVector;
419                 PartialDesc->u.Interrupt.Affinity = 0xFFFFFFFF;
420 
421                 ResourceList->List[0].PartialResourceList.Count++;
422 
423                 break;
424             }
425         }
426 
427         /* Return resources and success */
428         *Resources = ResourceList;
429 
430         ExFreePoolWithTag(RequirementsList, TAG_HAL);
431 
432         return STATUS_SUCCESS;
433     }
434     else if (DeviceExtension->PdoType == WdPdo)
435     {
436         /* Watchdog doesn't */
437         return STATUS_NOT_SUPPORTED;
438     }
439     else
440     {
441         /* This shouldn't happen */
442         return STATUS_UNSUCCESSFUL;
443     }
444 }
445 
446 NTSTATUS
447 NTAPI
448 HalpQueryResourceRequirements(IN PDEVICE_OBJECT DeviceObject,
449                               OUT PIO_RESOURCE_REQUIREMENTS_LIST *Requirements)
450 {
451     PPDO_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
452     PAGED_CODE();
453 
454     /* Only the ACPI PDO has requirements */
455     if (DeviceExtension->PdoType == AcpiPdo)
456     {
457         /* Query ACPI requirements */
458         return HalpQueryAcpiResourceRequirements(Requirements);
459     }
460     else if (DeviceExtension->PdoType == WdPdo)
461     {
462         /* Watchdog doesn't */
463         return STATUS_NOT_SUPPORTED;
464     }
465     else
466     {
467         /* This shouldn't happen */
468         return STATUS_UNSUCCESSFUL;
469     }
470 }
471 
472 NTSTATUS
473 NTAPI
474 HalpQueryIdPdo(IN PDEVICE_OBJECT DeviceObject,
475                IN BUS_QUERY_ID_TYPE IdType,
476                OUT PUSHORT *BusQueryId)
477 {
478     PPDO_EXTENSION PdoExtension;
479     PDO_TYPE PdoType;
480     PWCHAR CurrentId;
481     WCHAR Id[100];
482     NTSTATUS Status;
483     SIZE_T Length = 0;
484     PWCHAR Buffer;
485 
486     /* Get the PDO type */
487     PdoExtension = DeviceObject->DeviceExtension;
488     PdoType = PdoExtension->PdoType;
489 
490     /* What kind of ID is being requested? */
491     DPRINT("ID: %d\n", IdType);
492     switch (IdType)
493     {
494         case BusQueryDeviceID:
495         case BusQueryHardwareIDs:
496 
497             /* What kind of PDO is this? */
498             if (PdoType == AcpiPdo)
499             {
500                 /* ACPI ID */
501                 CurrentId = L"ACPI_HAL\\PNP0C08";
502                 RtlCopyMemory(Id, CurrentId, (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL));
503                 Length += (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
504 
505                 CurrentId = L"*PNP0C08";
506                 RtlCopyMemory(&Id[wcslen(Id) + 1], CurrentId, (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL));
507                 Length += (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
508             }
509             else if (PdoType == WdPdo)
510             {
511                 /* WatchDog ID */
512                 CurrentId = L"ACPI_HAL\\PNP0C18";
513                 RtlCopyMemory(Id, CurrentId, (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL));
514                 Length += (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
515 
516                 CurrentId = L"*PNP0C18";
517                 RtlCopyMemory(&Id[wcslen(Id) + 1], CurrentId, (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL));
518                 Length += (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
519             }
520             else
521             {
522                 /* Unknown */
523                 return STATUS_NOT_SUPPORTED;
524             }
525             break;
526 
527         case BusQueryInstanceID:
528 
529             /* Instance ID */
530             CurrentId = L"0";
531             RtlCopyMemory(Id, CurrentId, (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL));
532             Length += (wcslen(CurrentId) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
533             break;
534 
535         case BusQueryCompatibleIDs:
536         default:
537 
538             /* We don't support anything else */
539             return STATUS_NOT_SUPPORTED;
540     }
541 
542 
543     /* Allocate the buffer */
544     Buffer = ExAllocatePoolWithTag(PagedPool,
545                                    Length + sizeof(UNICODE_NULL),
546                                    TAG_HAL);
547     if (Buffer)
548     {
549         /* Copy the string and null-terminate it */
550         RtlCopyMemory(Buffer, Id, Length);
551         Buffer[Length / sizeof(WCHAR)] = UNICODE_NULL;
552 
553         /* Return string */
554         *BusQueryId = Buffer;
555         Status = STATUS_SUCCESS;
556         DPRINT("Returning: %S\n", *BusQueryId);
557     }
558     else
559     {
560         /* Fail */
561         Status = STATUS_INSUFFICIENT_RESOURCES;
562     }
563 
564     /* Return status */
565     return Status;
566 }
567 
568 NTSTATUS
569 NTAPI
570 HalpQueryIdFdo(IN PDEVICE_OBJECT DeviceObject,
571                IN BUS_QUERY_ID_TYPE IdType,
572                OUT PUSHORT *BusQueryId)
573 {
574     NTSTATUS Status;
575     SIZE_T Length;
576     PWCHAR Id;
577     PWCHAR Buffer;
578 
579     /* What kind of ID is being requested? */
580     DPRINT("ID: %d\n", IdType);
581     switch (IdType)
582     {
583         case BusQueryDeviceID:
584         case BusQueryHardwareIDs:
585 
586             /* This is our hardware ID */
587             Id = HalHardwareIdString;
588             break;
589 
590         case BusQueryInstanceID:
591 
592             /* And our instance ID */
593             Id = L"0";
594             break;
595 
596         default:
597 
598             /* We don't support anything else */
599             return STATUS_NOT_SUPPORTED;
600     }
601 
602     /* Calculate the length */
603     Length = (wcslen(Id) * sizeof(WCHAR)) + sizeof(UNICODE_NULL);
604 
605     /* Allocate the buffer */
606     Buffer = ExAllocatePoolWithTag(PagedPool,
607                                    Length + sizeof(UNICODE_NULL),
608                                    TAG_HAL);
609     if (Buffer)
610     {
611         /* Copy the string and null-terminate it */
612         RtlCopyMemory(Buffer, Id, Length);
613         Buffer[Length / sizeof(WCHAR)] = UNICODE_NULL;
614 
615         /* Return string */
616         *BusQueryId = Buffer;
617         Status = STATUS_SUCCESS;
618         DPRINT("Returning: %S\n", *BusQueryId);
619     }
620     else
621     {
622         /* Fail */
623         Status = STATUS_INSUFFICIENT_RESOURCES;
624     }
625 
626     /* Return status */
627     return Status;
628 }
629 
630 NTSTATUS
631 NTAPI
632 HalpDispatchPnp(IN PDEVICE_OBJECT DeviceObject,
633                 IN PIRP Irp)
634 {
635     PIO_STACK_LOCATION IoStackLocation;
636     //PPDO_EXTENSION PdoExtension;
637     PFDO_EXTENSION FdoExtension;
638     NTSTATUS Status;
639     UCHAR Minor;
640 
641     /* Get the device extension and stack location */
642     FdoExtension = DeviceObject->DeviceExtension;
643     IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
644     Minor = IoStackLocation->MinorFunction;
645 
646     /* FDO? */
647     if (FdoExtension->ExtensionType == FdoExtensionType)
648     {
649         /* Query the IRP type */
650         switch (Minor)
651         {
652             case IRP_MN_QUERY_DEVICE_RELATIONS:
653 
654                 /* Call the worker */
655                 DPRINT("Querying device relations for FDO\n");
656                 Status = HalpQueryDeviceRelations(DeviceObject,
657                                                   IoStackLocation->Parameters.QueryDeviceRelations.Type,
658                                                   (PVOID)&Irp->IoStatus.Information);
659                 break;
660 
661             case IRP_MN_QUERY_INTERFACE:
662 
663                 /* Call the worker */
664                 DPRINT("Querying interface for FDO\n");
665                 Status = HalpQueryInterface(DeviceObject,
666                                             IoStackLocation->Parameters.QueryInterface.InterfaceType,
667                                             IoStackLocation->Parameters.QueryInterface.Size,
668                                             IoStackLocation->Parameters.QueryInterface.InterfaceSpecificData,
669                                             IoStackLocation->Parameters.QueryInterface.Version,
670                                             IoStackLocation->Parameters.QueryInterface.Interface,
671                                             (PVOID)&Irp->IoStatus.Information);
672                 break;
673 
674 
675             case IRP_MN_QUERY_ID:
676 
677                 /* Call the worker */
678                 DPRINT("Querying ID for FDO\n");
679                 Status = HalpQueryIdFdo(DeviceObject,
680                                         IoStackLocation->Parameters.QueryId.IdType,
681                                         (PVOID)&Irp->IoStatus.Information);
682                 break;
683 
684             case IRP_MN_QUERY_CAPABILITIES:
685 
686                 /* Call the worker */
687                 DPRINT("Querying the capabilities for the FDO\n");
688                 Status = HalpQueryCapabilities(DeviceObject,
689                                                IoStackLocation->Parameters.DeviceCapabilities.Capabilities);
690                 break;
691 
692             default:
693 
694                 DPRINT("Other IRP: %lx\n", Minor);
695                 Status = STATUS_NOT_SUPPORTED;
696                 break;
697         }
698 
699         /* What happpened? */
700         if ((NT_SUCCESS(Status)) || (Status == STATUS_NOT_SUPPORTED))
701         {
702             /* Set the IRP status, unless this isn't understood */
703             if (Status != STATUS_NOT_SUPPORTED)
704             {
705                 Irp->IoStatus.Status = Status;
706             }
707 
708             /* Pass it on */
709             IoSkipCurrentIrpStackLocation(Irp);
710             return IoCallDriver(FdoExtension->AttachedDeviceObject, Irp);
711         }
712 
713         /* Otherwise, we failed, so set the status and complete the request */
714         DPRINT1("IRP failed with status: %lx\n", Status);
715         Irp->IoStatus.Status = Status;
716         IoCompleteRequest(Irp, IO_NO_INCREMENT);
717         return Status;
718     }
719     else
720     {
721         /* This is a PDO instead */
722         ASSERT(FdoExtension->ExtensionType == PdoExtensionType);
723         //PdoExtension = (PPDO_EXTENSION)FdoExtension;
724         /* Query the IRP type */
725         Status = STATUS_SUCCESS;
726         switch (Minor)
727         {
728             case IRP_MN_START_DEVICE:
729 
730                 /* We only care about a PCI PDO */
731                 DPRINT1("Start device received\n");
732                 /* Complete the IRP normally */
733                 break;
734 
735             case IRP_MN_REMOVE_DEVICE:
736 
737                 /* Check if this is a PCI device */
738                 DPRINT1("Remove device received\n");
739 
740                 /* We're done */
741                 Status = STATUS_SUCCESS;
742                 break;
743 
744             case IRP_MN_SURPRISE_REMOVAL:
745 
746                 /* Inherit whatever status we had */
747                 DPRINT1("Surprise removal IRP\n");
748                 Status = Irp->IoStatus.Status;
749                 break;
750 
751             case IRP_MN_QUERY_DEVICE_RELATIONS:
752 
753                 /* Query the device relations */
754                 DPRINT("Querying PDO relations\n");
755                 Status = HalpQueryDeviceRelations(DeviceObject,
756                                                   IoStackLocation->Parameters.QueryDeviceRelations.Type,
757                                                   (PVOID)&Irp->IoStatus.Information);
758                 break;
759 
760             case IRP_MN_QUERY_INTERFACE:
761 
762                 /* Call the worker */
763                 DPRINT("Querying interface for PDO\n");
764                 Status = HalpQueryInterface(DeviceObject,
765                                             IoStackLocation->Parameters.QueryInterface.InterfaceType,
766                                             IoStackLocation->Parameters.QueryInterface.Size,
767                                             IoStackLocation->Parameters.QueryInterface.InterfaceSpecificData,
768                                             IoStackLocation->Parameters.QueryInterface.Version,
769                                             IoStackLocation->Parameters.QueryInterface.Interface,
770                                             (PVOID)&Irp->IoStatus.Information);
771                 break;
772 
773             case IRP_MN_QUERY_CAPABILITIES:
774 
775                 /* Call the worker */
776                 DPRINT("Querying the capabilities for the PDO\n");
777                 Status = HalpQueryCapabilities(DeviceObject,
778                                                IoStackLocation->Parameters.DeviceCapabilities.Capabilities);
779                 break;
780 
781             case IRP_MN_QUERY_RESOURCES:
782 
783                 /* Call the worker */
784                 DPRINT("Querying the resources for the PDO\n");
785                 Status = HalpQueryResources(DeviceObject, (PVOID)&Irp->IoStatus.Information);
786                 break;
787 
788             case IRP_MN_QUERY_RESOURCE_REQUIREMENTS:
789 
790                 /* Call the worker */
791                 DPRINT("Querying the resource requirements for the PDO\n");
792                 Status = HalpQueryResourceRequirements(DeviceObject,
793                                                        (PVOID)&Irp->IoStatus.Information);
794                 break;
795 
796             case IRP_MN_QUERY_ID:
797 
798                 /* Call the worker */
799                 DPRINT("Query the ID for the PDO\n");
800                 Status = HalpQueryIdPdo(DeviceObject,
801                                         IoStackLocation->Parameters.QueryId.IdType,
802                                         (PVOID)&Irp->IoStatus.Information);
803                 break;
804 
805             default:
806 
807                 /* We don't handle anything else, so inherit the old state */
808                 DPRINT("Illegal IRP: %lx\n", Minor);
809                 Status = Irp->IoStatus.Status;
810                 break;
811         }
812 
813         /* If it's not supported, inherit the old status */
814         if (Status == STATUS_NOT_SUPPORTED) Status = Irp->IoStatus.Status;
815 
816         /* Complete the IRP */
817         DPRINT("IRP completed with status: %lx\n", Status);
818         Irp->IoStatus.Status = Status;
819         IoCompleteRequest(Irp, IO_NO_INCREMENT);
820         return Status;
821     }
822 }
823 
824 NTSTATUS
825 NTAPI
826 HalpDispatchWmi(IN PDEVICE_OBJECT DeviceObject,
827                 IN PIRP Irp)
828 {
829     UNIMPLEMENTED_DBGBREAK("HAL: PnP Driver WMI!\n");
830     return STATUS_SUCCESS;
831 }
832 
833 NTSTATUS
834 NTAPI
835 HalpDispatchPower(IN PDEVICE_OBJECT DeviceObject,
836                   IN PIRP Irp)
837 {
838     PFDO_EXTENSION FdoExtension;
839 
840     DPRINT("HAL: PnP Driver Power!\n");
841     FdoExtension = DeviceObject->DeviceExtension;
842     if (FdoExtension->ExtensionType == FdoExtensionType)
843     {
844         PoStartNextPowerIrp(Irp);
845         IoSkipCurrentIrpStackLocation(Irp);
846         return PoCallDriver(FdoExtension->AttachedDeviceObject, Irp);
847     }
848     else
849     {
850         PoStartNextPowerIrp(Irp);
851         Irp->IoStatus.Status = STATUS_SUCCESS;
852         IoCompleteRequest(Irp, IO_NO_INCREMENT);
853         return STATUS_SUCCESS;
854     }
855 }
856 
857 NTSTATUS
858 NTAPI
859 HalpDriverEntry(IN PDRIVER_OBJECT DriverObject,
860                 IN PUNICODE_STRING RegistryPath)
861 {
862     NTSTATUS Status;
863     PDEVICE_OBJECT TargetDevice = NULL;
864 
865     DPRINT("HAL: PnP Driver ENTRY!\n");
866 
867     /* This is us */
868     HalpDriverObject = DriverObject;
869 
870     /* Set up add device */
871     DriverObject->DriverExtension->AddDevice = HalpAddDevice;
872 
873     /* Set up the callouts */
874     DriverObject->MajorFunction[IRP_MJ_PNP] = HalpDispatchPnp;
875     DriverObject->MajorFunction[IRP_MJ_POWER] = HalpDispatchPower;
876     DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = HalpDispatchWmi;
877 
878     /* Create the PDO and tell the PnP manager about us*/
879     Status = IoReportDetectedDevice(DriverObject,
880                                     InterfaceTypeUndefined,
881                                     -1,
882                                     -1,
883                                     NULL,
884                                     NULL,
885                                     FALSE,
886                                     &TargetDevice);
887     if (!NT_SUCCESS(Status))
888         return Status;
889 
890     TargetDevice->Flags &= ~DO_DEVICE_INITIALIZING;
891 
892     /* Set up the device stack */
893     Status = HalpAddDevice(DriverObject, TargetDevice);
894     if (!NT_SUCCESS(Status))
895     {
896         IoDeleteDevice(TargetDevice);
897         return Status;
898     }
899 
900     /* Return to kernel */
901     return Status;
902 }
903 
904 NTSTATUS
905 NTAPI
906 HaliInitPnpDriver(VOID)
907 {
908     NTSTATUS Status;
909     UNICODE_STRING DriverString;
910     PAGED_CODE();
911 
912     /* Create the driver */
913     RtlInitUnicodeString(&DriverString, L"\\Driver\\ACPI_HAL");
914     Status = IoCreateDriver(&DriverString, HalpDriverEntry);
915 
916     /* Return status */
917     return Status;
918 }
919 
920 /* EOF */
921