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 * 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 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 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 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 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 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 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 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 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 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 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