1 /*
2 * pnpdump - PnP BIOS information dumper
3 */
4
5 #include <windows.h>
6 #include <string.h>
7 #include <stdio.h>
8 #include <stdlib.h>
9 #include <conio.h>
10
11 #include <ntddk.h>
12
13 #include <pshpack1.h>
14
15 typedef struct _CM_PNP_BIOS_DEVICE_NODE
16 {
17 USHORT Size;
18 UCHAR Node;
19 ULONG ProductId;
20 UCHAR DeviceType[3];
21 USHORT DeviceAttributes;
22 } CM_PNP_BIOS_DEVICE_NODE,*PCM_PNP_BIOS_DEVICE_NODE;
23
24 typedef struct _CM_PNP_BIOS_INSTALLATION_CHECK
25 {
26 UCHAR Signature[4]; // $PnP (ascii)
27 UCHAR Revision;
28 UCHAR Length;
29 USHORT ControlField;
30 UCHAR Checksum;
31 ULONG EventFlagAddress; // Physical address
32 USHORT RealModeEntryOffset;
33 USHORT RealModeEntrySegment;
34 USHORT ProtectedModeEntryOffset;
35 ULONG ProtectedModeCodeBaseAddress;
36 ULONG OemDeviceId;
37 USHORT RealModeDataBaseAddress;
38 ULONG ProtectedModeDataBaseAddress;
39 } CM_PNP_BIOS_INSTALLATION_CHECK, *PCM_PNP_BIOS_INSTALLATION_CHECK;
40
41 #include <poppack.h>
42
43 typedef struct _PNP_ID_NAME_
44 {
45 char *PnpId;
46 char *DeviceName;
47 } PNP_ID_NAME, *PPNP_ID_NAME;
48
49
50 static char Hex[] = "0123456789ABCDEF";
51
52 static PNP_ID_NAME PnpName[] =
53 {
54 /* Interrupt Controllers */
55 {"PNP0000", "AT Interrupt Controller"},
56 {"PNP0001", "EISA Interrupt Controller"},
57 {"PNP0002", "MCA Interrupt Controller"},
58 {"PNP0003", "APIC"},
59 {"PNP0004", "Cyrix SLiC MP Interrupt Controller"},
60
61 /* Timers */
62 {"PNP0100", "AT Timer"},
63 {"PNP0101", "EISA Timer"},
64 {"PNP0102", "MCA Timer"},
65
66 /* DMA Controllers */
67 {"PNP0200", "AT DMA Controller"},
68 {"PNP0201", "EISA DMA Controller"},
69 {"PNP0202", "MCA DMA Controller"},
70
71 /* Keyboards */
72 {"PNP0300", "IBM PC/XT Keyboard (83 keys)"},
73 {"PNP0301", "IBM PC/AT Keyboard (86 keys)"},
74 {"PNP0302", "IBM PC/XT Keyboard (84 keys)"},
75 {"PNP0303", "IBM Enhanced (101/102 keys)"},
76 {"PNP0304", "Olivetti Keyboard (83 keys)"},
77 {"PNP0305", "Olivetti Keyboard (102 keys)"},
78 {"PNP0306", "Olivetti Keyboard (86 keys)"},
79 {"PNP0307", "Microsoft Windows(R) Keyboard"},
80 {"PNP0308", "General Input Device Emulation Interface (GIDEI) legacy"},
81 {"PNP0309", "Olivetti Keyboard (A101/102 key)"},
82 {"PNP030A", "AT&T 302 keyboard"},
83 {"PNP030B", "Reserved by Microsoft"},
84 {"PNP0320", "Japanese 101-key keyboard"},
85 {"PNP0321", "Japanese AX keyboard"},
86 {"PNP0322", "Japanese 106-key keyboard A01"},
87 {"PNP0323", "Japanese 106-key keyboard 002/003"},
88 {"PNP0324", "Japanese 106-key keyboard 001"},
89 {"PNP0325", "Japanese Toshiba Desktop keyboard"},
90 {"PNP0326", "Japanese Toshiba Laptop keyboard"},
91 {"PNP0327", "Japanese Toshiba Notebook keyboard"},
92 {"PNP0340", "Korean 84-key keyboard"},
93 {"PNP0341", "Korean 86-key keyboard"},
94 {"PNP0342", "Korean Enhanced keyboard"},
95 {"PNP0343", "Korean Enhanced keyboard 101b"},
96 {"PNP0343", "Korean Enhanced keyboard 101c"},
97 {"PNP0344", "Korean Enhanced keyboard 103"},
98
99 /* Parallel Ports */
100 {"PNP0400", "Standard LPT printer port"},
101 {"PNP0401", "ECP printer port"},
102
103 /* Serial Ports */
104 {"PNP0500", "Standard PC COM port"},
105 {"PNP0501", "16550A-compatible COM port"},
106 {"PNP0510", "Generic IRDA-compatible port"},
107
108 /* Harddisk Controllers */
109 {"PNP0600", "Generic ESDI/ATA/IDE harddisk controller"},
110 {"PNP0601", "Plus Hardcard II"},
111 {"PNP0602", "Plus Hardcard IIXL/EZ"},
112 {"PNP0603", "Generic IDE supporting Microsoft Device Bay Specification"},
113
114 /* Floppy Controllers */
115 {"PNP0700", "PC standard floppy disk controller"},
116 {"PNP0701", "Standard floppy controller supporting MS Device Bay Specification"},
117
118 /* obsolete devices */
119 {"PNP0800", "Microsoft Sound System compatible device"},
120
121 /* Display Adapters */
122 {"PNP0900", "VGA Compatible"},
123 {"PNP0901", "Video Seven VRAM/VRAM II/1024i"},
124 {"PNP0902", "8514/A Compatible"},
125 {"PNP0903", "Trident VGA"},
126 {"PNP0904", "Cirrus Logic Laptop VGA"},
127 {"PNP0905", "Cirrus Logic VGA"},
128 {"PNP0906", "Tseng ET4000"},
129 {"PNP0907", "Western Digital VGA"},
130 {"PNP0908", "Western Digital Laptop VGA"},
131 {"PNP0909", "S3 Inc. 911/924"},
132 {"PNP090A", "ATI Ultra Pro/Plus (Mach 32)"},
133 {"PNP090B", "ATI Ultra (Mach 8)"},
134 {"PNP090C", "XGA Compatible"},
135 {"PNP090D", "ATI VGA Wonder"},
136 {"PNP090E", "Weitek P9000 Graphics Adapter"},
137 {"PNP090F", "Oak Technology VGA"},
138 {"PNP0910", "Compaq QVision"},
139 {"PNP0911", "XGA/2"},
140 {"PNP0912", "Tseng Labs W32/W32i/W32p"},
141 {"PNP0913", "S3 Inc. 801/928/964"},
142 {"PNP0914", "Cirrus Logic 5429/5434 (memory mapped)"},
143 {"PNP0915", "Compaq Advanced VGA (AVGA)"},
144 {"PNP0916", "ATI Ultra Pro Turbo (Mach64)"},
145 {"PNP0917", "Reserved by Microsoft"},
146 {"PNP0918", "Matrox MGA"},
147 {"PNP0919", "Compaq QVision 2000"},
148 {"PNP091A", "Tseng W128"},
149 {"PNP0930", "Chips & Technologies Super VGA"},
150 {"PNP0931", "Chips & Technologies Accelerator"},
151 {"PNP0940", "NCR 77c22e Super VGA"},
152 {"PNP0941", "NCR 77c32blt"},
153 {"PNP09FF", "Plug and Play Monitors (VESA DDC)"},
154
155 /* Peripheral Buses */
156 {"PNP0A00", "ISA Bus"},
157 {"PNP0A01", "EISA Bus"},
158 {"PNP0A02", "MCA Bus"},
159 {"PNP0A03", "PCI Bus"},
160 {"PNP0A04", "VESA/VL Bus"},
161 {"PNP0A05", "Generic ACPI Bus"},
162 {"PNP0A06", "Generic ACPI Extended-IO Bus (EIO bus)"},
163
164 /* System devices */
165 {"PNP0800", "AT-style speaker sound"},
166 {"PNP0B00", "AT Real-Time Clock"},
167 {"PNP0C00", "Plug and Play BIOS (only created by the root enumerator)"},
168 {"PNP0C01", "System Board"},
169 {"PNP0C02", "General Plug and Play motherboard registers."},
170 {"PNP0C03", "Plug and Play BIOS Event Notification Interrupt"},
171 {"PNP0C04", "Math Coprocessor"},
172 {"PNP0C05", "APM BIOS (Version independent)"},
173 {"PNP0C06", "Reserved for identification of early Plug and Play BIOS implementation"},
174 {"PNP0C07", "Reserved for identification of early Plug and Play BIOS implementation"},
175 {"PNP0C08", "ACPI system board hardware"},
176 {"PNP0C09", "ACPI Embedded Controller"},
177 {"PNP0C0A", "ACPI Control Method Battery"},
178 {"PNP0C0B", "ACPI Fan"},
179 {"PNP0C0C", "ACPI power button device"},
180 {"PNP0C0D", "ACPI lid device"},
181 {"PNP0C0E", "ACPI sleep button device"},
182 {"PNP0C0F", "PCI interrupt link device"},
183 {"PNP0C10", "ACPI system indicator device"},
184 {"PNP0C11", "ACPI thermal zone"},
185 {"PNP0C12", "Device Bay Controller"},
186
187 /* PCMCIA Controllers */
188 {"PNP0E00", "Intel 82365-Compatible PCMCIA Controller"},
189 {"PNP0E01", "Cirrus Logic CL-PD6720 PCMCIA Controller"},
190 {"PNP0E02", "VLSI VL82C146 PCMCIA Controller"},
191 {"PNP0E03", "Intel 82365-compatible CardBus controller"},
192
193 /* Mice */
194 {"PNP0F00", "Microsoft Bus Mouse"},
195 {"PNP0F01", "Microsoft Serial Mouse"},
196 {"PNP0F02", "Microsoft InPort Mouse"},
197 {"PNP0F03", "Microsoft PS/2-style Mouse"},
198 {"PNP0F04", "Mouse Systems Mouse"},
199 {"PNP0F05", "Mouse Systems 3-Button Mouse (COM2)"},
200 {"PNP0F06", "Genius Mouse (COM1)"},
201 {"PNP0F07", "Genius Mouse (COM2)"},
202 {"PNP0F08", "Logitech Serial Mouse"},
203 {"PNP0F09", "Microsoft BallPoint Serial Mouse"},
204 {"PNP0F0A", "Microsoft Plug and Play Mouse"},
205 {"PNP0F0B", "Microsoft Plug and Play BallPoint Mouse"},
206 {"PNP0F0C", "Microsoft-compatible Serial Mouse"},
207 {"PNP0F0D", "Microsoft-compatible InPort-compatible Mouse"},
208 {"PNP0F0E", "Microsoft-compatible PS/2-style Mouse"},
209 {"PNP0F0F", "Microsoft-compatible Serial BallPoint-compatible Mouse"},
210 {"PNP0F10", "Texas Instruments QuickPort Mouse"},
211 {"PNP0F11", "Microsoft-compatible Bus Mouse"},
212 {"PNP0F12", "Logitech PS/2-style Mouse"},
213 {"PNP0F13", "PS/2 Port for PS/2-style Mice"},
214 {"PNP0F14", "Microsoft Kids Mouse"},
215 {"PNP0F15", "Logitech bus mouse"},
216 {"PNP0F16", "Logitech SWIFT device"},
217 {"PNP0F17", "Logitech-compatible serial mouse"},
218 {"PNP0F18", "Logitech-compatible bus mouse"},
219 {"PNP0F19", "Logitech-compatible PS/2-style Mouse"},
220 {"PNP0F1A", "Logitech-compatible SWIFT Device"},
221 {"PNP0F1B", "HP Omnibook Mouse"},
222 {"PNP0F1C", "Compaq LTE Trackball PS/2-style Mouse"},
223 {"PNP0F1D", "Compaq LTE Trackball Serial Mouse"},
224 {"PNP0F1E", "Microsoft Kids Trackball Mouse"},
225 {"PNP0F1F", "Reserved by Microsoft Input Device Group"},
226 {"PNP0F20", "Reserved by Microsoft Input Device Group"},
227 {"PNP0F21", "Reserved by Microsoft Input Device Group"},
228 {"PNP0F22", "Reserved by Microsoft Input Device Group"},
229 {"PNP0F23", "Reserved by Microsoft Input Device Group"},
230 {"PNP0FFF", "Reserved by Microsoft Systems"},
231
232 /* List Terminator */
233 {NULL, NULL}
234 };
235
236
237 /* FUNCTIONS ****************************************************************/
238
239 static char *
GetDeviceName(char * PnpId)240 GetDeviceName(char *PnpId)
241 {
242 PPNP_ID_NAME IdName;
243
244 IdName = PnpName;
245 while (IdName->PnpId != NULL)
246 {
247 if (!strcmp(IdName->PnpId, PnpId))
248 return IdName->DeviceName;
249
250 IdName++;
251 }
252
253 return "Unknown Device";
254 }
255
256
257 LONG
GetPnpKey(PHKEY PnpKey)258 GetPnpKey(PHKEY PnpKey)
259 {
260 LONG lError;
261 char szBuffer[80];
262 HKEY hAdapterKey;
263 HKEY hBusKey;
264 DWORD dwBus;
265 DWORD dwType;
266 DWORD dwSize;
267
268 *PnpKey = 0;
269
270 lError = RegOpenKey(HKEY_LOCAL_MACHINE,
271 "HARDWARE\\DESCRIPTION\\System\\MultifunctionAdapter",
272 &hAdapterKey);
273 if (lError != ERROR_SUCCESS)
274 return 0;
275
276 /* Enumerate buses */
277 for (dwBus = 0; ; dwBus++)
278 {
279 sprintf(szBuffer, "%lu", dwBus);
280
281 lError = RegOpenKey(hAdapterKey,
282 szBuffer,
283 &hBusKey);
284 if (lError != ERROR_SUCCESS)
285 {
286 RegCloseKey(hAdapterKey);
287 return lError;
288 }
289
290 dwSize = 80;
291 lError = RegQueryValueEx(hBusKey,
292 "Identifier",
293 NULL,
294 &dwType,
295 (LPBYTE)szBuffer,
296 &dwSize);
297 if (lError != ERROR_SUCCESS)
298 {
299 RegCloseKey(hBusKey);
300 RegCloseKey(hAdapterKey);
301 return lError;
302 }
303
304 if (dwType == REG_SZ && _stricmp(szBuffer, "pnp bios") == 0)
305 {
306 *PnpKey = hBusKey;
307 RegCloseKey(hAdapterKey);
308 return ERROR_SUCCESS;
309 }
310
311 RegCloseKey(hBusKey);
312 }
313
314 return 1;
315 }
316
317
318 static VOID
PnpDecodeIrq(unsigned char * Ptr)319 PnpDecodeIrq(unsigned char *Ptr)
320 {
321 USHORT IrqMask;
322 int i;
323
324 IrqMask = *Ptr;
325 Ptr++;
326 IrqMask |= (*Ptr << 8);
327
328 printf(" IRQs:");
329
330 for (i = 0; i < 16; i++)
331 {
332 if (IrqMask & (1 << i))
333 {
334 printf(" %u", i);
335 }
336 }
337
338 printf("\n");
339 }
340
341
342 static VOID
PnpDecodeDma(unsigned char * Ptr)343 PnpDecodeDma(unsigned char *Ptr)
344 {
345 unsigned char DmaChannel;
346 unsigned char DmaStatus;
347 int i;
348
349 DmaChannel = *Ptr;
350 Ptr++;
351 DmaStatus = *Ptr;
352
353 printf(" DMAs:");
354
355 for (i = 0; i < 8; i++)
356 {
357 if (DmaChannel & (1 << i))
358 {
359 printf(" %u", i);
360 }
361 }
362
363 printf("\n");
364 }
365
366
367 static VOID
PnpDecodeIoPort(unsigned char * Ptr)368 PnpDecodeIoPort(unsigned char *Ptr)
369 {
370 USHORT MinBase;
371 USHORT MaxBase;
372 UCHAR Align;
373 UCHAR Length;
374
375 // Info = *Ptr;
376 Ptr++;
377 MinBase = *Ptr;
378 Ptr++;
379 MinBase += (*Ptr << 8);
380 Ptr++;
381 MaxBase = *Ptr;
382 Ptr++;
383 MaxBase += (*Ptr << 8);
384 Ptr++;
385 Align = *Ptr;
386 Ptr++;
387 Length = *Ptr;
388
389 printf(" I/O Port descriptor\n");
390 printf(" MinBase 0x%x MaxBase 0x%x Align %u Length %u\n",
391 MinBase, MaxBase, Align, Length);
392 }
393
394
395 static VOID
PnpDecodeFixedIoPort(unsigned char * Ptr)396 PnpDecodeFixedIoPort(unsigned char *Ptr)
397 {
398 USHORT IoPort;
399 UCHAR Length;
400
401 IoPort = *Ptr;
402 Ptr++;
403 IoPort += (*Ptr << 8);
404 Ptr++;
405 Length = *Ptr;
406
407 printf(" Fixed I/O Port descriptor\n");
408 printf(" PortBase 0x%hx Length 0x%x\n",
409 IoPort, Length);
410
411 #if 0
412 if (Length == 1)
413 {
414 printf(" Fixed location I/O Port descriptor: 0x%x\n",
415 IoPort);
416 }
417 else
418 {
419 printf(" Fixed location I/O Port descriptor: 0x%x - 0x%x\n",
420 IoPort,
421 IoPort + Length - 1);
422 }
423 #endif
424 }
425
426
427 static VOID
PnpDecodeMemory16(unsigned char * Ptr)428 PnpDecodeMemory16(unsigned char *Ptr)
429 {
430 UCHAR Info;
431 USHORT MinBase;
432 USHORT MaxBase;
433 USHORT Align;
434 USHORT Length;
435
436 Info = *Ptr;
437 Ptr++;
438
439 MinBase = *Ptr;
440 Ptr++;
441 MinBase += (*Ptr << 8);
442 Ptr++;
443
444 MaxBase = *Ptr;
445 Ptr++;
446 MaxBase += (*Ptr << 8);
447 Ptr++;
448
449 Align = *Ptr;
450 Ptr++;
451 Align += (*Ptr << 8);
452 Ptr++;
453
454 Length = *Ptr;
455 Ptr++;
456 Length += (*Ptr << 8);
457
458 printf(" 16-Bit memory range descriptor\n");
459 printf(" MinBase 0x%hx MaxBase 0x%hx Align 0x%hx Length 0x%hx Flags 0x%02x\n",
460 MinBase, MaxBase, Align,Length, Info);
461 }
462
463
464 static VOID
PnpDecodeMemory32(unsigned char * Ptr)465 PnpDecodeMemory32(unsigned char *Ptr)
466 {
467 UCHAR Info;
468 ULONG MinBase;
469 ULONG MaxBase;
470 ULONG Align;
471 ULONG Length;
472
473 Info = *Ptr;
474 Ptr++;
475
476 MinBase = *Ptr;
477 Ptr++;
478 MinBase += (*Ptr << 8);
479 Ptr++;
480 MinBase += (*Ptr << 16);
481 Ptr++;
482 MinBase += (*Ptr << 24);
483 Ptr++;
484
485 MaxBase = *Ptr;
486 Ptr++;
487 MaxBase += (*Ptr << 8);
488 Ptr++;
489 MaxBase += (*Ptr << 16);
490 Ptr++;
491 MaxBase += (*Ptr << 24);
492 Ptr++;
493
494 Align = *Ptr;
495 Ptr++;
496 Align += (*Ptr << 8);
497 Ptr++;
498 Align += (*Ptr << 16);
499 Ptr++;
500 Align += (*Ptr << 24);
501 Ptr++;
502
503 Length = *Ptr;
504 Ptr++;
505 Length += (*Ptr << 8);
506 Ptr++;
507 Length += (*Ptr << 16);
508 Ptr++;
509 Length += (*Ptr << 24);
510
511 printf(" 32-Bit memory range descriptor\n");
512 printf(" MinBase 0x%lx MaxBase 0x%lx Align 0x%lx Length 0x%lx Flags 0x%02x\n",
513 MinBase, MaxBase, Align,Length, Info);
514 }
515
516
517 static VOID
PnpDecodeFixedMemory(unsigned char * Ptr)518 PnpDecodeFixedMemory(unsigned char *Ptr)
519 {
520 UCHAR Info;
521 ULONG Base;
522 ULONG Length;
523
524 Info = *Ptr;
525 Ptr++;
526
527 Base = *Ptr;
528 Ptr++;
529 Base += (*Ptr << 8);
530 Ptr++;
531 Base += (*Ptr << 16);
532 Ptr++;
533 Base += (*Ptr << 24);
534 Ptr++;
535
536 Length = *Ptr;
537 Ptr++;
538 Length += (*Ptr << 8);
539 Ptr++;
540 Length += (*Ptr << 16);
541 Ptr++;
542 Length += (*Ptr << 24);
543
544 printf(" 32-Bit fixed location memory range descriptor\n");
545 printf(" Base 0x%lx Length 0x%lx Flags 0x%02x\n",
546 Base, Length, Info);
547 }
548
549
PrintDeviceData(PCM_PNP_BIOS_DEVICE_NODE DeviceNode)550 void PrintDeviceData (PCM_PNP_BIOS_DEVICE_NODE DeviceNode)
551 {
552 char PnpId[8];
553 unsigned char *Ptr;
554 unsigned int TagSize;
555 unsigned int TagType;
556
557 unsigned char Id[4];
558
559 printf ("Node: %x Size %hu (0x%hx)\n",
560 DeviceNode->Node,
561 DeviceNode->Size,
562 DeviceNode->Size);
563
564 memcpy(Id, &DeviceNode->ProductId, 4);
565
566 PnpId[0] = ((Id[0] >> 2) & 0x1F) + 0x40;
567 PnpId[1] = ((Id[0] << 3) & 0x18) +
568 ((Id[1] >> 5) & 0x07) + 0x40;
569 PnpId[2] = (Id[1] & 0x1F) + 0x40;
570
571 PnpId[3] = Hex[(Id[2] >> 4) & 0xF];
572 PnpId[4] = Hex[Id[2] & 0x0F];
573
574 PnpId[5] = Hex[(Id[3] >> 4) & 0x0F];
575 PnpId[6] = Hex[Id[3] & 0x0F];
576 PnpId[7] = 0;
577
578 printf(" '%s' (%s)\n",
579 PnpId, GetDeviceName(PnpId));
580
581 if (DeviceNode->Size > sizeof(CM_PNP_BIOS_DEVICE_NODE))
582 {
583 Ptr = (unsigned char *)(DeviceNode + 1);
584 while (TRUE)
585 {
586 if (*Ptr & 0x80)
587 {
588 TagType = *Ptr & 0x7F;
589 Ptr++;
590 TagSize = *Ptr;
591 Ptr++;
592 TagSize += (*Ptr << 16);
593 Ptr++;
594
595
596 switch (TagType)
597 {
598 case 1:
599 PnpDecodeMemory16(Ptr);
600 break;
601
602 case 5:
603 PnpDecodeMemory32(Ptr);
604 break;
605
606 case 6:
607 PnpDecodeFixedMemory(Ptr);
608 break;
609
610 default:
611 printf(" Large tag: type %u size %u\n",
612 TagType,
613 TagSize);
614 break;
615 }
616 }
617 else
618 {
619 TagType = (*Ptr >> 3) & 0x0F;
620 TagSize = *Ptr & 0x07;
621 Ptr++;
622
623 switch (TagType)
624 {
625 case 2:
626 printf(" Logical device ID\n");
627 break;
628
629 case 3:
630 printf(" Compatible device ID\n");
631 break;
632
633 case 4:
634 PnpDecodeIrq(Ptr);
635 break;
636
637 case 5:
638 PnpDecodeDma(Ptr);
639 break;
640
641 case 8:
642 PnpDecodeIoPort(Ptr);
643 break;
644
645 case 9:
646 PnpDecodeFixedIoPort(Ptr);
647 break;
648
649 case 0x0F: /* end tag */
650 break;
651
652 default:
653 printf(" Small tag: type %u size %u\n",
654 TagType,
655 TagSize);
656 break;
657 }
658
659 /* end tag */
660 if (TagType == 0x0F)
661 break;
662 }
663
664 Ptr = Ptr + TagSize;
665 }
666 }
667 }
668
669
main(int argc,char * argv[])670 int main (int argc, char *argv[])
671 {
672 LONG lError;
673 HKEY hPnpKey;
674 DWORD dwType;
675 DWORD dwSize;
676 BOOL Ask;
677 PCM_FULL_RESOURCE_DESCRIPTOR lpBuffer;
678 PCM_PNP_BIOS_INSTALLATION_CHECK lpPnpInst;
679 PCM_PNP_BIOS_DEVICE_NODE lpDevNode;
680 DWORD dwDataSize;
681 DWORD dwResourceSize;
682
683 hPnpKey = 0;
684
685 Ask = TRUE;
686 if (argc >1 && (!strcmp(argv[1],"/S") || !strcmp(argv[1],"/s")))
687 {
688 Ask = FALSE;
689 }
690
691 if (argc >1 && !strcmp(argv[1],"/?"))
692 {
693 printf("This utility prints the PnP-nodes from the registry\n");
694 printf("\"/s\" prevents the \"Press any key\"\n\n");
695 return 0;
696 }
697
698 lError = GetPnpKey(&hPnpKey);
699 if (lError != ERROR_SUCCESS)
700 {
701 printf("Failed to get PnP-BIOS key\n");
702 return 0;
703 }
704
705 if (hPnpKey != 0)
706 {
707 printf("Found PnP-BIOS key\n");
708 }
709
710 /* Allocate buffer */
711 dwSize = 2048;
712 lpBuffer = malloc(dwSize);
713 if (lpBuffer == NULL)
714 {
715 printf("Error: malloc() failed\n");
716 RegCloseKey(hPnpKey);
717 return 0;
718 }
719
720 do
721 {
722 lError = RegQueryValueEx(hPnpKey,
723 "Configuration Data",
724 NULL,
725 &dwType,
726 (LPBYTE)lpBuffer,
727 &dwSize);
728 if (lError == ERROR_MORE_DATA)
729 {
730 lpBuffer = realloc(lpBuffer, dwSize);
731 if (lpBuffer == NULL)
732 {
733 printf("Error: realloc() of %u bytes failed\n", (unsigned) dwSize);
734 RegCloseKey(hPnpKey);
735 return 0;
736 }
737 }
738 }
739 while (lError == ERROR_MORE_DATA);
740 if (lError != ERROR_SUCCESS)
741 {
742 printf("Failed to read 'Configuration Data' value\n");
743 free(lpBuffer);
744 RegCloseKey(hPnpKey);
745 return 0;
746 }
747
748 // printf ("Data size: %lu\n", dwSize);
749
750 RegCloseKey(hPnpKey);
751
752 // printf("Resource count %lu\n", lpBuffer->PartialResourceList.Count);
753
754 if (lpBuffer->PartialResourceList.Count == 0)
755 {
756 printf("Invalid resource count!\n");
757 free(lpBuffer);
758 return 0;
759 }
760
761 // printf("lpBuffer %p\n", lpBuffer);
762
763 dwResourceSize = lpBuffer->PartialResourceList.PartialDescriptors[0].u.DeviceSpecificData.DataSize;
764 // printf("ResourceSize: %lu\n", dwResourceSize);
765
766 lpPnpInst = (PCM_PNP_BIOS_INSTALLATION_CHECK)
767 ((ULONG_PTR)(&lpBuffer->PartialResourceList.PartialDescriptors[0]) +
768 sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR));
769
770 // printf("lpPnpInst %p\n", lpPnpInst);
771
772 printf("Signature '%.4s'\n", lpPnpInst->Signature);
773 if (strncmp((PCHAR)lpPnpInst->Signature, "$PnP", 4))
774 {
775 printf("Error: Invalid PnP signature\n");
776 free(lpBuffer);
777 return 0;
778 }
779
780 // printf("InstCheck length: %lu\n", lpPnpInst->Length);
781
782 dwDataSize = sizeof(CM_PNP_BIOS_INSTALLATION_CHECK);
783 lpDevNode = (PCM_PNP_BIOS_DEVICE_NODE)((DWORD)lpPnpInst + sizeof(CM_PNP_BIOS_INSTALLATION_CHECK));
784
785 if (lpDevNode->Size == 0)
786 {
787 printf("Error: Device node size is zero!\n");
788 return 0;
789 }
790
791 #if 0
792 printf("Node: %x Size %hu (0x%hx)\n",
793 lpDevNode->Node,
794 lpDevNode->Size,
795 lpDevNode->Size);
796
797 printf("Done.\n");
798 return 0;
799 #endif
800
801
802 while (dwDataSize < dwResourceSize)
803 {
804 if (lpDevNode->Size == 0)
805 break;
806
807 printf("Node: %x Size %hu (0x%hx)\n",
808 lpDevNode->Node,
809 lpDevNode->Size,
810 lpDevNode->Size);
811
812 dwDataSize += lpDevNode->Size;
813 lpDevNode = (PCM_PNP_BIOS_DEVICE_NODE)((DWORD)lpDevNode + lpDevNode->Size);
814 }
815
816 if (Ask)
817 {
818 printf("\n Press any key...\n");
819 getch();
820 }
821 else
822 {
823 printf("\n");
824 }
825
826 dwDataSize = sizeof(CM_PNP_BIOS_INSTALLATION_CHECK);
827 lpDevNode = (PCM_PNP_BIOS_DEVICE_NODE)((DWORD)lpPnpInst + sizeof(CM_PNP_BIOS_INSTALLATION_CHECK));
828
829 while (dwDataSize < dwResourceSize)
830 {
831 if (lpDevNode->Size == 0)
832 break;
833
834 PrintDeviceData(lpDevNode);
835
836 if (Ask)
837 {
838 printf("\n Press any key...\n");
839 getch();
840 }
841 else
842 {
843 printf("\n");
844 }
845
846 dwDataSize += lpDevNode->Size;
847 lpDevNode = (PCM_PNP_BIOS_DEVICE_NODE)((DWORD)lpDevNode + lpDevNode->Size);
848 }
849
850 free(lpBuffer);
851
852 return 0;
853 }
854
855 /* EOF */
856