Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

6210 lines
178 KiB

  1. /*++
  2. Copyright (c) 1997-2000 Microsoft Corporation
  3. Module Name:
  4. devnode.c
  5. Abstract:
  6. WinDbg Extension Api
  7. Revision History:
  8. --*/
  9. #include "precomp.h"
  10. //#include "pci.h"
  11. #pragma hdrstop
  12. #include "stddef.h"
  13. #include <initguid.h>
  14. #define FLAG_NAME(flag) {flag, #flag}
  15. FLAG_NAME DeviceNodeFlags[] = {
  16. FLAG_NAME(DNF_MADEUP), // 00000001
  17. FLAG_NAME(DNF_DUPLICATE), // 00000002
  18. FLAG_NAME(DNF_HAL_NODE), // 00000004
  19. FLAG_NAME(DNF_REENUMERATE), // 00000008
  20. FLAG_NAME(DNF_ENUMERATED), // 00000010
  21. FLAG_NAME(DNF_IDS_QUERIED), // 00000020
  22. FLAG_NAME(DNF_HAS_BOOT_CONFIG), // 00000040
  23. FLAG_NAME(DNF_BOOT_CONFIG_RESERVED), // 00000080
  24. FLAG_NAME(DNF_NO_RESOURCE_REQUIRED), // 00000100
  25. FLAG_NAME(DNF_RESOURCE_REQUIREMENTS_NEED_FILTERED), // 00000200
  26. FLAG_NAME(DNF_RESOURCE_REQUIREMENTS_CHANGED), // 00000400
  27. FLAG_NAME(DNF_NON_STOPPED_REBALANCE), // 00000800
  28. FLAG_NAME(DNF_LEGACY_DRIVER), // 00001000
  29. FLAG_NAME(DNF_HAS_PROBLEM), // 00002000
  30. FLAG_NAME(DNF_HAS_PRIVATE_PROBLEM), // 00004000
  31. FLAG_NAME(DNF_HARDWARE_VERIFICATION), // 00008000
  32. FLAG_NAME(DNF_DEVICE_GONE), // 00010000
  33. FLAG_NAME(DNF_LEGACY_RESOURCE_DEVICENODE), // 00020000
  34. FLAG_NAME(DNF_NEEDS_REBALANCE), // 00040000
  35. FLAG_NAME(DNF_LOCKED_FOR_EJECT), // 00080000
  36. FLAG_NAME(DNF_DRIVER_BLOCKED), // 00100000
  37. {0,0}
  38. };
  39. FLAG_NAME DeviceNodeUserFlags[] = {
  40. FLAG_NAME(DNUF_WILL_BE_REMOVED), // 00000001
  41. FLAG_NAME(DNUF_DONT_SHOW_IN_UI), // 00000002
  42. FLAG_NAME(DNUF_NEED_RESTART), // 00000004
  43. FLAG_NAME(DNUF_NOT_DISABLEABLE), // 00000008
  44. FLAG_NAME(DNUF_SHUTDOWN_QUERIED), // 00000010
  45. FLAG_NAME(DNUF_SHUTDOWN_SUBTREE_DONE), // 00000020
  46. {0,0}
  47. };
  48. #define DeviceD1 0x00000001
  49. #define DeviceD2 0x00000002
  50. #define LockSupported 0x00000004
  51. #define EjectSupported 0x00000008
  52. #define Removable 0x00000010
  53. #define DockDevice 0x00000020
  54. #define UniqueID 0x00000040
  55. #define SilentInstall 0x00000080
  56. #define RawDeviceOK 0x00000100
  57. #define SurpriseRemovalOK 0x00000200
  58. #define WakeFromD0 0x00000400
  59. #define WakeFromD1 0x00000800
  60. #define WakeFromD2 0x00001000
  61. #define WakeFromD3 0x00002000
  62. #define HardwareDisabled 0x00004000
  63. #define NonDynamic 0x00008000
  64. #define WarmEjectSupported 0x00010000
  65. #define NoDisplayInUI 0x00020000
  66. FLAG_NAME DeviceNodeCapabilityFlags[] = {
  67. FLAG_NAME(DeviceD1), // 00000001
  68. FLAG_NAME(DeviceD2), // 00000002
  69. FLAG_NAME(LockSupported), // 00000004
  70. FLAG_NAME(EjectSupported), // 00000008
  71. FLAG_NAME(Removable), // 00000010
  72. FLAG_NAME(DockDevice), // 00000020
  73. FLAG_NAME(UniqueID), // 00000040
  74. FLAG_NAME(SilentInstall), // 00000080
  75. FLAG_NAME(RawDeviceOK), // 00000100
  76. FLAG_NAME(SurpriseRemovalOK), // 00000200
  77. FLAG_NAME(WakeFromD0), // 00000400
  78. FLAG_NAME(WakeFromD1), // 00000800
  79. FLAG_NAME(WakeFromD2), // 00001000
  80. FLAG_NAME(WakeFromD3), // 00002000
  81. FLAG_NAME(HardwareDisabled), // 00004000
  82. FLAG_NAME(NonDynamic), // 00008000
  83. FLAG_NAME(WarmEjectSupported), // 00010000
  84. FLAG_NAME(NoDisplayInUI), // 00020000
  85. {0,0}
  86. };
  87. #define PROBLEM_MAP_SIZE 0x32
  88. #if PROBLEM_MAP_SIZE != NUM_CM_PROB
  89. #error Add new problem code to DevNodeProblems and update PROBLEM_MAP_SIZE.
  90. #endif
  91. PUCHAR DevNodeProblems[] = {
  92. NULL,
  93. "CM_PROB_NOT_CONFIGURED",
  94. "CM_PROB_DEVLOADER_FAILED",
  95. "CM_PROB_OUT_OF_MEMORY",
  96. "CM_PROB_ENTRY_IS_WRONG_TYPE",
  97. "CM_PROB_LACKED_ARBITRATOR",
  98. "CM_PROB_BOOT_CONFIG_CONFLICT",
  99. "CM_PROB_FAILED_FILTER",
  100. "CM_PROB_DEVLOADER_NOT_FOUND",
  101. "CM_PROB_INVALID_DATA",
  102. "CM_PROB_FAILED_START",
  103. "CM_PROB_LIAR",
  104. "CM_PROB_NORMAL_CONFLICT",
  105. "CM_PROB_NOT_VERIFIED",
  106. "CM_PROB_NEED_RESTART",
  107. "CM_PROB_REENUMERATION",
  108. "CM_PROB_PARTIAL_LOG_CONF",
  109. "CM_PROB_UNKNOWN_RESOURCE",
  110. "CM_PROB_REINSTALL",
  111. "CM_PROB_REGISTRY",
  112. "CM_PROB_VXDLDR",
  113. "CM_PROB_WILL_BE_REMOVED",
  114. "CM_PROB_DISABLED",
  115. "CM_PROB_DEVLOADER_NOT_READY",
  116. "CM_PROB_DEVICE_NOT_THERE",
  117. "CM_PROB_MOVED",
  118. "CM_PROB_TOO_EARLY",
  119. "CM_PROB_NO_VALID_LOG_CONF",
  120. "CM_PROB_FAILED_INSTALL",
  121. "CM_PROB_HARDWARE_DISABLED",
  122. "CM_PROB_CANT_SHARE_IRQ",
  123. "CM_PROB_FAILED_ADD",
  124. "CM_PROB_DISABLED_SERVICE",
  125. "CM_PROB_TRANSLATION_FAILED",
  126. "CM_PROB_NO_SOFTCONFIG",
  127. "CM_PROB_BIOS_TABLE",
  128. "CM_PROB_IRQ_TRANSLATION_FAILED",
  129. "CM_PROB_FAILED_DRIVER_ENTRY",
  130. "CM_PROB_DRIVER_FAILED_PRIOR_UNLOAD",
  131. "CM_PROB_DRIVER_FAILED_LOAD",
  132. "CM_PROB_DRIVER_SERVICE_KEY_INVALID",
  133. "CM_PROB_LEGACY_SERVICE_NO_DEVICES",
  134. "CM_PROB_DUPLICATE_DEVICE",
  135. "CM_PROB_FAILED_POST_START",
  136. "CM_PROB_HALTED",
  137. "CM_PROB_PHANTOM",
  138. "CM_PROB_SYSTEM_SHUTDOWN",
  139. "CM_PROB_HELD_FOR_EJECT",
  140. "CM_PROB_DRIVER_BLOCKED",
  141. "CM_PROB_REGISTRY_TOO_LARGE"
  142. };
  143. extern
  144. VOID
  145. DevExtIsapnp(
  146. ULONG64 Extension
  147. );
  148. extern
  149. VOID
  150. DevExtPcmcia(
  151. ULONG64 Extension
  152. );
  153. extern
  154. VOID
  155. DevExtUsbd(
  156. ULONG64 Extension
  157. );
  158. extern
  159. VOID
  160. DevExtOpenHCI(
  161. ULONG64 Extension
  162. );
  163. extern
  164. VOID
  165. DevExtUHCD(
  166. ULONG64 Extension
  167. );
  168. extern
  169. VOID
  170. DevExtHID(
  171. ULONG64 Extension
  172. );
  173. extern
  174. VOID
  175. DevExtUsbhub(
  176. ULONG64 Extension
  177. );
  178. VOID
  179. DumpResourceDescriptorHeader(
  180. ULONG Depth,
  181. ULONG Number,
  182. UCHAR Option,
  183. UCHAR Type,
  184. UCHAR SharingDisposition,
  185. USHORT Flags
  186. );
  187. BOOLEAN
  188. DumpDevNode(
  189. ULONG64 DeviceNode,
  190. ULONG Depth,
  191. BOOLEAN DumpSibling,
  192. BOOLEAN DumpChild,
  193. BOOLEAN DumpCmRes,
  194. BOOLEAN DumpCmResReqList,
  195. BOOLEAN DumpCmResTrans,
  196. BOOLEAN DumpOnlyDevnodesWithProblems,
  197. BOOLEAN DumpOnlyNonStartedDevnodes,
  198. PUNICODE_STRING ServiceName,
  199. BOOLEAN PrintDefault
  200. );
  201. BOOLEAN
  202. DumpPendingEjects(
  203. BOOLEAN DumpCmRes,
  204. BOOLEAN DumpCmResReqList,
  205. BOOLEAN DumpCmResTrans
  206. );
  207. BOOLEAN
  208. DumpPendingRemovals(
  209. BOOLEAN DumpCmRes,
  210. BOOLEAN DumpCmResReqList,
  211. BOOLEAN DumpCmResTrans
  212. );
  213. BOOLEAN
  214. DumpDeviceEventEntry(
  215. ULONG64 DeviceEvent,
  216. ULONG64 ListHead,
  217. BOOL FollowLinks
  218. );
  219. VOID
  220. DumpPlugPlayEventBlock(
  221. ULONG64 PlugPlayEventBlock,
  222. ULONG Size
  223. );
  224. BOOLEAN
  225. DumpResourceList64(
  226. LPSTR Description,
  227. ULONG Depth,
  228. ULONG64 ResourceList
  229. );
  230. BOOLEAN
  231. DumpResourceRequirementList64(
  232. ULONG Depth,
  233. ULONG64 ResourceRequirementList
  234. );
  235. BOOLEAN
  236. DumpE820ResourceList64(
  237. ULONG Depth,
  238. ULONG64 ResourceList
  239. );
  240. BOOLEAN
  241. DumpArbiter(
  242. IN DWORD Depth,
  243. IN ULONG64 ArbiterAddr,
  244. IN BOOLEAN DisplayAliases
  245. );
  246. #define ARBITER_DUMP_IO 0x1
  247. #define ARBITER_DUMP_MEMORY 0x2
  248. #define ARBITER_DUMP_IRQ 0x4
  249. #define ARBITER_DUMP_DMA 0x8
  250. #define ARBITER_DUMP_BUS_NUMBER 0x10
  251. #define ARBITER_NO_DUMP_ALIASES 0x100
  252. #define ARBITER_DUMP_ALL (ARBITER_DUMP_IO | \
  253. ARBITER_DUMP_MEMORY | \
  254. ARBITER_DUMP_IRQ | \
  255. ARBITER_DUMP_DMA | \
  256. ARBITER_DUMP_BUS_NUMBER)
  257. BOOLEAN
  258. DumpArbitersForDevNode(
  259. IN DWORD Depth,
  260. IN ULONG64 DevNodeAddr,
  261. IN DWORD Flags
  262. );
  263. BOOLEAN
  264. DumpRangeList(
  265. IN DWORD Depth,
  266. IN ULONG64 RangeListHead,
  267. IN BOOLEAN IsMerged,
  268. IN BOOLEAN OwnerIsDevObj,
  269. IN BOOLEAN DisplayAliases
  270. );
  271. VOID
  272. PrintDevNodeState(
  273. IN ULONG Depth,
  274. IN PUCHAR Field,
  275. IN ULONG State
  276. );
  277. DECLARE_API( ioreslist )
  278. /*++
  279. Routine Description:
  280. Dump a device object.
  281. Arguments:
  282. args - the location of the device object to dump.
  283. Return Value:
  284. None
  285. --*/
  286. {
  287. ULONG64 requirementList;
  288. requirementList = GetExpression(args);
  289. DumpResourceRequirementList64(0, requirementList);
  290. return S_OK;
  291. }
  292. DECLARE_API( cmreslist )
  293. /*++
  294. Routine Description:
  295. Dump a device object.
  296. Arguments:
  297. args - the location of the device object to dump.
  298. Return Value:
  299. None
  300. --*/
  301. {
  302. ULONG64 resourceList;
  303. resourceList = GetExpression(args);
  304. DumpResourceList64("CmResourceList", 0, resourceList);
  305. return S_OK;
  306. }
  307. DECLARE_API( e820reslist )
  308. /*++
  309. Routine Description:
  310. Dump an e820 resource table
  311. Arguments:
  312. args - the location of the resources list to dump
  313. Return Value:
  314. None
  315. --*/
  316. {
  317. ULONG64 resourceList;
  318. resourceList = GetExpression(args);
  319. DumpE820ResourceList64(0, resourceList );
  320. return S_OK;
  321. }
  322. #define DUMP_CHILD 1
  323. #define DUMP_CM_RES 2
  324. #define DUMP_CM_RES_REQ_LIST 4
  325. #define DUMP_CM_RES_TRANS 8
  326. #define DUMP_NOT_STARTED 0x10
  327. #define DUMP_PROBLEMS 0x20
  328. BOOLEAN
  329. xReadMemory(
  330. ULONG64 S,
  331. PVOID D,
  332. ULONG Len
  333. )
  334. {
  335. ULONG result;
  336. return (ReadMemory((S), D, Len, &result) && (result == Len));
  337. }
  338. DECLARE_API( devnode )
  339. /*++
  340. Routine Description:
  341. Dump a device object.
  342. Arguments:
  343. args - the location of the device object to dump.
  344. Return Value:
  345. None
  346. --*/
  347. {
  348. ULONG64 deviceNode=0;
  349. ULONG verbose = 0;
  350. PUCHAR buffer = NULL;
  351. UNICODE_STRING serviceName;
  352. ANSI_STRING ansiString;
  353. BOOLEAN serviceNameAllocated = FALSE;
  354. NTSTATUS status;
  355. ULONG64 tmp;
  356. buffer = LocalAlloc(LPTR, 256);
  357. if (buffer) {
  358. RtlZeroMemory(buffer, 256);
  359. if (GetExpressionEx(args, &deviceNode, &args)) {
  360. if (sscanf(args, "%lx %s", &verbose, buffer)) {
  361. if (buffer [0] != '\0') {
  362. RtlInitAnsiString(&ansiString, (PCSZ)buffer);
  363. status = RtlAnsiStringToUnicodeString(&serviceName,
  364. &ansiString,
  365. TRUE);
  366. if (NT_SUCCESS(status)) {
  367. serviceNameAllocated = TRUE;
  368. }
  369. }
  370. }
  371. }
  372. LocalFree(buffer);
  373. } else if (GetExpressionEx(args, &deviceNode, &args)) {
  374. if (sscanf(args,"%lx", &tmp)) {
  375. verbose = (ULONG) tmp;
  376. }
  377. }
  378. if (deviceNode == 0) {
  379. ULONG64 addr = GetExpression( "nt!IopRootDeviceNode" );
  380. ULONG64 dummy;
  381. if (addr == 0) {
  382. dprintf("Error retrieving address of IopRootDeviceNode\n");
  383. return E_INVALIDARG;
  384. }
  385. if (!ReadPointer(addr, &deviceNode)) {
  386. dprintf("Error reading value of IopRootDeviceNode (%#010p)\n", addr);
  387. return E_INVALIDARG;
  388. }
  389. dprintf("Dumping IopRootDeviceNode (= %#08p)\n", deviceNode);
  390. } else if (deviceNode == 1) {
  391. DumpPendingRemovals((BOOLEAN) ((verbose & DUMP_CM_RES) != 0),
  392. (BOOLEAN) ((verbose & DUMP_CM_RES_REQ_LIST) != 0),
  393. (BOOLEAN) ((verbose & DUMP_CM_RES_TRANS) != 0)
  394. );
  395. return E_INVALIDARG;
  396. } else if (deviceNode == 2) {
  397. DumpPendingEjects((BOOLEAN) ((verbose & DUMP_CM_RES) != 0),
  398. (BOOLEAN) ((verbose & DUMP_CM_RES_REQ_LIST) != 0),
  399. (BOOLEAN) ((verbose & DUMP_CM_RES_TRANS) != 0)
  400. );
  401. return E_INVALIDARG;
  402. }
  403. if (serviceNameAllocated) {
  404. DumpDevNode( deviceNode,
  405. 0,
  406. FALSE,
  407. (BOOLEAN) ((verbose & DUMP_CHILD) != 0),
  408. (BOOLEAN) ((verbose & DUMP_CM_RES) != 0),
  409. (BOOLEAN) ((verbose & DUMP_CM_RES_REQ_LIST) != 0),
  410. (BOOLEAN) ((verbose & DUMP_CM_RES_TRANS) != 0),
  411. (BOOLEAN) ((verbose & DUMP_PROBLEMS) != 0),
  412. (BOOLEAN) ((verbose & DUMP_NOT_STARTED) != 0),
  413. &serviceName,
  414. FALSE
  415. );
  416. RtlFreeUnicodeString(&serviceName);
  417. } else {
  418. DumpDevNode( deviceNode,
  419. 0,
  420. FALSE,
  421. (BOOLEAN) ((verbose & DUMP_CHILD) != 0),
  422. (BOOLEAN) ((verbose & DUMP_CM_RES) != 0),
  423. (BOOLEAN) ((verbose & DUMP_CM_RES_REQ_LIST) != 0),
  424. (BOOLEAN) ((verbose & DUMP_CM_RES_TRANS) != 0),
  425. (BOOLEAN) ((verbose & DUMP_PROBLEMS) != 0),
  426. (BOOLEAN) ((verbose & DUMP_NOT_STARTED) != 0),
  427. NULL,
  428. FALSE
  429. );
  430. }
  431. return S_OK;
  432. }
  433. static CCHAR DebugBuffer[300];
  434. VOID
  435. xdprintf(
  436. ULONG Depth,
  437. PCCHAR S,
  438. ...
  439. )
  440. {
  441. va_list ap;
  442. ULONG i;
  443. for (i=0; i<Depth; i++) {
  444. dprintf (" ");
  445. }
  446. va_start(ap, S);
  447. vsprintf(DebugBuffer, S, ap);
  448. dprintf(DebugBuffer);
  449. va_end(ap);
  450. }
  451. VOID
  452. DumpFlags(
  453. ULONG Depth,
  454. LPSTR FlagDescription,
  455. ULONG Flags,
  456. PFLAG_NAME FlagTable
  457. )
  458. {
  459. ULONG i;
  460. ULONG mask = 0;
  461. ULONG count = 0;
  462. UCHAR prolog[64];
  463. sprintf(prolog, "%s (%#010x) ", FlagDescription, Flags);
  464. xdprintf(Depth, "%s", prolog);
  465. if (Flags == 0) {
  466. dprintf("\n");
  467. return;
  468. }
  469. memset(prolog, ' ', strlen(prolog));
  470. for (i = 0; FlagTable[i].Name != 0; i++) {
  471. PFLAG_NAME flag = &(FlagTable[i]);
  472. mask |= flag->Flag;
  473. if ((Flags & flag->Flag) == flag->Flag) {
  474. //
  475. // print trailing comma
  476. //
  477. if (count != 0) {
  478. dprintf(", ");
  479. //
  480. // Only print two flags per line.
  481. //
  482. if ((count % 2) == 0) {
  483. dprintf("\n");
  484. xdprintf(Depth, "%s", prolog);
  485. }
  486. }
  487. dprintf("%s", flag->Name);
  488. count++;
  489. }
  490. }
  491. dprintf("\n");
  492. if ((Flags & (~mask)) != 0) {
  493. xdprintf(Depth, "%sUnknown flags %#010lx\n", prolog, (Flags & (~mask)));
  494. }
  495. return;
  496. }
  497. BOOLEAN
  498. DumpE820ResourceList64(
  499. ULONG Depth,
  500. ULONG64 ResourceList
  501. )
  502. {
  503. BOOLEAN continueDump = TRUE;
  504. CHAR Buff[80];
  505. ULONG i;
  506. ULONG ListOffset;
  507. ULONG64 Count;
  508. ULONG64 Type;
  509. ULONG64 V1;
  510. ULONG64 V2;
  511. #define Res2(F, V) GetFieldValue(ResourceList, "nt!_ACPI_BIOS_MULTI_NODE", #F, V)
  512. #define Res(F) GetFieldValue(ResourceList, "nt!_ACPI_BIOS_MULTI_NODE", #F, F)
  513. #define Entry(F,V) (sprintf(Buff,"E820Entry[%lx].%s",i,#F),\
  514. GetFieldValue(ResourceList,"nt!_ACPI_BIOS_MULTI_NODE", Buff, V))
  515. if (ResourceList == 0) {
  516. goto GetOut;
  517. }
  518. if (Res(Count)) {
  519. goto GetOut;
  520. }
  521. if (Count == 0) {
  522. goto GetOut;
  523. }
  524. xdprintf(Depth, "ACPI_BIOS_MULTI_NODE (an E820 Resource List) at %#010p\n", ResourceList);
  525. Depth++;
  526. for (i = 0; i < Count; i++) {
  527. if (CheckControlC()) {
  528. continueDump = FALSE;
  529. break;
  530. }
  531. Entry(Type,Type);
  532. switch(Type) {
  533. case 1: // AcpiAddressRangeMemory
  534. xdprintf(Depth,"Memory - ");
  535. break;
  536. case 2: // AcpiAddressRangeReserved
  537. xdprintf(Depth,"Reserved - ");
  538. break;
  539. case 3: // AcpiAddressRangeACPI
  540. xdprintf(Depth,"ACPI - ");
  541. break;
  542. case 4: // AcpiAddressRangeNVS
  543. xdprintf(Depth,"NVS - ");
  544. break;
  545. case 5: // AcpiAddressRangeMaximum
  546. default:
  547. xdprintf(Depth,"Unknown - ");
  548. break;
  549. }
  550. Entry(Base.QuadPart, V1);
  551. Entry(Length.QuadPart, V2);
  552. dprintf( "%16I64x - %#16I64x (Length %#10I64x)", V1, (V1 + V2 - 1), V2 );
  553. if (Type == 2) {
  554. dprintf(" [Ignored]");
  555. }
  556. if (Type == 3 || Type == 4) {
  557. if (V2 > 0xFFFFFFFF) {
  558. dprintf(" [Illegal Range]");
  559. }
  560. }
  561. dprintf("\n");
  562. }
  563. GetOut:
  564. #undef Res
  565. #undef Res2
  566. #undef Entry
  567. return continueDump;
  568. }
  569. BOOLEAN
  570. DumpResourceRequirementList64(
  571. ULONG Depth,
  572. ULONG64 ResourceRequirementList
  573. )
  574. {
  575. ULONG64 requirementList = 0;
  576. ULONG64 resourceList;
  577. ULONG64 descriptors;
  578. ULONG ListSize;
  579. ULONG i;
  580. ULONG j;
  581. ULONG k;
  582. ULONG result;
  583. BOOLEAN continueDump = TRUE;
  584. ULONG InterfaceType, BusNumber, SlotNumber, Reserved1, Reserved2, Reserved3;
  585. ULONG AlternativeLists;
  586. ULONG ListOffset, SizeOfIoResList, SizeofIoDesc;
  587. #define Res2(F, V) GetFieldValue(ResourceRequirementList, "nt!_IO_RESOURCE_REQUIREMENTS_LIST", #F, V)
  588. #define Res(F) GetFieldValue(ResourceRequirementList, "nt!_IO_RESOURCE_REQUIREMENTS_LIST", #F, F)
  589. if (ResourceRequirementList == 0) {
  590. goto GetOut;
  591. }
  592. if (Res(ListSize)) {
  593. goto GetOut;
  594. }
  595. if (ListSize == 0) {
  596. goto GetOut;
  597. }
  598. Res(InterfaceType); Res(BusNumber); Res(SlotNumber);
  599. xdprintf(Depth, "IoResList at ");
  600. dprintf("%#010p : Interface %#x Bus %#x Slot %#x\n",
  601. ResourceRequirementList,
  602. InterfaceType,
  603. BusNumber,
  604. SlotNumber);
  605. Res2(Reserved[0], Reserved1); Res2(Reserved[1], Reserved2);
  606. Res2(Reserved[2], Reserved3);
  607. if ((Reserved1 |
  608. Reserved2 |
  609. Reserved3) != 0) {
  610. xdprintf(Depth, "Reserved Values = {%#010lx, %#010lx, %#010lx}\n",
  611. Reserved1,
  612. Reserved2,
  613. Reserved3);
  614. }
  615. Depth++;
  616. Res(AlternativeLists);
  617. GetFieldOffset("nt!_IO_RESOURCE_REQUIREMENTS_LIST", "List", &ListOffset);
  618. resourceList = ResourceRequirementList + ListOffset;
  619. SizeOfIoResList = GetTypeSize("IO_RESOURCE_LIST");
  620. SizeofIoDesc = GetTypeSize("IO_RESOURCE_DESCRIPTOR");
  621. for (i = 0; i < AlternativeLists; i++) {
  622. ULONG Version, Revision, Count;
  623. if (CheckControlC()) {
  624. continueDump = FALSE;
  625. break;
  626. }
  627. GetFieldValue(resourceList, "nt!_IO_RESOURCE_LIST", "Version", Version);
  628. GetFieldValue(resourceList, "nt!_IO_RESOURCE_LIST", "Revision", Revision);
  629. GetFieldValue(resourceList, "nt!_IO_RESOURCE_LIST", "Count", Count);
  630. xdprintf(Depth, "Alternative %d (Version %d.%d)\n",
  631. i,
  632. Version,
  633. Revision);
  634. Depth++;
  635. for (j = 0; j < Count; j++) {
  636. CHAR Buff[80];
  637. ULONG64 Q1, Q2;
  638. ULONG Option, Type, ShareDisposition, Flags, V1, V2, V3, Data[3];
  639. #define Desc(F,V) (sprintf(Buff,"Descriptors[%lx].%s",j, #F), \
  640. GetFieldValue(resourceList, "nt!_IO_RESOURCE_LIST", Buff, V))
  641. if (CheckControlC()) {
  642. continueDump = FALSE;
  643. goto GetOut;
  644. }
  645. // descriptors = resourceList->Descriptors + j;
  646. Desc(Option,Option); Desc(Type,Type);
  647. Desc(ShareDisposition,ShareDisposition);
  648. Desc(Flags,Flags);
  649. DumpResourceDescriptorHeader(Depth,
  650. j,
  651. (UCHAR)Option,
  652. (UCHAR) Type,
  653. (UCHAR) ShareDisposition,
  654. (USHORT) Flags);
  655. Depth++;
  656. switch (Type) {
  657. case CmResourceTypeBusNumber:
  658. Desc(u.BusNumber.MinBusNumber, V1);
  659. Desc(u.BusNumber.MaxBusNumber, V2);
  660. Desc(u.BusNumber.Length, V3);
  661. xdprintf(Depth, "0x%x - 0x%x for length 0x%x\n",
  662. V1, V2, V3);
  663. break;
  664. case CmResourceTypePort:
  665. case CmResourceTypeMemory:
  666. Desc(u.Generic.Length, V1);
  667. Desc(u.Generic.Alignment, V2);
  668. Desc(u.Port.MinimumAddress.QuadPart, Q1);
  669. Desc(u.Port.MaximumAddress.QuadPart, Q2);
  670. xdprintf(Depth, "%#08lx byte range with alignment %#08lx\n", V1, V2);
  671. xdprintf(Depth, "%I64x - %#I64x\n", Q1, Q2);
  672. break;
  673. case CmResourceTypeInterrupt:
  674. Desc(u.Interrupt.MinimumVector,V1);
  675. Desc(u.Interrupt.MaximumVector,V2);
  676. xdprintf(Depth, "0x%x - 0x%x\n", V1,V2);
  677. break;
  678. default:
  679. Desc(u.DevicePrivate.Data, Data);
  680. xdprintf(Depth,
  681. "Data: : 0x%x 0x%x 0x%x\n",
  682. Data[0],
  683. Data[1],
  684. Data[2]);
  685. break;
  686. }
  687. Depth--;
  688. #undef Desc
  689. }
  690. Depth--;
  691. resourceList += (SizeOfIoResList + SizeofIoDesc*(j-1));
  692. }
  693. GetOut:
  694. xdprintf(Depth, "\n");
  695. #undef Res
  696. #undef Res2
  697. return continueDump;
  698. }
  699. BOOLEAN
  700. DumpResourceList64(
  701. LPSTR Description,
  702. ULONG Depth,
  703. ULONG64 ResourceList
  704. )
  705. {
  706. ULONG64 resourceListWalker;
  707. ULONG i;
  708. ULONG j;
  709. ULONG k;
  710. BOOLEAN continueDump = TRUE;
  711. ULONG Count, ListSize, ListOffset;
  712. ULONG64 ListAddr;
  713. if (ResourceList == 0) {
  714. goto GetOut;
  715. }
  716. resourceListWalker = ResourceList;
  717. if (GetFieldOffset("nt!_CM_RESOURCE_LIST", "List", &ListOffset)) {
  718. goto GetOut;
  719. }
  720. ListAddr = ResourceList + ListOffset;
  721. GetFieldValue(ResourceList, "nt!_CM_RESOURCE_LIST", "Count", Count);
  722. resourceListWalker = ListAddr;
  723. for (i = 0; i < Count; i++) {
  724. ULONG Partial_Count=0, PartialListAddr, BusNumber=0, Partial_Version=0, Partial_Revision=0;
  725. CHAR ResListType[] = "nt!_CM_FULL_RESOURCE_DESCRIPTOR";
  726. ULONG64 InterfaceType=0;
  727. ULONG PartDescOffset;
  728. if (CheckControlC()) {
  729. continueDump = FALSE;
  730. break;
  731. }
  732. GetFieldValue(resourceListWalker, ResListType, "InterfaceType", InterfaceType);
  733. GetFieldValue(resourceListWalker, ResListType, "BusNumber", BusNumber);
  734. GetFieldValue(resourceListWalker, ResListType, "PartialResourceList.Version", Partial_Version);
  735. GetFieldValue(resourceListWalker, ResListType, "PartialResourceList.Revision", Partial_Revision);
  736. GetFieldValue(resourceListWalker, ResListType, "PartialResourceList.Count", Partial_Count);
  737. GetFieldOffset(ResListType, "PartialResourceList.PartialDescriptors", &PartDescOffset);
  738. resourceListWalker = resourceListWalker + PartDescOffset;
  739. xdprintf(Depth, "%s at ", Description);
  740. dprintf("%#010p Version %d.%d Interface %#x Bus #%#x\n",
  741. ResourceList,
  742. Partial_Version,
  743. Partial_Revision,
  744. (ULONG) InterfaceType,
  745. BusNumber);
  746. Depth++;
  747. for (j = 0; j < Partial_Count; j++) {
  748. CHAR PartResType[] = "nt!_CM_PARTIAL_RESOURCE_DESCRIPTOR";
  749. ULONG Type=0, ShareDisposition=0, Flags=0, sz, u_Port_Length=0, u_Memory_Length=0,
  750. u_Interrupt_Level=0, u_Interrupt_Vector=0,
  751. u_Dma_Channel=0, u_Dma_Port=0, u_Dma_Reserved1=0, u_BusNumber_Start=0,
  752. u_BusNumber_Length=0, u_DevicePrivate_Data[3];
  753. ULONG64 u_Port_QuadPart=0, u_Memory_QuadPart=0, u_Interrupt_Affinity=0;
  754. if (CheckControlC()) {
  755. continueDump = FALSE;
  756. goto GetOut;
  757. }
  758. sz = GetTypeSize(PartResType);
  759. if (GetFieldValue(resourceListWalker,
  760. PartResType,
  761. "Type",
  762. Type)) {
  763. goto GetOut;
  764. }
  765. GetFieldValue(resourceListWalker, PartResType, "ShareDisposition", ShareDisposition);
  766. GetFieldValue(resourceListWalker, PartResType, "Flags", Flags);
  767. xdprintf(Depth, "Entry %d - %s (%#x) %s (%#x)\n",
  768. j,
  769. (Type == CmResourceTypeNull ? "Null" :
  770. Type == CmResourceTypePort ? "Port" :
  771. Type == CmResourceTypeInterrupt ? "Interrupt" :
  772. Type == CmResourceTypeMemory ? "Memory" :
  773. Type == CmResourceTypeDma ? "Dma" :
  774. Type == CmResourceTypeDeviceSpecific ? "DeviceSpecific" :
  775. Type == CmResourceTypeBusNumber ? "BusNumber" :
  776. Type == CmResourceTypeDevicePrivate ? "DevicePrivate" :
  777. Type == CmResourceTypeConfigData ? "ConfigData" : "Unknown"),
  778. Type,
  779. (ShareDisposition == CmResourceShareUndetermined ? "Undetermined Sharing" :
  780. ShareDisposition == CmResourceShareDeviceExclusive ? "Device Exclusive" :
  781. ShareDisposition == CmResourceShareDriverExclusive ? "Driver Exclusive" :
  782. ShareDisposition == CmResourceShareShared ? "Shared" : "Unknown Sharing"),
  783. ShareDisposition);
  784. Depth++;
  785. xdprintf(Depth, "Flags (%#04wx) - ", Flags);
  786. switch (Type) {
  787. case CmResourceTypePort:
  788. //
  789. // CM_RESOURCE_PORT_MEMORY = ~CM_RESOURCE_PORT_IO
  790. //
  791. if ( (~Flags) & ~CM_RESOURCE_PORT_MEMORY) {
  792. dprintf("PORT_MEMORY ");
  793. }
  794. if (Flags & CM_RESOURCE_PORT_IO) {
  795. dprintf("PORT_IO ");
  796. }
  797. if (Flags & CM_RESOURCE_PORT_10_BIT_DECODE) {
  798. dprintf("10_BIT_DECODE ");
  799. }
  800. if (Flags & CM_RESOURCE_PORT_12_BIT_DECODE) {
  801. dprintf("12_BIT_DECODE ");
  802. }
  803. if (Flags & CM_RESOURCE_PORT_16_BIT_DECODE) {
  804. dprintf("16_BIT_DECODE ");
  805. }
  806. if (Flags & CM_RESOURCE_PORT_POSITIVE_DECODE) {
  807. dprintf("POSITIVE_DECODE ");
  808. }
  809. if (Flags & CM_RESOURCE_PORT_PASSIVE_DECODE) {
  810. dprintf ("PASSIVE_DECODE ");
  811. }
  812. if (Flags & CM_RESOURCE_PORT_WINDOW_DECODE) {
  813. dprintf ("WINDOW_DECODE ");
  814. }
  815. dprintf("\n");
  816. GetFieldValue(resourceListWalker, PartResType, "u.Port.Start.QuadPart", u_Port_QuadPart);
  817. GetFieldValue(resourceListWalker, PartResType, "u.Port.Length", u_Port_Length);
  818. xdprintf(Depth, "Range starts at %#I64lx for %#x bytes\n",
  819. u_Port_QuadPart,
  820. u_Port_Length);
  821. break;
  822. case CmResourceTypeMemory:
  823. if (Flags == CM_RESOURCE_MEMORY_READ_WRITE) {
  824. dprintf("READ_WRITE ");
  825. }
  826. if (Flags & CM_RESOURCE_MEMORY_READ_ONLY) {
  827. dprintf("READ_ONLY ");
  828. }
  829. if (Flags & CM_RESOURCE_MEMORY_WRITE_ONLY) {
  830. dprintf("WRITE_ONLY ");
  831. }
  832. if (Flags & CM_RESOURCE_MEMORY_PREFETCHABLE) {
  833. dprintf("PREFETCHABLE ");
  834. }
  835. if (Flags & CM_RESOURCE_MEMORY_COMBINEDWRITE) {
  836. dprintf("COMBINEDWRITE ");
  837. }
  838. if (Flags & CM_RESOURCE_MEMORY_24) {
  839. dprintf("MEMORY_24 ");
  840. }
  841. dprintf("\n");
  842. GetFieldValue(resourceListWalker, PartResType, "u.Memory.Start.QuadPart", u_Memory_QuadPart);
  843. GetFieldValue(resourceListWalker, PartResType, "u.Memory.Length", u_Memory_Length);
  844. xdprintf(Depth, "Range starts at %#018I64lx for %#x bytes\n",
  845. u_Memory_QuadPart,
  846. u_Memory_Length);
  847. break;
  848. case CmResourceTypeInterrupt:
  849. if (Flags == CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE) {
  850. dprintf("LEVEL_SENSITIVE ");
  851. }
  852. if (Flags & CM_RESOURCE_INTERRUPT_LATCHED) {
  853. dprintf("LATCHED ");
  854. }
  855. dprintf("\n");
  856. GetFieldValue(resourceListWalker, PartResType, "u.Interrupt.Level", u_Interrupt_Level);
  857. GetFieldValue(resourceListWalker, PartResType, "u.Interrupt.Vector", u_Interrupt_Vector);
  858. GetFieldValue(resourceListWalker, PartResType, "u.Interrupt.Affinity", u_Interrupt_Affinity);
  859. xdprintf(Depth, "Level %#x, Vector %#x, Affinity %#I64x\n",
  860. u_Interrupt_Level,
  861. u_Interrupt_Vector,
  862. u_Interrupt_Affinity);
  863. break;
  864. case CmResourceTypeDma:
  865. if (Flags & CM_RESOURCE_DMA_8) {
  866. dprintf("DMA_8 ");
  867. }
  868. if (Flags & CM_RESOURCE_DMA_16) {
  869. dprintf("DMA_16 ");
  870. }
  871. if (Flags & CM_RESOURCE_DMA_32) {
  872. dprintf("DMA_32 ");
  873. }
  874. if (Flags & CM_RESOURCE_DMA_8_AND_16) {
  875. dprintf("DMA_8_AND_16 ");
  876. }
  877. if (Flags & CM_RESOURCE_DMA_BUS_MASTER) {
  878. dprintf("DMA_BUS_MASTER ");
  879. }
  880. if (Flags & CM_RESOURCE_DMA_TYPE_A) {
  881. dprintf("DMA_A ");
  882. }
  883. if (Flags & CM_RESOURCE_DMA_TYPE_B) {
  884. dprintf("DMA_B ");
  885. }
  886. if (Flags & CM_RESOURCE_DMA_TYPE_F) {
  887. dprintf("DMA_F ");
  888. }
  889. dprintf("\n");
  890. GetFieldValue(resourceListWalker, PartResType, "u.Dma.Channel", u_Dma_Channel);
  891. GetFieldValue(resourceListWalker, PartResType, "u.Dma.Port", u_Dma_Port);
  892. GetFieldValue(resourceListWalker, PartResType, "u.Dma.Reserved1", u_Dma_Reserved1);
  893. xdprintf(Depth, "Channel %#x Port %#x (Reserved1 %#x)\n",
  894. u_Dma_Channel,
  895. u_Dma_Port,
  896. u_Dma_Reserved1);
  897. break;
  898. case CmResourceTypeBusNumber:
  899. dprintf("\n");
  900. GetFieldValue(resourceListWalker, PartResType, "u.BusNumber.Start", u_BusNumber_Start);
  901. GetFieldValue(resourceListWalker, PartResType, "u.BusNumber.Length", u_BusNumber_Length);
  902. xdprintf(Depth, "Range starts at %#010lx for %#x values\n",
  903. u_BusNumber_Start,
  904. u_BusNumber_Length);
  905. break;
  906. default:
  907. dprintf("\n");
  908. GetFieldValue(resourceListWalker, PartResType, "u.DevicePrivate.Data", u_DevicePrivate_Data);
  909. xdprintf(Depth, "Data - {%#010lx, %#010lx, %#010lx}\n",
  910. u_DevicePrivate_Data[0],
  911. u_DevicePrivate_Data[1],
  912. u_DevicePrivate_Data[2]);
  913. break;
  914. }
  915. Depth--;
  916. resourceListWalker += sz; // sizeof(partialDescriptors);
  917. }
  918. }
  919. GetOut:
  920. xdprintf(Depth, "\n");
  921. return continueDump;
  922. }
  923. BOOLEAN
  924. DumpRelationsList(
  925. ULONG64 RelationsList,
  926. BOOLEAN DumpCmRes,
  927. BOOLEAN DumpCmResReqList,
  928. BOOLEAN DumpCmResTrans
  929. )
  930. {
  931. ULONG64 listEntryAddr;
  932. ULONG64 deviceObjectAddr;
  933. ULONG i, j, EntrySize;
  934. BOOLEAN continueDump = TRUE;
  935. CHAR RelListType[] = "nt!_RELATION_LIST", RelListentryType[] = "nt!_RELATION_LIST_ENTRY";
  936. ULONG64 EntryAddr;
  937. ULONG Count=0, TagCount=0, FirstLevel=0, MaxLevel=0;
  938. ULONG EntryOffset;
  939. if (GetFieldValue(RelationsList, RelListType, "Count",Count)) {
  940. xdprintf(1, "Error reading relation list @ ");
  941. dprintf("0x%08p)\n", RelationsList);
  942. return continueDump;
  943. }
  944. EntrySize = DBG_PTR_SIZE;
  945. GetFieldOffset(RelListType, "Entries", &EntryOffset);
  946. EntryAddr=RelationsList + EntryOffset;
  947. GetFieldValue(RelationsList, RelListType, "TagCount", TagCount);
  948. GetFieldValue(RelationsList, RelListType, "FirstLevel", FirstLevel);
  949. GetFieldValue(RelationsList, RelListType, "MaxLevel", MaxLevel);
  950. xdprintf(1, "Relation List @ "); dprintf("0x%08p\n", RelationsList);
  951. xdprintf(2, "Count = %d, TagCount = %d, FirstLevel = %d, MaxLevel = %d\n",
  952. Count,
  953. TagCount,
  954. FirstLevel,
  955. MaxLevel);
  956. for (i = 0; i <= (MaxLevel - FirstLevel); i++) {
  957. ULONG res,Entry_Count=0, MaxCount=0, DeviceSize;
  958. ULONG64 DeviceAddr;
  959. FIELD_INFO relListEntryFields[] = {
  960. {"Devices", "", 0, DBG_DUMP_FIELD_RETURN_ADDRESS, 0, NULL},
  961. };
  962. if (CheckControlC()) {
  963. continueDump = FALSE;
  964. break;
  965. }
  966. listEntryAddr = 0;
  967. if (ReadPointer( EntryAddr + i*EntrySize,
  968. &listEntryAddr)) {
  969. if (!listEntryAddr) {
  970. xdprintf(2, "Null relation list entry\n");
  971. }else if (!GetFieldValue(listEntryAddr, RelListentryType,
  972. "Count",Entry_Count)) {
  973. ULONG DeviceOffset;
  974. GetFieldOffset(RelListentryType, "Devices", &DeviceOffset);
  975. DeviceAddr = listEntryAddr + DeviceOffset;
  976. DeviceSize = EntrySize; // == pointer size
  977. GetFieldValue(listEntryAddr, RelListentryType, "MaxCount", MaxCount);
  978. xdprintf(2, "Relation List Entry @ "); dprintf("0x%08p\n", listEntryAddr);
  979. xdprintf(3, "Count = %d, MaxCount = %d\n", Entry_Count, MaxCount);
  980. xdprintf(3, "Devices at level %d\n", FirstLevel + i);
  981. for (j = 0; j < Entry_Count; j++) {
  982. if (CheckControlC()) {
  983. continueDump = FALSE;
  984. break;
  985. }
  986. if (ReadPointer( DeviceAddr + j*DeviceSize,
  987. &deviceObjectAddr)) {
  988. ULONG64 DeviceNode=0;
  989. FIELD_INFO DevObjFields[] = {
  990. // Implicitly reads the pointer DeviceObjectExtension and its field DeviceNode
  991. {"DeviceObjectExtension", "", 0, DBG_DUMP_FIELD_RECUR_ON_THIS, 0, NULL},
  992. {"DeviceObjectExtension.DeviceNode", "", 0, DBG_DUMP_FIELD_COPY_FIELD_DATA, 0, (PVOID) &DeviceNode}
  993. };
  994. if (!deviceObjectAddr) {
  995. continue;
  996. }
  997. xdprintf(4, "Device Object = ");
  998. dprintf("0x%08p (%sagged, %sirect)\n",
  999. deviceObjectAddr & ~3,
  1000. (deviceObjectAddr & 1) ? "T" : "Not t",
  1001. (deviceObjectAddr & 2) ? "D" : "Ind");
  1002. if (!GetFieldValue(deviceObjectAddr, "nt!_DEVICE_OBJECT",
  1003. "DeviceObjectExtension.DeviceNode",
  1004. DeviceNode)) {
  1005. if (DeviceNode) {
  1006. continueDump = DumpDevNode(DeviceNode,
  1007. 5,
  1008. FALSE,
  1009. FALSE,
  1010. DumpCmRes,
  1011. DumpCmResReqList,
  1012. DumpCmResTrans,
  1013. FALSE,
  1014. FALSE,
  1015. (PUNICODE_STRING)NULL,
  1016. TRUE);
  1017. }
  1018. } else {
  1019. xdprintf(5, "Error reading device object @ ");
  1020. dprintf("0x%08p)\n",deviceObjectAddr);
  1021. }
  1022. } else {
  1023. xdprintf(4, "Error reading relation list entry device ");
  1024. dprintf("%d @ 0x%08p)\n",
  1025. j,
  1026. DeviceAddr + j*DeviceSize);
  1027. }
  1028. if (!continueDump) {
  1029. break;
  1030. }
  1031. }
  1032. } else {
  1033. xdprintf(2, "Error reading relation list entry @ ");
  1034. dprintf("0x%08p)\n",
  1035. listEntryAddr);
  1036. }
  1037. } else {
  1038. xdprintf(1, "Error reading relation list entry ");
  1039. dprintf("%d @ 0x%08p)\n",
  1040. i,
  1041. EntryAddr + i*EntrySize);
  1042. }
  1043. if (!continueDump) {
  1044. break;
  1045. }
  1046. }
  1047. return continueDump;
  1048. }
  1049. BOOLEAN
  1050. DumpPendingRemovals(
  1051. BOOLEAN DumpCmRes,
  1052. BOOLEAN DumpCmResReqList,
  1053. BOOLEAN DumpCmResTrans
  1054. )
  1055. {
  1056. ULONG64 listHeadAddr, pendingEntryAddr;
  1057. BOOLEAN continueDump = TRUE;
  1058. ULONG64 Head_Blink=0, Head_Flink=0, Link_Flink=0;
  1059. listHeadAddr = GetExpression( "nt!IopPendingSurpriseRemovals" );
  1060. if (listHeadAddr == 0) {
  1061. dprintf("Error retrieving address of IopPendingSurpriseRemovals\n");
  1062. return continueDump;
  1063. }
  1064. if (GetFieldValue(listHeadAddr, "nt!_LIST_ENTRY", "Flink", Head_Flink) ||
  1065. GetFieldValue(listHeadAddr, "nt!_LIST_ENTRY", "Blink", Head_Blink)) {
  1066. dprintf("Error reading value of IopPendingSurpriseRemovals (0x%08p)\n", listHeadAddr);
  1067. return continueDump;
  1068. }
  1069. if (Head_Flink == listHeadAddr) {
  1070. dprintf("There are no removals currently pending\n");
  1071. return continueDump;
  1072. }
  1073. for (pendingEntryAddr = Head_Flink;
  1074. pendingEntryAddr != listHeadAddr;
  1075. pendingEntryAddr = Link_Flink) {
  1076. ULONG64 RelationsList=0, DeviceObject=0;
  1077. if (CheckControlC()) {
  1078. continueDump = FALSE;
  1079. break;
  1080. }
  1081. if (GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "DeviceObject", DeviceObject)){
  1082. dprintf("Error reading pending relations list entry (0x%08p)\n", pendingEntryAddr);
  1083. return continueDump;
  1084. }
  1085. GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "Link.Flink", Link_Flink);
  1086. GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "RelationsList", RelationsList);
  1087. dprintf("Pending Removal of PDO 0x%08p, Pending Relations List Entry @ 0x%08p\n",
  1088. DeviceObject,
  1089. pendingEntryAddr
  1090. );
  1091. continueDump = DumpRelationsList( RelationsList,
  1092. DumpCmRes,
  1093. DumpCmResReqList,
  1094. DumpCmResTrans
  1095. );
  1096. if (!continueDump) {
  1097. break;
  1098. }
  1099. }
  1100. return continueDump;
  1101. }
  1102. BOOLEAN
  1103. DumpPendingEjects(
  1104. BOOLEAN DumpCmRes,
  1105. BOOLEAN DumpCmResReqList,
  1106. BOOLEAN DumpCmResTrans
  1107. )
  1108. {
  1109. ULONG64 listHeadAddr, pendingEntryAddr;
  1110. BOOLEAN continueDump = TRUE;
  1111. ULONG64 Head_Blink=0, Head_Flink=0, Link_Flink=0;
  1112. listHeadAddr = GetExpression( "nt!IopPendingEjects" );
  1113. if (listHeadAddr == 0) {
  1114. dprintf("Error retrieving address of IopEjects\n");
  1115. return continueDump;
  1116. }
  1117. if (GetFieldValue(listHeadAddr, "nt!_LIST_ENTRY", "Flink", Head_Flink) ||
  1118. GetFieldValue(listHeadAddr, "nt!_LIST_ENTRY", "Blink", Head_Blink)) {
  1119. dprintf("Error reading value of IopPendingSurpriseRemovals (0x%08p)\n", listHeadAddr);
  1120. return continueDump;
  1121. }
  1122. if (Head_Flink == listHeadAddr) {
  1123. dprintf("There are no ejects currently pending\n");
  1124. return continueDump;
  1125. }
  1126. for (pendingEntryAddr = Head_Flink;
  1127. pendingEntryAddr != listHeadAddr;
  1128. pendingEntryAddr = Link_Flink) {
  1129. ULONG64 RelationsList=0, DeviceObject=0, EjectIrp=0;
  1130. if (CheckControlC()) {
  1131. continueDump = FALSE;
  1132. break;
  1133. }
  1134. if (GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "DeviceObject", DeviceObject)){
  1135. dprintf("Error reading pending relations list entry (0x%08p)\n", pendingEntryAddr);
  1136. return continueDump;
  1137. }
  1138. GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "Link.Flink", Link_Flink);
  1139. GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "RelationsList", RelationsList);
  1140. GetFieldValue(pendingEntryAddr, "nt!_PENDING_RELATIONS_LIST_ENTRY", "EjectIrp", EjectIrp);
  1141. dprintf("Pending Eject of PDO 0x%08p, EjectIrp = 0x%08p, \nPending Relations List Entry @ 0x%08p\n",
  1142. DeviceObject,
  1143. EjectIrp,
  1144. pendingEntryAddr
  1145. );
  1146. continueDump = DumpRelationsList( RelationsList,
  1147. DumpCmRes,
  1148. DumpCmResReqList,
  1149. DumpCmResTrans
  1150. );
  1151. if (!continueDump) {
  1152. break;
  1153. }
  1154. }
  1155. return continueDump;
  1156. }
  1157. BOOLEAN
  1158. DumpDevNode(
  1159. ULONG64 DeviceNode,
  1160. ULONG Depth,
  1161. BOOLEAN DumpSibling,
  1162. BOOLEAN DumpChild,
  1163. BOOLEAN DumpCmRes,
  1164. BOOLEAN DumpCmResReqList,
  1165. BOOLEAN DumpCmResTrans,
  1166. BOOLEAN DumpOnlyDevnodesWithProblems,
  1167. BOOLEAN DumpOnlyNonStartedDevnodes,
  1168. PUNICODE_STRING ServiceName OPTIONAL,
  1169. BOOLEAN PrintDefault
  1170. )
  1171. /*++
  1172. Routine Description:
  1173. Displays the driver name for the device object if FullDetail == FALSE.
  1174. Otherwise displays more information about the device and the device queue.
  1175. Arguments:
  1176. DeviceAddress - address of device object to dump.
  1177. FullDetail - TRUE means the device object name, driver name, and
  1178. information about Irps queued to the device.
  1179. DumpOnlyDevnodesWithFlags -
  1180. if set, then the only devnodes that will be dumped are
  1181. those with one of the FlagsSet bits set or one of the
  1182. FlagsNotSet bits not set.
  1183. Return Value:
  1184. None
  1185. --*/
  1186. {
  1187. ULONG result;
  1188. // DEVICE_NODE deviceNode;
  1189. BOOLEAN print = TRUE;
  1190. BOOLEAN continueDump = TRUE;
  1191. CHAR DevNodeType[]="nt!_DEVICE_NODE";
  1192. USHORT ServiceName_Len=0, ServiceName_MaxLen=0, InstancePath_Len=0, InstancePath_Max=0;
  1193. ULONG Flags=0, BusNumber=0, Header_SignalState=0, UserFlags=0;
  1194. ULONG CapabilityFlags=0, DisableableDepends=0, Problem=0;
  1195. ULONG FailureStatus=0, State=0, PreviousState=0;
  1196. ULONG StateHistoryEntry = 0;
  1197. ULONG StateHistoryValue = 0;
  1198. ULONG64 ServiceName_Buff=0, PhysicalDeviceObject=0, Parent=0, Sibling=0,
  1199. Child=0, InterfaceType=0, DuplicatePDO=0, TargetDeviceNotify_Blink=0, TargetDeviceNotify_Flink=0,
  1200. PreviousResourceList=0, ResourceList=0, BootResources=0, ResourceListTranslated=0,
  1201. ResourceRequirements=0, InstancePath_Buff=0, PreviousResourceRequirements=0;
  1202. ULONG64 TargetDeviceNotifyAddress;
  1203. ULONG AmountRead;
  1204. PNP_DEVNODE_STATE StateHistory[STATE_HISTORY_SIZE];
  1205. if (CheckControlC()) {
  1206. return FALSE;
  1207. }
  1208. GetFieldOffset("nt!_DEVICE_NODE", "TargetDeviceNotify", &result);
  1209. TargetDeviceNotifyAddress = DeviceNode + result;
  1210. #define DevFld(F) GetFieldValue(DeviceNode, DevNodeType, #F, F)
  1211. #define DevFld2(F, V) GetFieldValue(DeviceNode, DevNodeType, #F, V)
  1212. if (DevFld2(ServiceName.Length, ServiceName_Len)) {
  1213. xdprintf(Depth, "");
  1214. dprintf("%08p: Could not read device node\n", DeviceNode);
  1215. return TRUE;
  1216. }
  1217. DevFld2(ServiceName.MaximumLength, ServiceName_MaxLen);
  1218. DevFld2(ServiceName.Buffer, ServiceName_Buff);
  1219. DevFld(Flags); DevFld(PhysicalDeviceObject); DevFld(Parent);
  1220. DevFld(Sibling); DevFld(Child); DevFld(InterfaceType);
  1221. DevFld(BusNumber);DevFld(DuplicatePDO); DevFld(State);
  1222. DevFld(PreviousState);
  1223. DevFld2(InstancePath.Length, InstancePath_Len);
  1224. DevFld2(InstancePath.MaximumLength, InstancePath_Max);
  1225. DevFld2(InstancePath.Buffer, InstancePath_Buff);
  1226. DevFld2(TargetDeviceNotify.Blink, TargetDeviceNotify_Blink);
  1227. DevFld2(TargetDeviceNotify.Flink, TargetDeviceNotify_Flink);
  1228. DevFld(UserFlags); DevFld(CapabilityFlags); DevFld(DisableableDepends);
  1229. DevFld(Problem); DevFld(FailureStatus);
  1230. DevFld(PreviousResourceList); DevFld(PreviousResourceRequirements);
  1231. DevFld(ResourceList);
  1232. DevFld(BootResources); DevFld(ResourceListTranslated);
  1233. DevFld(ResourceRequirements); DevFld(StateHistoryEntry);
  1234. #undef DevFld
  1235. #undef DevFld2
  1236. //
  1237. // Check if printing should be suppressed
  1238. //
  1239. if (!PrintDefault && ServiceName != NULL) {
  1240. if ((ServiceName_Buff == 0) || (ServiceName_MaxLen <= 0)) {
  1241. print = FALSE;
  1242. } else {
  1243. //
  1244. // Compare the service names
  1245. //
  1246. UNICODE_STRING v;
  1247. v.Buffer = LocalAlloc(LPTR, ServiceName_MaxLen);
  1248. if (v.Buffer != NULL) {
  1249. v.MaximumLength = ServiceName_MaxLen;
  1250. v.Length = ServiceName_Len;
  1251. if (ReadMemory(ServiceName_Buff,
  1252. v.Buffer,
  1253. ServiceName_Len,
  1254. (PULONG) &AmountRead)) {
  1255. if (RtlCompareUnicodeString(ServiceName,
  1256. &v,
  1257. TRUE) != 0) {
  1258. //
  1259. // We are not interested in this devnode
  1260. //
  1261. print = FALSE;
  1262. }
  1263. }
  1264. LocalFree(v.Buffer);
  1265. }
  1266. }
  1267. }
  1268. if (DumpOnlyDevnodesWithProblems) {
  1269. print = (BOOLEAN) ((Flags & (DNF_HAS_PROBLEM | DNF_HAS_PRIVATE_PROBLEM)) != 0);
  1270. } else if (DumpOnlyNonStartedDevnodes) {
  1271. print = (BOOLEAN) (State != DeviceNodeStarted);
  1272. }
  1273. if (print) {
  1274. xdprintf(Depth, "");
  1275. dprintf("DevNode %#010p for PDO %#010p\n",
  1276. DeviceNode,
  1277. PhysicalDeviceObject);
  1278. Depth++;
  1279. if (!DumpChild) {
  1280. xdprintf(Depth, "");
  1281. dprintf("Parent %#010p Sibling %#010p Child %#010p\n",
  1282. Parent,
  1283. Sibling,
  1284. Child);
  1285. if ((LONG) InterfaceType != -1 ||
  1286. (BusNumber != -1 && BusNumber != -16)) {
  1287. xdprintf(Depth, "InterfaceType %#x Bus Number %#x\n",
  1288. (LONG) InterfaceType,
  1289. BusNumber);
  1290. }
  1291. if (DuplicatePDO != 0) {
  1292. dprintf("Duplicate PDO %#010p", DuplicatePDO);
  1293. }
  1294. }
  1295. if (InstancePath_Buff != 0) {
  1296. UNICODE_STRING64 v;
  1297. xdprintf(Depth, "InstancePath is \"");
  1298. v.MaximumLength = InstancePath_Max;
  1299. v.Length = InstancePath_Len;
  1300. v.Buffer = InstancePath_Buff;
  1301. DumpUnicode64(v);
  1302. dprintf("\"\n");
  1303. }
  1304. if (ServiceName_Buff != 0) {
  1305. UNICODE_STRING64 v;
  1306. xdprintf(Depth, "ServiceName is \"");
  1307. v.MaximumLength = ServiceName_MaxLen;
  1308. v.Length = ServiceName_Len;
  1309. v.Buffer = ServiceName_Buff;
  1310. DumpUnicode64(v);
  1311. dprintf("\"\n");
  1312. }
  1313. if ((TargetDeviceNotify_Flink != 0 ||
  1314. TargetDeviceNotify_Blink != 0) &&
  1315. (TargetDeviceNotify_Blink != TargetDeviceNotifyAddress ||
  1316. TargetDeviceNotify_Flink != TargetDeviceNotifyAddress)) {
  1317. xdprintf(Depth, "TargetDeviceNotify List - ");
  1318. dprintf ("f %#010p b %#010p\n",
  1319. TargetDeviceNotify_Flink,
  1320. TargetDeviceNotify_Blink);
  1321. }
  1322. PrintDevNodeState(Depth, "State", State);
  1323. PrintDevNodeState(Depth, "Previous State", PreviousState);
  1324. if (!DumpChild) {
  1325. GetFieldOffset("nt!_DEVICE_NODE", "StateHistory", &result);
  1326. if (ReadMemory(DeviceNode+result,
  1327. StateHistory,
  1328. sizeof(PNP_DEVNODE_STATE) * STATE_HISTORY_SIZE,
  1329. &AmountRead)) {
  1330. UCHAR Description[20];
  1331. ULONG index;
  1332. INT i;
  1333. for (i = STATE_HISTORY_SIZE - 1; i >= 0; i--) {
  1334. index = (i + StateHistoryEntry) % STATE_HISTORY_SIZE;
  1335. sprintf(Description, "StateHistory[%02d]", index);
  1336. PrintDevNodeState(Depth, Description, StateHistory[index]);
  1337. }
  1338. }
  1339. DumpFlags(Depth, "Flags", Flags, DeviceNodeFlags);
  1340. if (UserFlags != 0) {
  1341. DumpFlags(Depth, "UserFlags", UserFlags, DeviceNodeUserFlags);
  1342. }
  1343. if (CapabilityFlags != 0) {
  1344. DumpFlags(Depth, "CapabilityFlags", CapabilityFlags, DeviceNodeCapabilityFlags);
  1345. }
  1346. if (DisableableDepends != 0) {
  1347. xdprintf(Depth, "DisableableDepends = %d (%s)\n", DisableableDepends,
  1348. (UserFlags&DNUF_NOT_DISABLEABLE)?"including self":"from children");
  1349. }
  1350. }
  1351. if (Flags & DNF_HAS_PROBLEM) {
  1352. if (Problem < NUM_CM_PROB && DevNodeProblems[Problem] != NULL) {
  1353. xdprintf(Depth, "Problem = %s\n", DevNodeProblems[Problem]);
  1354. } else {
  1355. xdprintf(Depth, "Problem = Unknown problem (%d)\n", Problem);
  1356. }
  1357. if (Problem == CM_PROB_FAILED_START) {
  1358. xdprintf(Depth, "Failure Status %#010lx\n", FailureStatus);
  1359. continueDump = DumpResourceList64( "Previous CmResourceList",
  1360. Depth,
  1361. PreviousResourceList);
  1362. if (continueDump) {
  1363. continueDump = DumpResourceRequirementList64( Depth,
  1364. PreviousResourceRequirements);
  1365. }
  1366. }
  1367. }
  1368. if (continueDump && DumpCmRes) {
  1369. continueDump = DumpResourceList64( "CmResourceList",
  1370. Depth,
  1371. ResourceList);
  1372. if (continueDump) {
  1373. continueDump = DumpResourceList64( "BootResourcesList",
  1374. Depth,
  1375. BootResources);
  1376. }
  1377. }
  1378. if (continueDump && DumpCmResTrans) {
  1379. continueDump = DumpResourceList64( "TranslatedResourceList",
  1380. Depth,
  1381. ResourceListTranslated);
  1382. }
  1383. if (continueDump && DumpCmResReqList) {
  1384. continueDump = DumpResourceRequirementList64( Depth,
  1385. ResourceRequirements);
  1386. }
  1387. }
  1388. if (continueDump && DumpChild && Child) {
  1389. continueDump = DumpDevNode( Child,
  1390. Depth,
  1391. DumpChild, // whem dumping a child, dump its siblings, too
  1392. DumpChild,
  1393. DumpCmRes,
  1394. DumpCmResReqList,
  1395. DumpCmResTrans,
  1396. DumpOnlyDevnodesWithProblems,
  1397. DumpOnlyNonStartedDevnodes,
  1398. ServiceName,
  1399. print
  1400. );
  1401. }
  1402. if (continueDump && DumpSibling && Sibling) {
  1403. continueDump = DumpDevNode( Sibling,
  1404. (print ? (Depth - 1):Depth),
  1405. DumpSibling,
  1406. DumpChild,
  1407. DumpCmRes,
  1408. DumpCmResReqList,
  1409. DumpCmResTrans,
  1410. DumpOnlyDevnodesWithProblems,
  1411. DumpOnlyNonStartedDevnodes,
  1412. ServiceName,
  1413. PrintDefault
  1414. );
  1415. }
  1416. return continueDump;
  1417. }
  1418. LPSTR
  1419. GetCmResourceTypeString(
  1420. IN UCHAR Type
  1421. )
  1422. {
  1423. LPSTR typeString;
  1424. switch (Type) {
  1425. case CmResourceTypeNull:
  1426. typeString = "Null";
  1427. break;
  1428. case CmResourceTypePort:
  1429. typeString = "Port";
  1430. break;
  1431. case CmResourceTypeInterrupt:
  1432. typeString = "Interrupt";
  1433. break;
  1434. case CmResourceTypeMemory:
  1435. typeString = "Memory";
  1436. break;
  1437. case CmResourceTypeDma:
  1438. typeString = "Dma";
  1439. break;
  1440. case CmResourceTypeDeviceSpecific:
  1441. typeString = "DeviceSpeific";
  1442. break;
  1443. case CmResourceTypeBusNumber:
  1444. typeString = "BusNumber";
  1445. break;
  1446. case CmResourceTypeMaximum:
  1447. typeString = "Maximum";
  1448. break;
  1449. case CmResourceTypeNonArbitrated:
  1450. typeString = "NonArbitrated/ConfigData";
  1451. break;
  1452. case CmResourceTypeDevicePrivate:
  1453. typeString = "DevicePrivate";
  1454. break;
  1455. case CmResourceTypePcCardConfig:
  1456. typeString = "PcCardConfig";
  1457. break;
  1458. default:
  1459. typeString = "Unknown";
  1460. break;
  1461. }
  1462. return typeString;
  1463. }
  1464. VOID
  1465. DumpResourceDescriptorHeader(
  1466. ULONG Depth,
  1467. ULONG Number,
  1468. UCHAR Option,
  1469. UCHAR Type,
  1470. UCHAR SharingDisposition,
  1471. USHORT Flags
  1472. )
  1473. {
  1474. PUCHAR typeString;
  1475. PUCHAR sharingString;
  1476. xdprintf (Depth, "");
  1477. if (Option & IO_RESOURCE_PREFERRED) {
  1478. dprintf("Preferred ");
  1479. }
  1480. if (Option & IO_RESOURCE_DEFAULT) {
  1481. dprintf("Default ");
  1482. }
  1483. if (Option & IO_RESOURCE_ALTERNATIVE) {
  1484. dprintf("Alternative ");
  1485. }
  1486. dprintf ("Descriptor %d - ", Number);
  1487. typeString = GetCmResourceTypeString(Type);
  1488. switch (SharingDisposition) {
  1489. case CmResourceShareUndetermined:
  1490. sharingString = "Undetermined Sharing";
  1491. break;
  1492. case CmResourceShareDeviceExclusive:
  1493. sharingString = "Device Exclusive";
  1494. break;
  1495. case CmResourceShareDriverExclusive:
  1496. sharingString = "Driver Exclusive";
  1497. break;
  1498. case CmResourceShareShared:
  1499. sharingString = "Shared";
  1500. break;
  1501. default:
  1502. sharingString = "Unknown Sharing";
  1503. break;
  1504. }
  1505. dprintf("%s (%#x) %s (%#x)\n",
  1506. typeString,
  1507. Type,
  1508. sharingString,
  1509. SharingDisposition);
  1510. Depth++;
  1511. xdprintf (Depth, "Flags (%#04wx) - ", Flags);
  1512. switch (Type) {
  1513. case CmResourceTypePort:
  1514. if (Flags == CM_RESOURCE_PORT_MEMORY) {
  1515. dprintf ("PORT_MEMORY ");
  1516. }
  1517. if (Flags & CM_RESOURCE_PORT_IO) {
  1518. dprintf ("PORT_IO ");
  1519. }
  1520. if (Flags & CM_RESOURCE_PORT_10_BIT_DECODE) {
  1521. dprintf ("10_BIT_DECODE ");
  1522. }
  1523. if (Flags & CM_RESOURCE_PORT_12_BIT_DECODE) {
  1524. dprintf ("12_BIT_DECODE ");
  1525. }
  1526. if (Flags & CM_RESOURCE_PORT_16_BIT_DECODE) {
  1527. dprintf ("16_BIT_DECODE ");
  1528. }
  1529. if (Flags & CM_RESOURCE_PORT_POSITIVE_DECODE) {
  1530. dprintf ("POSITIVE_DECODE ");
  1531. }
  1532. break;
  1533. case CmResourceTypeMemory:
  1534. if (Flags == CM_RESOURCE_MEMORY_READ_WRITE) {
  1535. dprintf ("READ_WRITE ");
  1536. }
  1537. if (Flags & CM_RESOURCE_MEMORY_READ_ONLY) {
  1538. dprintf ("READ_ONLY ");
  1539. }
  1540. if (Flags & CM_RESOURCE_MEMORY_WRITE_ONLY) {
  1541. dprintf ("WRITE_ONLY ");
  1542. }
  1543. if (Flags & CM_RESOURCE_MEMORY_PREFETCHABLE) {
  1544. dprintf ("PREFETCHABLE ");
  1545. }
  1546. if (Flags & CM_RESOURCE_MEMORY_COMBINEDWRITE) {
  1547. dprintf ("COMBINEDWRITE ");
  1548. }
  1549. if (Flags & CM_RESOURCE_MEMORY_24) {
  1550. dprintf ("MEMORY_24 ");
  1551. }
  1552. break;
  1553. case CmResourceTypeInterrupt:
  1554. if (Flags == CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE) {
  1555. dprintf ("LEVEL_SENSITIVE ");
  1556. }
  1557. if (Flags & CM_RESOURCE_INTERRUPT_LATCHED) {
  1558. dprintf ("LATCHED ");
  1559. }
  1560. break;
  1561. case CmResourceTypeDma:
  1562. if (Flags & CM_RESOURCE_DMA_8) {
  1563. dprintf ("DMA_8 ");
  1564. }
  1565. if (Flags & CM_RESOURCE_DMA_16) {
  1566. dprintf ("DMA_16 ");
  1567. }
  1568. if (Flags & CM_RESOURCE_DMA_32) {
  1569. dprintf ("DMA_32 ");
  1570. }
  1571. if (Flags & CM_RESOURCE_DMA_8_AND_16) {
  1572. dprintf ("DMA_8_AND_16");
  1573. }
  1574. if (Flags & CM_RESOURCE_DMA_TYPE_A) {
  1575. dprintf ("DMA_TYPE_A");
  1576. }
  1577. if (Flags & CM_RESOURCE_DMA_TYPE_B) {
  1578. dprintf ("DMA_TYPE_B");
  1579. }
  1580. if (Flags & CM_RESOURCE_DMA_TYPE_F) {
  1581. dprintf ("DMA_TYPE_F");
  1582. }
  1583. break;
  1584. }
  1585. dprintf("\n");
  1586. }
  1587. DECLARE_API( arbiter )
  1588. /*++
  1589. Routine Description:
  1590. Dumps all the arbiters in the system, their allocated ranges, and
  1591. the owners of the ranges.
  1592. !arbiter [flags]
  1593. flags 1 - I/O port arbiters
  1594. 2 - memory arbiters
  1595. 4 - IRQ arbiters
  1596. 8 - DMA arbiters
  1597. If no flags are specified, all arbiters are dumped.
  1598. Arguments:
  1599. args - optional flags specifying which arbiters to dump
  1600. Return Value:
  1601. None
  1602. --*/
  1603. {
  1604. DWORD Flags=0;
  1605. ULONG64 addr;
  1606. ULONG64 deviceNode;
  1607. Flags = (ULONG) GetExpression(args);
  1608. if (Flags == 0) {
  1609. Flags = ARBITER_DUMP_ALL;
  1610. } else if (Flags == ARBITER_NO_DUMP_ALIASES) {
  1611. Flags = ARBITER_NO_DUMP_ALIASES + ARBITER_DUMP_ALL;
  1612. }
  1613. //
  1614. // Find the root devnode and dump its arbiters
  1615. //
  1616. addr = GetExpression( "nt!IopRootDeviceNode" );
  1617. if (addr == 0) {
  1618. dprintf("Error retrieving address of IopRootDeviceNode\n");
  1619. return E_INVALIDARG;
  1620. }
  1621. if (!ReadPointer(addr, &deviceNode)) {
  1622. dprintf("Error reading value of IopRootDeviceNode (%#010p)\n", addr);
  1623. return E_INVALIDARG;
  1624. }
  1625. DumpArbitersForDevNode(0, deviceNode, Flags);
  1626. return S_OK;
  1627. }
  1628. BOOLEAN
  1629. DumpArbitersForDevNode(
  1630. IN DWORD Depth,
  1631. IN ULONG64 DevNodeAddr,
  1632. IN DWORD Flags
  1633. )
  1634. /*++
  1635. Routine Description:
  1636. Dumps all the arbiters for the specified devnode. Recursively calls
  1637. itself on the specified devnode's children.
  1638. Arguments:
  1639. Depth - Supplies the print depth.
  1640. DevNodeAddr - Supplies the address of the devnode.
  1641. Flags - Supplies the type of arbiters to dump:
  1642. ARBITER_DUMP_IO
  1643. ARBITER_DUMP_MEMORY
  1644. ARBITER_DUMP_IRQ
  1645. ARBITER_DUMP_DMA
  1646. Return Value:
  1647. None
  1648. --*/
  1649. {
  1650. ULONG result;
  1651. ULONG64 NextArbiter, startAddress;
  1652. ULONG64 ArbiterAddr;
  1653. BOOLEAN PrintedHeader = FALSE;
  1654. BOOLEAN continueDump = TRUE;
  1655. UCHAR devNodeType[] = "nt!_DEVICE_NODE";
  1656. ULONG DevArbOffset, DevNodeOffset;
  1657. ULONG64 DeviceArbiterList_Flink=0, InstancePath_Buffer=0, Child=0, Sibling=0;
  1658. UNICODE_STRING64 InstancePath;
  1659. // Get offset for listing
  1660. if (GetFieldOffset("nt!_PI_RESOURCE_ARBITER_ENTRY", "DeviceArbiterList.Flink", &DevArbOffset)) {
  1661. return TRUE;
  1662. }
  1663. if (GetFieldValue(DevNodeAddr, devNodeType, "DeviceArbiterList.Flink", DeviceArbiterList_Flink)) {
  1664. xdprintf(Depth, ""); dprintf("%08p: Could not read device node\n", DevNodeAddr);
  1665. return TRUE;
  1666. }
  1667. GetFieldOffset("nt!_DEVICE_NODE", "DeviceArbiterList.Flink", &DevNodeOffset);
  1668. GetFieldValue(DevNodeAddr, devNodeType, "Child", Child);
  1669. GetFieldValue(DevNodeAddr, devNodeType, "Sibling", Sibling);
  1670. GetFieldValue(DevNodeAddr, devNodeType, "InstancePath.Buffer", InstancePath.Buffer);
  1671. GetFieldValue(DevNodeAddr, devNodeType, "InstancePath.Length", InstancePath.Length);
  1672. GetFieldValue(DevNodeAddr, devNodeType, "InstancePath.MaximumLength", InstancePath.MaximumLength);
  1673. //
  1674. // Dump the list of arbiters
  1675. //
  1676. startAddress = DevNodeAddr + DevNodeOffset;
  1677. NextArbiter = DeviceArbiterList_Flink;
  1678. while (NextArbiter != startAddress) {
  1679. ULONG ResourceType=0;
  1680. if (CheckControlC()) {
  1681. return FALSE;
  1682. }
  1683. ArbiterAddr = NextArbiter - DevArbOffset;
  1684. // CONTAINING_RECORD(NextArbiter, PI_RESOURCE_ARBITER_ENTRY, DeviceArbiterList);
  1685. if (GetFieldValue(ArbiterAddr, "nt!_PI_RESOURCE_ARBITER_ENTRY", "ResourceType", ResourceType)) {
  1686. xdprintf(Depth, "Could not read arbiter entry ");
  1687. dprintf("%08p for devnode %08p\n", ArbiterAddr, DevNodeAddr);
  1688. break;
  1689. }
  1690. if (((ResourceType == CmResourceTypePort) && (Flags & ARBITER_DUMP_IO)) ||
  1691. ((ResourceType == CmResourceTypeInterrupt) && (Flags & ARBITER_DUMP_IRQ)) ||
  1692. ((ResourceType == CmResourceTypeMemory) && (Flags & ARBITER_DUMP_MEMORY)) ||
  1693. ((ResourceType == CmResourceTypeDma) && (Flags & ARBITER_DUMP_DMA)) ||
  1694. ((ResourceType == CmResourceTypeBusNumber) && (Flags & ARBITER_DUMP_BUS_NUMBER))) {
  1695. if (!PrintedHeader) {
  1696. dprintf("\n");
  1697. xdprintf(Depth, ""); dprintf("DEVNODE %08p ", DevNodeAddr);
  1698. if (InstancePath.Buffer != 0) {
  1699. dprintf("(");
  1700. DumpUnicode64(InstancePath);
  1701. dprintf(")");
  1702. }
  1703. dprintf("\n");
  1704. PrintedHeader = TRUE;
  1705. }
  1706. continueDump = DumpArbiter( Depth+1,
  1707. ArbiterAddr,
  1708. // &ArbiterEntry,
  1709. (BOOLEAN)!(Flags & ARBITER_NO_DUMP_ALIASES)
  1710. );
  1711. if (!continueDump) {
  1712. break;
  1713. }
  1714. }
  1715. GetFieldValue(ArbiterAddr, "nt!_PI_RESOURCE_ARBITER_ENTRY", "DeviceArbiterList.Flink", NextArbiter);
  1716. }
  1717. //
  1718. // Dump this devnode's children (if any)
  1719. //
  1720. if (continueDump && Child) {
  1721. continueDump = DumpArbitersForDevNode( Depth + 1, Child, Flags);
  1722. }
  1723. //
  1724. // Dump this devnode's siblings (if any)
  1725. //
  1726. if (continueDump && Sibling) {
  1727. continueDump = DumpArbitersForDevNode( Depth, Sibling, Flags);
  1728. }
  1729. return continueDump;
  1730. }
  1731. BOOLEAN
  1732. DumpArbiter(
  1733. IN DWORD Depth,
  1734. IN ULONG64 ArbiterAddr,
  1735. IN BOOLEAN DisplayAliases
  1736. )
  1737. /*++
  1738. Routine Description:
  1739. Dumps out the specified arbiter.
  1740. Arguments:
  1741. Depth - Supplies the print depth.
  1742. ArbiterAddr - Supplies the original address of the arbiter (on the target)
  1743. Arbiter - Supplies the arbiter
  1744. Return Value:
  1745. None.
  1746. --*/
  1747. {
  1748. ULONG64 PciInstance;
  1749. ULONG64 InstanceAddr;
  1750. BOOL IsPciArbiter = FALSE;
  1751. WCHAR NameBuffer[64];
  1752. BOOLEAN continueDump = TRUE;
  1753. UCHAR ArbiterTyp[] = "nt!_PI_RESOURCE_ARBITER_ENTRY";
  1754. UCHAR ArbIntfTyp[] = "nt!_ARBITER_INTERFACE";
  1755. ULONG64 ArbiterInterface=0, Context=0, NamePtr=0, ResourceType=0;
  1756. ULONG ListHeadOff;
  1757. //
  1758. // First obtain the ARBITER_INTERFACE.
  1759. //
  1760. if (GetFieldValue(ArbiterAddr, ArbiterTyp, "ArbiterInterface", ArbiterInterface)) {
  1761. dprintf("Error reading ArbiterInterface %08p for ArbiterEntry at %08p\n",
  1762. ArbiterInterface,
  1763. ArbiterAddr);
  1764. return continueDump;
  1765. }
  1766. if (!ArbiterInterface ||
  1767. GetFieldValue(ArbiterInterface, ArbIntfTyp, "Context", Context)) {
  1768. dprintf("Error reading ArbiterInterface %08p for ArbiterEntry at %08p\n",
  1769. ArbiterInterface,
  1770. ArbiterAddr);
  1771. return continueDump;
  1772. }
  1773. //
  1774. // Now that we have the ARBITER_INTERFACE we need to get the ARBITER_INSTANCE.
  1775. // This is not as easy as it should be since PeterJ is paranoid and encrypts the
  1776. // pointer to it. Luckily, we know his secret encryption method and can decrypt it.
  1777. // First we have to figure out if this is PCI's arbiter. Quick hack check is to see
  1778. // if the low bit is set.
  1779. //
  1780. if (Context & 1) {
  1781. ULONG Offset;
  1782. GetFieldOffset("nt!_PCI_ARBITER_INSTANCE", "CommonInstance", &Offset);
  1783. IsPciArbiter = TRUE;
  1784. PciInstance = (Context ^ PciFdoExtensionType);
  1785. InstanceAddr = PciInstance + Offset;
  1786. } else {
  1787. //
  1788. // We will assume that the context is just a pointer to an ARBITER_INSTANCE structure
  1789. //
  1790. InstanceAddr = Context;
  1791. }
  1792. //
  1793. // Read the instance
  1794. //
  1795. if (GetFieldValue(InstanceAddr, "nt!_ARBITER_INSTANCE", "Name", NamePtr)) {
  1796. dprintf("Error reading ArbiterInstance %08p for ArbiterInterface at %08p\n",
  1797. InstanceAddr,
  1798. ArbiterInterface);
  1799. return continueDump;
  1800. }
  1801. GetFieldValue(InstanceAddr, "nt!_ARBITER_INSTANCE", "ResourceType", ResourceType);
  1802. xdprintf(Depth,
  1803. "%s Arbiter \"",
  1804. GetCmResourceTypeString((UCHAR)ResourceType));
  1805. if (NamePtr != 0) {
  1806. ULONG result;
  1807. if (ReadMemory(NamePtr,NameBuffer,sizeof(NameBuffer),&result)) {
  1808. dprintf("%ws",NameBuffer);
  1809. }
  1810. }
  1811. dprintf("\" at %08p\n",InstanceAddr);
  1812. xdprintf(Depth+1,"Allocated ranges:\n");
  1813. GetFieldOffset("nt!_RTL_RANGE_LIST", "ListHead", &ListHeadOff);
  1814. InitTypeRead(InstanceAddr, nt!_ARBITER_INSTANCE);
  1815. continueDump = DumpRangeList( Depth+2,
  1816. ReadField(Allocation) + ListHeadOff,
  1817. FALSE,
  1818. TRUE,
  1819. DisplayAliases);
  1820. if (continueDump) {
  1821. InitTypeRead(InstanceAddr, nt!_ARBITER_INSTANCE);
  1822. xdprintf(Depth+1,"Possible allocation:\n");
  1823. continueDump = DumpRangeList(Depth+2, ReadField(PossibleAllocation) + ListHeadOff,
  1824. FALSE, TRUE, DisplayAliases);
  1825. }
  1826. return continueDump;
  1827. }
  1828. VOID
  1829. DumpRangeEntry(
  1830. IN DWORD Depth,
  1831. IN ULONG64 RangeEntry,
  1832. IN BOOLEAN OwnerIsDevObj,
  1833. IN BOOLEAN DisplayAliases
  1834. )
  1835. /*++
  1836. Routine Description:
  1837. Dumps out the specified RTLP_RANGE_LIST_ENTRY
  1838. Arguments:
  1839. Depth - Supplies the print depth.
  1840. RangeList - Supplies the address of the RTLP_RANGE_LIST_ENTRY
  1841. OwnerIsDevObj - Indicates that the owner field is a pointer to a DEVICE_OBJECT
  1842. Return Value:
  1843. None.
  1844. --*/
  1845. {
  1846. ULONG PrivateFlags=0, Attributes=0, PublicFlags=0;
  1847. ULONG64 Owner=0, Start=0, End=0;
  1848. if (GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "PrivateFlags", PrivateFlags)) {
  1849. dprintf("Cannot find _RTLP_RANGE_LIST_ENTRY type.\n");
  1850. return ;
  1851. }
  1852. // dprintf("Range Entry %p\n", RangeEntry);
  1853. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Attributes", Attributes);
  1854. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Start", Start);
  1855. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "End", End);
  1856. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "PublicFlags", PublicFlags);
  1857. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Allocated.Owner", Owner);
  1858. if (!(PrivateFlags & RTLP_RANGE_LIST_ENTRY_MERGED)) {
  1859. ULONG64 DeviceObjectExtension=0, DeviceNode=0;
  1860. if ((Attributes & ARBITER_RANGE_ALIAS) && !DisplayAliases) {
  1861. return;
  1862. }
  1863. if (OwnerIsDevObj && Owner) {
  1864. if (GetFieldValue(Owner, "nt!_DEVICE_OBJECT", "DeviceObjectExtension", DeviceObjectExtension)) {
  1865. dprintf("Error reading DeviceObject %08p\n",
  1866. Owner
  1867. );
  1868. return;
  1869. }
  1870. if (GetFieldValue(DeviceObjectExtension, "nt!_DEVOBJ_EXTENSION", "DeviceNode", DeviceNode)) {
  1871. dprintf("Error reading DeviceObjectExtension %08p\n",
  1872. DeviceObjectExtension
  1873. );
  1874. return;
  1875. }
  1876. }
  1877. xdprintf(Depth,
  1878. "%016I64x - %016I64x %c%c%c%c%c",
  1879. Start,
  1880. End,
  1881. PublicFlags & RTL_RANGE_SHARED ? 'S' : ' ',
  1882. PublicFlags & RTL_RANGE_CONFLICT ? 'C' : ' ',
  1883. (Attributes & ARBITER_RANGE_BOOT_ALLOCATED)? 'B' : (Attributes & ARBITER_RANGE_SHARE_DRIVER_EXCLUSIVE)? 'D' : ' ',
  1884. Attributes & ARBITER_RANGE_ALIAS ? 'A' : ' ',
  1885. Attributes & ARBITER_RANGE_POSITIVE_DECODE ? 'P' : ' ');
  1886. dprintf(" %08p ",Owner);
  1887. if (Owner) {
  1888. UNICODE_STRING64 ServiceName;
  1889. GetFieldValue(DeviceNode, "nt!_DEVICE_NODE", "ServiceName.Buffer", ServiceName.Buffer);
  1890. if (OwnerIsDevObj && ServiceName.Buffer) {
  1891. GetFieldValue(DeviceNode, "nt!_DEVICE_NODE", "ServiceName.Length", ServiceName.Length);
  1892. GetFieldValue(DeviceNode, "nt!_DEVICE_NODE", "ServiceName.MaximumLength", ServiceName.MaximumLength);
  1893. dprintf(" (");
  1894. DumpUnicode64(ServiceName);
  1895. dprintf(")");
  1896. }
  1897. } else {
  1898. dprintf("<Not on bus>");
  1899. }
  1900. } else {
  1901. xdprintf(Depth,
  1902. "%016I64x - %016I64x %c%c ",
  1903. Start,
  1904. End,
  1905. PublicFlags & RTL_RANGE_SHARED ? 'S' : ' ',
  1906. PublicFlags & RTL_RANGE_CONFLICT ? 'C' : ' '
  1907. );
  1908. }
  1909. dprintf("\n");
  1910. }
  1911. BOOLEAN
  1912. DumpRangeList(
  1913. IN DWORD Depth,
  1914. IN ULONG64 RangeListHead,
  1915. IN BOOLEAN IsMerged,
  1916. IN BOOLEAN OwnerIsDevObj,
  1917. IN BOOLEAN DisplayAliases
  1918. )
  1919. /*++
  1920. Routine Description:
  1921. Dumps out the specified RTL_RANGE_LIST
  1922. Arguments:
  1923. Depth - Supplies the print depth.
  1924. RangeListHead - Supplies the address of the LIST_ENTRY containing the RTLP_RANGE_LIST_ENTRYs
  1925. IsMerged - Indicates whether this list is in a merged range
  1926. OwnerIsDevObj - Indicates that the owner field is a pointer to a DEVICE_OBJECT
  1927. Return Value:
  1928. None.
  1929. --*/
  1930. {
  1931. ULONG64 ListEntry, EntryAddr;
  1932. ULONG ListEntryOffset, ListHeadOffset;
  1933. BOOLEAN continueDump = TRUE;
  1934. GetFieldOffset("nt!_RTLP_RANGE_LIST_ENTRY", "ListEntry", &ListEntryOffset);
  1935. GetFieldOffset("nt!_RTLP_RANGE_LIST_ENTRY", "Merged.ListHead", &ListHeadOffset);
  1936. //
  1937. // Read the range list
  1938. //
  1939. if (GetFieldValue(RangeListHead, "nt!_LIST_ENTRY", "Flink", ListEntry)) {
  1940. dprintf("Error reading RangeList %08p\n", RangeListHead);
  1941. return TRUE;
  1942. }
  1943. if (ListEntry == RangeListHead) {
  1944. xdprintf(Depth, "< none >\n");
  1945. return TRUE;
  1946. }
  1947. while (ListEntry != RangeListHead) {
  1948. ULONG PrivateFlags=0;
  1949. ULONG64 ListHeadAddr=0;
  1950. if (CheckControlC()) {
  1951. continueDump = FALSE;
  1952. break;
  1953. }
  1954. EntryAddr = ListEntry - ListEntryOffset;
  1955. if (GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY", "PrivateFlags", PrivateFlags)) {
  1956. dprintf("Error reading RangeEntry %08p from RangeList %08p\n",
  1957. EntryAddr,
  1958. RangeListHead);
  1959. return TRUE;
  1960. }
  1961. if (PrivateFlags & RTLP_RANGE_LIST_ENTRY_MERGED) {
  1962. //
  1963. // This is a merged range list, call ourselves recursively
  1964. //
  1965. DumpRangeEntry(Depth, EntryAddr, FALSE, DisplayAliases);
  1966. continueDump = DumpRangeList(Depth + 1, EntryAddr + ListHeadOffset, TRUE, TRUE, DisplayAliases);
  1967. if (!continueDump) {
  1968. break;
  1969. }
  1970. } else {
  1971. DumpRangeEntry(Depth, EntryAddr, OwnerIsDevObj, DisplayAliases);
  1972. }
  1973. GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY", "ListEntry.Flink", ListEntry);
  1974. }
  1975. return continueDump;
  1976. }
  1977. DECLARE_API( range )
  1978. /*++
  1979. Routine Description:
  1980. Dumps an RTL_RANGE_LIST
  1981. Arguments:
  1982. args - specifies the address of the RTL_RANGE_LIST
  1983. Return Value:
  1984. None
  1985. --*/
  1986. {
  1987. ULONG ListHeadOffset;
  1988. BOOLEAN continueDump = TRUE;
  1989. ULONG64 RangeList;
  1990. if (GetFieldOffset("nt!_RTL_RANGE_LIST", "ListHead", &ListHeadOffset)) {
  1991. dprintf("Cannot find _RTL_RANGE_LIST type.\n");
  1992. return E_INVALIDARG;
  1993. }
  1994. RangeList = GetExpression(args);
  1995. DumpRangeList(0, RangeList + ListHeadOffset, FALSE, FALSE, TRUE);
  1996. return S_OK;
  1997. }
  1998. ULONG
  1999. DumpArbList(
  2000. PFIELD_INFO ListElement,
  2001. PVOID Context
  2002. )
  2003. {
  2004. UCHAR andOr = ' ';
  2005. ULONG resourceType = 0xffffffff;
  2006. ULONG remaining;
  2007. PULONG data;
  2008. PCHAR headingDefault = " Owner Data";
  2009. PCHAR formatDefault = "%c %s %c %08x %08x %08x %08x %08x %08x\n";
  2010. PCHAR headingMemIo = " Owner Length Alignment Minimum Address - Maximum Address\n";
  2011. PCHAR formatMemIo = "%c %s %c %8x %8x %08x%08x - %08x%08x\n";
  2012. PCHAR headingIrqDma = " Owner Minimum - Maximum\n";
  2013. PCHAR formatIrqDma = "%c %s %c %8x - %-8x\n";
  2014. PCHAR heading = headingDefault;
  2015. PCHAR format = formatDefault;
  2016. CHAR shared;
  2017. CHAR ownerBuffer[20];
  2018. PCHAR ownerBlank = " ";
  2019. PCHAR ownerText = ownerBlank;
  2020. ULONG64 pcurrent=ListElement->address;
  2021. ULONG64 palternative;
  2022. static ULONG64 previousOwner = 0;
  2023. BOOLEAN doHeading = TRUE;
  2024. BOOLEAN verbose = FALSE;
  2025. ULONG Flag = *((PULONG) Context);
  2026. ULONG AlternativeCount=0, resDescSize;
  2027. ULONG64 PhysicalDeviceObject, Alternatives;
  2028. ULONG Flags=0, InterfaceType=0, RequestSource=0;
  2029. //
  2030. // Check if user wants out.
  2031. //
  2032. if (CheckControlC()) {
  2033. dprintf("User terminated with <control>C\n");
  2034. return TRUE;
  2035. }
  2036. if ((LONG64)pcurrent >= 0) {
  2037. dprintf("List broken, forward entry is not a valid kernel address.\n");
  2038. return TRUE;
  2039. }
  2040. if (Flag & 1) {
  2041. verbose = TRUE;
  2042. }
  2043. //
  2044. // Get the arbitration list entry from host memory.
  2045. //
  2046. if (GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "PhysicalDeviceObject", PhysicalDeviceObject)) {
  2047. dprintf("Couldn't read _ARBITER_LIST_ENTRY at %p\n", pcurrent);
  2048. return TRUE;
  2049. }
  2050. //
  2051. // Check if we've changed device objects.
  2052. //
  2053. if (previousOwner != PhysicalDeviceObject) {
  2054. previousOwner = PhysicalDeviceObject;
  2055. sprintf(ownerBuffer, "%08p", previousOwner);
  2056. ownerText = ownerBuffer;
  2057. andOr = ' ';
  2058. if (verbose) {
  2059. doHeading = TRUE;
  2060. }
  2061. }
  2062. //
  2063. // Run the alternatives for this device object.
  2064. //
  2065. GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "AlternativeCount", AlternativeCount);
  2066. GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "Alternatives", Alternatives);
  2067. GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "Flags", Flags);
  2068. GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "InterfaceType", InterfaceType);
  2069. GetFieldValue(pcurrent, "nt!_ARBITER_LIST_ENTRY", "RequestSource", RequestSource);
  2070. resDescSize = GetTypeSize("nt!_IO_RESOURCE_DESCRIPTOR");
  2071. for (remaining = AlternativeCount, palternative = Alternatives;
  2072. remaining;
  2073. remaining--, palternative+=resDescSize) {
  2074. ULONG Type=0, ShareDisposition=0, Gen_Length=0, Gen_Alignment=0;
  2075. ULONG Gen_Min_Low, Gen_Min_Hi=0, Gen_Max_Low=0, Gen_Max_Hi=0;
  2076. ULONG Int_Min=0, Int_Max=0, DevData[6]={0};
  2077. //
  2078. // Check if user wants out.
  2079. //
  2080. if (CheckControlC()) {
  2081. dprintf("User terminated with <control>C\n");
  2082. return TRUE;
  2083. }
  2084. //
  2085. // Read this resource descriptor from memory. (An optimization
  2086. // would be to read the entire array or to buffer it).
  2087. //
  2088. if (GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "Type", Type)){
  2089. dprintf("Couldn't read IO_RESOURCE_DESCRIPTOR at %08p, quitting.\n",
  2090. palternative
  2091. );
  2092. return TRUE;
  2093. }
  2094. if (Type != resourceType) {
  2095. if (resourceType == 0xffffffff) {
  2096. //
  2097. // Go look at the first alternative to figure out what
  2098. // resource type is being arbitrated.
  2099. //
  2100. resourceType = Type;
  2101. switch (resourceType) {
  2102. case CmResourceTypeNull:
  2103. dprintf("Arbitration list resource type is NULL. Seems odd.\n");
  2104. break;
  2105. case CmResourceTypePort:
  2106. dprintf("Arbitrating CmResourceTypePort resources\n");
  2107. format = formatMemIo;
  2108. heading = headingMemIo;
  2109. break;
  2110. case CmResourceTypeInterrupt:
  2111. dprintf("Arbitrating CmResourceTypeInterrupt resources\n");
  2112. format = formatIrqDma;
  2113. heading = headingIrqDma;
  2114. break;
  2115. case CmResourceTypeMemory:
  2116. dprintf("Arbitrating CmResourceTypeMemory resources\n");
  2117. format = formatMemIo;
  2118. heading = headingMemIo;
  2119. break;
  2120. case CmResourceTypeDma:
  2121. dprintf("Arbitrating CmResourceTypeDma resources\n");
  2122. format = formatIrqDma;
  2123. heading = headingIrqDma;
  2124. break;
  2125. default:
  2126. dprintf("Arbitrating resource type 0x%x\n");
  2127. }
  2128. } else {
  2129. dprintf("error: Resource Type change, arbiters don't do multiple\n");
  2130. dprintf("types. Resource Type was 0x%x, new type 0x%x, quitting.\n", Type);
  2131. return TRUE;
  2132. }
  2133. }
  2134. if (doHeading) {
  2135. if (verbose) {
  2136. dprintf("Owning object %08p, Flags 0x%x, Interface 0x%x, Source 0x%x\n",
  2137. PhysicalDeviceObject,
  2138. Flags,
  2139. InterfaceType,
  2140. RequestSource
  2141. );
  2142. }
  2143. dprintf(heading);
  2144. doHeading = FALSE;
  2145. }
  2146. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "ShareDisposition", ShareDisposition);
  2147. shared = ShareDisposition == CmResourceShareShared ? 'S' : ' ';
  2148. switch (resourceType) {
  2149. case CmResourceTypePort:
  2150. case CmResourceTypeMemory:
  2151. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.Length", Gen_Length);
  2152. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.Alignment", Gen_Alignment);
  2153. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.MinimumAddress.HighPart", Gen_Min_Hi);
  2154. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.MinimumAddress.LowPart", Gen_Min_Low);
  2155. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.MaximumAddress.HighPart", Gen_Max_Hi);
  2156. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Generic.MaximumAddress.LowPart", Gen_Max_Low );
  2157. dprintf(format,
  2158. andOr,
  2159. ownerText,
  2160. shared,
  2161. Gen_Length,
  2162. Gen_Alignment,
  2163. Gen_Min_Hi,
  2164. Gen_Min_Low,
  2165. Gen_Max_Hi,
  2166. Gen_Max_Low
  2167. );
  2168. break;
  2169. case CmResourceTypeInterrupt:
  2170. case CmResourceTypeDma:
  2171. //
  2172. // Dma and Interrupt are same format,... overload.
  2173. //
  2174. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Interrupt.MinimumVector", Int_Min);
  2175. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.Interrupt.MaximumVector", Int_Max);
  2176. dprintf(format,
  2177. andOr,
  2178. ownerText,
  2179. shared,
  2180. Int_Min,
  2181. Int_Max
  2182. );
  2183. break;
  2184. default:
  2185. GetFieldValue(palternative, "nt!_IO_RESOURCE_DESCRIPTOR", "u.DevicePrivate.Data", DevData);
  2186. data = DevData;
  2187. dprintf(format,
  2188. andOr,
  2189. ownerText,
  2190. shared,
  2191. *data,
  2192. *(data + 0),
  2193. *(data + 1),
  2194. *(data + 2),
  2195. *(data + 3),
  2196. *(data + 4),
  2197. *(data + 5)
  2198. );
  2199. break;
  2200. }
  2201. ownerText = ownerBlank;
  2202. andOr = '|';
  2203. }
  2204. if (AlternativeCount == 0) {
  2205. //
  2206. // Get here if the alternatives list was empty.
  2207. //
  2208. if (doHeading) {
  2209. dprintf("Owning object %08p, Flags 0x%x, Interface 0x%x\n",
  2210. PhysicalDeviceObject,
  2211. Flags,
  2212. InterfaceType
  2213. );
  2214. }
  2215. dprintf("Arbiter list entry (@%08p) has 0 resources. Seems odd.\n",
  2216. pcurrent
  2217. );
  2218. }
  2219. andOr = '&';
  2220. return FALSE; // Continue dumping the list
  2221. }
  2222. DECLARE_API( arblist )
  2223. /*++
  2224. Routine Description:
  2225. Dumps an arbitration list (2nd parameter to PnpTestAllocation).
  2226. !arblist address
  2227. If no address specified we try to tell you what to do.
  2228. Arguments:
  2229. address of the arbitration list.
  2230. Return Value:
  2231. None
  2232. --*/
  2233. {
  2234. ULONG64 ArbitrationList = 0;
  2235. ULONG Flags = 0;
  2236. ULONG64 pcurrent;
  2237. ULONG64 palternative;
  2238. ULONG64 KModeCheck;
  2239. BOOL Ptr64 = IsPtr64();
  2240. if (GetExpressionEx(args, &ArbitrationList, &args)) {
  2241. Flags = (ULONG) GetExpression(args);
  2242. }
  2243. if (ArbitrationList == 0) {
  2244. dprintf("!arblist <address> [flags]\n");
  2245. dprintf(" <address> is the address of an arbitration list.\n");
  2246. dprintf(" [flags] Bitmask used to adjust what is printed.\n");
  2247. dprintf(" 0001 Include interface, flags and source for each device\n");
  2248. dprintf("\n");
  2249. dprintf(" Prints out an arbitration list. An arbitration list\n");
  2250. dprintf(" is the set of resources for which arbitration is\n");
  2251. dprintf(" needed. (eg second parameter to PnpTestAllocation).\n");
  2252. return E_INVALIDARG;
  2253. }
  2254. if (Ptr64) {
  2255. KModeCheck = 0x8000000000000000L;
  2256. } else {
  2257. KModeCheck = 0xffffffff80000000L;
  2258. }
  2259. //
  2260. // The argument is a pointer to a list head.
  2261. //
  2262. if (!((LONG64)ArbitrationList & KModeCheck)) {
  2263. dprintf("The arbitration list pointer must be a kernel mode address.\n");
  2264. return E_INVALIDARG;
  2265. }
  2266. if (GetFieldValue(ArbitrationList, "nt!_LIST_ENTRY", "Flink", pcurrent)) {
  2267. dprintf("error reading arbitration list ListHead, cannot continue.\n");
  2268. return E_INVALIDARG;
  2269. }
  2270. //
  2271. // A little validation.
  2272. //
  2273. if (!(pcurrent & KModeCheck)) {
  2274. dprintf("%08p does not point to a valid list head.\n", ArbitrationList);
  2275. return E_INVALIDARG;
  2276. }
  2277. if (pcurrent == ArbitrationList) {
  2278. dprintf("%08p points to an empty list.\n", ArbitrationList);
  2279. return E_INVALIDARG;
  2280. }
  2281. //
  2282. // Run the list.
  2283. //
  2284. ListType("_ARBITER_LIST_ENTRY", pcurrent, TRUE, "ListEntry.Flink", (PVOID) &Flags, DumpArbList);
  2285. return S_OK;
  2286. }
  2287. ULONG
  2288. TokenizeString(
  2289. PUCHAR Input,
  2290. PUCHAR *Output,
  2291. ULONG Max
  2292. )
  2293. /*++
  2294. Routine Description:
  2295. Convert an input string of white space delimited tokens into
  2296. an array of strings. The routine will produce an array of
  2297. pointers to strings where the (maximum) size of the array is
  2298. given by Max. Note, if Max is too small to fully tokenize the
  2299. string, the last element of the array will point to the remainder
  2300. of the string.
  2301. Arguments:
  2302. Input Input string
  2303. Output Pointer to an array which will be filled in with
  2304. pointers to each new token.
  2305. Max Maximum number of strings.
  2306. Return Value:
  2307. Number of tokens found.
  2308. --*/
  2309. {
  2310. ULONG count = 0;
  2311. PUCHAR tok;
  2312. BOOLEAN inToken = FALSE;
  2313. UCHAR c;
  2314. if (Max == 0) {
  2315. return 0;
  2316. }
  2317. while ((c = *Input++) != '\0') {
  2318. if (isspace(c)) {
  2319. if (!inToken) {
  2320. //
  2321. // Continue skipping characters.
  2322. //
  2323. continue;
  2324. }
  2325. //
  2326. // Found end of Token, delimit it and change state.
  2327. //
  2328. inToken = FALSE;
  2329. *(Input - 1) = '\0';
  2330. } else {
  2331. //
  2332. // Non-blank character.
  2333. //
  2334. if (inToken) {
  2335. //
  2336. // Continue search for end of token.
  2337. //
  2338. continue;
  2339. }
  2340. //
  2341. // New Token.
  2342. //
  2343. inToken = TRUE;
  2344. *Output++ = Input - 1;
  2345. count++;
  2346. if (count == Max) {
  2347. //
  2348. // We can't find any more so return what we still
  2349. // have as the last thing.
  2350. //
  2351. return count;
  2352. }
  2353. }
  2354. }
  2355. return count;
  2356. }
  2357. //
  2358. // DevExtPCI
  2359. //
  2360. // Data structures and functions to implement !devext pci.
  2361. //
  2362. PUCHAR DevExtPciDeviceState[] = {
  2363. "PciNotStarted",
  2364. "PciStarted",
  2365. "PciDeleted",
  2366. "PciStopped",
  2367. "PciSurpriseRemoved",
  2368. "PciSynchronizedOperation"
  2369. };
  2370. PUCHAR DevExtPciSystemPowerState[] = {
  2371. "Unspecified",
  2372. "Working",
  2373. "Sleeping1",
  2374. "Sleeping2",
  2375. "Sleeping3",
  2376. "Hibernate",
  2377. "Shutdown"
  2378. };
  2379. PUCHAR DevExtPciDevicePowerState[] = {
  2380. "Unspecified",
  2381. "D0",
  2382. "D1",
  2383. "D2",
  2384. "D3"
  2385. };
  2386. struct {
  2387. USHORT VendorId;
  2388. PUCHAR VendorName;
  2389. } DevExtPciVendors[] = {
  2390. { 0x003D, "LOCKHEED MARTIN"},
  2391. { 0x0E11, "COMPAQ"},
  2392. { 0x1002, "ATI TECHNOLOGIES INC"},
  2393. { 0x1003, "ULSI SYSTEMS"},
  2394. { 0x1004, "VLSI TECHNOLOGY INC"},
  2395. { 0x1006, "REPLY GROUP"},
  2396. { 0x1007, "NETFRAME SYSTEMS INC"},
  2397. { 0x1008, "EPSON"},
  2398. { 0x100A, "PHOENIX TECHNOLOGIES"},
  2399. { 0x100B, "NATIONAL SEMICONDUCTOR CORPORATION"},
  2400. { 0x100C, "TSENG LABS INC"},
  2401. { 0x100D, "AST RESEARCH INC"},
  2402. { 0x1010, "VIDEO LOGIC LTD"},
  2403. { 0x1011, "DIGITAL EQUIPMENT CORPORATION"},
  2404. { 0x1012, "MICRONICS COMPUTERS INC"},
  2405. { 0x1013, "CIRRUS LOGIC"},
  2406. { 0x1014, "IBM"},
  2407. { 0x1015, "LSI LOGIC CORP OF CANADA"},
  2408. { 0x1016, "ICL PERSONAL SYSTEMS"},
  2409. { 0x1017, "SPEA SOFTWARE AG"},
  2410. { 0x1018, "UNISYS SYSTEMS"},
  2411. { 0x1019, "ELITEGROUP COMPUTER SYS"},
  2412. { 0x101A, "NCR"},
  2413. { 0x101C, "WESTERN DIGITAL"},
  2414. { 0x101E, "AMERICAN MEGATRENDS"},
  2415. { 0x101F, "PICTURETEL"},
  2416. { 0x1021, "OKI ELECTRIC INDUSTRY"},
  2417. { 0x1022, "ADVANCED MICRO DEVICES"},
  2418. { 0x1023, "TRIDENT MICROSYSTEMS"},
  2419. { 0x1025, "ACER INCORPORATED"},
  2420. { 0x1028, "DELL COMPUTER CORPORATION"},
  2421. { 0x102B, "MATROX"},
  2422. { 0x102C, "CHIPS AND TECHNOLOGIES"},
  2423. { 0x102E, "OLIVETTI ADVANCED TECHNOLOGY"},
  2424. { 0x102F, "TOSHIBA AMERICA ELECT."},
  2425. { 0x1030, "TMC RESEARCH"},
  2426. { 0x1031, "MIRO COMPUTER PRODUCTS AG"},
  2427. { 0x1033, "NEC CORPORATION"},
  2428. { 0x1033, "NEC CORPORATION"},
  2429. { 0x1034, "BURNDY CORPORATION"},
  2430. { 0x1035, "COMP.&COMM. RESH. LAB"},
  2431. { 0x1036, "FUTURE DOMAIN"},
  2432. { 0x1037, "HITACHI MICRO SYSTEMS"},
  2433. { 0x1038, "AMP, INC"},
  2434. { 0x1039, "SILICON INTEGRATED SYSTEM"},
  2435. { 0x103A, "SEIKO EPSON CORPORATION"},
  2436. { 0x103B, "TATUNG CO. OF AMERICA"},
  2437. { 0x103C, "HEWLETT PACKARD"},
  2438. { 0x103E, "SOLLIDAY ENGINEERING"},
  2439. { 0x103F, "SYNOPSYS INC."},
  2440. { 0x1040, "ACCELGRAPHICS INC."},
  2441. { 0x1042, "PC TECHNOLOGY INC"},
  2442. { 0x1043, "ASUSTEK COMPUTER, INC."},
  2443. { 0x1044, "DISTRIBUTED PROCESSING TECHNOLOGY"},
  2444. { 0x1045, "OPTI"},
  2445. { 0x1046, "IPC CORPORATION, LTD."},
  2446. { 0x1047, "GENOA SYSTEMS CORP"},
  2447. { 0x1048, "ELSA GMBH"},
  2448. { 0x1049, "FOUNTAIN TECHNOLOGY"},
  2449. { 0x104A, "SGS THOMSON"},
  2450. { 0x104B, "BUSLOGIC"},
  2451. { 0x104C, "TEXAS INSTRUMENTS"},
  2452. { 0x104D, "SONY CORPORATION"},
  2453. { 0x104E, "OAK TECHNOLOGY, INC"},
  2454. { 0x1050, "WINBOND ELECTRONICS CORP"},
  2455. { 0x1051, "ANIGMA, INC."},
  2456. { 0x1055, "EFAR MICROSYSTEMS"},
  2457. { 0x1057, "MOTOROLA"},
  2458. { 0x1058, "ELECTRONICS & TELEC. RSH"},
  2459. { 0x1059, "TEKNOR INDUSTRIAL COMPUTERS INC"},
  2460. { 0x105A, "PROMISE TECHNOLOGY"},
  2461. { 0x105B, "FOXCONN INTERNATIONAL"},
  2462. { 0x105C, "WIPRO INFOTECH LIMITED"},
  2463. { 0x105D, "NUMBER 9 COMPUTER COMPANY"},
  2464. { 0x105E, "VTECH COMPUTERS LTD"},
  2465. { 0x105F, "INFOTRONIC AMERICA INC"},
  2466. { 0x1060, "UNITED MICROELECTRONICS"},
  2467. { 0x1061, "I.T.T."},
  2468. { 0x1062, "MASPAR COMPUTER CORP"},
  2469. { 0x1063, "OCEAN OFFICE AUTOMATION"},
  2470. { 0x1064, "ALCATEL CIT"},
  2471. { 0x1065, "TEXAS MICROSYSTEMS"},
  2472. { 0x1066, "PICOPOWER TECHNOLOGY"},
  2473. { 0x1067, "MITSUBISHI ELECTRONICS"},
  2474. { 0x1068, "DIVERSIFIED TECHNOLOGY"},
  2475. { 0x1069, "MYLEX CORPORATION"},
  2476. { 0x106B, "APPLE COMPUTER INC."},
  2477. { 0x106C, "HYUNDAI ELECTRONICS AMERI"},
  2478. { 0x106D, "SEQUENT"},
  2479. { 0x106E, "DFI, INC"},
  2480. { 0x106F, "CITY GATE DEVELOPMENT LTD"},
  2481. { 0x1070, "DAEWOO TELECOM LTD"},
  2482. { 0x1073, "YAMAHA CORPORATION"},
  2483. { 0x1074, "NEXGEN MICROSYSTEME"},
  2484. { 0x1075, "ADVANCED INTEGRATION RES."},
  2485. { 0x1076, "CHAINTECH COMPUTER CO. LTD"},
  2486. { 0x1077, "Q LOGIC"},
  2487. { 0x1078, "CYRIX CORPORATION"},
  2488. { 0x1079, "I-BUS"},
  2489. { 0x107B, "GATEWAY 2000"},
  2490. { 0x107C, "LG ELECTRONICS"},
  2491. { 0x107D, "LEADTEK RESEARCH INC."},
  2492. { 0x107E, "INTERPHASE CORPORATION"},
  2493. { 0x107F, "DATA TECHNOLOGY CORPORATION"},
  2494. { 0x1080, "CYPRESS SEMICONDUCTOR"},
  2495. { 0x1081, "RADIUS"},
  2496. { 0x1083, "FOREX COMPUTER CORPORATION"},
  2497. { 0x1085, "TULIP COMPUTERS INT.B.V."},
  2498. { 0x1089, "DATA GENERAL CORPORATION"},
  2499. { 0x108A, "BIT 3 COMPUTER"},
  2500. { 0x108C, "OAKLEIGH SYSTEMS INC."},
  2501. { 0x108D, "OLICOM"},
  2502. { 0x108E, "SUN MICROSYSTEMS"},
  2503. { 0x108F, "SYSTEMSOFT"},
  2504. { 0x1090, "ENCORE COMPUTER CORPORATION"},
  2505. { 0x1091, "INTERGRAPH CORPORATION"},
  2506. { 0x1092, "DIAMOND MULTMEDIA SYSTEMS"},
  2507. { 0x1093, "NATIONAL INSTRUMENTS"},
  2508. { 0x1094, "FIRST INT'L COMPUTERS"},
  2509. { 0x1095, "CMD TECHNOLOGY INC"},
  2510. { 0x1096, "ALACRON"},
  2511. { 0x1098, "QUANTUM DESIGNS (H.K.) LTD"},
  2512. { 0x109B, "GEMLIGHT COMPUTER LTD."},
  2513. { 0x109E, "BROOKTREE CORPORATION"},
  2514. { 0x109F, "TRIGEM COMPUTER INC."},
  2515. { 0x10A0, "MEIDENSHA CORPORATION"},
  2516. { 0x10A2, "QUANTUM CORPORATION"},
  2517. { 0x10A3, "EVEREX SYSTEMS INC"},
  2518. { 0x10A5, "RACAL INTERLAN"},
  2519. { 0x10A8, "SIERRA SEMICONDUCTOR"},
  2520. { 0x10A9, "SILICON GRAPHICS"},
  2521. { 0x10AA, "ACC MICROELECTRONICS"},
  2522. { 0x10AD, "SYMPHONY LABS"},
  2523. { 0x10AE, "CORNERSTONE TECHNOLOGY"},
  2524. { 0x10B0, "CARDEXPERT TECHNOLOGY"},
  2525. { 0x10B1, "CABLETRON SYSTEMS INC"},
  2526. { 0x10B2, "RAYTHEON COMPANY"},
  2527. { 0x10B3, "DATABOOK INC"},
  2528. { 0x10B4, "STB SYSTEMS INC"},
  2529. { 0x10B5, "PLX TECHNOLOGY"},
  2530. { 0x10B6, "MADGE NETWORKS"},
  2531. { 0x10B7, "3COM CORPORATION"},
  2532. { 0x10B8, "STANDARD MICROSYSTEMS"},
  2533. { 0x10B9, "ACER LABS"},
  2534. { 0x10BA, "MITSUBISHI ELECTRONICS CORP."},
  2535. { 0x10BC, "ADVANCED LOGIC RESEARCH"},
  2536. { 0x10BD, "SURECOM TECHNOLOGY"},
  2537. { 0x10C0, "BOCA RESEARCH INC."},
  2538. { 0x10C1, "ICM CO., LTD."},
  2539. { 0x10C2, "AUSPEX SYSTEMS INC."},
  2540. { 0x10C3, "SAMSUNG SEMICONDUCTORS"},
  2541. { 0x10C4, "AWARD SOFTWARE INTERNATIONAL INC."},
  2542. { 0x10C5, "XEROX CORPORATION"},
  2543. { 0x10C6, "RAMBUS INC."},
  2544. { 0x10C8, "NEOMAGIC CORPORATION"},
  2545. { 0x10C9, "DATAEXPERT CORPORATION"},
  2546. { 0x10CB, "OMRON CORPORATION"},
  2547. { 0x10CC, "MENTOR ARC INC"},
  2548. { 0x10CD, "ADVANCED SYSTEM PRODUCTS, INC"},
  2549. { 0x10D0, "FUJITSU LIMITED"},
  2550. { 0x10D1, "FUTURE+ SYSTEMS"},
  2551. { 0x10D2, "MOLEX INCORPORATED"},
  2552. { 0x10D3, "JABIL CIRCUIT INC"},
  2553. { 0x10D4, "HUALON MICROELECTRONICS"},
  2554. { 0x10D6, "CETIA"},
  2555. { 0x10D7, "BCM ADVANCED"},
  2556. { 0x10D8, "ADVANCED PERIPHERALS LABS"},
  2557. { 0x10DA, "THOMAS-CONRAD CORPORATION"},
  2558. { 0x10DC, "CERN/ECP/EDU"},
  2559. { 0x10DD, "EVANS & SUTHERLAND"},
  2560. { 0x10DE, "NVIDIA CORPORATION"},
  2561. { 0x10DF, "EMULEX CORPORATION"},
  2562. { 0x10E0, "INTEGRATED MICRO SOLUTIONS INC."},
  2563. { 0x10E1, "TEKRAM TECHNOLOGY CO.,LTD."},
  2564. { 0x10E2, "APTIX CORPORATION"},
  2565. { 0x10E3, "TUNDRA SEMICONDUCTOR (formerly NEWBRIDGE MICROSYSTEMS)"},
  2566. { 0x10E4, "TANDEM COMPUTERS"},
  2567. { 0x10E5, "MICRO INDUSTRIES CORPORATION"},
  2568. { 0x10E6, "GAINBERY COMPUTER PRODUCTS INC."},
  2569. { 0x10E7, "VADEM"},
  2570. { 0x10E8, "APPLIED MICRO CIRCUITS CORPORATION"},
  2571. { 0x10E9, "ALPS ELECTRIC CO. LTD."},
  2572. { 0x10EB, "ARTISTS GRAPHICS"},
  2573. { 0x10EC, "REALTEK SEMICONDUCTOR CO., LTD."},
  2574. { 0x10ED, "ASCII CORPORATION"},
  2575. { 0x10EE, "XILINX CORPORATION"},
  2576. { 0x10EF, "RACORE COMPUTER PRODUCTS, INC."},
  2577. { 0x10F0, "PERITEK CORPORATION"},
  2578. { 0x10F1, "TYAN COMPUTER"},
  2579. { 0x10F3, "ALARIS, INC."},
  2580. { 0x10F4, "S-MOS SYSTEMS"},
  2581. { 0x10F5, "NKK CORPORATION"},
  2582. { 0x10F6, "CREATIVE ELECTRONIC SYSTEMS SA"},
  2583. { 0x10F7, "MATSUSHITA ELECTRIC INDUSTRIAL CO., LTD."},
  2584. { 0x10F8, "ALTOS INDIA LTD"},
  2585. { 0x10F9, "PC DIRECT"},
  2586. { 0x10FA, "TRUEVISION"},
  2587. { 0x10FB, "THESYS GES. F. MIKROELEKTRONIK MGH"},
  2588. { 0x10FC, "I-O DATA DEVICE, INC."},
  2589. { 0x10FD, "SOYO COMPUTER INC."},
  2590. { 0x10FE, "FAST ELECTRONIC GMBH"},
  2591. { 0x10FF, "NCUBE"},
  2592. { 0x1100, "JAZZ MULTIMEDIA"},
  2593. { 0x1101, "INITIO CORPORATION"},
  2594. { 0x1102, "CREATIVE LABS"},
  2595. { 0x1103, "TRIONES TECHNOLOGIES, INC."},
  2596. { 0x1104, "RASTEROPS"},
  2597. { 0x1105, "SIGMA DESIGNS, INC"},
  2598. { 0x1106, "VIA TECHNOLOGIES, INC."},
  2599. { 0x1107, "STRATUS COMPUTERS"},
  2600. { 0x1108, "PROTEON, INC."},
  2601. { 0x1109, "COGENT DATA TECHNOLOGIES"},
  2602. { 0x110A, "SIEMENS NIXDORF IS/AG"},
  2603. { 0x110B, "CHROMATIC RESEARCH INC."},
  2604. { 0x110C, "MINI-MAX TECHNOLOGY, INC."},
  2605. { 0x110D, "ZNYX ADVANCED SYSTEMS"},
  2606. { 0x110E, "CPU TECHNOLOGY"},
  2607. { 0x110F, "ROSS TECHNOLOGY"},
  2608. { 0x1110, "Fire Power Systems, Inc."},
  2609. { 0x1111, "SANTA CRUZ OPERATION"},
  2610. { 0x1112, "RNS (formerly ROCKWELL NETWORK SYSTEMS)"},
  2611. { 0x1114, "ATMEL CORPORATION"},
  2612. { 0x1117, "DATACUBE, INC"},
  2613. { 0x1118, "BERG ELECTRONICS"},
  2614. { 0x1119, "VORTEX COMPUTERSYSTEME GMBH"},
  2615. { 0x111A, "EFFICIENT NETWORKS"},
  2616. { 0x111B, "LITTON GCS"},
  2617. { 0x111C, "TRICORD SYSTEMS"},
  2618. { 0x111D, "INTEGRATED DEVICE TECH"},
  2619. { 0x111F, "PRECISION DIGITAL IMAGES"},
  2620. { 0x1120, "EMC CORPORATION"},
  2621. { 0x1122, "MULTI-TECH SYSTEMS, INC."},
  2622. { 0x1123, "EXCELLENT DESIGN, INC."},
  2623. { 0x1124, "LEUTRON VISION AG"},
  2624. { 0x1127, "FORE SYSTEMS INC"},
  2625. { 0x1129, "FIRMWORKS"},
  2626. { 0x112A, "HERMES ELECTRONICS COMPANY, LTD."},
  2627. { 0x112B, "LINOTYPE - HELL AG"},
  2628. { 0x112C, "ZENITH DATA SYSTEMS"},
  2629. { 0x112E, "INFOMEDIA MICROELECTRONICS INC."},
  2630. { 0x112F, "IMAGING TECHNLOGY INC"},
  2631. { 0x1130, "COMPUTERVISION"},
  2632. { 0x1131, "PHILIPS SEMICONDUCTORS"},
  2633. { 0x1132, "MITEL CORP."},
  2634. { 0x1133, "EICON TECHNOLOGY CORPORATION"},
  2635. { 0x1134, "MERCURY COMPUTER SYSTEMS"},
  2636. { 0x1135, "FUJI XEROX CO LTD"},
  2637. { 0x1136, "MOMENTUM DATA SYSTEMS"},
  2638. { 0x1137, "CISCO SYSTEMS INC"},
  2639. { 0x1138, "ZIATECH CORPORATION"},
  2640. { 0x1139, "DYNAMIC PICTURES INC"},
  2641. { 0x113A, "FWB INC"},
  2642. { 0x113B, "NETWORK COMPUTING DEVICES"},
  2643. { 0x113C, "CYCLONE MICROSYSTEMS"},
  2644. { 0x113D, "LEADING EDGE PRODUCTS INC"},
  2645. { 0x113E, "SANYO ELECTRIC CO"},
  2646. { 0x113F, "EQUINOX SYSTEMS"},
  2647. { 0x1140, "INTERVOICE INC"},
  2648. { 0x1141, "CREST MICROSYSTEM INC"},
  2649. { 0x1142, "ALLIANCE SEMICONDUCTOR CORPORATION"},
  2650. { 0x1143, "NETPOWER, INC"},
  2651. { 0x1144, "CINCINNATI MILACRON"},
  2652. { 0x1145, "WORKBIT CORP"},
  2653. { 0x1146, "FORCE COMPUTERS"},
  2654. { 0x1147, "INTERFACE CORP"},
  2655. { 0x1148, "SYSKONNECT"},
  2656. { 0x1149, "WIN SYSTEM CORPORATION"},
  2657. { 0x114A, "VMIC"},
  2658. { 0x114B, "CANOPUS CO., LTD"},
  2659. { 0x114C, "ANNABOOKS"},
  2660. { 0x114D, "IC CORPORATION"},
  2661. { 0x114E, "NIKON SYSTEMS INC"},
  2662. { 0x114F, "DIGI INTERNATIONAL"},
  2663. { 0x1151, "JAE ELECTRONICS INC."},
  2664. { 0x1152, "MEGATEK"},
  2665. { 0x1153, "LAND WIN ELECTRONIC CORP"},
  2666. { 0x1154, "MELCO INC"},
  2667. { 0x1155, "PINE TECHNOLOGY LTD"},
  2668. { 0x1156, "PERISCOPE ENGINEERING"},
  2669. { 0x1157, "AVSYS CORPORATION"},
  2670. { 0x1158, "VOARX R & D INC"},
  2671. { 0x1159, "MUTECH CORP"},
  2672. { 0x115A, "HARLEQUIN LTD"},
  2673. { 0x115B, "PARALLAX GRAPHICS"},
  2674. { 0x115C, "PHOTRON LTD."},
  2675. { 0x115D, "XIRCOM"},
  2676. { 0x115E, "PEER PROTOCOLS INC"},
  2677. { 0x115F, "MAXTOR CORPORATION"},
  2678. { 0x1160, "MEGASOFT INC"},
  2679. { 0x1161, "PFU LIMITED"},
  2680. { 0x1162, "OA LABORATORY CO LTD"},
  2681. { 0x1163, "RENDITION"},
  2682. { 0x1164, "ADVANCED PERIPHERALS TECH"},
  2683. { 0x1165, "IMAGRAPH"},
  2684. { 0x1166, "RELIANCE COMPUTER CORP."},
  2685. { 0x1167, "MUTOH INDUSTRIES INC"},
  2686. { 0x1168, "THINE ELECTRONICS INC"},
  2687. { 0x116A, "POLARIS COMMUNICATIONS"},
  2688. { 0x116B, "CONNECTWARE INC"},
  2689. { 0x116C, "INTELLIGENT RESOURCES"},
  2690. { 0x116E, "ELECTRONICS FOR IMAGING"},
  2691. { 0x116F, "WORKSTATION TECHNOLOGY"},
  2692. { 0x1170, "INVENTEC CORPORATION"},
  2693. { 0x1171, "LOUGHBOROUGH SOUND IMAGES"},
  2694. { 0x1172, "ALTERA CORPORATION"},
  2695. { 0x1173, "ADOBE SYSTEMS, INC"},
  2696. { 0x1174, "BRIDGEPORT MACHINES"},
  2697. { 0x1175, "MITRON COMPUTER INC."},
  2698. { 0x1176, "SBE INCORPORATED"},
  2699. { 0x1177, "SILICON ENGINEERING"},
  2700. { 0x1178, "ALFA, INC."},
  2701. { 0x117A, "A-TREND TECHNOLOGY"},
  2702. { 0x117C, "ATTO TECHNOLOGY"},
  2703. { 0x117D, "BECTON & DICKINSON"},
  2704. { 0x117E, "T/R SYSTEMS"},
  2705. { 0x117F, "INTEGRATED CIRCUIT SYSTEMS"},
  2706. { 0x1180, "RICOH CO LTD"},
  2707. { 0x1181, "TELMATICS INTERNATIONAL"},
  2708. { 0x1183, "FUJIKURA LTD"},
  2709. { 0x1184, "FORKS INC"},
  2710. { 0x1185, "DATAWORLD"},
  2711. { 0x1186, "D-LINK SYSTEM INC"},
  2712. { 0x1187, "ADVANCED TECHNOLOGY LABORATORIES"},
  2713. { 0x1188, "SHIMA SEIKI MANUFACTURING LTD."},
  2714. { 0x1189, "MATSUSHITA ELECTRONICS CO LTD"},
  2715. { 0x118A, "HILEVEL TECHNOLOGY"},
  2716. { 0x118B, "HYPERTEC PTY LIMITED"},
  2717. { 0x118C, "COROLLARY, INC"},
  2718. { 0x118D, "BITFLOW INC"},
  2719. { 0x118E, "HERMSTEDT GMBH"},
  2720. { 0x118F, "GREEN LOGIC"},
  2721. { 0x1191, "ARTOP ELECTRONIC CORP"},
  2722. { 0x1192, "DENSAN CO., LTD"},
  2723. { 0x1193, "ZEITNET INC."},
  2724. { 0x1194, "TOUCAN TECHNOLOGY"},
  2725. { 0x1195, "RATOC SYSTEM INC"},
  2726. { 0x1196, "HYTEC ELECTRONICS LTD"},
  2727. { 0x1197, "GAGE APPLIED SCIENCES, INC."},
  2728. { 0x1198, "LAMBDA SYSTEMS INC"},
  2729. { 0x1199, "ATTACHMATE CORPORATION"},
  2730. { 0x1199, "DIGITAL COMMUNICATIONS ASSOCIATES INC"},
  2731. { 0x119A, "MIND SHARE, INC."},
  2732. { 0x119B, "OMEGA MICRO INC."},
  2733. { 0x119C, "INFORMATION TECHNOLOGY INST."},
  2734. { 0x119D, "BUG SAPPORO JAPAN"},
  2735. { 0x119E, "FUJITSU"},
  2736. { 0x119F, "BULL HN INFORMATION SYSTEMS"},
  2737. { 0x11A0, "CONVEX COMPUTER CORPORATION"},
  2738. { 0x11A1, "HAMAMATSU PHOTONICS K.K."},
  2739. { 0x11A2, "SIERRA RESEARCH AND TECHNOLOGY"},
  2740. { 0x11A4, "BARCO GRAPHICS NV"},
  2741. { 0x11A5, "MICROUNITY SYSTEMS ENG. INC"},
  2742. { 0x11A6, "PURE DATA"},
  2743. { 0x11A7, "POWER COMPUTING CORP."},
  2744. { 0x11A8, "SYSTECH CORP."},
  2745. { 0x11A9, "INNOSYS"},
  2746. { 0x11AA, "ACTEL"},
  2747. { 0x11AB, "GALILEO TECHNOLOGY LTD."},
  2748. { 0x11AC, "CANON INFORMATION SYSTEMS"},
  2749. { 0x11AD, "LITE-ON COMMUNICATIONS INC"},
  2750. { 0x11AE, "AZTECH SYSTEM LTD"},
  2751. { 0x11AE, "SCITEX CORPORATION"},
  2752. { 0x11AF, "AVID TECHNOLOGY INC"},
  2753. { 0x11AF, "PRO-LOG CORPORATION"},
  2754. { 0x11B0, "V3 SEMICONDUCTOR"},
  2755. { 0x11B1, "APRICOT COMPUTERS"},
  2756. { 0x11B2, "EASTMAN KODAK"},
  2757. { 0x11B3, "BARR SYSTEMS INC."},
  2758. { 0x11B4, "LEITCH TECHNOLOGY INTERNATIONAL"},
  2759. { 0x11B5, "RADSTONE TECHNOLOGY PLC"},
  2760. { 0x11B6, "UNITED VIDEO CORP"},
  2761. { 0x11B8, "XPOINT TECHNOLOGIES INC"},
  2762. { 0x11B9, "PATHLIGHT TECHNOLOGY INC"},
  2763. { 0x11BA, "VIDEOTRON CORP"},
  2764. { 0x11BB, "PYRAMID TECHNOLOGY"},
  2765. { 0x11BC, "NETWORK PERIPHERALS INC"},
  2766. { 0x11BD, "PINNACLE SYSTEMS INC."},
  2767. { 0x11BE, "INTERNATIONAL MICROCIRCUITS INC"},
  2768. { 0x11BF, "ASTRODESIGN, INC."},
  2769. { 0x11C0, "HEWLETT PACKARD"},
  2770. { 0x11C1, "AT&T MICROELECTRONICS"},
  2771. { 0x11C2, "SAND MICROELECTRONICS"},
  2772. { 0x11C4, "DOCUMENT TECHNOLOGIES, IND"},
  2773. { 0x11C5, "SHIVA CORPORATION"},
  2774. { 0x11C6, "DAINIPPON SCREEN MFG. CO. LTD"},
  2775. { 0x11C7, "D.C.M. DATA SYSTEMS"},
  2776. { 0x11C8, "DOLPHIN INTERCONNECT"},
  2777. { 0x11C9, "MAGMA"},
  2778. { 0x11CA, "LSI SYSTEMS, INC"},
  2779. { 0x11CB, "SPECIALIX RESEARCH LTD"},
  2780. { 0x11CC, "MICHELS & KLEBERHOFF COMPUTER GMBH"},
  2781. { 0x11CD, "HAL COMPUTER SYSTEMS, INC."},
  2782. { 0x11CE, "PRIMARY RATE INC"},
  2783. { 0x11CF, "PIONEER ELECTRONIC CORPORATION"},
  2784. { 0x11D0, "LORAL FREDERAL SYSTEMS - MANASSAS"},
  2785. { 0x11D1, "AURAVISION"},
  2786. { 0x11D2, "INTERCOM INC."},
  2787. { 0x11D3, "TRANCELL SYSTEMS INC"},
  2788. { 0x11D4, "ANALOG DEVICES"},
  2789. { 0x11D5, "IKON CORPORATION"},
  2790. { 0x11D6, "TEKELEC TECHNOLOGIES"},
  2791. { 0x11D7, "TRENTON TERMINALS INC"},
  2792. { 0x11D8, "IMAGE TECHNOLOGIES DEVELOPMENT"},
  2793. { 0x11D9, "TEC CORPORATION"},
  2794. { 0x11DA, "NOVELL"},
  2795. { 0x11DB, "SEGA ENTERPRISES LITD"},
  2796. { 0x11DC, "QUESTRA CORPORATION"},
  2797. { 0x11DD, "CROSFIELD ELECTRONICS LIMITED"},
  2798. { 0x11DE, "ZORAN CORPORATION"},
  2799. { 0x11DF, "NEW WAVE PDG"},
  2800. { 0x11E0, "CRAY COMMUNICATIONS A/S"},
  2801. { 0x11E1, "GEC PLESSEY SEMI INC."},
  2802. { 0x11E2, "SAMSUNG INFORMATION SYSTEMS AMERICA"},
  2803. { 0x11E3, "QUICKLOGIC CORPORATION"},
  2804. { 0x11E4, "SECOND WAVE INC"},
  2805. { 0x11E5, "IIX CONSULTING"},
  2806. { 0x11E6, "MITSUI-ZOSEN SYSTEM RESEARCH"},
  2807. { 0x11E7, "TOSHIBA AMERICA, ELEC. COMPANY"},
  2808. { 0x11E8, "DIGITAL PROCESSING SYSTEMS INC."},
  2809. { 0x11E9, "HIGHWATER DESIGNS LTD."},
  2810. { 0x11EA, "ELSAG BAILEY"},
  2811. { 0x11EB, "FORMATION INC."},
  2812. { 0x11EC, "CORECO INC"},
  2813. { 0x11ED, "MEDIAMATICS"},
  2814. { 0x11EE, "DOME IMAGING SYSTEMS INC"},
  2815. { 0x11EF, "NICOLET TECHNOLOGIES B.V."},
  2816. { 0x11F0, "COMPU-SHACK GMBH"},
  2817. { 0x11F1, "SYMBIOS LOGIC INC"},
  2818. { 0x11F2, "PICTURE TEL JAPAN K.K."},
  2819. { 0x11F3, "KEITHLEY METRABYTE"},
  2820. { 0x11F4, "KINETIC SYSTEMS CORPORATION"},
  2821. { 0x11F5, "COMPUTING DEVICES INTERNATIONAL"},
  2822. { 0x11F6, "POWERMATIC DATA SYSTEMS LTD"},
  2823. { 0x11F7, "SCIENTIFIC ATLANTA"},
  2824. { 0x11F8, "PMC-SIERRA INC"},
  2825. { 0x11F9, "I-CUBE INC"},
  2826. { 0x11FA, "KASAN ELECTRONICS COMPANY, LTD."},
  2827. { 0x11FB, "DATEL INC"},
  2828. { 0x11FC, "SILICON MAGIC"},
  2829. { 0x11FD, "HIGH STREET CONSULTANTS"},
  2830. { 0x11FE, "COMTROL CORPORATION"},
  2831. { 0x11FF, "SCION CORPORATION"},
  2832. { 0x1200, "CSS CORPORATION"},
  2833. { 0x1201, "VISTA CONTROLS CORP"},
  2834. { 0x1202, "NETWORK GENERAL CORP."},
  2835. { 0x1203, "BAYER CORPORATION, AGFA DIVISION"},
  2836. { 0x1204, "LATTICE SEMICONDUCTOR CORPORATION"},
  2837. { 0x1205, "ARRAY CORPORATION"},
  2838. { 0x1206, "AMDAHL CORPORATION"},
  2839. { 0x1208, "PARSYTEC GMBH"},
  2840. { 0x1209, "SCI SYSTEMS INC"},
  2841. { 0x120A, "SYNAPTEL"},
  2842. { 0x120B, "ADAPTIVE SOLUTIONS"},
  2843. { 0x120D, "COMPRESSION LABS, INC."},
  2844. { 0x120E, "CYCLADES CORPORATION"},
  2845. { 0x120F, "ESSENTIAL COMMUNICATIONS"},
  2846. { 0x1210, "HYPERPARALLEL TECHNOLOGIES"},
  2847. { 0x1211, "BRAINTECH INC"},
  2848. { 0x1212, "KINGSTON TECHNOLOGY CORP."},
  2849. { 0x1213, "APPLIED INTELLIGENT SYSTEMS, INC."},
  2850. { 0x1214, "PERFORMANCE TECHNOLOGIES, INC."},
  2851. { 0x1215, "INTERWARE CO., LTD"},
  2852. { 0x1216, "PURUP PREPRESS A/S"},
  2853. { 0x1217, "2 MICRO, INC."},
  2854. { 0x1218, "HYBRICON CORP."},
  2855. { 0x1219, "FIRST VIRTUAL CORPORATION"},
  2856. { 0x121A, "3DFX INTERACTIVE, INC."},
  2857. { 0x121B, "ADVANCED TELECOMMUNICATIONS MODULES"},
  2858. { 0x121C, "NIPPON TEXA CO., LTD"},
  2859. { 0x121D, "LIPPERT AUTOMATIONSTECHNIK GMBH"},
  2860. { 0x121E, "CSPI"},
  2861. { 0x121F, "ARCUS TECHNOLOGY, INC."},
  2862. { 0x1220, "ARIEL CORPORATION"},
  2863. { 0x1221, "CONTEC CO., LTD"},
  2864. { 0x1222, "ANCOR COMMUNICATIONS, INC."},
  2865. { 0x1223, "HEURIKON/COMPUTER PRODUCTS"},
  2866. { 0x122C, "SICAN GMBH"},
  2867. { 0x1234, "TECHNICAL CORP."},
  2868. { 0x1239, "THE 3DO Company"},
  2869. { 0x124F, "INFORTREND TECHNOLOGY INC."},
  2870. { 0x126F, "SILICON MOTION"},
  2871. { 0x1275, "NETWORK APPLIANCE"},
  2872. { 0x1278, "TRANSTECH PARALLEL SYSTEMS"},
  2873. { 0x127B, "PIXERA CORPORATION"},
  2874. { 0x1297, "HOLCO ENTERPRISE"},
  2875. { 0x12A1, "SIMPACT, INC"},
  2876. { 0x12BA, "BITTWARE RESEARCH SYSTEMS, INC."},
  2877. { 0x12E7, "SEBRING SYSTEMS, INC."},
  2878. { 0x12EB, "AUREAL SEMICONDUCTOR"},
  2879. { 0x3D3D, "3DLABS"},
  2880. { 0x4005, "AVANCE LOGIC INC"},
  2881. { 0x5333, "S3 INC."},
  2882. { 0x8086, "INTEL"},
  2883. { 0x8E0E, "COMPUTONE CORPORATION"},
  2884. { 0x9004, "ADAPTEC"},
  2885. { 0xE159, "TIGER JET NETWORK INC."},
  2886. { 0xEDD8, "ARK LOGIC INC"}
  2887. };
  2888. //
  2889. // Taken from the PCI driver
  2890. //
  2891. #define PCI_TYPE0_RANGE_COUNT ((PCI_TYPE0_ADDRESSES) + 1)
  2892. #define PCI_TYPE1_RANGE_COUNT ((PCI_TYPE1_ADDRESSES) + 4)
  2893. #define PCI_TYPE2_RANGE_COUNT ((PCI_TYPE2_ADDRESSES) + 1)
  2894. #if PCI_TYPE0_RANGE_COUNT > PCI_TYPE1_RANGE_COUNT
  2895. #if PCI_TYPE0_RANGE_COUNT > PCI_TYPE2_RANGE_COUNT
  2896. #define PCI_MAX_RANGE_COUNT PCI_TYPE0_RANGE_COUNT
  2897. #else
  2898. #define PCI_MAX_RANGE_COUNT PCI_TYPE2_RANGE_COUNT
  2899. #endif
  2900. #else
  2901. #if PCI_TYPE1_RANGE_COUNT > PCI_TYPE2_RANGE_COUNT
  2902. #define PCI_MAX_RANGE_COUNT PCI_TYPE1_RANGE_COUNT
  2903. #else
  2904. #define PCI_MAX_RANGE_COUNT PCI_TYPE2_RANGE_COUNT
  2905. #endif
  2906. #endif
  2907. VOID
  2908. DevExtUsage()
  2909. {
  2910. dprintf("!devext <addess> <type>\n");
  2911. dprintf(" <address> is the address of a device extension to\n");
  2912. dprintf(" be dumped.\n");
  2913. dprintf(" <type> is the type of the object owning this extension:\n");
  2914. dprintf(" PCI if it is a PCI device extension\n");
  2915. dprintf(" ISAPNP if it is an ISAPNP device extension\n");
  2916. dprintf(" PCMCIA if it a PCMCIA device extension\n");
  2917. dprintf(" USBD OPENHCI UHCD if it is a USB Host Controller extension\n");
  2918. dprintf(" USBHUB if it is a USB Hub extension\n");
  2919. dprintf(" HID if it is a HID device extension\n");
  2920. return;
  2921. }
  2922. VOID
  2923. DevExtPciFindClassCodes(
  2924. ULONG PdoxBaseClass,
  2925. ULONG PdoxSubClass,
  2926. PUCHAR * BaseClass,
  2927. PUCHAR * SubClass
  2928. )
  2929. {
  2930. PUCHAR bc = "Unknown Base Class";
  2931. PUCHAR sc = "Unknown Sub Class";
  2932. switch (PdoxBaseClass) {
  2933. case PCI_CLASS_PRE_20:
  2934. // Class 00 - PCI_CLASS_PRE_20:
  2935. bc = "Pre PCI 2.0";
  2936. switch (PdoxSubClass) {
  2937. case PCI_SUBCLASS_PRE_20_NON_VGA:
  2938. sc = "Pre PCI 2.0 Non-VGA Device";
  2939. break;
  2940. case PCI_SUBCLASS_PRE_20_VGA:
  2941. sc = "Pre PCI 2.0 VGA Device";
  2942. break;
  2943. }
  2944. break;
  2945. case PCI_CLASS_MASS_STORAGE_CTLR:
  2946. // Class 01 - PCI_CLASS_MASS_STORAGE_CTLR:
  2947. bc = "Mass Storage Controller";
  2948. switch (PdoxSubClass) {
  2949. case PCI_SUBCLASS_MSC_SCSI_BUS_CTLR:
  2950. sc = "SCSI";
  2951. break;
  2952. case PCI_SUBCLASS_MSC_IDE_CTLR:
  2953. sc = "IDE";
  2954. break;
  2955. case PCI_SUBCLASS_MSC_FLOPPY_CTLR:
  2956. sc = "Floppy";
  2957. break;
  2958. case PCI_SUBCLASS_MSC_IPI_CTLR:
  2959. sc = "IPI";
  2960. break;
  2961. case PCI_SUBCLASS_MSC_RAID_CTLR:
  2962. sc = "RAID";
  2963. break;
  2964. case PCI_SUBCLASS_MSC_OTHER:
  2965. sc = "'Other'";
  2966. break;
  2967. }
  2968. break;
  2969. case PCI_CLASS_NETWORK_CTLR:
  2970. // Class 02 - PCI_CLASS_NETWORK_CTLR:
  2971. bc = "Network Controller";
  2972. switch (PdoxSubClass) {
  2973. case PCI_SUBCLASS_NET_ETHERNET_CTLR:
  2974. sc = "Ethernet";
  2975. break;
  2976. case PCI_SUBCLASS_NET_TOKEN_RING_CTLR:
  2977. sc = "Token Ring";
  2978. break;
  2979. case PCI_SUBCLASS_NET_FDDI_CTLR:
  2980. sc = "FDDI";
  2981. break;
  2982. case PCI_SUBCLASS_NET_ATM_CTLR:
  2983. sc = "ATM";
  2984. break;
  2985. case PCI_SUBCLASS_NET_OTHER:
  2986. sc = "'Other'";
  2987. break;
  2988. }
  2989. break;
  2990. case PCI_CLASS_DISPLAY_CTLR:
  2991. // Class 03 - PCI_CLASS_DISPLAY_CTLR:
  2992. // N.B. Sub Class 00 could be VGA or 8514 depending on Interface byte:
  2993. bc = "Display Controller";
  2994. switch (PdoxSubClass) {
  2995. case PCI_SUBCLASS_VID_VGA_CTLR:
  2996. sc = "VGA";
  2997. break;
  2998. case PCI_SUBCLASS_VID_XGA_CTLR:
  2999. sc = "XGA";
  3000. break;
  3001. case PCI_SUBCLASS_VID_OTHER:
  3002. sc = "'Other'";
  3003. break;
  3004. }
  3005. break;
  3006. case PCI_CLASS_MULTIMEDIA_DEV:
  3007. // Class 04 - PCI_CLASS_MULTIMEDIA_DEV:
  3008. bc = "Multimedia Device";
  3009. switch (PdoxSubClass) {
  3010. case PCI_SUBCLASS_MM_VIDEO_DEV:
  3011. sc = "Video";
  3012. break;
  3013. case PCI_SUBCLASS_MM_AUDIO_DEV:
  3014. sc = "Audio";
  3015. break;
  3016. case PCI_SUBCLASS_MM_OTHER:
  3017. sc = "'Other'";
  3018. break;
  3019. }
  3020. break;
  3021. case PCI_CLASS_MEMORY_CTLR:
  3022. // Class 05 - PCI_CLASS_MEMORY_CTLR:
  3023. bc = "Memory Controller";
  3024. switch (PdoxSubClass) {
  3025. case PCI_SUBCLASS_MEM_RAM:
  3026. sc = "RAM";
  3027. break;
  3028. case PCI_SUBCLASS_MEM_FLASH:
  3029. sc = "FLASH";
  3030. break;
  3031. case PCI_SUBCLASS_MEM_OTHER:
  3032. sc = "'Other'";
  3033. break;
  3034. }
  3035. break;
  3036. case PCI_CLASS_BRIDGE_DEV:
  3037. // Class 06 - PCI_CLASS_BRIDGE_DEV:
  3038. bc = "Bridge";
  3039. switch (PdoxSubClass) {
  3040. case PCI_SUBCLASS_BR_HOST:
  3041. sc = "HOST to PCI";
  3042. break;
  3043. case PCI_SUBCLASS_BR_ISA:
  3044. sc = "PCI to ISA";
  3045. break;
  3046. case PCI_SUBCLASS_BR_EISA:
  3047. sc = "PCI to EISA";
  3048. break;
  3049. case PCI_SUBCLASS_BR_MCA:
  3050. sc = "PCI to MCA";
  3051. break;
  3052. case PCI_SUBCLASS_BR_PCI_TO_PCI:
  3053. sc = "PCI to PCI";
  3054. break;
  3055. case PCI_SUBCLASS_BR_PCMCIA:
  3056. sc = "PCI to PCMCIA";
  3057. break;
  3058. case PCI_SUBCLASS_BR_NUBUS:
  3059. sc = "PCI to NUBUS";
  3060. break;
  3061. case PCI_SUBCLASS_BR_CARDBUS:
  3062. sc = "PCI to CardBus";
  3063. break;
  3064. case PCI_SUBCLASS_BR_OTHER:
  3065. sc = "PCI to 'Other'";
  3066. break;
  3067. }
  3068. break;
  3069. case PCI_CLASS_SIMPLE_COMMS_CTLR:
  3070. // Class 07 - PCI_CLASS_SIMPLE_COMMS_CTLR:
  3071. // N.B. Sub Class 00 and 01 additional info in Interface byte:
  3072. bc = "Simple Serial Communications Controller";
  3073. switch (PdoxSubClass) {
  3074. case PCI_SUBCLASS_COM_SERIAL:
  3075. sc = "Serial Port";
  3076. break;
  3077. case PCI_SUBCLASS_COM_PARALLEL:
  3078. sc = "Parallel Port";
  3079. break;
  3080. case PCI_SUBCLASS_COM_OTHER:
  3081. sc = "'Other'";
  3082. break;
  3083. }
  3084. break;
  3085. case PCI_CLASS_BASE_SYSTEM_DEV:
  3086. // Class 08 - PCI_CLASS_BASE_SYSTEM_DEV:
  3087. // N.B. See Interface byte for additional info.:
  3088. bc = "Base System Device";
  3089. switch (PdoxSubClass) {
  3090. case PCI_SUBCLASS_SYS_INTERRUPT_CTLR:
  3091. sc = "Interrupt Controller";
  3092. break;
  3093. case PCI_SUBCLASS_SYS_DMA_CTLR:
  3094. sc = "DMA Controller";
  3095. break;
  3096. case PCI_SUBCLASS_SYS_SYSTEM_TIMER:
  3097. sc = "System Timer";
  3098. break;
  3099. case PCI_SUBCLASS_SYS_REAL_TIME_CLOCK:
  3100. sc = "Real Time Clock";
  3101. break;
  3102. case PCI_SUBCLASS_SYS_OTHER:
  3103. sc = "'Other' base system device";
  3104. break;
  3105. }
  3106. break;
  3107. case PCI_CLASS_INPUT_DEV:
  3108. // Class 09 - PCI_CLASS_INPUT_DEV:
  3109. bc = "Input Device";
  3110. switch (PdoxSubClass) {
  3111. case PCI_SUBCLASS_INP_KEYBOARD:
  3112. sc = "Keyboard";
  3113. break;
  3114. case PCI_SUBCLASS_INP_DIGITIZER:
  3115. sc = "Digitizer";
  3116. break;
  3117. case PCI_SUBCLASS_INP_MOUSE:
  3118. sc = "Mouse";
  3119. break;
  3120. case PCI_SUBCLASS_INP_OTHER:
  3121. sc = "'Other'";
  3122. break;
  3123. }
  3124. break;
  3125. case PCI_CLASS_DOCKING_STATION:
  3126. // Class 0a - PCI_CLASS_DOCKING_STATION:
  3127. bc = "Docking Station";
  3128. switch (PdoxSubClass) {
  3129. case PCI_SUBCLASS_DOC_GENERIC:
  3130. sc = "Generic";
  3131. break;
  3132. case PCI_SUBCLASS_DOC_OTHER:
  3133. sc = "'Other'";
  3134. break;
  3135. }
  3136. break;
  3137. case PCI_CLASS_PROCESSOR:
  3138. // Class 0b - PCI_CLASS_PROCESSOR:
  3139. bc = "Processor";
  3140. switch (PdoxSubClass) {
  3141. case PCI_SUBCLASS_PROC_386:
  3142. sc = "386";
  3143. break;
  3144. case PCI_SUBCLASS_PROC_486:
  3145. sc = "486";
  3146. break;
  3147. case PCI_SUBCLASS_PROC_PENTIUM:
  3148. sc = "Pentium";
  3149. break;
  3150. case PCI_SUBCLASS_PROC_ALPHA:
  3151. sc = "Alpha";
  3152. break;
  3153. case PCI_SUBCLASS_PROC_POWERPC:
  3154. sc = "PowerPC";
  3155. break;
  3156. case PCI_SUBCLASS_PROC_COPROCESSOR:
  3157. sc = "CoProcessor";
  3158. break;
  3159. }
  3160. break;
  3161. case PCI_CLASS_SERIAL_BUS_CTLR:
  3162. // Class 0c - PCI_CLASS_SERIAL_BUS_CTLR:
  3163. bc = "Serial Bus Controller";
  3164. switch (PdoxSubClass) {
  3165. case PCI_SUBCLASS_SB_IEEE1394:
  3166. sc = "1394";
  3167. break;
  3168. case PCI_SUBCLASS_SB_ACCESS:
  3169. sc = "Access Bus";
  3170. break;
  3171. case PCI_SUBCLASS_SB_SSA:
  3172. sc = "SSA";
  3173. break;
  3174. case PCI_SUBCLASS_SB_USB:
  3175. sc = "USB";
  3176. break;
  3177. case PCI_SUBCLASS_SB_FIBRE_CHANNEL:
  3178. sc = "Fibre Channel";
  3179. break;
  3180. }
  3181. break;
  3182. case PCI_CLASS_NOT_DEFINED:
  3183. bc = "(Explicitly) Undefined";
  3184. break;
  3185. }
  3186. *BaseClass = bc;
  3187. *SubClass = sc;
  3188. }
  3189. VOID
  3190. DevExtPciPrintDeviceState(
  3191. PCI_OBJECT_STATE State
  3192. )
  3193. {
  3194. if ((sizeof(DevExtPciDeviceState) / sizeof(DevExtPciDeviceState[0])) !=
  3195. PciMaxObjectState) {
  3196. dprintf("\nWARNING: PCI_OBJECT_STATE enumeration changed, please review.\n");
  3197. }
  3198. if (State >= PciMaxObjectState) {
  3199. dprintf("Unknown PCI Object state.\n");
  3200. return;
  3201. }
  3202. dprintf(" Driver State = %s\n", DevExtPciDeviceState[State]);
  3203. }
  3204. //#if 0
  3205. VOID
  3206. DevExtPciPrintPowerState(
  3207. ULONG64 Power
  3208. )
  3209. {
  3210. #define MAXSYSPOWSTATES \
  3211. (sizeof(DevExtPciSystemPowerState) / sizeof(DevExtPciSystemPowerState[0]))
  3212. #define MAXDEVPOWSTATES \
  3213. (sizeof(DevExtPciDevicePowerState) / sizeof(DevExtPciDevicePowerState[0]))
  3214. SYSTEM_POWER_STATE index;
  3215. DEVICE_POWER_STATE systemStateMapping[PowerSystemMaximum];
  3216. ULONG64 pwrState, waitWakeIrp=0;
  3217. PUCHAR sw, sc, dd, dc, dw;
  3218. //
  3219. // A little sanity.
  3220. //
  3221. if (MAXSYSPOWSTATES != PowerSystemMaximum) {
  3222. dprintf("WARNING: System Power State definition has changed, ext dll is out of date.\n");
  3223. }
  3224. if (MAXDEVPOWSTATES != PowerDeviceMaximum) {
  3225. dprintf("WARNING: Device Power State definition has changed, ext dll is out of date.\n");
  3226. }
  3227. sw = sc = dc = dw = "<*BAD VAL*>";
  3228. GetFieldValue(Power, "pci!PCI_POWER_STATE", "CurrentSystemState", pwrState);
  3229. if (pwrState < PowerSystemMaximum) {
  3230. sc = DevExtPciSystemPowerState[pwrState];
  3231. }
  3232. GetFieldValue(Power, "pci!PCI_POWER_STATE", "CurrentDeviceState", pwrState);
  3233. if (pwrState < PowerDeviceMaximum) {
  3234. dc = DevExtPciDevicePowerState[pwrState];
  3235. }
  3236. GetFieldValue(Power, "pci!PCI_POWER_STATE", "DeviceWakeLevel", pwrState);
  3237. if (pwrState < PowerDeviceMaximum) {
  3238. dw = DevExtPciDevicePowerState[pwrState];
  3239. }
  3240. GetFieldValue(Power, "pci!PCI_POWER_STATE", "SystemWakeLevel", pwrState);
  3241. if (pwrState < PowerSystemMaximum) {
  3242. sw = DevExtPciSystemPowerState[pwrState];
  3243. }
  3244. dprintf(
  3245. " CurrentState: System %s, Device %s\n"
  3246. " WakeLevel: System %s, Device %s\n",
  3247. sc,
  3248. dc,
  3249. sw,
  3250. dw
  3251. );
  3252. GetFieldValue(Power, "pci!PCI_POWER_STATE", "SystemStateMapping", systemStateMapping);
  3253. for (index = PowerSystemWorking; index < PowerSystemMaximum; index++) {
  3254. if (systemStateMapping[index] != PowerDeviceUnspecified) {
  3255. if (systemStateMapping[index] < PowerDeviceMaximum) {
  3256. dc = DevExtPciDevicePowerState[systemStateMapping[index]];
  3257. } else {
  3258. dc = "<*BAD VAL*>";
  3259. }
  3260. dprintf(" %s:\t%s\n",
  3261. DevExtPciSystemPowerState[index],
  3262. dc
  3263. );
  3264. }
  3265. }
  3266. GetFieldValue(Power, "pci!PCI_POWER_STATE", "WaitWakeIrp", waitWakeIrp);
  3267. if (waitWakeIrp) {
  3268. dprintf(" WaitWakeIrp: %#08p\n",
  3269. waitWakeIrp
  3270. );
  3271. }
  3272. #undef MAXSYSPOWSTATES
  3273. #undef MAXDEVPOWSTATES
  3274. }
  3275. VOID
  3276. DevExtPciPrintSecondaryExtensions(
  3277. ULONG64 ListEntry
  3278. )
  3279. {
  3280. ULONG64 extension = 0, extype = 0;
  3281. if (!ListEntry) {
  3282. return;
  3283. }
  3284. xdprintf(1, "Secondary Extensions\n");
  3285. do {
  3286. GetFieldValue(ListEntry, "pci!PCI_SECONDARY_EXTENSION", "ExtensionType", extype);
  3287. switch (extype) {
  3288. case PciArb_Io:
  3289. xdprintf(2, "Arbiter - IO Port");
  3290. break;
  3291. case PciArb_Memory:
  3292. xdprintf(2, "Arbiter - Memory");
  3293. break;
  3294. case PciArb_Interrupt:
  3295. xdprintf(2, "Arbiter - Interrupt");
  3296. break;
  3297. case PciArb_BusNumber:
  3298. xdprintf(2, "Arbiter - Bus Number");
  3299. break;
  3300. case PciTrans_Interrupt:
  3301. xdprintf(2, "Translator - Interrupt");
  3302. break;
  3303. case PciInterface_BusHandler:
  3304. xdprintf(2, "Interface - Bus Handler");
  3305. break;
  3306. case PciInterface_IntRouteHandler:
  3307. xdprintf(2, "Interface - Interrupt Route Handler");
  3308. break;
  3309. default:
  3310. xdprintf(2, "Unknown Extension Type.");
  3311. break;
  3312. }
  3313. dprintf("\n");
  3314. GetFieldValue(ListEntry, "nt!_SINGLE_LIST_ENTRY", "Next", ListEntry);
  3315. } while (ListEntry);
  3316. }
  3317. PUCHAR
  3318. DevExtPciFindVendorId(
  3319. USHORT VendorId
  3320. )
  3321. /*++
  3322. Routine Description:
  3323. Get the vendor name given the vendor ID (assuming we know it).
  3324. Arguments:
  3325. VendorId 16 bit PCI Vendor ID.
  3326. Return Value:
  3327. Pointer to the vendor's name (or NULL).
  3328. --*/
  3329. {
  3330. #define PCIIDCOUNT (sizeof(DevExtPciVendors) / sizeof(DevExtPciVendors[0]))
  3331. ULONG index;
  3332. //
  3333. // Yep, a binary search would be much faster, good thing we only
  3334. // do this once per user iteration.
  3335. //
  3336. for (index = 0; index < PCIIDCOUNT; index++) {
  3337. if (DevExtPciVendors[index].VendorId == VendorId) {
  3338. return DevExtPciVendors[index].VendorName;
  3339. }
  3340. }
  3341. return NULL;
  3342. #undef PCIIDCOUNT
  3343. }
  3344. #define GET_PCI_PDO_FIELD(Pdo,Field,Value) \
  3345. GetPciExtensionField(TRUE, Pdo, Field, sizeof(Value), (PVOID)(&(Value)))
  3346. #define GET_PCI_FDO_FIELD(Fdo,Field,Value) \
  3347. GetPciExtensionField(FALSE, Fdo, Field, sizeof(Value), (PVOID)(&(Value)))
  3348. ULONG
  3349. GetPciExtensionField(
  3350. IN BOOLEAN IsPdo, // TRUE for PDO, false for FDO
  3351. IN ULONG64 TypeAddress,
  3352. IN PUCHAR Field,
  3353. IN ULONG OutSize,
  3354. OUT PVOID OutValue
  3355. )
  3356. {
  3357. ULONG ret;
  3358. static PCHAR pciFDO = "nt!_PCI_FDO_EXTENSION", pciPDO = "nt!_PCI_PDO_EXTENSION";
  3359. //
  3360. // Try the post Win2k name for a pci extension (includes a PCI_ prefix)
  3361. //
  3362. // This would work for public symbols
  3363. ret = GetFieldData(TypeAddress,
  3364. IsPdo ? pciPDO : pciFDO,
  3365. Field,
  3366. OutSize,
  3367. OutValue
  3368. );
  3369. if (ret) {
  3370. pciFDO = "pci!PCI_FDO_EXTENSION";
  3371. pciPDO = "pci!PCI_PDO_EXTENSION";
  3372. //
  3373. // If we have private symbols, nt won't have those types, so try in pci
  3374. //
  3375. ret = GetFieldData(TypeAddress,
  3376. IsPdo ? pciPDO : pciFDO,
  3377. Field,
  3378. OutSize,
  3379. OutValue
  3380. );
  3381. }
  3382. //
  3383. // If that failed then fall back on the name without the PCI_ prefix. This
  3384. // allows debugging Win2k clients with the post Win2k debugger
  3385. //
  3386. if (ret) {
  3387. ret = GetFieldData(TypeAddress,
  3388. IsPdo ? "pci!PDO_EXTENSION" : "pci!FDO_EXTENSION",
  3389. Field,
  3390. OutSize,
  3391. OutValue
  3392. );
  3393. }
  3394. if (ret) {
  3395. // didn't find anything, reste these
  3396. pciFDO = "nt!_PCI_FDO_EXTENSION";
  3397. pciPDO = "nt!_PCI_PDO_EXTENSION";
  3398. }
  3399. return ret;
  3400. }
  3401. typedef struct _PCI_SLOT_NUMBER {
  3402. union {
  3403. struct {
  3404. ULONG DeviceNumber:5;
  3405. ULONG FunctionNumber:3;
  3406. ULONG Reserved:24;
  3407. } bits;
  3408. ULONG AsULONG;
  3409. } u;
  3410. } PCI_SLOT_NUMBER, *PPCI_SLOT_NUMBER;
  3411. VOID
  3412. DevExtPciGetBusDevFunc(
  3413. IN ULONG64 PdoExt,
  3414. OUT PULONG Bus,
  3415. OUT PULONG Dev,
  3416. OUT PULONG Func
  3417. )
  3418. {
  3419. ULONG BaseBus=0;
  3420. ULONG64 ParentFdoExtension=0;
  3421. PCI_SLOT_NUMBER slot;
  3422. GET_PCI_PDO_FIELD(PdoExt, "ParentFdoExtension", ParentFdoExtension);
  3423. if (GET_PCI_FDO_FIELD(ParentFdoExtension, "BaseBus", BaseBus)) {
  3424. dprintf(
  3425. "Failed to read memory at %08x for sizeof FDO Extension.\n",
  3426. ParentFdoExtension
  3427. );
  3428. }
  3429. if (GET_PCI_PDO_FIELD(PdoExt, "Slot", slot)){
  3430. dprintf(
  3431. "Failed to read memory at %08x\n",
  3432. PdoExt
  3433. );
  3434. }
  3435. *Bus = BaseBus;
  3436. *Dev = slot.u.bits.DeviceNumber;
  3437. *Func = slot.u.bits.FunctionNumber;
  3438. }
  3439. VOID
  3440. DevExtPci(
  3441. ULONG64 Extension
  3442. )
  3443. /*++
  3444. Routine Description:
  3445. Dump a PCI Device extension.
  3446. Arguments:
  3447. Extension Address of the extension to be dumped.
  3448. Return Value:
  3449. None.
  3450. --*/
  3451. {
  3452. ULONG64 pdo=0, fdoX=0, pdoX=0, pcifield=0;
  3453. ULONG Signature=0, fieldoffset=0;
  3454. ULONG bus=0xff,dev=0xff,func=0xff;
  3455. USHORT vendorId=0xffff, deviceId=0xffff, subsystemVendorId=0xfff, subsystemId=0xffff;
  3456. UCHAR deviceState, baseClass, subClass, progIf, revisionId, interruptPin, rawInterruptLine, adjustedInterruptLine;
  3457. UCHAR secBus=0xff, subBus=0xff;
  3458. PUCHAR s;
  3459. PUCHAR bc;
  3460. PUCHAR sc;
  3461. ULONG i;
  3462. BOOLEAN found, subDecode=FALSE, isaBit=FALSE, vgaBit=FALSE;
  3463. if (GET_PCI_PDO_FIELD(Extension, "ExtensionType", Signature)) {
  3464. dprintf(
  3465. "Failed to read PCI object signature at %008p, giving up.\n",
  3466. Extension
  3467. );
  3468. return;
  3469. }
  3470. switch (Signature) {
  3471. case PciPdoExtensionType:
  3472. //
  3473. // PDO Extension.
  3474. //
  3475. DevExtPciGetBusDevFunc(Extension, &bus, &dev, &func);
  3476. dprintf("PDO Extension, Bus 0x%x, Device %x, Function %d.\n",
  3477. bus, dev, func
  3478. );
  3479. GET_PCI_PDO_FIELD(Extension, "PhysicalDeviceObject", pdo);
  3480. GET_PCI_PDO_FIELD(Extension, "ParentFdoExtension", fdoX);
  3481. dprintf(" DevObj %#08p PCI Parent Bus FDO DevExt %#08p\n",
  3482. pdo, fdoX
  3483. );
  3484. GET_PCI_PDO_FIELD(Extension, "DeviceState", deviceState);
  3485. GET_PCI_PDO_FIELD(Extension, "VendorId", vendorId);
  3486. GET_PCI_PDO_FIELD(Extension, "DeviceId", deviceId);
  3487. DevExtPciPrintDeviceState(deviceState);
  3488. s = DevExtPciFindVendorId(vendorId);
  3489. if (s != NULL) {
  3490. dprintf(" Vendor ID %04x (%s) Device ID %04X\n",
  3491. vendorId,
  3492. s,
  3493. deviceId
  3494. );
  3495. } else {
  3496. dprintf(
  3497. " Vendor ID %04x, Device ID %04X\n",
  3498. vendorId,
  3499. deviceId
  3500. );
  3501. }
  3502. GET_PCI_PDO_FIELD(Extension, "SubsystemVendorId", subsystemVendorId);
  3503. GET_PCI_PDO_FIELD(Extension, "SubsystemId", subsystemId);
  3504. if (subsystemVendorId || subsystemId) {
  3505. s = DevExtPciFindVendorId(subsystemVendorId);
  3506. if (s != NULL) {
  3507. dprintf(
  3508. " Subsystem Vendor ID %04x (%s) Subsystem ID %04X\n",
  3509. subsystemVendorId,
  3510. s,
  3511. subsystemId
  3512. );
  3513. } else {
  3514. dprintf(" Subsystem Vendor ID %04x, Subsystem ID %04X\n",
  3515. subsystemVendorId,
  3516. subsystemId
  3517. );
  3518. }
  3519. }
  3520. GET_PCI_PDO_FIELD(Extension, "BaseClass", baseClass);
  3521. GET_PCI_PDO_FIELD(Extension, "SubClass", subClass);
  3522. DevExtPciFindClassCodes(baseClass, subClass, &bc, &sc);
  3523. dprintf(" Class Base/Sub %02x/%02x (%s/%s)\n",
  3524. baseClass,
  3525. subClass,
  3526. bc,
  3527. sc);
  3528. GET_PCI_PDO_FIELD(Extension, "ProgIf", progIf);
  3529. GET_PCI_PDO_FIELD(Extension, "RevisionId", revisionId);
  3530. GET_PCI_PDO_FIELD(Extension, "InterruptPin", interruptPin);
  3531. GET_PCI_PDO_FIELD(Extension, "RawInterruptLine", rawInterruptLine);
  3532. GET_PCI_PDO_FIELD(Extension, "AdjustedInterruptLine", adjustedInterruptLine);
  3533. dprintf(" Programming Interface: %02x, Revision: %02x, IntPin: %02x, Line Raw/Adj %02x/%02x\n",
  3534. progIf,
  3535. revisionId,
  3536. interruptPin,
  3537. rawInterruptLine,
  3538. adjustedInterruptLine
  3539. );
  3540. {
  3541. USHORT enables=0;
  3542. GET_PCI_PDO_FIELD(Extension, "CommandEnables", enables);
  3543. dprintf(" Enables ((cmd & 7) = %x): %s%s%s",
  3544. enables,
  3545. enables == 0 ? "<none>" :
  3546. (enables & PCI_ENABLE_BUS_MASTER) ? "B" : "",
  3547. (enables & PCI_ENABLE_MEMORY_SPACE) ? "M" : "",
  3548. (enables & PCI_ENABLE_IO_SPACE) ? "I" : ""
  3549. );
  3550. GET_PCI_PDO_FIELD(Extension, "CapabilitiesPtr", pcifield);
  3551. if (pcifield) {
  3552. dprintf(" Capabilities Pointer = %02x\n", pcifield);
  3553. } else {
  3554. dprintf(" Capabilities Pointer = <none>\n");
  3555. }
  3556. }
  3557. GetFieldOffset("pci!PCI_PDO_EXTENSION", "PowerState", &fieldoffset);
  3558. DevExtPciPrintPowerState(Extension+fieldoffset);
  3559. if (baseClass == PCI_CLASS_BRIDGE_DEV) {
  3560. if ((subClass == PCI_SUBCLASS_BR_PCI_TO_PCI) ||
  3561. (subClass == PCI_SUBCLASS_BR_CARDBUS)) {
  3562. GetFieldOffset("pci!PCI_PDO_EXTENSION", "Dependent", &fieldoffset);
  3563. GetFieldValue(Extension+fieldoffset,
  3564. "pci!PCI_HEADER_TYPE_DEPENDENT",
  3565. "type1.SecondaryBus",
  3566. secBus);
  3567. GetFieldValue(Extension+fieldoffset,
  3568. "pci!PCI_HEADER_TYPE_DEPENDENT",
  3569. "type1.SecondaryBus",
  3570. subBus);
  3571. dprintf(" Bridge PCI-%s Secondary Bus = 0x%x, Subordinate Bus = 0x%x\n",
  3572. subClass == PCI_SUBCLASS_BR_PCI_TO_PCI
  3573. ? "PCI" : "CardBus",
  3574. secBus,
  3575. subBus
  3576. );
  3577. GetFieldValue(Extension+fieldoffset,
  3578. "pci!PCI_HEADER_TYPE_DEPENDENT",
  3579. "type1.SubtractiveDecode",
  3580. subDecode);
  3581. GetFieldValue(Extension+fieldoffset,
  3582. "pci!PCI_HEADER_TYPE_DEPENDENT",
  3583. "type1.IsaBitSet",
  3584. isaBit);
  3585. GetFieldValue(Extension+fieldoffset,
  3586. "pci!PCI_HEADER_TYPE_DEPENDENT",
  3587. "type1.VgaBitSet",
  3588. vgaBit);
  3589. if (subDecode || isaBit || vgaBit) {
  3590. CHAR preceed = ' ';
  3591. dprintf(" Bridge Flags:");
  3592. if (subDecode) {
  3593. dprintf("%c Subtractive Decode", preceed);
  3594. preceed = ',';
  3595. }
  3596. if (isaBit) {
  3597. dprintf("%c ISA", preceed);
  3598. preceed = ',';
  3599. }
  3600. if (vgaBit) {
  3601. dprintf("%c VGA", preceed);
  3602. preceed = ',';
  3603. }
  3604. dprintf("\n");
  3605. }
  3606. }
  3607. }
  3608. //
  3609. // Now lets parse our PCI_FUNCTION_RESOURCES struct
  3610. //
  3611. pcifield = 0;
  3612. GET_PCI_PDO_FIELD(Extension, "Resources", pcifield);
  3613. dprintf(" Requirements:");
  3614. if (pcifield) {
  3615. ULONG64 limit = 0, current=0;
  3616. ULONG ioSize=0, cmSize=0, type=0;
  3617. ULONG alignment, length, minhighpart, minlowpart, maxhighpart, maxlowpart;
  3618. ioSize = GetTypeSize("nt!IO_RESOURCE_DESCRIPTOR");
  3619. cmSize = GetTypeSize("nt!CM_PARTIAL_RESOURCE_DESCRIPTOR");
  3620. //
  3621. // Limit begins at offset 0
  3622. //
  3623. limit = pcifield;
  3624. found = FALSE;
  3625. dprintf(" Alignment Length Minimum Maximum\n");
  3626. for (i = 0; i < PCI_MAX_RANGE_COUNT; i++) {
  3627. type=0;
  3628. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "Type", type);
  3629. if (type == CmResourceTypeNull) {
  3630. continue;
  3631. }
  3632. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.Alignment", alignment);
  3633. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.Length", length);
  3634. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.MinimumAddress.HighPart", minhighpart);
  3635. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.MinimumAddress.LowPart", minlowpart);
  3636. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.MaximumAddress.HighPart", maxhighpart);
  3637. GetFieldValue(limit, "nt!IO_RESOURCE_DESCRIPTOR", "u.Generic.MaximumAddress.LowPart", maxlowpart);
  3638. dprintf(" %s: %08x %08x %08x%08x %08x%08x\n",
  3639. type == CmResourceTypeMemory ? "Memory" : " Io",
  3640. alignment,
  3641. length,
  3642. minhighpart,
  3643. minlowpart,
  3644. maxhighpart,
  3645. maxlowpart
  3646. );
  3647. limit += ioSize;
  3648. }
  3649. //
  3650. // Get the offset for Current so we can dump it next
  3651. //
  3652. GetFieldOffset("pci!PCI_FUNCTION_RESOURCES", "Current", &fieldoffset);
  3653. current = pcifield+fieldoffset;
  3654. dprintf(" Current:");
  3655. for (i = 0; i < PCI_MAX_RANGE_COUNT; i++) {
  3656. type=0;
  3657. length=0;
  3658. maxhighpart=0;
  3659. maxlowpart=0;
  3660. GetFieldValue(current, "nt!CM_PARTIAL_RESOURCE_DESCRIPTOR", "Type", type);
  3661. if (type == CmResourceTypeNull) {
  3662. continue;
  3663. }
  3664. if (!found) {
  3665. dprintf(" Start Length\n");
  3666. found = TRUE;
  3667. }
  3668. GetFieldValue(current, "nt!CM_PARTIAL_RESOURCE_DESCRIPTOR", "u.Generic.Start.HighPart", maxhighpart);
  3669. GetFieldValue(current, "nt!CM_PARTIAL_RESOURCE_DESCRIPTOR", "u.Generic.Start.LowPart", maxlowpart);
  3670. GetFieldValue(current, "nt!CM_PARTIAL_RESOURCE_DESCRIPTOR", "u.Generic.Length", length);
  3671. dprintf(" %s: %08x%08x %08x\n",
  3672. type == CmResourceTypeMemory ? "Memory" : " Io",
  3673. maxhighpart,
  3674. maxlowpart,
  3675. length
  3676. );
  3677. current += cmSize;
  3678. }
  3679. if (!found) {
  3680. dprintf(" <none>\n");
  3681. }
  3682. } else {
  3683. dprintf(" <none>\n");
  3684. }
  3685. GetFieldOffset("pci!PCI_PDO_EXTENSION", "SecondaryExtension", &fieldoffset);
  3686. GetFieldValue(Extension+fieldoffset, "nt!_SINGLE_LIST_ENTRY", "Next", pcifield);
  3687. DevExtPciPrintSecondaryExtensions(pcifield);
  3688. break;
  3689. case PciFdoExtensionType:
  3690. //
  3691. // FDO Extension
  3692. //
  3693. GET_PCI_FDO_FIELD(Extension, "BaseBus", bus);
  3694. GET_PCI_FDO_FIELD(Extension, "PhysicalDeviceObject", pdo);
  3695. dprintf("FDO Extension, Bus 0x%x, Parent PDO %#08p Bus is ", bus,pdo);
  3696. GET_PCI_FDO_FIELD(Extension, "BusRootFdoExtension", pcifield);
  3697. if (pcifield == Extension) {
  3698. dprintf("a root bus.\n");
  3699. } else {
  3700. GET_PCI_FDO_FIELD(Extension, "PhysicalDeviceObject", pcifield);
  3701. GetFieldValue(pcifield, "nt!_DEVICE_OBJECT", "DeviceExtension", pdoX);
  3702. if (!pdoX) {
  3703. dprintf("a child bus.\n");
  3704. }else{
  3705. DevExtPciGetBusDevFunc(pdoX, &bus, &dev, &func);
  3706. dprintf("a child of bus 0x%x.\n", bus);
  3707. }
  3708. }
  3709. GET_PCI_FDO_FIELD(Extension, "DeviceState", deviceState);
  3710. DevExtPciPrintDeviceState(deviceState);
  3711. GetFieldOffset("pci!PCI_FDO_EXTENSION", "PowerState", &fieldoffset);
  3712. DevExtPciPrintPowerState(Extension+fieldoffset);
  3713. GET_PCI_FDO_FIELD(Extension, "ChildWaitWakeCount", pcifield);
  3714. if (pcifield) {
  3715. dprintf("Number of PDO's with outstanding WAIT_WAKEs = %d\n", pcifield);
  3716. }
  3717. {
  3718. PUCHAR heading = " Child PDOXs:";
  3719. ULONG counter = 0;
  3720. ULONG64 list = 0;
  3721. GET_PCI_FDO_FIELD(Extension, "ChildPdoList", list);
  3722. while (list) {
  3723. if (counter == 0) {
  3724. dprintf(heading);
  3725. heading = "\n ";
  3726. counter = 1;
  3727. } else if (counter == 2) {
  3728. counter = 0;
  3729. } else {
  3730. counter++;
  3731. }
  3732. dprintf(" %#08p", list);
  3733. GET_PCI_PDO_FIELD(list, "Next", list);
  3734. }
  3735. dprintf("\n");
  3736. }
  3737. GetFieldOffset("pci!PCI_FDO_EXTENSION", "SecondaryExtension", &fieldoffset);
  3738. GetFieldValue(Extension+fieldoffset, "nt!_SINGLE_LIST_ENTRY", "Next", pcifield);
  3739. DevExtPciPrintSecondaryExtensions(pcifield);
  3740. break;
  3741. default:
  3742. dprintf("%08x doesn't seem to point to a PCI PDO or FDO extension,\n");
  3743. dprintf("the PCI Device object extension signatures don't match.\n");
  3744. return;
  3745. }
  3746. }
  3747. DECLARE_API( devext )
  3748. /*++
  3749. Routine Description:
  3750. Dumps a device extension.
  3751. Arguments:
  3752. address Address of the device extension to be dumped.
  3753. type Type of device extension.
  3754. Return Value:
  3755. None
  3756. --*/
  3757. {
  3758. #define DEVEXT_MAXTOKENS 3
  3759. #define DEVEXT_MAXBUFFER 80
  3760. #define DEVEXT_USAGE() { DevExtUsage(); return E_INVALIDARG; }
  3761. ULONG64 Extension = 0;
  3762. PUCHAR ExtensionType;
  3763. PUCHAR Tokens[DEVEXT_MAXTOKENS];
  3764. UCHAR Buffer[DEVEXT_MAXBUFFER];
  3765. PUCHAR s;
  3766. UCHAR c;
  3767. ULONG count;
  3768. //
  3769. // Validate parameters. Tokenize the incoming string, the first
  3770. // argument should be a (kernel mode) address, the second a string.
  3771. //
  3772. //
  3773. // args is const, we need to modify the buffer, copy it.
  3774. //
  3775. for (count = 0; count < DEVEXT_MAXBUFFER; count++) {
  3776. if ((Buffer[count] = args[count]) == '\0') {
  3777. break;
  3778. }
  3779. }
  3780. if (count == DEVEXT_MAXBUFFER) {
  3781. dprintf("Buffer to small to contain input arguments\n");
  3782. DEVEXT_USAGE();
  3783. }
  3784. if (TokenizeString(Buffer, Tokens, DEVEXT_MAXTOKENS) !=
  3785. (DEVEXT_MAXTOKENS - 1)) {
  3786. DEVEXT_USAGE();
  3787. }
  3788. if ((Extension = GetExpression(Tokens[0])) == 0) {
  3789. DEVEXT_USAGE();
  3790. }
  3791. // Signextend it
  3792. if (DBG_PTR_SIZE == sizeof(ULONG)) {
  3793. Extension = (ULONG64) (LONG64) (LONG) Extension;
  3794. }
  3795. if ((LONG64)Extension >= 0) {
  3796. DEVEXT_USAGE();
  3797. }
  3798. //
  3799. // The second argument should be a string telling us what kind of
  3800. // device extension to dump. Convert it to upper case to make life
  3801. // easier.
  3802. //
  3803. s = Tokens[1];
  3804. while ((c = *s) != '\0') {
  3805. *s++ = (UCHAR)toupper(c);
  3806. }
  3807. s = Tokens[1];
  3808. if (!strcmp(s, "PCI")) {
  3809. //
  3810. // It's a PCI device extension.
  3811. //
  3812. DevExtPci(Extension);
  3813. } else if (!strcmp(s, "PCMCIA")) {
  3814. //
  3815. // It's a PCMCIA device extension
  3816. //
  3817. DevExtPcmcia(Extension);
  3818. } else if (!strcmp(s, "USBD")) {
  3819. //
  3820. // It's a USBD device extension
  3821. //
  3822. DevExtUsbd(Extension);
  3823. } else if (!strcmp(s, "OPENHCI")) {
  3824. //
  3825. // It's a OpenHCI device extension
  3826. //
  3827. DevExtOpenHCI(Extension);
  3828. } else if (!strcmp(s, "UHCD")) {
  3829. //
  3830. // It's a UHCD device extension
  3831. //
  3832. DevExtUHCD(Extension);
  3833. } else if (!strcmp(s, "HID")) {
  3834. //
  3835. // It's a HID device extension
  3836. //
  3837. DevExtHID(Extension);
  3838. } else if (!strcmp(s, "USBHUB")) {
  3839. //
  3840. // It's a usbhub device extension
  3841. //
  3842. DevExtUsbhub(Extension);
  3843. } else if (!strcmp(s, "ISAPNP")) {
  3844. //
  3845. // Congratulations! It's an ISAPNP device extension
  3846. //
  3847. DevExtIsapnp(Extension);
  3848. #if 0
  3849. } else if (!strcmp(s, "SOME_OTHER_EXTENSION_TYPE")) {
  3850. //
  3851. // It's some other extension type.
  3852. //
  3853. DevExtSomeOther(Extension);
  3854. #endif
  3855. } else {
  3856. dprintf("Device extension type '%s' is not handled by !devext.\n", s);
  3857. }
  3858. return S_OK;
  3859. }
  3860. //
  3861. // pcitree data structures and functions
  3862. //
  3863. BOOLEAN
  3864. pcitreeProcessBus(
  3865. ULONG Indent,
  3866. ULONG64 InMemoryBusFdoX,
  3867. ULONG64 TreeTop
  3868. )
  3869. {
  3870. ULONG64 BusFdoX=InMemoryBusFdoX;
  3871. ULONG64 next;
  3872. PUCHAR bc, sc, vendor;
  3873. ULONG BaseBus, BaseClass, SubClass;
  3874. ULONG64 ChildPdoList;
  3875. #define PdoxFld(F,V) GET_PCI_PDO_FIELD(next, #F, V)
  3876. GET_PCI_FDO_FIELD(BusFdoX, "BaseBus", BaseBus);
  3877. GET_PCI_FDO_FIELD(BusFdoX, "ChildPdoList", next);
  3878. xdprintf(Indent,"");
  3879. dprintf( "Bus 0x%x (FDO Ext %08p)\n",
  3880. BaseBus,
  3881. InMemoryBusFdoX);
  3882. if (next == 0) {
  3883. xdprintf(Indent, "No devices have been enumerated on this bus.\n");
  3884. return TRUE;
  3885. }
  3886. Indent++;
  3887. do {
  3888. ULONG DeviceId, VendorId, DeviceNumber, FunctionNumber;
  3889. ULONG64 BridgeFdoExtension;
  3890. if (CheckControlC()) {
  3891. return FALSE;
  3892. }
  3893. if (PdoxFld(BaseClass,BaseClass)) {
  3894. xdprintf(Indent,"FAILED to read PDO Ext at ");
  3895. dprintf("%08p, cannot continue this bus.\n",
  3896. next
  3897. );
  3898. return TRUE;
  3899. }
  3900. PdoxFld(SubClass,SubClass);
  3901. PdoxFld(DeviceId,DeviceId);
  3902. PdoxFld(VendorId,VendorId);
  3903. PdoxFld(Slot.u.bits.DeviceNumber,DeviceNumber);
  3904. PdoxFld(Slot.u.bits.FunctionNumber,FunctionNumber);
  3905. DevExtPciFindClassCodes(BaseClass, SubClass, &bc, &sc);
  3906. xdprintf(Indent, "");
  3907. dprintf("%02x%02x %04x%04x (d=%x,%s f=%d) devext %08p %s/%s\n",
  3908. BaseClass,
  3909. SubClass,
  3910. DeviceId,
  3911. VendorId,
  3912. DeviceNumber,
  3913. (DeviceNumber > 9 ? "" : " "),
  3914. FunctionNumber,
  3915. next,
  3916. bc,
  3917. sc
  3918. );
  3919. #if 0
  3920. // removed, too talkative. plj.
  3921. vendor = DevExtPciFindVendorId(PdoX.VendorId);
  3922. if (vendor) {
  3923. xdprintf(Indent, " (%s)\n", vendor);
  3924. }
  3925. #endif
  3926. if (BaseClass == PCI_CLASS_BRIDGE_DEV && SubClass == PCI_SUBCLASS_BR_PCI_TO_PCI) {
  3927. //
  3928. // Process child bus.
  3929. //
  3930. PdoxFld(BridgeFdoExtension,BridgeFdoExtension);
  3931. //
  3932. // Recurse.
  3933. //
  3934. if (pcitreeProcessBus(Indent, BridgeFdoExtension, TreeTop) == FALSE) {
  3935. //
  3936. // Error, exit.
  3937. //
  3938. return FALSE;
  3939. }
  3940. }
  3941. PdoxFld(Next, next);
  3942. } while (next);
  3943. #undef PdoxFld
  3944. return TRUE;
  3945. }
  3946. DECLARE_API( pcitree )
  3947. /*++
  3948. Routine Description:
  3949. Dumps the pci tree. (Why is this here? Because the other PCI stuff
  3950. ended up in here because devnode was such a useful base).
  3951. Arguments:
  3952. None
  3953. Return Value:
  3954. None
  3955. --*/
  3956. {
  3957. ULONG Signature;
  3958. ULONG64 TreeTop;
  3959. ULONG64 list;
  3960. ULONG64 addr;
  3961. ULONG indent = 0;
  3962. ULONG rootCount = 0;
  3963. addr = GetExpression("Pci!PciFdoExtensionListHead");
  3964. if (addr == 0) {
  3965. dprintf("Error retrieving address of PciFdoExtensionListHead\n");
  3966. return E_INVALIDARG;
  3967. }
  3968. if (!ReadPointer(addr, &list)) {
  3969. dprintf("Error reading PciFdoExtensionListHead (@%08p)\n", addr);
  3970. return E_INVALIDARG;
  3971. }
  3972. if (list == 0) {
  3973. dprintf(
  3974. "PciFdoExtensionListHead forward pointer is NULL, list is empty\n"
  3975. );
  3976. return E_INVALIDARG;
  3977. }
  3978. TreeTop = list;
  3979. do {
  3980. ULONG64 BusRootFdoExtension;
  3981. if (CheckControlC()) {
  3982. dprintf("User terminated with <control>C\n");
  3983. break;
  3984. }
  3985. if (GET_PCI_FDO_FIELD(list, "BusRootFdoExtension", BusRootFdoExtension)) {
  3986. dprintf(
  3987. "Error reading PCI FDO extension at %08p, quitting.\n",
  3988. list
  3989. );
  3990. return E_INVALIDARG;
  3991. }
  3992. if (BusRootFdoExtension == list) {
  3993. //
  3994. // This is the FDO for a root bus.
  3995. //
  3996. rootCount++;
  3997. if (pcitreeProcessBus(0, list, TreeTop) == FALSE) {
  3998. //
  3999. // Asked not to continue;
  4000. //
  4001. dprintf("User terminated output.\n");
  4002. break;
  4003. }
  4004. }
  4005. GET_PCI_FDO_FIELD(list, "List.Next", list);
  4006. } while (list != 0);
  4007. dprintf("Total PCI Root busses processed = %d\n", rootCount);
  4008. return S_OK;
  4009. }
  4010. VOID
  4011. DumpDeviceCapabilities(
  4012. ULONG64 caps
  4013. ) {
  4014. FIELD_INFO fields [] = {
  4015. { "DeviceD1", "Spare1", 0, 0, 0, NULL},
  4016. { "DeviceD2", "\tDeviceD2", 0, 0, 0, NULL},
  4017. { "LockSupported", "\tLock", 0, 0, 0, NULL},
  4018. { "EjectSupported","\tEject", 0, 0, 0, NULL},
  4019. { "Removeable", "\tRemoveable", 0, 0, 0, NULL},
  4020. { "DockDevice", "\nDock Device",0, 0, 0, NULL},
  4021. { "UniqueID", "Unique ID", 0, 0, 0, NULL},
  4022. { "SilentInstall", "\tSilent Install", 0, 0, 0, NULL},
  4023. { "RawDeviceOK", "\tRawDeviceOK",0, 0, 0, NULL},
  4024. { "Address", "\nAddress", 0, 0, 0, NULL},
  4025. { "UINumber", "", 0, 0, 0, NULL},
  4026. { "DeviceState", "\nDevice State : ", 0, DBG_DUMP_FIELD_ARRAY , 0, NULL},
  4027. { "SystemWake", "\nSystemWake", 0, 0, 0, NULL},
  4028. { "DeviceWake", "", 0, 0, 0, NULL},
  4029. };
  4030. UCHAR nm[] = "_DEVICE_CAPABILITIES";
  4031. SYM_DUMP_PARAM Sym = {
  4032. sizeof (SYM_DUMP_PARAM),
  4033. &nm[0],
  4034. DBG_DUMP_NO_OFFSET | DBG_DUMP_COMPACT_OUT,
  4035. caps,
  4036. NULL,
  4037. NULL,
  4038. NULL,
  4039. sizeof (fields) / sizeof (FIELD_INFO),
  4040. &fields[0],
  4041. };
  4042. dprintf ("PnP CAPS: \n");
  4043. Ioctl(IG_DUMP_SYMBOL_INFO, &Sym, sizeof (Sym));
  4044. }
  4045. #if 0
  4046. VOID
  4047. DumpDeviceCapabilities(
  4048. PDEVICE_CAPABILITIES caps
  4049. )
  4050. {
  4051. ULONG i,j,k;
  4052. ULONG bitmask;
  4053. PCHAR bits [] = {
  4054. "Spare1",
  4055. "Spare1",
  4056. "Lock",
  4057. "Eject",
  4058. "Removeable",
  4059. "Dock Device",
  4060. "Unique ID",
  4061. "Silent Install",
  4062. "RawDeviceOK"
  4063. };
  4064. bitmask = * (((PULONG) caps) + 1);
  4065. dprintf ("PnP CAPS: ");
  4066. for (i = 0, j = 1, k = 0; i < sizeof (bits); i++, j <<= 1) {
  4067. if (bitmask & j) {
  4068. if (k) {
  4069. dprintf("\n ");
  4070. }
  4071. k ^= 1;
  4072. dprintf(bits [i]);
  4073. }
  4074. }
  4075. dprintf("\n");
  4076. dprintf(" Address %x UINumber %x \n",
  4077. caps->Address,
  4078. caps->UINumber);
  4079. dprintf(" Device State [");
  4080. for (i = 0; i < PowerSystemMaximum; i ++) {
  4081. dprintf ("%02x ", (caps->DeviceState [i] & 0xFF));
  4082. }
  4083. dprintf("]\n");
  4084. dprintf(" System Wake %x Device Wake %x\n",
  4085. caps->SystemWake,
  4086. caps->DeviceWake);
  4087. dprintf("\n");
  4088. }
  4089. #endif
  4090. VOID
  4091. DumpTranslator(
  4092. IN DWORD Depth,
  4093. IN ULONG64 TranslatorAddr
  4094. )
  4095. /*++
  4096. Routine Description:
  4097. Dumps out the specified translator.
  4098. Arguments:
  4099. Depth - Supplies the print depth.
  4100. TranslatorAddr - Supplies the original address of the arbiter (on the target)
  4101. Translator - Supplies the arbiter
  4102. Return Value:
  4103. None.
  4104. --*/
  4105. {
  4106. // TRANSLATOR_INTERFACE interface;
  4107. UCHAR buffer[256];
  4108. ULONG64 displacement;
  4109. ULONG ResourceType;
  4110. ULONG64 TranslatorInterface, TranslateResources, TranslateResourceRequirements;
  4111. if (GetFieldValue(TranslatorAddr,
  4112. "nt!_PI_RESOURCE_TRANSLATOR_ENTRY",
  4113. "ResourceType",
  4114. ResourceType)) {
  4115. xdprintf( Depth, "");
  4116. dprintf( "Could not read translator entry %08p\n",
  4117. TranslatorAddr
  4118. );
  4119. return;
  4120. }
  4121. GetFieldValue(TranslatorAddr,"nt!_PI_RESOURCE_TRANSLATOR_ENTRY","TranslatorInterface", TranslatorInterface);
  4122. xdprintf(Depth,
  4123. "%s Translator\n",
  4124. GetCmResourceTypeString((UCHAR)ResourceType)
  4125. );
  4126. if (GetFieldValue(TranslatorInterface,
  4127. "nt!_TRANSLATOR_INTERFACE",
  4128. "TranslateResources",
  4129. TranslateResources)) {
  4130. dprintf("Error reading TranslatorInterface %08p for TranslatorEntry at %08p\n",
  4131. TranslatorInterface,
  4132. TranslatorAddr);
  4133. return;
  4134. }
  4135. GetSymbol(TranslateResources, buffer, &displacement);
  4136. xdprintf(Depth+1, "Resources: %s", buffer, displacement);
  4137. if (displacement != 0) {
  4138. dprintf("+0x%1p\n", displacement);
  4139. } else {
  4140. dprintf("\n");
  4141. }
  4142. GetFieldValue(TranslatorInterface,"nt!_TRANSLATOR_INTERFACE",
  4143. "TranslateResourceRequirements",TranslateResourceRequirements);
  4144. GetSymbol(TranslateResourceRequirements, buffer, &displacement);
  4145. xdprintf(Depth+1, "Requirements: %s", buffer, displacement);
  4146. if (displacement != 0) {
  4147. dprintf("+0x%1p\n", displacement);
  4148. } else {
  4149. dprintf("\n");
  4150. }
  4151. return;
  4152. }
  4153. BOOLEAN
  4154. DumpTranslatorsForDevNode(
  4155. IN DWORD Depth,
  4156. IN ULONG64 DevNodeAddr,
  4157. IN DWORD Flags
  4158. )
  4159. /*++
  4160. Routine Description:
  4161. Dumps all the translators for the specified devnode. Recursively calls
  4162. itself on the specified devnode's children.
  4163. Arguments:
  4164. Depth - Supplies the print depth.
  4165. DevNodeAddr - Supplies the address of the devnode.
  4166. Flags - Supplies the type of arbiters to dump:
  4167. ARBITER_DUMP_IO
  4168. ARBITER_DUMP_MEMORY
  4169. ARBITER_DUMP_IRQ
  4170. ARBITER_DUMP_DMA
  4171. Return Value:
  4172. None
  4173. --*/
  4174. {
  4175. ULONG result;
  4176. ULONG64 nextTranslator;
  4177. ULONG64 translatorAddr;
  4178. BOOLEAN PrintedHeader = FALSE;
  4179. BOOLEAN continueDump = TRUE;
  4180. ULONG DeviceTranslatorListOffset, TransEntryOffset;
  4181. ULONG64 Child, Sibling;
  4182. if (GetFieldOffset("nt!_DEVICE_NODE", "DeviceTranslatorList", &DeviceTranslatorListOffset)) {
  4183. dprintf("_DEVICE_NODE type not found.\n");
  4184. return continueDump;
  4185. }
  4186. if (GetFieldOffset("nt!_PI_RESOURCE_TRANSLATOR_ENTRY", "DeviceTranslatorList", &TransEntryOffset)) {
  4187. dprintf("_PI_RESOURCE_TRANSLATOR_ENTRY type not found.\n");
  4188. return continueDump;
  4189. }
  4190. if (GetFieldValue(DevNodeAddr,
  4191. "nt!_DEVICE_NODE",
  4192. "DeviceTranslatorList.Flink",
  4193. nextTranslator)) {
  4194. xdprintf(Depth, ""); dprintf("%08p: Could not read device node\n", DevNodeAddr);
  4195. return continueDump;
  4196. }
  4197. // dprintf("Next %p, Devnode %p, Trans Off %x", nextTranslator, DevNodeAddr, DeviceTranslatorListOffset);
  4198. //
  4199. // Dump the list of translators
  4200. //
  4201. while (nextTranslator != DevNodeAddr+DeviceTranslatorListOffset) {
  4202. ULONG ResourceType;
  4203. if (CheckControlC()) {
  4204. continueDump = FALSE;
  4205. break;
  4206. }
  4207. translatorAddr = (nextTranslator-TransEntryOffset);
  4208. if (GetFieldValue(translatorAddr,
  4209. "nt!_PI_RESOURCE_TRANSLATOR_ENTRY",
  4210. "ResourceType",
  4211. ResourceType)) {
  4212. xdprintf( Depth, "");
  4213. dprintf ("Could not read translator entry %08p for devnode %08p\n",
  4214. translatorAddr,
  4215. DevNodeAddr);
  4216. break;
  4217. }
  4218. if (!PrintedHeader) {
  4219. UNICODE_STRING64 u;
  4220. dprintf("\n");
  4221. xdprintf(Depth, ""); dprintf("DEVNODE %08p ", DevNodeAddr);
  4222. GetFieldValue(DevNodeAddr,"nt!_DEVICE_NODE","InstancePath.Buffer", u.Buffer);
  4223. GetFieldValue(DevNodeAddr,"nt!_DEVICE_NODE","InstancePath.Length", u.Length);
  4224. GetFieldValue(DevNodeAddr,"nt!_DEVICE_NODE","InstancePath.MaximumLength", u.MaximumLength);
  4225. if (u.Buffer != 0) {
  4226. dprintf("(");
  4227. DumpUnicode64(u);
  4228. dprintf(")");
  4229. }
  4230. dprintf("\n");
  4231. PrintedHeader = TRUE;
  4232. }
  4233. if (((ResourceType == CmResourceTypePort) && (Flags & ARBITER_DUMP_IO)) ||
  4234. ((ResourceType == CmResourceTypeInterrupt) && (Flags & ARBITER_DUMP_IRQ)) ||
  4235. ((ResourceType == CmResourceTypeMemory) && (Flags & ARBITER_DUMP_MEMORY)) ||
  4236. ((ResourceType == CmResourceTypeDma) && (Flags & ARBITER_DUMP_DMA)) ||
  4237. ((ResourceType == CmResourceTypeBusNumber) && (Flags & ARBITER_DUMP_BUS_NUMBER))) {
  4238. DumpTranslator( Depth+1, translatorAddr);
  4239. }
  4240. GetFieldValue(translatorAddr,"nt!_PI_RESOURCE_TRANSLATOR_ENTRY","DeviceTranslatorList.Flink",nextTranslator);
  4241. }
  4242. GetFieldValue(DevNodeAddr,"nt!_DEVICE_NODE","Sibling", Sibling);
  4243. GetFieldValue(DevNodeAddr,"nt!_DEVICE_NODE","Child", Child);
  4244. //
  4245. // Dump this devnode's children (if any)
  4246. //
  4247. if (continueDump && Child) {
  4248. continueDump = DumpTranslatorsForDevNode( Depth+1, Child, Flags);
  4249. }
  4250. //
  4251. // Dump this devnode's siblings (if any)
  4252. //
  4253. if (continueDump && Sibling) {
  4254. continueDump = DumpTranslatorsForDevNode( Depth, Sibling, Flags);
  4255. }
  4256. return continueDump;
  4257. }
  4258. DECLARE_API( translator )
  4259. /*++
  4260. Routine Description:
  4261. Dumps all the translators in the system
  4262. !translator [flags]
  4263. flags 1 - I/O port arbiters
  4264. 2 - memory arbiters
  4265. 4 - IRQ arbiters
  4266. 8 - DMA arbiters
  4267. If no flags are specified, all translators are dumped.
  4268. Arguments:
  4269. args - optional flags specifying which arbiters to dump
  4270. Return Value:
  4271. None
  4272. --*/
  4273. {
  4274. DWORD Flags=0;
  4275. ULONG64 addr;
  4276. ULONG64 deviceNode;
  4277. if (!(Flags = (ULONG) GetExpression(args))) {
  4278. Flags = ARBITER_DUMP_ALL;
  4279. }
  4280. //
  4281. // Find the root devnode and dump its arbiters
  4282. //
  4283. addr = GetExpression( "nt!IopRootDeviceNode" );
  4284. if (addr == 0) {
  4285. dprintf("Error retrieving address of IopRootDeviceNode\n");
  4286. return E_INVALIDARG;
  4287. }
  4288. if (!ReadPointer(addr, &deviceNode)) {
  4289. dprintf("Error reading value of IopRootDeviceNode (%#010p)\n", addr);
  4290. return E_INVALIDARG;
  4291. }
  4292. DumpTranslatorsForDevNode(0, deviceNode, Flags);
  4293. return S_OK;
  4294. }
  4295. #define RAW_RANGE_DUMP_STATS_ONLY 0x00000001
  4296. VOID
  4297. DumpRawRangeEntry(
  4298. IN DWORD Depth,
  4299. IN ULONG64 RangeEntry
  4300. )
  4301. /*++
  4302. Routine Description:
  4303. Dumps out the specified RTLP_RANGE_LIST_ENTRY
  4304. Arguments:
  4305. Depth - Supplies the print depth.
  4306. RangeList - Supplies the address of the RTLP_RANGE_LIST_ENTRY
  4307. OwnerIsDevObj - Indicates that the owner field is a pointer to a DEVICE_OBJECT
  4308. Return Value:
  4309. None.
  4310. --*/
  4311. {
  4312. ULONG PublicFlags;
  4313. ULONG64 UserData, Owner;
  4314. ULONG64 Start, End;
  4315. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Start", Start);
  4316. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "End", End);
  4317. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "PublicFlags", PublicFlags);
  4318. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Allocated.UserData", UserData);
  4319. GetFieldValue(RangeEntry, "nt!_RTLP_RANGE_LIST_ENTRY", "Allocated.Owner", Owner);
  4320. xdprintf(Depth, "");
  4321. dprintf ("%016I64x - %016I64x %c%c %08p %08p\n",
  4322. Start,
  4323. End,
  4324. PublicFlags & RTL_RANGE_SHARED ? 'S' : ' ',
  4325. PublicFlags & RTL_RANGE_CONFLICT ? 'C' : ' ',
  4326. UserData,
  4327. Owner
  4328. );
  4329. }
  4330. BOOLEAN
  4331. DumpRawRangeList(
  4332. IN DWORD Depth,
  4333. IN ULONG64 RangeListHead,
  4334. IN ULONG Flags,
  4335. IN PULONG MergedCount OPTIONAL,
  4336. IN PULONG EntryCount
  4337. )
  4338. /*++
  4339. Routine Description:
  4340. Dumps out the specified RTL_RANGE_LIST
  4341. Arguments:
  4342. Depth - Supplies the print depth.
  4343. RangeListHead - Supplies the address of the LIST_ENTRY containing the RTLP_RANGE_LIST_ENTRYs
  4344. Return Value:
  4345. None.
  4346. --*/
  4347. {
  4348. ULONG64 EntryAddr;
  4349. ULONG64 ListEntry;
  4350. ULONG ListEntryOffset;
  4351. BOOLEAN continueDump = TRUE;
  4352. //
  4353. // Read the range list
  4354. //
  4355. if (GetFieldValue(RangeListHead, "nt!_LIST_ENTRY", "Flink", ListEntry)) {
  4356. dprintf("Error reading RangeList %08p\n", RangeListHead);
  4357. return continueDump;
  4358. }
  4359. if (ListEntry == RangeListHead) {
  4360. xdprintf(Depth, "< none >\n");
  4361. return continueDump;
  4362. }
  4363. GetFieldOffset("nt!RTLP_RANGE_LIST_ENTRY", "ListEntry", &ListEntryOffset);
  4364. while (ListEntry != RangeListHead) {
  4365. ULONG PrivateFlags;
  4366. ULONG64 Start, End;
  4367. if (CheckControlC()) {
  4368. continueDump = FALSE;
  4369. break;
  4370. }
  4371. EntryAddr = ( ListEntry - ListEntryOffset);
  4372. if (GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY",
  4373. "PrivateFlags", PrivateFlags)) {
  4374. dprintf("Error reading RangeEntry %08p from RangeList %08p\n",
  4375. EntryAddr,
  4376. RangeListHead);
  4377. return continueDump;
  4378. }
  4379. (*EntryCount)++;
  4380. if (PrivateFlags & RTLP_RANGE_LIST_ENTRY_MERGED) {
  4381. ULONG MergedOffset;
  4382. //
  4383. // This is a merged range list, call ourselves recursively
  4384. //
  4385. if (MergedCount) {
  4386. (*MergedCount)++;
  4387. }
  4388. if (!(Flags & RAW_RANGE_DUMP_STATS_ONLY)) {
  4389. GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY", "Start", Start);
  4390. GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY", "End", End);
  4391. xdprintf(Depth, "Merged Range - %I64x-%I64x\n", Start, End);
  4392. }
  4393. GetFieldOffset("nt!_RTLP_RANGE_LIST_ENTRY", "Merged.ListHead", &MergedOffset);
  4394. continueDump = DumpRawRangeList( Depth+1,
  4395. EntryAddr + MergedOffset,
  4396. Flags,
  4397. MergedCount,
  4398. EntryCount);
  4399. if (!continueDump) {
  4400. break;
  4401. }
  4402. } else if (!(Flags & RAW_RANGE_DUMP_STATS_ONLY)) {
  4403. DumpRawRangeEntry(Depth, EntryAddr);
  4404. }
  4405. GetFieldValue(EntryAddr, "nt!_RTLP_RANGE_LIST_ENTRY", "ListEntry.Flink", ListEntry);
  4406. }
  4407. return continueDump;
  4408. }
  4409. DECLARE_API( rawrange )
  4410. /*++
  4411. Routine Description:
  4412. Dumps an RTL_RANGE_LIST
  4413. Arguments:
  4414. args - specifies the address of the RTL_RANGE_LIST
  4415. Return Value:
  4416. None
  4417. --*/
  4418. {
  4419. ULONG64 RangeList=0;
  4420. ULONG mergedCount = 0, entryCount = 0, flags=0, Offset;
  4421. if (GetExpressionEx(args, &RangeList, &args)) {
  4422. flags = (ULONG) GetExpression(args);
  4423. }
  4424. GetFieldOffset("nt!_RTLP_RANGE_LIST_ENTRY", "ListHead", &Offset);
  4425. DumpRawRangeList(0, RangeList + Offset, flags, &mergedCount, &entryCount);
  4426. dprintf("Stats\nMerged = %i\nEntries = %i\n", mergedCount, entryCount);
  4427. return S_OK;
  4428. }
  4429. VOID
  4430. DumpIoResourceDescriptor(
  4431. IN ULONG Depth,
  4432. IN ULONG64 Descriptor
  4433. )
  4434. {
  4435. ULONG64 Q1, Q2;
  4436. ULONG Option, Type, ShareDisposition, Flags, V1, V2, V3, Data[3];
  4437. #define Desc(F,V) GetFieldValue(Descriptor, "nt!IO_RESOURCE_LIST", #F, V)
  4438. Desc(Option,Option); Desc(Type,Type);
  4439. Desc(ShareDisposition,ShareDisposition);
  4440. Desc(Flags,Flags);
  4441. DumpResourceDescriptorHeader(Depth,
  4442. 0,
  4443. (UCHAR) Option,
  4444. (UCHAR) Type,
  4445. (UCHAR) ShareDisposition,
  4446. (USHORT) Flags);
  4447. Depth++;
  4448. switch (Type) {
  4449. case CmResourceTypeBusNumber:
  4450. Desc(u.BusNumber.MinBusNumber, V1);
  4451. Desc(u.BusNumber.MaxBusNumber, V2);
  4452. Desc(u.BusNumber.Length, V3);
  4453. xdprintf(Depth, "0x%x - 0x%x for length 0x%x\n",
  4454. V1, V2, V3);
  4455. break;
  4456. case CmResourceTypePort:
  4457. case CmResourceTypeMemory:
  4458. Desc(u.Generic.Length, V1);
  4459. Desc(u.Generic.Alignment, V2);
  4460. Desc(u.Port.MinimumAddress.QuadPart, Q1);
  4461. Desc(u.Port.MaximumAddress.QuadPart, Q2);
  4462. xdprintf(Depth, "%#08lx byte range with alignment %#08lx\n", V1, V2);
  4463. xdprintf(Depth, "%I64x - %#I64x\n", Q1, Q2);
  4464. case CmResourceTypeInterrupt:
  4465. Desc(u.Interrupt.MinimumVector,V1);
  4466. Desc(u.Interrupt.MaximumVector,V2);
  4467. xdprintf(Depth, "0x%x - 0x%x\n", V1,V2);
  4468. break;
  4469. case CmResourceTypeDma:
  4470. Desc(u.Dma.MinimumChannel,V1);
  4471. Desc(u.Dma.MaximumChannel,V2);
  4472. xdprintf(Depth, "0x%x - 0x%x\n",V1, V2);
  4473. break;
  4474. default:
  4475. Desc(u.DevicePrivate.Data, Data);
  4476. xdprintf(Depth,
  4477. "Data: : 0x%x 0x%x 0x%x\n",
  4478. Data[0],
  4479. Data[1],
  4480. Data[2]);
  4481. break;
  4482. }
  4483. #undef Desc
  4484. }
  4485. DECLARE_API( ioresdes )
  4486. {
  4487. ULONG64 descriptorAddr;
  4488. descriptorAddr = GetExpression(args);
  4489. DumpIoResourceDescriptor(0, descriptorAddr);
  4490. return S_OK;
  4491. }
  4492. DECLARE_API( pnpevent )
  4493. /*++
  4494. Routine Description:
  4495. Dump a device object.
  4496. Arguments:
  4497. args - the location of the device object to dump.
  4498. Return Value:
  4499. None
  4500. --*/
  4501. {
  4502. ULONG64 deviceEventListAddr;
  4503. ULONG64 listHead;
  4504. ULONG64 deviceEvent = 0;
  4505. ULONG verbose = 0;
  4506. ULONG64 address;
  4507. ULONG offset;
  4508. BOOL followLinks;
  4509. ULONG64 status = 0, flink = 0, blink = 0;
  4510. ULONG eventQ_Header_SignalState = 0, lk_Event_Hdr_SignalState = 0;
  4511. address = GetExpression( "nt!PpDeviceEventList" );
  4512. if (address == 0) {
  4513. dprintf("Error retrieving address of PpDeviceEventList\n");
  4514. return E_INVALIDARG;
  4515. }
  4516. if (!ReadPointer(address, &deviceEventListAddr)) {
  4517. dprintf("Error reading value of PpDeviceEventList (%#010lx)\n", address);
  4518. return E_INVALIDARG;
  4519. }
  4520. if (GetFieldOffset("nt!_PNP_DEVICE_EVENT_LIST", "List", &offset)) {
  4521. dprintf("Cannot find _PNP_DEVICE_EVENT_LIST.List type.\n");
  4522. return E_INVALIDARG;
  4523. }
  4524. listHead = deviceEventListAddr + offset;
  4525. dprintf("Dumping PpDeviceEventList @ 0x%08p\n", deviceEventListAddr);
  4526. if (GetFieldValue(deviceEventListAddr, "nt!_PNP_DEVICE_EVENT_LIST", "Status", status)) {
  4527. dprintf("Error reading PpDeviceEventList->Status (%#010p)\n", deviceEventListAddr);
  4528. return E_INVALIDARG;
  4529. }
  4530. if (GetFieldValue(deviceEventListAddr, "nt!_PNP_DEVICE_EVENT_LIST", "EventQueueMutex.Header.SignalState", eventQ_Header_SignalState)) {
  4531. dprintf("Error reading PpDeviceEventList->EventQueueMutex.Header.SignalState (%#010p)\n", deviceEventListAddr);
  4532. return E_INVALIDARG;
  4533. }
  4534. if (GetFieldValue(deviceEventListAddr, "nt!_PNP_DEVICE_EVENT_LIST", "Lock.Event.Header.SignalState", lk_Event_Hdr_SignalState)) {
  4535. dprintf("Error reading PpDeviceEventList->Lock.Event.Header.SignalState (%#010p)\n", deviceEventListAddr);
  4536. return E_INVALIDARG;
  4537. }
  4538. if (GetFieldValue(deviceEventListAddr, "nt!_PNP_DEVICE_EVENT_LIST", "List.Flink", flink)) {
  4539. dprintf("Error reading PpDeviceEventList->List.Flink (%#010p)\n", deviceEventListAddr);
  4540. return E_INVALIDARG;
  4541. }
  4542. if (GetFieldValue(deviceEventListAddr, "nt!_PNP_DEVICE_EVENT_LIST", "List.Blink", blink)) {
  4543. dprintf("Error reading PpDeviceEventList->List.Blink (%#010p)\n", deviceEventListAddr);
  4544. return E_INVALIDARG;
  4545. }
  4546. dprintf(" Status = 0x%08I64X, EventQueueMutex is %sheld, Lock is %sheld\n",
  4547. status,
  4548. eventQ_Header_SignalState ? "not " : "",
  4549. lk_Event_Hdr_SignalState ? "not " : "",
  4550. flink, blink);
  4551. dprintf(" List = 0x%08p, 0x%08p\n", flink, blink);
  4552. if (GetExpressionEx(args, &deviceEvent, &args)) {
  4553. verbose = (ULONG) GetExpression(args);
  4554. }
  4555. if (deviceEvent == 0) {
  4556. ULONG listEntryOffset = 0;
  4557. if (flink == listHead) {
  4558. dprintf("Event list is empty\n");
  4559. return E_INVALIDARG;
  4560. }
  4561. //
  4562. // Now get addtess of PNP_DEVICE_EVENT_ENTRY
  4563. //
  4564. if (GetFieldOffset("nt!_PNP_DEVICE_EVENT_ENTRY", "ListEntry", &offset)) {
  4565. dprintf("Cannot find _PNP_DEVICE_EVENT_ENTRY type.\n");
  4566. return E_INVALIDARG;
  4567. }
  4568. deviceEvent = flink - offset; // CONTAINING_RECORD(deviceEventList.List.Flink, PNP_DEVICE_EVENT_ENTRY, ListEntry);
  4569. followLinks = TRUE;
  4570. } else {
  4571. followLinks = FALSE;
  4572. }
  4573. DumpDeviceEventEntry( deviceEvent, listHead, followLinks );
  4574. return S_OK;
  4575. }
  4576. BOOLEAN
  4577. DumpDeviceEventEntry(
  4578. ULONG64 DeviceEvent,
  4579. ULONG64 ListHead,
  4580. BOOL FollowLinks
  4581. )
  4582. {
  4583. BOOLEAN continueDump = TRUE;
  4584. ULONG64 flink=0, blink=0, argument=0, callerEvent=0, callback=0, context=0;
  4585. ULONG64 vetoType=0, vetoName=0;
  4586. ULONG dataSize=0;
  4587. ULONG offset;
  4588. ULONG64 dataAddress;
  4589. dprintf("\nDumping DeviceEventEntry @ 0x%08p\n", DeviceEvent);
  4590. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "ListEntry.Flink", flink)) {
  4591. dprintf("Error reading DeviceEvent->ListEntry.Flink (%#010p)\n", DeviceEvent);
  4592. return FALSE;
  4593. }
  4594. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "ListEntry.Blink", blink)) {
  4595. dprintf("Error reading DeviceEvent->ListEntry.Blink (%#010p)\n", DeviceEvent);
  4596. return FALSE;
  4597. }
  4598. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "CallerEvent", callerEvent)) {
  4599. dprintf("Error reading DeviceEvent->CallerEvent (%#010p)\n", DeviceEvent);
  4600. return FALSE;
  4601. }
  4602. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "Callback", callback)) {
  4603. dprintf("Error reading DeviceEvent->Callback (%#010p)\n", DeviceEvent);
  4604. return FALSE;
  4605. }
  4606. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "Context", context)) {
  4607. dprintf("Error reading DeviceEvent->Context (%#010p)\n", DeviceEvent);
  4608. return FALSE;
  4609. }
  4610. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "VetoType", vetoType)) {
  4611. dprintf("Error reading DeviceEvent->VetoType (%#010p)\n", DeviceEvent);
  4612. return FALSE;
  4613. }
  4614. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "VetoName", vetoName)) {
  4615. dprintf("Error reading DeviceEvent->VetoName (%#010p)\n", DeviceEvent);
  4616. return FALSE;
  4617. }
  4618. dprintf(" ListEntry = 0x%08p, 0x%08p, Argument = 0x%08I64x\n",
  4619. flink, blink, argument);
  4620. dprintf(" CallerEvent = 0x%08p, Callback = 0x%08p, Context = 0x%08p\n",
  4621. callerEvent, callback, context);
  4622. dprintf(" VetoType = 0x%08p, VetoName = 0x%08p\n",
  4623. vetoType, vetoName);
  4624. if (GetFieldOffset("nt!_PNP_DEVICE_EVENT_ENTRY", "Data", &offset)) {
  4625. dprintf("Cannot find offset of Data in _PNP_DEVICE_EVENT_ENTRY type.\n");
  4626. return FALSE;
  4627. }
  4628. dataAddress = DeviceEvent + offset;
  4629. if (GetFieldValue(DeviceEvent, "nt!_PNP_DEVICE_EVENT_ENTRY", "Data.TotalSize", dataSize)) {
  4630. dprintf("Error reading DeviceEvent->Data.TotalSize (%#010p)\n", DeviceEvent);
  4631. return FALSE;
  4632. }
  4633. DumpPlugPlayEventBlock( dataAddress, dataSize );
  4634. if (FollowLinks && flink != ListHead) {
  4635. if (CheckControlC()) {
  4636. continueDump = FALSE;
  4637. } else {
  4638. if (GetFieldOffset("nt!_PNP_DEVICE_EVENT_ENTRY", "ListEntry", &offset)) {
  4639. dprintf("Cannot find offset of ListEntry in _PNP_DEVICE_EVENT_ENTRY type.\n");
  4640. return FALSE;
  4641. }
  4642. continueDump = DumpDeviceEventEntry(flink - offset,
  4643. ListHead,
  4644. TRUE);
  4645. }
  4646. }
  4647. return continueDump;
  4648. }
  4649. PUCHAR DevNodeStateNames[] = {
  4650. "DeviceNodeUnspecified",
  4651. "DeviceNodeUninitialized",
  4652. "DeviceNodeInitialized",
  4653. "DeviceNodeDriversAdded",
  4654. "DeviceNodeResourcesAssigned",
  4655. "DeviceNodeStartPending",
  4656. "DeviceNodeStartCompletion",
  4657. "DeviceNodeStartPostWork",
  4658. "DeviceNodeStarted",
  4659. "DeviceNodeQueryStopped",
  4660. "DeviceNodeStopped",
  4661. "DeviceNodeRestartCompletion",
  4662. "DeviceNodeEnumeratePending",
  4663. "DeviceNodeEnumerateCompletion",
  4664. "DeviceNodeAwaitingQueuedDeletion",
  4665. "DeviceNodeAwaitingQueuedRemoval",
  4666. "DeviceNodeQueryRemoved",
  4667. "DeviceNodeRemovePendingCloses",
  4668. "DeviceNodeRemoved",
  4669. "DeviceNodeDeletePendingCloses",
  4670. "DeviceNodeDeleted"
  4671. };
  4672. #define DEVNODE_STATE_NAMES_SIZE (sizeof(DevNodeStateNames) / sizeof(DevNodeStateNames[0]))
  4673. VOID
  4674. PrintDevNodeState(
  4675. IN ULONG Depth,
  4676. IN PUCHAR Field,
  4677. IN ULONG State
  4678. )
  4679. {
  4680. UCHAR *Description;
  4681. ULONG stateIndex;
  4682. stateIndex = State - 0x300; //DeviceNodeUnspecified;
  4683. if (stateIndex < DEVNODE_STATE_NAMES_SIZE) {
  4684. Description = DevNodeStateNames[stateIndex];
  4685. } else {
  4686. Description = "Unknown State";
  4687. }
  4688. xdprintf(Depth, "%s = %s (0x%x)\n", Field, Description, State);
  4689. }
  4690. VOID
  4691. DumpMultiSz(
  4692. IN PWCHAR MultiSz
  4693. )
  4694. {
  4695. PWCHAR p = MultiSz;
  4696. ULONG length;
  4697. while (*p) {
  4698. length = wcslen(p);
  4699. dprintf(" %S\n", p);
  4700. p += length + 1;
  4701. }
  4702. }
  4703. struct {
  4704. CONST GUID *Guid;
  4705. PCHAR Name;
  4706. } EventGuidTable[] = {
  4707. // From wdmguid.h
  4708. { &GUID_HWPROFILE_QUERY_CHANGE, "GUID_HWPROFILE_QUERY_CHANGE" },
  4709. { &GUID_HWPROFILE_CHANGE_CANCELLED, "GUID_HWPROFILE_CHANGE_CANCELLED" },
  4710. { &GUID_HWPROFILE_CHANGE_COMPLETE, "GUID_HWPROFILE_CHANGE_COMPLETE" },
  4711. { &GUID_DEVICE_INTERFACE_ARRIVAL, "GUID_DEVICE_INTERFACE_ARRIVAL" },
  4712. { &GUID_DEVICE_INTERFACE_REMOVAL, "GUID_DEVICE_INTERFACE_REMOVAL" },
  4713. { &GUID_TARGET_DEVICE_QUERY_REMOVE, "GUID_TARGET_DEVICE_QUERY_REMOVE" },
  4714. { &GUID_TARGET_DEVICE_REMOVE_CANCELLED, "GUID_TARGET_DEVICE_REMOVE_CANCELLED" },
  4715. { &GUID_TARGET_DEVICE_REMOVE_COMPLETE, "GUID_TARGET_DEVICE_REMOVE_COMPLETE" },
  4716. { &GUID_PNP_CUSTOM_NOTIFICATION, "GUID_PNP_CUSTOM_NOTIFICATION" },
  4717. { &GUID_PNP_POWER_NOTIFICATION, "GUID_PNP_POWER_NOTIFICATION" },
  4718. // From pnpmgr.h
  4719. { &GUID_DEVICE_ARRIVAL, "GUID_DEVICE_ARRIVAL" },
  4720. { &GUID_DEVICE_ENUMERATED, "GUID_DEVICE_ENUMERATED" },
  4721. { &GUID_DEVICE_ENUMERATE_REQUEST, "GUID_DEVICE_ENUMERATE_REQUEST" },
  4722. { &GUID_DEVICE_START_REQUEST, "GUID_DEVICE_START_REQUEST" },
  4723. { &GUID_DEVICE_REMOVE_PENDING, "GUID_DEVICE_REMOVE_PENDING" },
  4724. { &GUID_DEVICE_QUERY_AND_REMOVE, "GUID_DEVICE_QUERY_AND_REMOVE" },
  4725. { &GUID_DEVICE_EJECT, "GUID_DEVICE_EJECT" },
  4726. { &GUID_DEVICE_NOOP, "GUID_DEVICE_NOOP" },
  4727. { &GUID_DEVICE_SURPRISE_REMOVAL, "GUID_DEVICE_SURPRISE_REMOVAL" },
  4728. { &GUID_DRIVER_BLOCKED, "GUID_DRIVER_BLOCKED" },
  4729. };
  4730. #define EVENT_GUID_TABLE_SIZE (sizeof(EventGuidTable) / sizeof(EventGuidTable[0]))
  4731. VOID
  4732. LookupGuid(
  4733. IN CONST GUID *Guid,
  4734. IN OUT PCHAR String,
  4735. IN ULONG StringLength
  4736. )
  4737. {
  4738. int i;
  4739. for (i = 0; i < EVENT_GUID_TABLE_SIZE; i++) {
  4740. if (memcmp(Guid, EventGuidTable[i].Guid, sizeof(Guid)) == 0) {
  4741. strncpy(String, EventGuidTable[i].Name, StringLength - 1);
  4742. String[StringLength - 1] = '\0';
  4743. return;
  4744. }
  4745. }
  4746. _snprintf( String, StringLength, "%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X",
  4747. Guid->Data1,
  4748. Guid->Data2,
  4749. Guid->Data3,
  4750. Guid->Data4[0],
  4751. Guid->Data4[1],
  4752. Guid->Data4[2],
  4753. Guid->Data4[3],
  4754. Guid->Data4[4],
  4755. Guid->Data4[5],
  4756. Guid->Data4[6],
  4757. Guid->Data4[7] );
  4758. }
  4759. PUCHAR EventCategoryStrings[] = {
  4760. "HardwareProfileChangeEvent",
  4761. "TargetDeviceChangeEvent",
  4762. "DeviceClassChangeEvent",
  4763. "CustomDeviceEvent",
  4764. "DeviceInstallEvent",
  4765. "DeviceArrivalEvent",
  4766. "PowerEvent",
  4767. "VetoEvent",
  4768. "BlockedDriverEvent"
  4769. };
  4770. #define N_EVENT_CATEGORIES (sizeof(EventCategoryStrings) / sizeof(EventCategoryStrings[0]))
  4771. VOID
  4772. DumpPlugPlayEventBlock(
  4773. ULONG64 PlugPlayEventBlock,
  4774. ULONG Size
  4775. )
  4776. {
  4777. GUID EventGuid;
  4778. UCHAR guidString[256];
  4779. UCHAR PnpEventTyp[] = "nt!PLUGPLAY_EVENT_BLOCK";
  4780. char categoryString[80];
  4781. ULONG64 Result=0, DeviceObject=0;
  4782. ULONG EventCategory=0, TotalSize=0, Flags=0;
  4783. ULONG64 PtrVal=0;
  4784. ULONG offset;
  4785. PUCHAR buffer;
  4786. ULONG powerData;
  4787. dprintf("\n Dumping PlugPlayEventBlock @ 0x%08X\n", PlugPlayEventBlock);
  4788. if (GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "EventGuid", EventGuid)) {
  4789. dprintf("Error reading PlugPlayEventBlock (%#010p)\n", PlugPlayEventBlock);
  4790. return;
  4791. }
  4792. LookupGuid(&EventGuid, guidString, sizeof(guidString));
  4793. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "EventCategory", EventCategory);
  4794. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "Result", Result );
  4795. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "Flags", Flags );
  4796. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "TotalSize", TotalSize );
  4797. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "DeviceObject", DeviceObject );
  4798. if ( EventCategory > N_EVENT_CATEGORIES) {
  4799. sprintf(categoryString, "Unknown category (%d)", EventCategory);
  4800. } else {
  4801. strcpy(categoryString, EventCategoryStrings[ EventCategory ]);
  4802. }
  4803. dprintf(" EventGuid = %s\n Category = %s\n",
  4804. guidString, categoryString);
  4805. dprintf(" Result = 0x%08p, Flags = 0x%08X, TotalSize = %d\n",
  4806. Result, Flags, TotalSize );
  4807. dprintf(" DeviceObject = 0x%08p\n", DeviceObject);
  4808. switch (EventCategory) {
  4809. case HardwareProfileChangeEvent:
  4810. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.ProfileNotification.Notification", PtrVal);
  4811. dprintf( " Notification = 0x%08p\n", PtrVal );
  4812. break;
  4813. case DeviceArrivalEvent:
  4814. break;
  4815. case TargetDeviceChangeEvent:
  4816. dprintf( " DeviceIds:\n");
  4817. GetFieldOffset(PnpEventTyp, "u.TargetDevice.DeviceIds", &offset);
  4818. buffer = LocalAlloc(LPTR, Size - offset);
  4819. xReadMemory(PlugPlayEventBlock + offset, buffer, Size - offset);
  4820. DumpMultiSz( (PWCHAR)buffer );
  4821. LocalFree(buffer);
  4822. break;
  4823. case DeviceClassChangeEvent:
  4824. #if DEBUGGER_GETS_FIXED
  4825. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.DeviceClass.ClassGuid", EventGuid);
  4826. #else
  4827. GetFieldOffset(PnpEventTyp, "u.DeviceClass.ClassGuid", &offset);
  4828. xReadMemory(PlugPlayEventBlock + offset, &EventGuid, sizeof(EventGuid));
  4829. #endif
  4830. LookupGuid(&EventGuid, guidString, sizeof(guidString));
  4831. dprintf( " ClassGuid = %s\n", guidString );
  4832. GetFieldOffset(PnpEventTyp, "u.DeviceClass.SymbolicLinkName", &offset);
  4833. buffer = LocalAlloc(LPTR, Size - offset);
  4834. xReadMemory(PlugPlayEventBlock + offset, buffer, Size - offset);
  4835. dprintf( " SymbolicLinkName = %S\n", buffer );
  4836. LocalFree(buffer);
  4837. break;
  4838. case CustomDeviceEvent:
  4839. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.CustomNotification.NotificationStructure", PtrVal);
  4840. dprintf( " NotificationStructure = 0x%08p\n DeviceIds:\n",
  4841. PtrVal);
  4842. GetFieldOffset(PnpEventTyp, "u.CustomNotification.DeviceIds", &offset);
  4843. buffer = LocalAlloc(LPTR, Size - offset);
  4844. xReadMemory(PlugPlayEventBlock + offset, buffer, Size - offset);
  4845. DumpMultiSz( (PWCHAR) buffer );
  4846. LocalFree(buffer);
  4847. break;
  4848. case DeviceInstallEvent:
  4849. GetFieldOffset(PnpEventTyp, "u.InstallDevice.DeviceId", &offset);
  4850. buffer = LocalAlloc(LPTR, Size - offset);
  4851. xReadMemory(PlugPlayEventBlock + offset, buffer, Size - offset);
  4852. dprintf( " DeviceId = %S\n", buffer );
  4853. LocalFree(buffer);
  4854. break;
  4855. case PowerEvent:
  4856. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.PowerNotification.NotificationCode", powerData);
  4857. dprintf( " NotificationCode = 0x%08X\n", powerData );
  4858. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.PowerNotification.NotificationData", powerData);
  4859. dprintf( " NotificationData = 0x%08X\n", powerData );
  4860. break;
  4861. case BlockedDriverEvent:
  4862. #if DEBUGGER_GETS_FIXED
  4863. GetFieldValue(PlugPlayEventBlock, PnpEventTyp, "u.BlockedDriverNotification.BlockedDriverGuid", EventGuid);
  4864. #else
  4865. GetFieldOffset(PnpEventTyp, "u.BlockedDriverNotification.BlockedDriverGuid", &offset);
  4866. xReadMemory(PlugPlayEventBlock + offset, &EventGuid, sizeof(EventGuid));
  4867. #endif
  4868. LookupGuid(&EventGuid, guidString, sizeof(guidString));
  4869. dprintf( " BlockedDriverGuid = %s\n", guidString );
  4870. break;
  4871. }
  4872. }
  4873. DECLARE_API( rellist )
  4874. /*++
  4875. Routine Description:
  4876. Dump a device object.
  4877. Arguments:
  4878. args - the location of the device object to dump.
  4879. Return Value:
  4880. None
  4881. --*/
  4882. {
  4883. ULONG64 relationList=0;
  4884. ULONG verbose=0;
  4885. if (GetExpressionEx(args, &relationList, &args)) {
  4886. verbose = (ULONG) GetExpression(args);
  4887. }
  4888. DumpRelationsList( relationList,
  4889. (BOOLEAN) (verbose & DUMP_CM_RES),
  4890. (BOOLEAN) (verbose & DUMP_CM_RES_REQ_LIST),
  4891. (BOOLEAN) (verbose & DUMP_CM_RES_TRANS)
  4892. );
  4893. return S_OK;
  4894. }
  4895. DECLARE_API( lbt )
  4896. /*++
  4897. Routine Description:
  4898. Dump the legacy bus information table.
  4899. Arguments:
  4900. args - none.
  4901. Return Value:
  4902. None
  4903. --*/
  4904. {
  4905. CHAR buffer[256];
  4906. ULONG size, deviceNodeOffset;
  4907. ULONG64 deviceNode;
  4908. ULONG64 listHeadAddr, legacyBusEntryAddr;
  4909. ULONG64 head_Flink, head_Blink, link_Flink;
  4910. USHORT instancePath_Len, instancePath_Max;
  4911. ULONG busNumber;
  4912. ULONG64 instancePath_Buff, interfaceType;
  4913. CHAR deviceNodeType[] ="nt!_DEVICE_NODE";
  4914. CHAR listEntryType[] = "nt!_LIST_ENTRY" ;
  4915. INTERFACE_TYPE Iface;
  4916. if ((size = GetTypeSize(listEntryType)) == 0) {
  4917. dprintf("Failed to get the size of %s\n", listEntryType);
  4918. return E_INVALIDARG;
  4919. }
  4920. if (GetFieldOffset(deviceNodeType, "LegacyBusListEntry.Flink", &deviceNodeOffset)) {
  4921. dprintf("Failed to get the offset of LegacyBusListEntry from %s\n", deviceNodeType);
  4922. return E_INVALIDARG;
  4923. }
  4924. dprintf("Legacy Bus Information Table...\n");
  4925. for (Iface = Internal; Iface < MaximumInterfaceType; Iface++) {
  4926. sprintf(buffer, "nt!IopLegacyBusInformationTable + %x", Iface * size);
  4927. if ((listHeadAddr = GetExpression(buffer)) == 0) {
  4928. dprintf("Error retrieving address of bus number list for interface %d\n", Iface);
  4929. continue;
  4930. }
  4931. if (GetFieldValue(listHeadAddr, listEntryType, "Flink", head_Flink) ||
  4932. GetFieldValue(listHeadAddr, listEntryType, "Blink", head_Blink)) {
  4933. dprintf("Error reading value of bus number list head (0x%08p) for interface %d\n", listHeadAddr, Iface);
  4934. continue;
  4935. }
  4936. if (head_Flink == listHeadAddr) {
  4937. continue;
  4938. }
  4939. for ( legacyBusEntryAddr = head_Flink;
  4940. legacyBusEntryAddr != listHeadAddr;
  4941. legacyBusEntryAddr = link_Flink) {
  4942. deviceNode = legacyBusEntryAddr - deviceNodeOffset;
  4943. if (GetFieldValue(deviceNode, deviceNodeType, "BusNumber", busNumber)) {
  4944. dprintf("Error reading BusNumber from (0x%08p)\n", deviceNode);
  4945. break;
  4946. }
  4947. if (GetFieldValue(deviceNode, deviceNodeType, "InstancePath.Length", instancePath_Len)) {
  4948. dprintf("Error reading InstancePath.Length from (0x%08p)\n", deviceNode);
  4949. break;
  4950. }
  4951. if (GetFieldValue(deviceNode, deviceNodeType, "InstancePath.MaximumLength", instancePath_Max)) {
  4952. dprintf("Error reading InstancePath.MaximumLength from (0x%08p)\n", deviceNode);
  4953. break;
  4954. }
  4955. if (GetFieldValue(deviceNode, deviceNodeType, "InstancePath.Buffer", instancePath_Buff)) {
  4956. dprintf("Error reading InstancePath.Buffer from (0x%08p)\n", deviceNode);
  4957. break;
  4958. }
  4959. if (instancePath_Buff != 0) {
  4960. UNICODE_STRING64 v;
  4961. switch (Iface) {
  4962. case InterfaceTypeUndefined: dprintf("InterfaceTypeUndefined"); break;
  4963. case Internal: dprintf("Internal"); break;
  4964. case Isa: dprintf("Isa"); break;
  4965. case Eisa: dprintf("Eisa"); break;
  4966. case MicroChannel: dprintf("Micro Channel"); break;
  4967. case TurboChannel: dprintf("Turbo Channel"); break;
  4968. case PCIBus: dprintf("PCI"); break;
  4969. case VMEBus: dprintf("VME"); break;
  4970. case NuBus: dprintf("NuBus"); break;
  4971. case PCMCIABus: dprintf("PCMCIA"); break;
  4972. case CBus: dprintf("CBus"); break;
  4973. case MPIBus: dprintf("MPIBus"); break;
  4974. case MPSABus: dprintf("MPSABus"); break;
  4975. case ProcessorInternal: dprintf("Processor Internal"); break;
  4976. case InternalPowerBus: dprintf("Internal Power Bus"); break;
  4977. case PNPISABus: dprintf("PnP Isa"); break;
  4978. case PNPBus: dprintf("PnP Bus"); break;
  4979. default: dprintf("** Unknown Interface Type **"); break;
  4980. }
  4981. dprintf("%d - !devnode 0x%08p\n", busNumber, deviceNode);
  4982. v.Buffer = instancePath_Buff;
  4983. v.Length = instancePath_Len;
  4984. v.MaximumLength = instancePath_Max;
  4985. dprintf("\t\""); DumpUnicode64(v); dprintf("\"\n");
  4986. }
  4987. if (GetFieldValue(legacyBusEntryAddr, listEntryType, "Flink", link_Flink)) {
  4988. dprintf("Error reading Flink from (0x%08p)\n", legacyBusEntryAddr);
  4989. break;
  4990. }
  4991. }
  4992. }
  4993. return S_OK;
  4994. }