xref: /reactos/hal/halx86/legacy/bus/pcibus.c (revision 734cd5e8)
1 /*
2  * PROJECT:         ReactOS HAL
3  * LICENSE:         GPL - See COPYING in the top level directory
4  * FILE:            hal/halx86/legacy/bus/pcibus.c
5  * PURPOSE:         PCI Bus Support (Configuration Space, Resource Allocation)
6  * PROGRAMMERS:     Alex Ionescu (alex.ionescu@reactos.org)
7  */
8 
9 /* INCLUDES ******************************************************************/
10 
11 #include <hal.h>
12 #define NDEBUG
13 #include <debug.h>
14 
15 /* GLOBALS *******************************************************************/
16 
17 extern BOOLEAN HalpPciLockSettings;
18 ULONG HalpBusType;
19 
20 BOOLEAN HalpPCIConfigInitialized;
21 ULONG HalpMinPciBus, HalpMaxPciBus;
22 KSPIN_LOCK HalpPCIConfigLock;
23 PCI_CONFIG_HANDLER PCIConfigHandler;
24 
25 /* PCI Operation Matrix */
26 UCHAR PCIDeref[4][4] =
27 {
28     {0, 1, 2, 2},   // ULONG-aligned offset
29     {1, 1, 1, 1},   // UCHAR-aligned offset
30     {2, 1, 2, 2},   // USHORT-aligned offset
31     {1, 1, 1, 1}    // UCHAR-aligned offset
32 };
33 
34 /* Type 1 PCI Bus */
35 PCI_CONFIG_HANDLER PCIConfigHandlerType1 =
36 {
37     /* Synchronization */
38     (FncSync)HalpPCISynchronizeType1,
39     (FncReleaseSync)HalpPCIReleaseSynchronzationType1,
40 
41     /* Read */
42     {
43         (FncConfigIO)HalpPCIReadUlongType1,
44         (FncConfigIO)HalpPCIReadUcharType1,
45         (FncConfigIO)HalpPCIReadUshortType1
46     },
47 
48     /* Write */
49     {
50         (FncConfigIO)HalpPCIWriteUlongType1,
51         (FncConfigIO)HalpPCIWriteUcharType1,
52         (FncConfigIO)HalpPCIWriteUshortType1
53     }
54 };
55 
56 /* Type 2 PCI Bus */
57 PCI_CONFIG_HANDLER PCIConfigHandlerType2 =
58 {
59     /* Synchronization */
60     (FncSync)HalpPCISynchronizeType2,
61     (FncReleaseSync)HalpPCIReleaseSynchronizationType2,
62 
63     /* Read */
64     {
65         (FncConfigIO)HalpPCIReadUlongType2,
66         (FncConfigIO)HalpPCIReadUcharType2,
67         (FncConfigIO)HalpPCIReadUshortType2
68     },
69 
70     /* Write */
71     {
72         (FncConfigIO)HalpPCIWriteUlongType2,
73         (FncConfigIO)HalpPCIWriteUcharType2,
74         (FncConfigIO)HalpPCIWriteUshortType2
75     }
76 };
77 
78 PCIPBUSDATA HalpFakePciBusData =
79 {
80     {
81         PCI_DATA_TAG,
82         PCI_DATA_VERSION,
83         HalpReadPCIConfig,
84         HalpWritePCIConfig,
85         NULL,
86         NULL,
87         {{{0, 0, 0}}},
88         {0, 0, 0, 0}
89     },
90     {{0, 0}},
91     32,
92 };
93 
94 BUS_HANDLER HalpFakePciBusHandler =
95 {
96     1,
97     PCIBus,
98     PCIConfiguration,
99     0,
100     NULL,
101     NULL,
102     &HalpFakePciBusData,
103     0,
104     NULL,
105     {0, 0, 0, 0},
106     (PGETSETBUSDATA)HalpGetPCIData,
107     (PGETSETBUSDATA)HalpSetPCIData,
108     NULL,
109     HalpAssignPCISlotResources,
110     NULL,
111     NULL
112 };
113 
114 /* TYPE 1 FUNCTIONS **********************************************************/
115 
116 VOID
117 NTAPI
HalpPCISynchronizeType1(IN PBUS_HANDLER BusHandler,IN PCI_SLOT_NUMBER Slot,OUT PKIRQL OldIrql,OUT PPCI_TYPE1_CFG_BITS PciCfg1)118 HalpPCISynchronizeType1(IN PBUS_HANDLER BusHandler,
119                         IN PCI_SLOT_NUMBER Slot,
120                         OUT PKIRQL OldIrql,
121                         OUT PPCI_TYPE1_CFG_BITS PciCfg1)
122 {
123     /* Setup the PCI Configuration Register */
124     PciCfg1->u.AsULONG = 0;
125     PciCfg1->u.bits.BusNumber = BusHandler->BusNumber;
126     PciCfg1->u.bits.DeviceNumber = Slot.u.bits.DeviceNumber;
127     PciCfg1->u.bits.FunctionNumber = Slot.u.bits.FunctionNumber;
128     PciCfg1->u.bits.Enable = TRUE;
129 
130     /* Acquire the lock */
131     KeRaiseIrql(HIGH_LEVEL, OldIrql);
132     KeAcquireSpinLockAtDpcLevel(&HalpPCIConfigLock);
133 }
134 
135 VOID
136 NTAPI
HalpPCIReleaseSynchronzationType1(IN PBUS_HANDLER BusHandler,IN KIRQL OldIrql)137 HalpPCIReleaseSynchronzationType1(IN PBUS_HANDLER BusHandler,
138                                   IN KIRQL OldIrql)
139 {
140     PCI_TYPE1_CFG_BITS PciCfg1;
141 
142     /* Clear the PCI Configuration Register */
143     PciCfg1.u.AsULONG = 0;
144     WRITE_PORT_ULONG(((PPCIPBUSDATA)BusHandler->BusData)->Config.Type1.Address,
145                      PciCfg1.u.AsULONG);
146 
147     /* Release the lock */
148     KeReleaseSpinLock(&HalpPCIConfigLock, OldIrql);
149 }
150 
TYPE1_READ(HalpPCIReadUcharType1,UCHAR)151 TYPE1_READ(HalpPCIReadUcharType1, UCHAR)
152 TYPE1_READ(HalpPCIReadUshortType1, USHORT)
153 TYPE1_READ(HalpPCIReadUlongType1, ULONG)
154 TYPE1_WRITE(HalpPCIWriteUcharType1, UCHAR)
155 TYPE1_WRITE(HalpPCIWriteUshortType1, USHORT)
156 TYPE1_WRITE(HalpPCIWriteUlongType1, ULONG)
157 
158 /* TYPE 2 FUNCTIONS **********************************************************/
159 
160 VOID
161 NTAPI
162 HalpPCISynchronizeType2(IN PBUS_HANDLER BusHandler,
163                         IN PCI_SLOT_NUMBER Slot,
164                         OUT PKIRQL OldIrql,
165                         OUT PPCI_TYPE2_ADDRESS_BITS PciCfg)
166 {
167     PCI_TYPE2_CSE_BITS PciCfg2Cse;
168     PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
169 
170     /* Setup the configuration register */
171     PciCfg->u.AsUSHORT = 0;
172     PciCfg->u.bits.Agent = (USHORT)Slot.u.bits.DeviceNumber;
173     PciCfg->u.bits.AddressBase = (USHORT)BusData->Config.Type2.Base;
174 
175     /* Acquire the lock */
176     KeRaiseIrql(HIGH_LEVEL, OldIrql);
177     KeAcquireSpinLockAtDpcLevel(&HalpPCIConfigLock);
178 
179     /* Setup the CSE Register */
180     PciCfg2Cse.u.AsUCHAR = 0;
181     PciCfg2Cse.u.bits.Enable = TRUE;
182     PciCfg2Cse.u.bits.FunctionNumber = (UCHAR)Slot.u.bits.FunctionNumber;
183     PciCfg2Cse.u.bits.Key = -1;
184 
185     /* Write the bus number and CSE */
186     WRITE_PORT_UCHAR(BusData->Config.Type2.Forward,
187                      (UCHAR)BusHandler->BusNumber);
188     WRITE_PORT_UCHAR(BusData->Config.Type2.CSE, PciCfg2Cse.u.AsUCHAR);
189 }
190 
191 VOID
192 NTAPI
HalpPCIReleaseSynchronizationType2(IN PBUS_HANDLER BusHandler,IN KIRQL OldIrql)193 HalpPCIReleaseSynchronizationType2(IN PBUS_HANDLER BusHandler,
194                                    IN KIRQL OldIrql)
195 {
196     PCI_TYPE2_CSE_BITS PciCfg2Cse;
197     PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
198 
199     /* Clear CSE and bus number */
200     PciCfg2Cse.u.AsUCHAR = 0;
201     WRITE_PORT_UCHAR(BusData->Config.Type2.CSE, PciCfg2Cse.u.AsUCHAR);
202     WRITE_PORT_UCHAR(BusData->Config.Type2.Forward, 0);
203 
204     /* Release the lock */
205     KeReleaseSpinLock(&HalpPCIConfigLock, OldIrql);
206 }
207 
TYPE2_READ(HalpPCIReadUcharType2,UCHAR)208 TYPE2_READ(HalpPCIReadUcharType2, UCHAR)
209 TYPE2_READ(HalpPCIReadUshortType2, USHORT)
210 TYPE2_READ(HalpPCIReadUlongType2, ULONG)
211 TYPE2_WRITE(HalpPCIWriteUcharType2, UCHAR)
212 TYPE2_WRITE(HalpPCIWriteUshortType2, USHORT)
213 TYPE2_WRITE(HalpPCIWriteUlongType2, ULONG)
214 
215 /* PCI CONFIGURATION SPACE ***************************************************/
216 
217 VOID
218 NTAPI
219 HalpPCIConfig(IN PBUS_HANDLER BusHandler,
220               IN PCI_SLOT_NUMBER Slot,
221               IN PUCHAR Buffer,
222               IN ULONG Offset,
223               IN ULONG Length,
224               IN FncConfigIO *ConfigIO)
225 {
226     KIRQL OldIrql;
227     ULONG i;
228     UCHAR State[20];
229 
230     /* Synchronize the operation */
231     PCIConfigHandler.Synchronize(BusHandler, Slot, &OldIrql, State);
232 
233     /* Loop every increment */
234     while (Length)
235     {
236         /* Find out the type of read/write we need to do */
237         i = PCIDeref[Offset % sizeof(ULONG)][Length % sizeof(ULONG)];
238 
239         /* Do the read/write and return the number of bytes */
240         i = ConfigIO[i]((PPCIPBUSDATA)BusHandler->BusData,
241                         State,
242                         Buffer,
243                         Offset);
244 
245         /* Increment the buffer position and offset, and decrease the length */
246         Offset += i;
247         Buffer += i;
248         Length -= i;
249     }
250 
251     /* Release the lock and PCI bus */
252     PCIConfigHandler.ReleaseSynchronzation(BusHandler, OldIrql);
253 }
254 
255 VOID
256 NTAPI
HalpReadPCIConfig(IN PBUS_HANDLER BusHandler,IN PCI_SLOT_NUMBER Slot,IN PVOID Buffer,IN ULONG Offset,IN ULONG Length)257 HalpReadPCIConfig(IN PBUS_HANDLER BusHandler,
258                   IN PCI_SLOT_NUMBER Slot,
259                   IN PVOID Buffer,
260                   IN ULONG Offset,
261                   IN ULONG Length)
262 {
263     /* Validate the PCI Slot */
264     if (!HalpValidPCISlot(BusHandler, Slot))
265     {
266         /* Fill the buffer with invalid data */
267         RtlFillMemory(Buffer, Length, -1);
268     }
269     else
270     {
271         /* Send the request */
272         HalpPCIConfig(BusHandler,
273                       Slot,
274                       Buffer,
275                       Offset,
276                       Length,
277                       PCIConfigHandler.ConfigRead);
278     }
279 }
280 
281 VOID
282 NTAPI
HalpWritePCIConfig(IN PBUS_HANDLER BusHandler,IN PCI_SLOT_NUMBER Slot,IN PVOID Buffer,IN ULONG Offset,IN ULONG Length)283 HalpWritePCIConfig(IN PBUS_HANDLER BusHandler,
284                    IN PCI_SLOT_NUMBER Slot,
285                    IN PVOID Buffer,
286                    IN ULONG Offset,
287                    IN ULONG Length)
288 {
289     /* Validate the PCI Slot */
290     if (HalpValidPCISlot(BusHandler, Slot))
291     {
292         /* Send the request */
293         HalpPCIConfig(BusHandler,
294                       Slot,
295                       Buffer,
296                       Offset,
297                       Length,
298                       PCIConfigHandler.ConfigWrite);
299     }
300 }
301 
302 #ifdef SARCH_XBOX
303 static
304 BOOLEAN
HalpXboxBlacklistedPCISlot(_In_ ULONG BusNumber,_In_ PCI_SLOT_NUMBER Slot)305 HalpXboxBlacklistedPCISlot(
306     _In_ ULONG BusNumber,
307     _In_ PCI_SLOT_NUMBER Slot)
308 {
309     /* Trying to get PCI config data from devices 0:0:1 and 0:0:2 will completely
310      * hang the Xbox. Also, the device number doesn't seem to be decoded for the
311      * video card, so it appears to be present on 1:0:0 - 1:31:0.
312      * We hack around these problems by indicating "device not present" for devices
313      * 0:0:1, 0:0:2, 1:1:0, 1:2:0, 1:3:0, ...., 1:31:0 */
314     if ((BusNumber == 0 && Slot.u.bits.DeviceNumber == 0 &&
315         (Slot.u.bits.FunctionNumber == 1 || Slot.u.bits.FunctionNumber == 2)) ||
316         (BusNumber == 1 && Slot.u.bits.DeviceNumber != 0))
317     {
318         DPRINT("Blacklisted PCI slot (%d:%d:%d)\n",
319                BusNumber, Slot.u.bits.DeviceNumber, Slot.u.bits.FunctionNumber);
320         return TRUE;
321     }
322 
323     return FALSE;
324 }
325 #endif
326 
327 BOOLEAN
328 NTAPI
HalpValidPCISlot(IN PBUS_HANDLER BusHandler,IN PCI_SLOT_NUMBER Slot)329 HalpValidPCISlot(IN PBUS_HANDLER BusHandler,
330                  IN PCI_SLOT_NUMBER Slot)
331 {
332     PCI_SLOT_NUMBER MultiSlot;
333     PPCIPBUSDATA BusData = (PPCIPBUSDATA)BusHandler->BusData;
334     UCHAR HeaderType;
335     //ULONG Device;
336 
337     /* Simple validation */
338     if (Slot.u.bits.Reserved) return FALSE;
339     if (Slot.u.bits.DeviceNumber >= BusData->MaxDevice) return FALSE;
340 
341 #ifdef SARCH_XBOX
342     if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
343         return FALSE;
344 #endif
345 
346     /* Function 0 doesn't need checking */
347     if (!Slot.u.bits.FunctionNumber) return TRUE;
348 
349     /* Functions 0+ need Multi-Function support, so check the slot */
350     //Device = Slot.u.bits.DeviceNumber;
351     MultiSlot = Slot;
352     MultiSlot.u.bits.FunctionNumber = 0;
353 
354     /* Send function 0 request to get the header back */
355     HalpReadPCIConfig(BusHandler,
356                       MultiSlot,
357                       &HeaderType,
358                       FIELD_OFFSET(PCI_COMMON_CONFIG, HeaderType),
359                       sizeof(UCHAR));
360 
361     /* Now make sure the header is multi-function */
362     if (!(HeaderType & PCI_MULTIFUNCTION) || (HeaderType == 0xFF)) return FALSE;
363     return TRUE;
364 }
365 
366 CODE_SEG("INIT")
367 ULONG
HalpPhase0GetPciDataByOffset(_In_ ULONG Bus,_In_ PCI_SLOT_NUMBER PciSlot,_Out_writes_bytes_all_ (Length)PVOID Buffer,_In_ ULONG Offset,_In_ ULONG Length)368 HalpPhase0GetPciDataByOffset(
369     _In_ ULONG Bus,
370     _In_ PCI_SLOT_NUMBER PciSlot,
371     _Out_writes_bytes_all_(Length) PVOID Buffer,
372     _In_ ULONG Offset,
373     _In_ ULONG Length)
374 {
375     ULONG BytesLeft = Length;
376     PUCHAR BufferPtr = Buffer;
377     PCI_TYPE1_CFG_BITS PciCfg;
378 
379 #ifdef SARCH_XBOX
380     if (HalpXboxBlacklistedPCISlot(Bus, PciSlot))
381     {
382         RtlFillMemory(Buffer, Length, 0xFF);
383         return Length;
384     }
385 #endif
386 
387     PciCfg.u.AsULONG = 0;
388     PciCfg.u.bits.BusNumber = Bus;
389     PciCfg.u.bits.DeviceNumber = PciSlot.u.bits.DeviceNumber;
390     PciCfg.u.bits.FunctionNumber = PciSlot.u.bits.FunctionNumber;
391     PciCfg.u.bits.Enable = TRUE;
392 
393     while (BytesLeft)
394     {
395         ULONG i;
396 
397         PciCfg.u.bits.RegisterNumber = Offset / sizeof(ULONG);
398         WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDRESS_PORT, PciCfg.u.AsULONG);
399 
400         i = PCIDeref[Offset % sizeof(ULONG)][BytesLeft % sizeof(ULONG)];
401         switch (i)
402         {
403             case 0:
404             {
405                 *(PULONG)BufferPtr = READ_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT);
406 
407                 /* Number of bytes read */
408                 i = sizeof(ULONG);
409                 break;
410             }
411             case 1:
412             {
413                 *BufferPtr = READ_PORT_UCHAR((PUCHAR)(PCI_TYPE1_DATA_PORT +
414                                              Offset % sizeof(ULONG)));
415                 break;
416             }
417             case 2:
418             {
419                 *(PUSHORT)BufferPtr = READ_PORT_USHORT((PUSHORT)(PCI_TYPE1_DATA_PORT +
420                                                                  Offset % sizeof(ULONG)));
421                 break;
422             }
423 
424             DEFAULT_UNREACHABLE;
425         }
426 
427         Offset += i;
428         BufferPtr += i;
429         BytesLeft -= i;
430     }
431 
432     return Length;
433 }
434 
435 CODE_SEG("INIT")
436 ULONG
HalpPhase0SetPciDataByOffset(_In_ ULONG Bus,_In_ PCI_SLOT_NUMBER PciSlot,_In_reads_bytes_ (Length)PVOID Buffer,_In_ ULONG Offset,_In_ ULONG Length)437 HalpPhase0SetPciDataByOffset(
438     _In_ ULONG Bus,
439     _In_ PCI_SLOT_NUMBER PciSlot,
440     _In_reads_bytes_(Length) PVOID Buffer,
441     _In_ ULONG Offset,
442     _In_ ULONG Length)
443 {
444     ULONG BytesLeft = Length;
445     PUCHAR BufferPtr = Buffer;
446     PCI_TYPE1_CFG_BITS PciCfg;
447 
448 #ifdef SARCH_XBOX
449     if (HalpXboxBlacklistedPCISlot(Bus, PciSlot))
450     {
451         return 0;
452     }
453 #endif
454 
455     PciCfg.u.AsULONG = 0;
456     PciCfg.u.bits.BusNumber = Bus;
457     PciCfg.u.bits.DeviceNumber = PciSlot.u.bits.DeviceNumber;
458     PciCfg.u.bits.FunctionNumber = PciSlot.u.bits.FunctionNumber;
459     PciCfg.u.bits.Enable = TRUE;
460 
461     while (BytesLeft)
462     {
463         ULONG i;
464 
465         PciCfg.u.bits.RegisterNumber = Offset / sizeof(ULONG);
466         WRITE_PORT_ULONG((PULONG)PCI_TYPE1_ADDRESS_PORT, PciCfg.u.AsULONG);
467 
468         i = PCIDeref[Offset % sizeof(ULONG)][BytesLeft % sizeof(ULONG)];
469         switch (i)
470         {
471             case 0:
472             {
473                 WRITE_PORT_ULONG((PULONG)PCI_TYPE1_DATA_PORT, *(PULONG)BufferPtr);
474 
475                 /* Number of bytes written */
476                 i = sizeof(ULONG);
477                 break;
478             }
479             case 1:
480             {
481                 WRITE_PORT_UCHAR((PUCHAR)(PCI_TYPE1_DATA_PORT + Offset % sizeof(ULONG)),
482                                  *BufferPtr);
483                 break;
484             }
485             case 2:
486             {
487                 WRITE_PORT_USHORT((PUSHORT)(PCI_TYPE1_DATA_PORT + Offset % sizeof(ULONG)),
488                                   *(PUSHORT)BufferPtr);
489                 break;
490             }
491 
492             DEFAULT_UNREACHABLE;
493         }
494 
495         Offset += i;
496         BufferPtr += i;
497         BytesLeft -= i;
498     }
499 
500     return Length;
501 }
502 
503 /* HAL PCI CALLBACKS *********************************************************/
504 
505 ULONG
506 NTAPI
HalpGetPCIData(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN ULONG SlotNumber,IN PVOID Buffer,IN ULONG Offset,IN ULONG Length)507 HalpGetPCIData(IN PBUS_HANDLER BusHandler,
508                IN PBUS_HANDLER RootHandler,
509                IN ULONG SlotNumber,
510                IN PVOID Buffer,
511                IN ULONG Offset,
512                IN ULONG Length)
513 {
514     PCI_SLOT_NUMBER Slot;
515     UCHAR PciBuffer[PCI_COMMON_HDR_LENGTH];
516     PPCI_COMMON_CONFIG PciConfig = (PPCI_COMMON_CONFIG)PciBuffer;
517     ULONG Len = 0;
518 
519     Slot.u.AsULONG = SlotNumber;
520 #ifdef SARCH_XBOX
521     if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
522     {
523         RtlFillMemory(Buffer, Length, 0xFF);
524         return Length;
525     }
526 #endif
527 
528     /* Normalize the length */
529     if (Length > sizeof(PCI_COMMON_CONFIG)) Length = sizeof(PCI_COMMON_CONFIG);
530 
531     /* Check if this is a vendor-specific read */
532     if (Offset >= PCI_COMMON_HDR_LENGTH)
533     {
534         /* Read the header */
535         HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, sizeof(ULONG));
536 
537         /* Make sure the vendor is valid */
538         if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
539     }
540     else
541     {
542         /* Read the entire header */
543         Len = PCI_COMMON_HDR_LENGTH;
544         HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, Len);
545 
546         /* Validate the vendor ID */
547         if (PciConfig->VendorID == PCI_INVALID_VENDORID)
548         {
549             /* It's invalid, but we want to return this much */
550             Len = sizeof(USHORT);
551         }
552 
553         /* Now check if there's space left */
554         if (Len < Offset) return 0;
555 
556         /* There is, so return what's after the offset and normalize */
557         Len -= Offset;
558         if (Len > Length) Len = Length;
559 
560         /* Copy the data into the caller's buffer */
561         RtlMoveMemory(Buffer, PciBuffer + Offset, Len);
562 
563         /* Update buffer and offset, decrement total length */
564         Offset += Len;
565         Buffer = (PVOID)((ULONG_PTR)Buffer + Len);
566         Length -= Len;
567     }
568 
569     /* Now we still have something to copy */
570     if (Length)
571     {
572         /* Check if it's vendor-specific data */
573         if (Offset >= PCI_COMMON_HDR_LENGTH)
574         {
575             /* Read it now */
576             HalpReadPCIConfig(BusHandler, Slot, Buffer, Offset, Length);
577             Len += Length;
578         }
579     }
580 
581     /* Update the total length read */
582     return Len;
583 }
584 
585 ULONG
586 NTAPI
HalpSetPCIData(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN ULONG SlotNumber,IN PVOID Buffer,IN ULONG Offset,IN ULONG Length)587 HalpSetPCIData(IN PBUS_HANDLER BusHandler,
588                IN PBUS_HANDLER RootHandler,
589                IN ULONG SlotNumber,
590                IN PVOID Buffer,
591                IN ULONG Offset,
592                IN ULONG Length)
593 {
594     PCI_SLOT_NUMBER Slot;
595     UCHAR PciBuffer[PCI_COMMON_HDR_LENGTH];
596     PPCI_COMMON_CONFIG PciConfig = (PPCI_COMMON_CONFIG)PciBuffer;
597     ULONG Len = 0;
598 
599     Slot.u.AsULONG = SlotNumber;
600 #ifdef SARCH_XBOX
601     if (HalpXboxBlacklistedPCISlot(BusHandler->BusNumber, Slot))
602         return 0;
603 #endif
604 
605     /* Normalize the length */
606     if (Length > sizeof(PCI_COMMON_CONFIG)) Length = sizeof(PCI_COMMON_CONFIG);
607 
608     /* Check if this is a vendor-specific read */
609     if (Offset >= PCI_COMMON_HDR_LENGTH)
610     {
611         /* Read the header */
612         HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, sizeof(ULONG));
613 
614         /* Make sure the vendor is valid */
615         if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
616     }
617     else
618     {
619         /* Read the entire header and validate the vendor ID */
620         Len = PCI_COMMON_HDR_LENGTH;
621         HalpReadPCIConfig(BusHandler, Slot, PciConfig, 0, Len);
622         if (PciConfig->VendorID == PCI_INVALID_VENDORID) return 0;
623 
624         /* Return what's after the offset and normalize */
625         Len -= Offset;
626         if (Len > Length) Len = Length;
627 
628         /* Copy the specific caller data */
629         RtlMoveMemory(PciBuffer + Offset, Buffer, Len);
630 
631         /* Write the actual configuration data */
632         HalpWritePCIConfig(BusHandler, Slot, PciBuffer + Offset, Offset, Len);
633 
634         /* Update buffer and offset, decrement total length */
635         Offset += Len;
636         Buffer = (PVOID)((ULONG_PTR)Buffer + Len);
637         Length -= Len;
638     }
639 
640     /* Now we still have something to copy */
641     if (Length)
642     {
643         /* Check if it's vendor-specific data */
644         if (Offset >= PCI_COMMON_HDR_LENGTH)
645         {
646             /* Read it now */
647             HalpWritePCIConfig(BusHandler, Slot, Buffer, Offset, Length);
648             Len += Length;
649         }
650     }
651 
652     /* Update the total length read */
653     return Len;
654 }
655 #ifndef _MINIHAL_
656 ULONG
657 NTAPI
HalpGetPCIIntOnISABus(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN ULONG BusInterruptLevel,IN ULONG BusInterruptVector,OUT PKIRQL Irql,OUT PKAFFINITY Affinity)658 HalpGetPCIIntOnISABus(IN PBUS_HANDLER BusHandler,
659                       IN PBUS_HANDLER RootHandler,
660                       IN ULONG BusInterruptLevel,
661                       IN ULONG BusInterruptVector,
662                       OUT PKIRQL Irql,
663                       OUT PKAFFINITY Affinity)
664 {
665     /* Validate the level first */
666     if (BusInterruptLevel < 1) return 0;
667 
668     /* PCI has its IRQs on top of ISA IRQs, so pass it on to the ISA handler */
669     return HalGetInterruptVector(Isa,
670                                  0,
671                                  BusInterruptLevel,
672                                  0,
673                                  Irql,
674                                  Affinity);
675 }
676 #endif // _MINIHAL_
677 
678 VOID
679 NTAPI
HalpPCIPin2ISALine(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN PCI_SLOT_NUMBER SlotNumber,IN PPCI_COMMON_CONFIG PciData)680 HalpPCIPin2ISALine(IN PBUS_HANDLER BusHandler,
681                    IN PBUS_HANDLER RootHandler,
682                    IN PCI_SLOT_NUMBER SlotNumber,
683                    IN PPCI_COMMON_CONFIG PciData)
684 {
685     UNIMPLEMENTED_DBGBREAK();
686 }
687 
688 VOID
689 NTAPI
HalpPCIISALine2Pin(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN PCI_SLOT_NUMBER SlotNumber,IN PPCI_COMMON_CONFIG PciNewData,IN PPCI_COMMON_CONFIG PciOldData)690 HalpPCIISALine2Pin(IN PBUS_HANDLER BusHandler,
691                    IN PBUS_HANDLER RootHandler,
692                    IN PCI_SLOT_NUMBER SlotNumber,
693                    IN PPCI_COMMON_CONFIG PciNewData,
694                    IN PPCI_COMMON_CONFIG PciOldData)
695 {
696     UNIMPLEMENTED_DBGBREAK();
697 }
698 
699 #ifndef _MINIHAL_
700 NTSTATUS
701 NTAPI
HalpGetISAFixedPCIIrq(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN PCI_SLOT_NUMBER PciSlot,OUT PSUPPORTED_RANGE * Range)702 HalpGetISAFixedPCIIrq(IN PBUS_HANDLER BusHandler,
703                       IN PBUS_HANDLER RootHandler,
704                       IN PCI_SLOT_NUMBER PciSlot,
705                       OUT PSUPPORTED_RANGE *Range)
706 {
707     PCI_COMMON_HEADER PciData;
708 
709     /* Read PCI configuration data */
710     HalGetBusData(PCIConfiguration,
711                   BusHandler->BusNumber,
712                   PciSlot.u.AsULONG,
713                   &PciData,
714                   PCI_COMMON_HDR_LENGTH);
715 
716     /* Make sure it's a real device */
717     if (PciData.VendorID == PCI_INVALID_VENDORID) return STATUS_UNSUCCESSFUL;
718 
719     /* Allocate the supported range structure */
720     *Range = ExAllocatePoolWithTag(PagedPool, sizeof(SUPPORTED_RANGE), TAG_HAL);
721     if (!*Range) return STATUS_INSUFFICIENT_RESOURCES;
722 
723     /* Set it up */
724     RtlZeroMemory(*Range, sizeof(SUPPORTED_RANGE));
725     (*Range)->Base = 1;
726 
727     /* If the PCI device has no IRQ, nothing to do */
728     if (!PciData.u.type0.InterruptPin) return STATUS_SUCCESS;
729 
730     /* FIXME: The PCI IRQ Routing Miniport should be called */
731 
732     /* Also if the INT# seems bogus, nothing to do either */
733     if ((PciData.u.type0.InterruptLine == 0) ||
734         (PciData.u.type0.InterruptLine == 255))
735     {
736         /* Fake success */
737         return STATUS_SUCCESS;
738     }
739 
740     /* Otherwise, the INT# should be valid, return it to the caller */
741     (*Range)->Base = PciData.u.type0.InterruptLine;
742     (*Range)->Limit = PciData.u.type0.InterruptLine;
743     return STATUS_SUCCESS;
744 }
745 #endif // _MINIHAL_
746 
747 static ULONG NTAPI
PciSize(ULONG Base,ULONG Mask)748 PciSize(ULONG Base, ULONG Mask)
749 {
750     ULONG Size = Mask & Base; /* Find the significant bits */
751     Size = Size & ~(Size - 1); /* Get the lowest of them to find the decode size */
752     return Size;
753 }
754 
755 NTSTATUS
756 NTAPI
HalpAdjustPCIResourceList(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN OUT PIO_RESOURCE_REQUIREMENTS_LIST * pResourceList)757 HalpAdjustPCIResourceList(IN PBUS_HANDLER BusHandler,
758                           IN PBUS_HANDLER RootHandler,
759                           IN OUT PIO_RESOURCE_REQUIREMENTS_LIST *pResourceList)
760 {
761     PPCIPBUSDATA BusData;
762     PCI_SLOT_NUMBER SlotNumber;
763     PSUPPORTED_RANGE Interrupt;
764     NTSTATUS Status;
765 
766     /* Get PCI bus data */
767     BusData = BusHandler->BusData;
768     SlotNumber.u.AsULONG = (*pResourceList)->SlotNumber;
769 
770     /* Get the IRQ supported range */
771     Status = BusData->GetIrqRange(BusHandler, RootHandler, SlotNumber, &Interrupt);
772     if (!NT_SUCCESS(Status)) return Status;
773 #ifndef _MINIHAL_
774     /* Handle the /PCILOCK feature */
775     if (HalpPciLockSettings)
776     {
777         /* /PCILOCK is not yet supported */
778         UNIMPLEMENTED_DBGBREAK("/PCILOCK boot switch is not yet supported.");
779     }
780 #endif
781     /* Now create the correct resource list based on the supported bus ranges */
782 #if 0
783     Status = HaliAdjustResourceListRange(BusHandler->BusAddresses,
784                                          Interrupt,
785                                          pResourceList);
786 #else
787     DPRINT1("HAL: No PCI Resource Adjustment done! Hardware may malfunction\n");
788     Status = STATUS_SUCCESS;
789 #endif
790 
791     /* Return to caller */
792     ExFreePool(Interrupt);
793     return Status;
794 }
795 
796 NTSTATUS
797 NTAPI
HalpAssignPCISlotResources(IN PBUS_HANDLER BusHandler,IN PBUS_HANDLER RootHandler,IN PUNICODE_STRING RegistryPath,IN PUNICODE_STRING DriverClassName OPTIONAL,IN PDRIVER_OBJECT DriverObject,IN PDEVICE_OBJECT DeviceObject OPTIONAL,IN ULONG Slot,IN OUT PCM_RESOURCE_LIST * AllocatedResources)798 HalpAssignPCISlotResources(IN PBUS_HANDLER BusHandler,
799                            IN PBUS_HANDLER RootHandler,
800                            IN PUNICODE_STRING RegistryPath,
801                            IN PUNICODE_STRING DriverClassName OPTIONAL,
802                            IN PDRIVER_OBJECT DriverObject,
803                            IN PDEVICE_OBJECT DeviceObject OPTIONAL,
804                            IN ULONG Slot,
805                            IN OUT PCM_RESOURCE_LIST *AllocatedResources)
806 {
807     PCI_COMMON_CONFIG PciConfig;
808     SIZE_T Address;
809     ULONG ResourceCount;
810     ULONG Size[PCI_TYPE0_ADDRESSES];
811     NTSTATUS Status = STATUS_SUCCESS;
812     UCHAR Offset;
813     PCM_PARTIAL_RESOURCE_DESCRIPTOR Descriptor;
814     PCI_SLOT_NUMBER SlotNumber;
815     ULONG WriteBuffer;
816     DPRINT1("WARNING: PCI Slot Resource Assignment is FOOBAR\n");
817 
818     /* FIXME: Should handle 64-bit addresses */
819 
820     /* Read configuration data */
821     SlotNumber.u.AsULONG = Slot;
822     HalpReadPCIConfig(BusHandler, SlotNumber, &PciConfig, 0, PCI_COMMON_HDR_LENGTH);
823 
824     /* Check if we read it correctly */
825     if (PciConfig.VendorID == PCI_INVALID_VENDORID)
826         return STATUS_NO_SUCH_DEVICE;
827 
828     /* Read the PCI configuration space for the device and store base address and
829     size information in temporary storage. Count the number of valid base addresses */
830     ResourceCount = 0;
831     for (Address = 0; Address < PCI_TYPE0_ADDRESSES; Address++)
832     {
833         if (0xffffffff == PciConfig.u.type0.BaseAddresses[Address])
834             PciConfig.u.type0.BaseAddresses[Address] = 0;
835 
836         /* Memory resource */
837         if (0 != PciConfig.u.type0.BaseAddresses[Address])
838         {
839             ResourceCount++;
840 
841             Offset = (UCHAR)FIELD_OFFSET(PCI_COMMON_CONFIG, u.type0.BaseAddresses[Address]);
842 
843             /* Write 0xFFFFFFFF there */
844             WriteBuffer = 0xffffffff;
845             HalpWritePCIConfig(BusHandler, SlotNumber, &WriteBuffer, Offset, sizeof(ULONG));
846 
847             /* Read that figure back from the config space */
848             HalpReadPCIConfig(BusHandler, SlotNumber, &Size[Address], Offset, sizeof(ULONG));
849 
850             /* Write back initial value */
851             HalpWritePCIConfig(BusHandler, SlotNumber, &PciConfig.u.type0.BaseAddresses[Address], Offset, sizeof(ULONG));
852         }
853     }
854 
855     /* Interrupt resource */
856     if (0 != PciConfig.u.type0.InterruptPin &&
857         0 != PciConfig.u.type0.InterruptLine &&
858         0xFF != PciConfig.u.type0.InterruptLine)
859         ResourceCount++;
860 
861     /* Allocate output buffer and initialize */
862     *AllocatedResources = ExAllocatePoolWithTag(
863         PagedPool,
864         sizeof(CM_RESOURCE_LIST) +
865         (ResourceCount - 1) * sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR),
866         TAG_HAL);
867 
868     if (NULL == *AllocatedResources)
869         return STATUS_NO_MEMORY;
870 
871     (*AllocatedResources)->Count = 1;
872     (*AllocatedResources)->List[0].InterfaceType = PCIBus;
873     (*AllocatedResources)->List[0].BusNumber = BusHandler->BusNumber;
874     (*AllocatedResources)->List[0].PartialResourceList.Version = 1;
875     (*AllocatedResources)->List[0].PartialResourceList.Revision = 1;
876     (*AllocatedResources)->List[0].PartialResourceList.Count = ResourceCount;
877     Descriptor = (*AllocatedResources)->List[0].PartialResourceList.PartialDescriptors;
878 
879     /* Store configuration information */
880     for (Address = 0; Address < PCI_TYPE0_ADDRESSES; Address++)
881     {
882         if (0 != PciConfig.u.type0.BaseAddresses[Address])
883         {
884             if (PCI_ADDRESS_MEMORY_SPACE ==
885                 (PciConfig.u.type0.BaseAddresses[Address] & 0x1))
886             {
887                 Descriptor->Type = CmResourceTypeMemory;
888                 Descriptor->ShareDisposition = CmResourceShareDeviceExclusive; /* FIXME I have no idea... */
889                 Descriptor->Flags = CM_RESOURCE_MEMORY_READ_WRITE;             /* FIXME Just a guess */
890                 Descriptor->u.Memory.Start.QuadPart = (PciConfig.u.type0.BaseAddresses[Address] & PCI_ADDRESS_MEMORY_ADDRESS_MASK);
891                 Descriptor->u.Memory.Length = PciSize(Size[Address], PCI_ADDRESS_MEMORY_ADDRESS_MASK);
892             }
893             else if (PCI_ADDRESS_IO_SPACE ==
894                 (PciConfig.u.type0.BaseAddresses[Address] & 0x1))
895             {
896                 Descriptor->Type = CmResourceTypePort;
897                 Descriptor->ShareDisposition = CmResourceShareDeviceExclusive; /* FIXME I have no idea... */
898                 Descriptor->Flags = CM_RESOURCE_PORT_IO;                       /* FIXME Just a guess */
899                 Descriptor->u.Port.Start.QuadPart = PciConfig.u.type0.BaseAddresses[Address] &= PCI_ADDRESS_IO_ADDRESS_MASK;
900                 Descriptor->u.Port.Length = PciSize(Size[Address], PCI_ADDRESS_IO_ADDRESS_MASK & 0xffff);
901             }
902             else
903             {
904                 ASSERT(FALSE);
905                 return STATUS_UNSUCCESSFUL;
906             }
907             Descriptor++;
908         }
909     }
910 
911     if (0 != PciConfig.u.type0.InterruptPin &&
912         0 != PciConfig.u.type0.InterruptLine &&
913         0xFF != PciConfig.u.type0.InterruptLine)
914     {
915         Descriptor->Type = CmResourceTypeInterrupt;
916         Descriptor->ShareDisposition = CmResourceShareShared;          /* FIXME Just a guess */
917         Descriptor->Flags = CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;     /* FIXME Just a guess */
918         Descriptor->u.Interrupt.Level = PciConfig.u.type0.InterruptLine;
919         Descriptor->u.Interrupt.Vector = PciConfig.u.type0.InterruptLine;
920         Descriptor->u.Interrupt.Affinity = 0xFFFFFFFF;
921 
922         Descriptor++;
923     }
924 
925     ASSERT(Descriptor == (*AllocatedResources)->List[0].PartialResourceList.PartialDescriptors + ResourceCount);
926 
927     /* FIXME: Should store the resources in the registry resource map */
928 
929     return Status;
930 }
931 
932 ULONG
933 NTAPI
HaliPciInterfaceReadConfig(IN PBUS_HANDLER RootBusHandler,IN ULONG BusNumber,IN PCI_SLOT_NUMBER SlotNumber,IN PVOID Buffer,IN ULONG Offset,IN ULONG Length)934 HaliPciInterfaceReadConfig(IN PBUS_HANDLER RootBusHandler,
935                            IN ULONG BusNumber,
936                            IN PCI_SLOT_NUMBER SlotNumber,
937                            IN PVOID Buffer,
938                            IN ULONG Offset,
939                            IN ULONG Length)
940 {
941     BUS_HANDLER BusHandler;
942 
943     /* Setup fake PCI Bus handler */
944     RtlCopyMemory(&BusHandler, &HalpFakePciBusHandler, sizeof(BUS_HANDLER));
945     BusHandler.BusNumber = BusNumber;
946 
947     /* Read configuration data */
948     HalpReadPCIConfig(&BusHandler, SlotNumber, Buffer, Offset, Length);
949 
950     /* Return length */
951     return Length;
952 }
953 
954 CODE_SEG("INIT")
955 PPCI_REGISTRY_INFO_INTERNAL
956 NTAPI
HalpQueryPciRegistryInfo(VOID)957 HalpQueryPciRegistryInfo(VOID)
958 {
959 #ifndef _MINIHAL_
960     WCHAR NameBuffer[8];
961     OBJECT_ATTRIBUTES  ObjectAttributes;
962     UNICODE_STRING KeyName, ConfigName, IdentName;
963     HANDLE KeyHandle, BusKeyHandle, CardListHandle;
964     NTSTATUS Status;
965     UCHAR KeyBuffer[sizeof(CM_FULL_RESOURCE_DESCRIPTOR) + 100];
966     PKEY_VALUE_FULL_INFORMATION ValueInfo = (PVOID)KeyBuffer;
967     UCHAR PartialKeyBuffer[sizeof(KEY_VALUE_PARTIAL_INFORMATION) +
968                            sizeof(PCI_CARD_DESCRIPTOR)];
969     PKEY_VALUE_PARTIAL_INFORMATION PartialValueInfo = (PVOID)PartialKeyBuffer;
970     KEY_FULL_INFORMATION KeyInformation;
971     ULONG ResultLength;
972     PWSTR Tag;
973     ULONG i, ElementCount;
974     PCM_FULL_RESOURCE_DESCRIPTOR FullDescriptor;
975     PCM_PARTIAL_RESOURCE_DESCRIPTOR PartialDescriptor;
976     PPCI_REGISTRY_INFO PciRegInfo;
977     PPCI_REGISTRY_INFO_INTERNAL PciRegistryInfo;
978     PPCI_CARD_DESCRIPTOR CardDescriptor;
979 
980     /* Setup the object attributes for the key */
981     RtlInitUnicodeString(&KeyName,
982                          L"\\Registry\\Machine\\Hardware\\Description\\"
983                          L"System\\MultiFunctionAdapter");
984     InitializeObjectAttributes(&ObjectAttributes,
985                                &KeyName,
986                                OBJ_CASE_INSENSITIVE,
987                                NULL,
988                                NULL);
989 
990     /* Open the key */
991     Status = ZwOpenKey(&KeyHandle, KEY_READ, &ObjectAttributes);
992     if (!NT_SUCCESS(Status)) return NULL;
993 
994     /* Setup the receiving string */
995     KeyName.Buffer = NameBuffer;
996     KeyName.MaximumLength = sizeof(NameBuffer);
997 
998     /* Setup the configuration and identifier key names */
999     RtlInitUnicodeString(&ConfigName, L"Configuration Data");
1000     RtlInitUnicodeString(&IdentName, L"Identifier");
1001 
1002     /* Keep looping for each ID */
1003     for (i = 0; TRUE; i++)
1004     {
1005         /* Setup the key name */
1006         RtlIntegerToUnicodeString(i, 10, &KeyName);
1007         InitializeObjectAttributes(&ObjectAttributes,
1008                                    &KeyName,
1009                                    OBJ_CASE_INSENSITIVE,
1010                                    KeyHandle,
1011                                    NULL);
1012 
1013         /* Open it */
1014         Status = ZwOpenKey(&BusKeyHandle, KEY_READ, &ObjectAttributes);
1015         if (!NT_SUCCESS(Status))
1016         {
1017             /* None left, fail */
1018             ZwClose(KeyHandle);
1019             return NULL;
1020         }
1021 
1022         /* Read the registry data */
1023         Status = ZwQueryValueKey(BusKeyHandle,
1024                                  &IdentName,
1025                                  KeyValueFullInformation,
1026                                  ValueInfo,
1027                                  sizeof(KeyBuffer),
1028                                  &ResultLength);
1029         if (!NT_SUCCESS(Status))
1030         {
1031             /* Failed, try the next one */
1032             ZwClose(BusKeyHandle);
1033             continue;
1034         }
1035 
1036         /* Get the PCI Tag and validate it */
1037         Tag = (PWSTR)((ULONG_PTR)ValueInfo + ValueInfo->DataOffset);
1038         if ((Tag[0] != L'P') ||
1039             (Tag[1] != L'C') ||
1040             (Tag[2] != L'I') ||
1041             (Tag[3]))
1042         {
1043             /* Not a valid PCI entry, skip it */
1044             ZwClose(BusKeyHandle);
1045             continue;
1046         }
1047 
1048         /* Now read our PCI structure */
1049         Status = ZwQueryValueKey(BusKeyHandle,
1050                                  &ConfigName,
1051                                  KeyValueFullInformation,
1052                                  ValueInfo,
1053                                  sizeof(KeyBuffer),
1054                                  &ResultLength);
1055         ZwClose(BusKeyHandle);
1056         if (!NT_SUCCESS(Status)) continue;
1057 
1058         /* We read it OK! Get the actual resource descriptors */
1059         FullDescriptor  = (PCM_FULL_RESOURCE_DESCRIPTOR)
1060                           ((ULONG_PTR)ValueInfo + ValueInfo->DataOffset);
1061         PartialDescriptor = (PCM_PARTIAL_RESOURCE_DESCRIPTOR)
1062                             ((ULONG_PTR)FullDescriptor->
1063                                         PartialResourceList.PartialDescriptors);
1064 
1065         /* Check if this is our PCI Registry Information */
1066         if (PartialDescriptor->Type == CmResourceTypeDeviceSpecific)
1067         {
1068             /* It is, stop searching */
1069             break;
1070         }
1071     }
1072 
1073     /* Close the key */
1074     ZwClose(KeyHandle);
1075 
1076     /* Save the PCI information for later */
1077     PciRegInfo = (PPCI_REGISTRY_INFO)(PartialDescriptor + 1);
1078 
1079     /* Assume no Card List entries */
1080     ElementCount = 0;
1081 
1082     /* Set up for checking the PCI Card List key */
1083     RtlInitUnicodeString(&KeyName,
1084                          L"\\Registry\\Machine\\System\\CurrentControlSet\\"
1085                          L"Control\\PnP\\PCI\\CardList");
1086     InitializeObjectAttributes(&ObjectAttributes,
1087                                &KeyName,
1088                                OBJ_CASE_INSENSITIVE,
1089                                NULL,
1090                                NULL);
1091 
1092     /* Attempt to open it */
1093     Status = ZwOpenKey(&CardListHandle, KEY_READ, &ObjectAttributes);
1094     if (NT_SUCCESS(Status))
1095     {
1096         /* It exists, so let's query it */
1097         Status = ZwQueryKey(CardListHandle,
1098                             KeyFullInformation,
1099                             &KeyInformation,
1100                             sizeof(KEY_FULL_INFORMATION),
1101                             &ResultLength);
1102         if (!NT_SUCCESS(Status))
1103         {
1104             /* Failed to query, so no info */
1105             PciRegistryInfo = NULL;
1106         }
1107         else
1108         {
1109             /* Allocate the full structure */
1110             PciRegistryInfo =
1111                 ExAllocatePoolWithTag(NonPagedPool,
1112                                       sizeof(PCI_REGISTRY_INFO_INTERNAL) +
1113                                       (KeyInformation.Values *
1114                                        sizeof(PCI_CARD_DESCRIPTOR)),
1115                                        TAG_HAL);
1116             if (PciRegistryInfo)
1117             {
1118                 /* Get the first card descriptor entry */
1119                 CardDescriptor = (PPCI_CARD_DESCRIPTOR)(PciRegistryInfo + 1);
1120 
1121                 /* Loop all the values */
1122                 for (i = 0; i < KeyInformation.Values; i++)
1123                 {
1124                     /* Attempt to get the value */
1125                     Status = ZwEnumerateValueKey(CardListHandle,
1126                                                  i,
1127                                                  KeyValuePartialInformation,
1128                                                  PartialValueInfo,
1129                                                  sizeof(PartialKeyBuffer),
1130                                                  &ResultLength);
1131                     if (!NT_SUCCESS(Status))
1132                     {
1133                         /* Something went wrong, stop the search */
1134                         break;
1135                     }
1136 
1137                     /* Make sure it is correctly sized */
1138                     if (PartialValueInfo->DataLength == sizeof(PCI_CARD_DESCRIPTOR))
1139                     {
1140                         /* Sure is, copy it over */
1141                         *CardDescriptor = *(PPCI_CARD_DESCRIPTOR)
1142                                            PartialValueInfo->Data;
1143 
1144                         /* One more Card List entry */
1145                         ElementCount++;
1146 
1147                         /* Move to the next descriptor */
1148                         CardDescriptor = (CardDescriptor + 1);
1149                     }
1150                 }
1151             }
1152         }
1153 
1154         /* Close the Card List key */
1155         ZwClose(CardListHandle);
1156     }
1157     else
1158     {
1159        /* No key, no Card List */
1160        PciRegistryInfo = NULL;
1161     }
1162 
1163     /* Check if we failed to get the full structure */
1164     if (!PciRegistryInfo)
1165     {
1166         /* Just allocate the basic structure then */
1167         PciRegistryInfo = ExAllocatePoolWithTag(NonPagedPool,
1168                                                 sizeof(PCI_REGISTRY_INFO_INTERNAL),
1169                                                 TAG_HAL);
1170         if (!PciRegistryInfo) return NULL;
1171     }
1172 
1173     /* Save the info we got */
1174     PciRegistryInfo->MajorRevision = PciRegInfo->MajorRevision;
1175     PciRegistryInfo->MinorRevision = PciRegInfo->MinorRevision;
1176     PciRegistryInfo->NoBuses = PciRegInfo->NoBuses;
1177     PciRegistryInfo->HardwareMechanism = PciRegInfo->HardwareMechanism;
1178     PciRegistryInfo->ElementCount = ElementCount;
1179 
1180     /* Return it */
1181     return PciRegistryInfo;
1182 #else
1183     return NULL;
1184 #endif
1185 }
1186 
1187 CODE_SEG("INIT")
1188 VOID
1189 NTAPI
HalpInitializePciStubs(VOID)1190 HalpInitializePciStubs(VOID)
1191 {
1192     PPCI_REGISTRY_INFO_INTERNAL PciRegistryInfo;
1193     UCHAR PciType;
1194     PPCIPBUSDATA BusData = (PPCIPBUSDATA)HalpFakePciBusHandler.BusData;
1195     ULONG i;
1196     PCI_SLOT_NUMBER j;
1197     ULONG VendorId = 0;
1198     ULONG MaxPciBusNumber;
1199 
1200     /* Query registry information */
1201     PciRegistryInfo = HalpQueryPciRegistryInfo();
1202     if (!PciRegistryInfo)
1203     {
1204         /* Assume type 1 */
1205         PciType = 1;
1206 
1207         /* Force a manual bus scan later */
1208         MaxPciBusNumber = MAXULONG;
1209     }
1210     else
1211     {
1212         /* Get the PCI type */
1213         PciType = PciRegistryInfo->HardwareMechanism & 0xF;
1214 
1215         /* Get MaxPciBusNumber and make it 0-based */
1216         MaxPciBusNumber = PciRegistryInfo->NoBuses - 1;
1217 
1218         /* Free the info structure */
1219         ExFreePoolWithTag(PciRegistryInfo, TAG_HAL);
1220     }
1221 
1222     /* Initialize the PCI lock */
1223     KeInitializeSpinLock(&HalpPCIConfigLock);
1224 
1225     /* Check the type of PCI bus */
1226     switch (PciType)
1227     {
1228         /* Type 1 PCI Bus */
1229         case 1:
1230 
1231             /* Copy the Type 1 handler data */
1232             RtlCopyMemory(&PCIConfigHandler,
1233                           &PCIConfigHandlerType1,
1234                           sizeof(PCIConfigHandler));
1235 
1236             /* Set correct I/O Ports */
1237             BusData->Config.Type1.Address = PCI_TYPE1_ADDRESS_PORT;
1238             BusData->Config.Type1.Data = PCI_TYPE1_DATA_PORT;
1239             break;
1240 
1241         /* Type 2 PCI Bus */
1242         case 2:
1243 
1244             /* Copy the Type 2 handler data */
1245             RtlCopyMemory(&PCIConfigHandler,
1246                           &PCIConfigHandlerType2,
1247                           sizeof (PCIConfigHandler));
1248 
1249             /* Set correct I/O Ports */
1250             BusData->Config.Type2.CSE = PCI_TYPE2_CSE_PORT;
1251             BusData->Config.Type2.Forward = PCI_TYPE2_FORWARD_PORT;
1252             BusData->Config.Type2.Base = PCI_TYPE2_ADDRESS_BASE;
1253 
1254             /* Only 16 devices supported, not 32 */
1255             BusData->MaxDevice = 16;
1256             break;
1257 
1258         default:
1259 
1260             /* Invalid type */
1261             DbgPrint("HAL: Unknown PCI type\n");
1262     }
1263 
1264     /* Run a forced bus scan if needed */
1265     if (MaxPciBusNumber == MAXULONG)
1266     {
1267         /* Initialize the max bus number to 0xFF */
1268         HalpMaxPciBus = 0xFF;
1269 
1270         /* Initialize the counter */
1271         MaxPciBusNumber = 0;
1272 
1273         /* Loop all possible buses */
1274         for (i = 0; i < HalpMaxPciBus; i++)
1275         {
1276             /* Loop all devices */
1277             for (j.u.AsULONG = 0; j.u.AsULONG < BusData->MaxDevice; j.u.AsULONG++)
1278             {
1279                 /* Query the interface */
1280                 if (HaliPciInterfaceReadConfig(NULL,
1281                                                i,
1282                                                j,
1283                                                &VendorId,
1284                                                0,
1285                                                sizeof(ULONG)))
1286                 {
1287                     /* Validate the vendor ID */
1288                     if ((VendorId & 0xFFFF) != PCI_INVALID_VENDORID)
1289                     {
1290                         /* Set this as the maximum ID */
1291                         MaxPciBusNumber = i;
1292                         break;
1293                     }
1294                 }
1295             }
1296         }
1297     }
1298 
1299     /* Set the real max bus number */
1300     HalpMaxPciBus = MaxPciBusNumber;
1301 
1302     /* We're done */
1303     HalpPCIConfigInitialized = TRUE;
1304 }
1305 
1306 /* EOF */
1307 
1308