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.

567 lines
17 KiB

  1. /*++
  2. Copyright (c) 1999-2000 Microsoft Corporation
  3. Module Name:
  4. A5Parse.c
  5. Abstract:
  6. This module contains the code that implements ACPI bugcheck parsing
  7. mechanism.
  8. Author:
  9. Vincent Geglia (vincentg) 13-Dec-1999
  10. Environment:
  11. Kernel mode
  12. Notes:
  13. Revision History:
  14. --*/
  15. #include "ntddk.h"
  16. #include "stdarg.h"
  17. #include "stdio.h"
  18. #include "inbv.h"
  19. #include "acpidbg.h"
  20. // Definitions
  21. #define ATTRIB_BLACK 0
  22. #define ATTRIB_RED 1
  23. #define ATTRIB_GREEN 2
  24. #define ATTRIB_YELLOW 3
  25. #define ATTRIB_BLUE 4
  26. #define ATTRIB_MAGENTA 5
  27. #define ATTRIB_CYAN 6
  28. #define ATTRIB_WHITE 7
  29. #define ATTRIB_BRIGHT 8
  30. #define A5PARSE_VERSION "1.0"
  31. #define HEADER ATTRIB_WHITE + ATTRIB_BRIGHT
  32. #define BODY ATTRIB_YELLOW + ATTRIB_BRIGHT
  33. // Globals
  34. KBUGCHECK_CALLBACK_RECORD A5ParseCallbackRecord;
  35. BOOLEAN VideoAvailable = FALSE;
  36. extern ULONG *KiBugCheckData;
  37. //
  38. // Define driver entry routine.
  39. //
  40. NTSTATUS
  41. DriverEntry(
  42. IN PDRIVER_OBJECT DriverObject,
  43. IN PUNICODE_STRING RegistryPath
  44. );
  45. //
  46. // Define the local routines used by this driver module.
  47. //
  48. VOID
  49. A5ParseBugcheckHandler (
  50. IN PVOID Buffer,
  51. IN ULONG Length
  52. );
  53. VOID
  54. A5ParseBugcheckParams (
  55. ULONG ArrayIndex
  56. );
  57. VOID
  58. A5ParseStallSystem (
  59. ULONG NumberOfSeconds
  60. );
  61. BOOLEAN
  62. A5ParseCheckBootVid (
  63. UCHAR Attribute
  64. );
  65. VOID
  66. A5ParseWriteToScreen (
  67. UCHAR Attribute,
  68. UCHAR *Text,
  69. ...
  70. );
  71. VOID
  72. A5ParseAcpiRootResourcesFailure (
  73. VOID
  74. );
  75. VOID
  76. A5ParseAcpiRootPciResourcesFailure (
  77. VOID
  78. );
  79. NTSTATUS
  80. A5ParseDispatch(
  81. IN PDEVICE_OBJECT DeviceObject,
  82. IN PIRP Irp
  83. );
  84. // Structures / type definitions
  85. typedef
  86. VOID
  87. (*PKA5PARSE_ROUTINE) (
  88. VOID
  89. );
  90. typedef struct _A5MESSAGE {
  91. ULONG ErrorCode;
  92. UCHAR ErrorTitle[80];
  93. UCHAR ErrorMessage[320];
  94. UCHAR InformationSource[160];
  95. PKA5PARSE_ROUTINE ParserFunction;
  96. } A5MESSAGE, *PA5MESSAGE;
  97. A5MESSAGE A5Messages[] = {
  98. ACPI_ROOT_RESOURCES_FAILURE, "ACPI Root Resources Failure", "ACPI cannot find the SCI Interrupt vector in the resources handed to it when\nACPI is started.", "(none)", A5ParseAcpiRootResourcesFailure,
  99. ACPI_ROOT_PCI_RESOURCE_FAILURE, "ACPI Root PCI Resources Failure", "ACPI could not process the resource list for the PCI root buses (_CRS method).", "See http://www.microsoft.com/hwdev/onnow for the _CRS whitepaper.", A5ParseAcpiRootPciResourcesFailure,
  100. ACPI_FAILED_MUST_SUCCEED_METHOD, "A Critical ACPI method could not be evaluated", "ACPI attempted to execute a control method while creating device extensions to\nrepresent the ACPI namespace, but was unable to.", "ACPI 1.0b", NULL,
  101. ACPI_PRW_PACKAGE_EXPECTED_INTEGER, "_PRW returned non-integer value", "ACPI evaluated a _PRW package, and expected an integer, but was returned a\ndifferent data type.", "ACPI 1.0b, section 7.2.1, page 154-155", NULL,
  102. ACPI_PRW_PACKAGE_TOO_SMALL, "_PRW package too small", "ACPI evaluated a _PRW pacakge, and expected at least two integers, but received less.", "ACPI 1.0b, section 7.2.1, page 154-155", NULL,
  103. ACPI_PRX_CANNOT_FIND_OBJECT, "_PRx package referenced an object that could not be found.", "ACPI found a _PRx method that referenced an object that was not found in the\nACPI namespace.", "ACPI 1.0b, section 7.2.2, page 155", NULL,
  104. ACPI_EXPECTED_BUFFER, "Object evaluated to unexpected data type", "ACPI evaluated a control method, and expected a data buffer to be returned, but received something else.", "ACPI 1.0b", NULL,
  105. ACPI_EXPECTED_INTEGER, "Object evaluated to unexpected data type", "ACPI evaluated a control method, and expected a data type of 'integer' to be\nreturned, but received something else.", "ACPI 1.0b", NULL,
  106. ACPI_EXPECTED_PACKAGE, "Object evaluated to unexpected data type", "ACPI evaluated a control method, and expected a data type of 'package' to be\nreturned, but received something else.", "ACPI 1.0b", NULL,
  107. ACPI_EXPECTED_STRING, "Object evaluated to unexpected data type", "ACPI evaluated a control method, and expected a data type of 'string' to be\nreturned, but received something else.", "ACPI 1.0b", NULL,
  108. ACPI_EJD_CANNOT_FIND_OBJECT, "_EJD method referenced a non-existent device", "ACPI evaluated an _EJD method, which returns the name of a namespace object, but that object could not be found.", "ACPI 1.0b, section 6.3.1, page 126", NULL,
  109. ACPI_CLAIMS_BOGUS_DOCK_SUPPORT, "Inconsistent/incomplete docking station information", "While enumerating the ACPI namespace, the OS found control methods indicating\nthe system is dockable, but other control methods were missing, or provided\nincorrect informatoin", "ACPI 1.0b, section 6.3, page 125-128", NULL,
  110. ACPI_REQUIRED_METHOD_NOT_PRESENT, "Missing _HID or _ADR method", "Physical devices described in the ACPI namespace must be identified using either a _HID or _ADR.", "ACPI 1.0b, section 6.1, page 119-121", NULL,
  111. ACPI_POWER_NODE_REQUIRED_METHOD_NOT_PRESENT, "Missing _ON, _OFF, and/or _STA method", "Power resources must have an _ON, _OFF, and _STA method.", "ACPI 1.0b, section 7.4, page 157-158", NULL,
  112. ACPI_PNP_RESOURCE_LIST_BUFFER_TOO_SMALL, "Unable to parse resource descriptors (_CRS/_PRS)", "The ACPI driver was unable to process a resource descriptor, which usually\nindicates an improperly formed resource descriptor returned by _CRS or _PRS.", "ACPI 1.0b, section 6.2/6.4, page 121-125, 128-150", NULL,
  113. ACPI_CANNOT_MAP_SYSTEM_TO_DEVICE_STATES, "Unable to map Sx to Dx states", "The ACPI driver was unable to determine correct power state (Sx-->Dx) mappings\ndue to a method (_PRx/_PRW) referencing a non-supported Sx state, or the lack\nof Dx support on a particular device.", "ACPI 1.0b, section 7.2, page 154-157\nSee also http://www.microsoft.com/hwdev/onnow for a whitepaper on this topic.", NULL,
  114. ACPI_SYSTEM_CANNOT_START_ACPI, "Unable to transition into ACPI mode", "The ACPI driver was unable to transition the system from non-ACPI mode to ACPI\nmode.", "ACPI 1.0b", NULL,
  115. ACPI_FAILED_PIC_METHOD, "Unable to evaluate _PIC method", "A required method for APIC/PIC capable machines, _PIC, could not be evaluated\nby the ACPI driver.", "See http://www.microsoft.com/hwdev/onnow for the 'PCI IRQ Routing on a\nMultiprocessor ACPI System' whitepaper.", NULL,
  116. ACPI_CANNOT_ROUTE_INTERRUPTS, "Unable to configure interrupt routing", "The ACPI driver attempted to configure IRQ routing on this system, but was\nunsuccessful.", "ACPI 1.0b", NULL,
  117. ACPI_PRT_CANNOT_FIND_LINK_NODE, "Unable to find IRQ routing link node", "The _PRT method referenced an IRQ routine link node that could not be located\nin the ACPI namespace.", "ACPI 1.0b, section 6.2.3, page 122-123", NULL,
  118. ACPI_PRT_CANNOT_FIND_DEVICE_ENTRY, "Unable to find entry in _PRT for PCI device", "The PCI bus driver enumerated a device for which there was no entry in the PCI\nbus _PRT method. Every PCI device using an interrupt must have an entry in\nthe _PRT.", "ACPI 1.0b, section 6.2.3, page 122-123", NULL,
  119. ACPI_PRT_HAS_INVALID_FUNCTION_NUMBERS, "_PRT did not comply with xxxxFFFF format for device address entry", "Windows 2000 / Windows 98 requires _PRT entries to include all function numbers when describing IRQ routine entries (e.g. 0007FFFF).", "(none)", NULL,
  120. ACPI_LINK_NODE_CANNOT_BE_DISABLED, "An IRQ routine link node could not be disabled", "The ACPI driver must be able to disable an IRQ routing link node in order to\nreprogram it - it was unable to do so.", "ACPI 1.0b", NULL,
  121. 0, "Unknown error", "ACPI called KeBugCheckEx with unknown parameters.", "(none)", NULL
  122. };
  123. //
  124. // Assign text sections for each routine.
  125. //
  126. #ifdef ALLOC_PRAGMA
  127. #pragma alloc_text(INIT, DriverEntry)
  128. #endif
  129. NTSTATUS
  130. DriverEntry(
  131. IN PDRIVER_OBJECT DriverObject,
  132. IN PUNICODE_STRING RegistryPath
  133. )
  134. /*++
  135. Routine Description:
  136. This is the initialization routine. It basically registers a bugcheck callback,
  137. nothing else.
  138. Arguments:
  139. DriverObject - Pointer to driver object created by the system.
  140. RegistryPath - Pointer to the registry path string
  141. Return Value:
  142. STATUS_SUCCESS if bugcheck callback registration successful, otherwise
  143. STATUS_UNSUCCESSFUL.
  144. --*/
  145. {
  146. UNICODE_STRING nameString;
  147. PDEVICE_OBJECT deviceObject;
  148. NTSTATUS status;
  149. BOOLEAN success = FALSE;
  150. //
  151. // Initialize the driver object with this device driver's entry points.
  152. //
  153. DriverObject->MajorFunction[IRP_MJ_CREATE] = A5ParseDispatch;
  154. DriverObject->MajorFunction[IRP_MJ_CLOSE] = A5ParseDispatch;
  155. DriverObject->MajorFunction[IRP_MJ_READ] = A5ParseDispatch;
  156. DriverObject->MajorFunction[IRP_MJ_WRITE] = A5ParseDispatch;
  157. DriverObject->MajorFunction[IRP_MJ_LOCK_CONTROL] = A5ParseDispatch;
  158. DriverObject->MajorFunction[IRP_MJ_QUERY_INFORMATION] = A5ParseDispatch;
  159. // Register bugcheck callback
  160. KeInitializeCallbackRecord (&A5ParseCallbackRecord);
  161. success = KeRegisterBugCheckCallback (&A5ParseCallbackRecord,
  162. (PVOID) A5ParseBugcheckHandler,
  163. NULL,
  164. 0,
  165. "A5PARSE");
  166. if (success == FALSE) {
  167. ASSERT (success);
  168. DbgPrint ("KeRegisterBugCheckCallback failed.\n");
  169. return STATUS_UNSUCCESSFUL;
  170. }
  171. /* KeBugCheckEx (0xA5,
  172. ACPI_ROOT_PCI_RESOURCE_FAILURE,
  173. 0,
  174. 0x81200023,
  175. 0);*/
  176. return STATUS_SUCCESS;
  177. }
  178. VOID
  179. A5ParseBugcheckHandler (
  180. IN PVOID Buffer,
  181. IN ULONG Length
  182. )
  183. /*++
  184. Routine Description:
  185. This is the bugcheck handler. It determines if the bugcheck is A5, and if so,
  186. modifies the bugcheck screen to display its message.
  187. Arguments:
  188. Buffer - Supplies a pointer to the bug check buffer (not used).
  189. Length - Supplies the length of the bug check buffer in bytes (not used).
  190. Return Value:
  191. None.
  192. --*/
  193. {
  194. char TextBuffer[2000];
  195. UCHAR count = 0;
  196. // We only parse bugcheck 0xA5
  197. if (KiBugCheckData[0] != 0xA5) {
  198. return;
  199. }
  200. // Set up video to display bugcheck text
  201. VideoAvailable = A5ParseCheckBootVid (ATTRIB_CYAN);
  202. // Show our banner
  203. A5ParseWriteToScreen (ATTRIB_WHITE + ATTRIB_BRIGHT,
  204. "** A5PARSE v%s, Copyright (C) Microsoft Corporation 1999-2000 **\n\n",
  205. A5PARSE_VERSION);
  206. count = 0;
  207. while (count < (sizeof(A5Messages) / sizeof (A5MESSAGE))) {
  208. if (A5Messages[count].ErrorCode == KiBugCheckData[1]) {
  209. // if (1) {
  210. A5ParseBugcheckParams(count);
  211. //_asm int 3;
  212. //A5ParseCheckBootVid (ATTRIB_CYAN);
  213. return;
  214. }
  215. count ++;
  216. }
  217. // Unknown (unhandled) error code - show unknown message
  218. A5ParseBugcheckParams (count-1);
  219. return;
  220. }
  221. VOID
  222. A5ParseStallSystem (
  223. ULONG NumberOfSeconds
  224. )
  225. /*++
  226. Routine Description:
  227. Stalls the system for a certain amount of time. Used to keep the bugcheck
  228. screen up on the system.
  229. Arguments:
  230. NumberOfSeconds - Number of seconds to wait.
  231. Return Value:
  232. None.
  233. --*/
  234. {
  235. while (NumberOfSeconds) {
  236. KeStallExecutionProcessor (1000000);
  237. NumberOfSeconds --;
  238. }
  239. }
  240. VOID
  241. A5ParseBugcheckParams (
  242. ULONG ArrayIndex
  243. )
  244. /*++
  245. Routine Description:
  246. Parses the A5 bugcheck codes and prints a meaningful message. Also calls any
  247. custom parsing function.
  248. Arguments:
  249. ArrayIndex - Index into message structure.
  250. Return Value:
  251. None.
  252. --*/
  253. {
  254. A5ParseWriteToScreen (ATTRIB_WHITE + ATTRIB_BRIGHT,
  255. "The system has crashed with a STOP 0xA5 (ACPI_BIOS_ERROR).\nA5PARSE will give additional information regarding this failure.\n\n");
  256. A5ParseWriteToScreen (HEADER,
  257. "Error code (ACPIDBG.H):");
  258. A5ParseWriteToScreen (BODY,
  259. "0x%08lX\n",
  260. A5Messages[ArrayIndex].ErrorCode);
  261. A5ParseWriteToScreen (HEADER,
  262. "Error title:");
  263. A5ParseWriteToScreen (BODY,
  264. "%s\n\n",
  265. A5Messages[ArrayIndex].ErrorTitle);
  266. A5ParseWriteToScreen (HEADER,
  267. "Error Description:\n");
  268. A5ParseWriteToScreen (BODY,
  269. "%s\n\n",
  270. A5Messages[ArrayIndex].ErrorMessage);
  271. A5ParseWriteToScreen (HEADER,
  272. "Additional Information can be found at:\n");
  273. A5ParseWriteToScreen (BODY,
  274. "%s\n\n",
  275. A5Messages[ArrayIndex].InformationSource);
  276. A5ParseWriteToScreen (HEADER,
  277. "Additional debug information:\n");
  278. if (A5Messages[ArrayIndex].ParserFunction) {
  279. (A5Messages[ArrayIndex].ParserFunction) ();
  280. }
  281. // Stall the system for 600 seconds.
  282. A5ParseStallSystem (600);
  283. return;
  284. }
  285. BOOLEAN
  286. A5ParseCheckBootVid (
  287. UCHAR Attribute
  288. )
  289. /*++
  290. Routine Description:
  291. Verifies the system screen is accessible and acquires ownership, sets the
  292. window size, and paints the background color.
  293. Arguments:
  294. Attribute - Text attribute for the background screen color.
  295. Return Value:
  296. TRUE if successful, FALSE if not successful.
  297. --*/
  298. {
  299. if (InbvIsBootDriverInstalled()) {
  300. InbvAcquireDisplayOwnership();
  301. InbvSolidColorFill(0,0,639,479,Attribute);
  302. InbvSetTextColor(15);
  303. InbvInstallDisplayStringFilter((INBV_DISPLAY_STRING_FILTER)NULL);
  304. InbvEnableDisplayString(TRUE);
  305. InbvSetScrollRegion(0,0,639,479);
  306. return TRUE;
  307. }
  308. return FALSE;
  309. }
  310. VOID
  311. A5ParseWriteToScreen (UCHAR Attribute,
  312. UCHAR *Text,
  313. ...
  314. )
  315. /*++
  316. Routine Description:
  317. Writes text to the console using the attribute specified.
  318. Arguments:
  319. Attribute - Text attribute to use to write text to the screen.
  320. Text - Pointer to the buffer containing the text to write to the screen.
  321. Return Value:
  322. None.
  323. --*/
  324. {
  325. UCHAR TextOut[2000];
  326. va_list va;
  327. va_start (va, Text);
  328. vsprintf (TextOut, Text, va);
  329. va_end (va);
  330. InbvSetTextColor(Attribute);
  331. InbvDisplayString (TextOut);
  332. DbgPrint (TextOut);
  333. }
  334. VOID
  335. A5ParseAcpiRootResourcesFailure(
  336. VOID
  337. )
  338. {
  339. if (!KiBugCheckData[3]) {
  340. A5ParseWriteToScreen (BODY,
  341. "No resource list provided to ACPI driver. This is likely caused by missing, or\nnon-compliant ACPI FADT/RSDT tables. Check to make sure the fixed tables are\ncorrect (see ACPI 1.0b chapters 4 and 5).\n\n");
  342. } else {
  343. A5ParseWriteToScreen (BODY,
  344. "A resource list was provided to the ACPI driver, but it did not contain an\ninterrupt vector. This could be cause by missing or non-compliant ACPI FADT/RSDT tables.\nCheck to make sure the fixed tables are correct (see ACPI 1.0b chapters 4\nand 5).\n\n");
  345. A5ParseWriteToScreen (BODY,
  346. "In the kernel debugger, type '!cmreslist %lx' to dump the resource\ndescriptor missing the interrupt resource.\n",
  347. KiBugCheckData[3]);
  348. }
  349. return;
  350. }
  351. VOID
  352. A5ParseAcpiRootPciResourcesFailure (
  353. VOID
  354. )
  355. {
  356. if (KiBugCheckData[3] == 2 && !KiBugCheckData[4]) {
  357. A5ParseWriteToScreen (BODY,
  358. "No resource list for the root PCI bus was provided to the ACPI driver. This is\ncaused by a missing, or non-compliant _CRS method for the root PCI bus.\n");
  359. return;
  360. }
  361. if (KiBugCheckData[3] == 3) {
  362. A5ParseWriteToScreen (BODY,
  363. "The current bus number was not found in the _CRS method for the root PCI bus.\nSee ACPI 1.0b, section 6.4.1, pg 128 for information on reporting bus numbering.\n");
  364. return;
  365. }
  366. if (KiBugCheckData[3] > 3) {
  367. A5ParseWriteToScreen (BODY,
  368. "The _CRS for the root PCI bus is incorrectly formed. Typically this means the\n_CRS claimed to decode memory that overlaps with system memory, which would\nindicate the _CRS is incorrect.\n");
  369. A5ParseWriteToScreen (BODY,
  370. "\nThere were %ld errors found in the _CRS.\n",
  371. KiBugCheckData[4]);
  372. A5ParseWriteToScreen (BODY,
  373. "\nThe resource list of the _CRS can be examined via the kernel debugger using the\ncommand '!ioreslist %lx'.\n",
  374. KiBugCheckData[3]);
  375. }
  376. return;
  377. }
  378. NTSTATUS
  379. A5ParseDispatch(
  380. IN PDEVICE_OBJECT DeviceObject,
  381. IN PIRP Irp
  382. )
  383. /*++
  384. Routine Description:
  385. Stub function for device driver entry points.
  386. Arguments:
  387. DeviceObject - Pointer to the device object for this driver.
  388. Irp - Pointer to the request packet representing the I/O request.
  389. Return Value:
  390. STATUS_NOT_SUPPORTED
  391. --*/
  392. {
  393. return STATUS_NOT_SUPPORTED;
  394. }