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.

1056 lines
30 KiB

  1. /*++
  2. Copyright (c) 1990 Microsoft Corporation
  3. Module Name:
  4. service.c
  5. Abstract:
  6. ACPI Embedded Controller Driver
  7. Author:
  8. Ken Reneris
  9. Environment:
  10. Notes:
  11. Revision History:
  12. --*/
  13. #include "ecp.h"
  14. #define NTMS 10000L // 1 millisecond is ten thousand 100ns
  15. #define NTSEC (NTMS * 1000L)
  16. LARGE_INTEGER AcpiEcWatchdogTimeout = {(NTSEC * -5L), -1};
  17. LARGE_INTEGER AcpiEcLastActionTime = {0,0};
  18. PUCHAR AcpiEcActionDescription [EC_ACTION_MAX >> 4] = {
  19. "Invalid ",
  20. "Read Status",
  21. "Read Data ",
  22. "Write Cmd ",
  23. "Write Data ",
  24. "Interrupt ",
  25. "Disable GPE",
  26. "Enable GPE ",
  27. "Clear GPE ",
  28. "Queued IO ",
  29. "Repeated Last action this many times:"
  30. };
  31. VOID
  32. AcpiEcServiceDevice (
  33. IN PECDATA EcData
  34. )
  35. /*++
  36. Routine Description:
  37. This routine starts or continues servicing the device's work queue
  38. Arguments:
  39. EcData - Pointer to embedded controller to service.
  40. Return Value:
  41. None
  42. --*/
  43. {
  44. KIRQL OldIrql;
  45. //
  46. // Even though the device is unloaded, there might still be a
  47. // service call which occurs until the timer is canceled
  48. //
  49. EcPrint(EC_TRACE, ("AcpiEcServiceDevice.\n"));
  50. if (EcData->DeviceState > EC_DEVICE_UNLOAD_PENDING) {
  51. return;
  52. }
  53. //
  54. // Acquire device lock and signal function was entered
  55. //
  56. KeAcquireSpinLock (&EcData->Lock, &OldIrql);
  57. EcData->InServiceLoop = TRUE;
  58. //
  59. // If not already in service, enter InService
  60. //
  61. if (!EcData->InService) {
  62. EcData->InService = TRUE;
  63. //
  64. // Disable the device's interrupt
  65. //
  66. if (EcData->InterruptEnabled) {
  67. EcData->InterruptEnabled = FALSE;
  68. //
  69. // Call ACPI to disable the device's interrupt
  70. //
  71. AcpiEcLogAction (EcData, EC_ACTION_DISABLE_GPE, 0);
  72. AcpiInterfaces.GpeDisableEvent (AcpiInterfaces.Context,
  73. EcData->GpeVectorObject);
  74. }
  75. //
  76. // While service invocation pending, loop
  77. //
  78. while (EcData->InServiceLoop) {
  79. EcData->InServiceLoop = FALSE;
  80. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  81. //
  82. // Dispatch service handler
  83. //
  84. AcpiEcServiceIoLoop (EcData);
  85. //
  86. // Loop and re-service
  87. //
  88. KeAcquireSpinLock (&EcData->Lock, &OldIrql);
  89. }
  90. //
  91. // No longer in service loop
  92. //
  93. EcData->InService = FALSE;
  94. //
  95. // If unload is pending, check to see if the device can be unloaded now
  96. //
  97. if (EcData->DeviceState > EC_DEVICE_WORKING) {
  98. AcpiEcUnloadPending (EcData);
  99. }
  100. //
  101. // Enable the device's interrupt
  102. //
  103. if (!EcData->InterruptEnabled) {
  104. EcData->InterruptEnabled = TRUE;
  105. //
  106. // Call ACPI to enable the device's interrupt
  107. //
  108. AcpiEcLogAction (EcData, EC_ACTION_ENABLE_GPE, 0);
  109. AcpiInterfaces.GpeEnableEvent (AcpiInterfaces.Context,
  110. EcData->GpeVectorObject);
  111. }
  112. }
  113. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  114. }
  115. VOID
  116. AcpiEcServiceIoLoop (
  117. IN PECDATA EcData
  118. )
  119. /*++
  120. Routine Description:
  121. Main embedded controller device service loop. Services EC events,
  122. and processes IO queue. Terminate when the controller is busy (e.g.,
  123. wait for interrupt to continue) or when all servicing has been completed.
  124. N.B. Caller must be the owner of the device InService flag
  125. Arguments:
  126. EcData - Pointer to embedded controller to service.
  127. Return Value:
  128. none
  129. --*/
  130. {
  131. PIO_STACK_LOCATION IrpSp;
  132. PLIST_ENTRY Link;
  133. PIRP Irp;
  134. PUCHAR WritePort;
  135. UCHAR WriteData;
  136. UCHAR Status;
  137. UCHAR Data;
  138. BOOLEAN EcBusy;
  139. BOOLEAN BurstEnabled;
  140. BOOLEAN ProcessQuery;
  141. ULONG NoWorkStall;
  142. ULONG StallAccumulator;
  143. PULONG Timeout;
  144. KIRQL OldIrql;
  145. LIST_ENTRY CompleteQueue;
  146. ULONG i, j;
  147. //
  148. // EcBusy flags that there may be work to do. Initialized to TRUE every time
  149. // The service loop is entered, or when a timeout almost occured, but then some
  150. // work was found. It is set to FALSE when the IO queue is empty and there are
  151. // no query events pending.
  152. //
  153. EcBusy = TRUE;
  154. //
  155. // Timeout points to the counter to be incremented as the loop exits. It is
  156. // also used as a flag to indicate that the loop should exit. The loop won't
  157. // exit until Timeout != NULL. When exiting because of idleness, it is set
  158. // to the local vailable i, so that we don't keep a permanent count of those
  159. // timeout conditions. We keep track of how many time we timeout waiting for
  160. // the EC. If we do, we expect an interrupt when the EC is ready.
  161. //
  162. Timeout = NULL;
  163. //
  164. // This is set (along with WriteData) to have write a command or data to the
  165. // EC at the appropriate point in the loop.
  166. //
  167. WritePort = NULL;
  168. //
  169. // NoWorkStall is incremented every time through the loop. It is reset to 0
  170. // whenever any work is done. If it gets too big, Timeout is set. If it
  171. // then gets cleared before the loop actually exits, Timeout is Cleared.
  172. //
  173. NoWorkStall = 0;
  174. //
  175. // BurstEnable keeps trakc of whethe we think Burst Mode is enabled. If
  176. // Burst Mode gets disabled automatically by the EC, we know that and pretend
  177. // that burst mode is enabled so that the driver can make forward progress.
  178. //
  179. BurstEnabled = FALSE;
  180. //
  181. // ProcessQuery is set if we need to go run some _Qxx methods as the loop exits.
  182. //
  183. ProcessQuery = FALSE;
  184. //
  185. // StallAccumulator counts how many ticks we've stalled for using
  186. // KeStallExecutionProcessor during one complete run of the service loop.
  187. //
  188. StallAccumulator = 0;
  189. EcPrint(EC_TRACE, ("AcpiEcServiceIoLoop.\n"));
  190. InitializeListHead (&CompleteQueue);
  191. //
  192. // Loop while busy
  193. //
  194. for (; ;) {
  195. //
  196. // If there's outgoing data write it, issue the device required
  197. // stall and indicate work is being done (clear noworkstall)
  198. //
  199. if (WritePort) {
  200. EcPrint(EC_IO, ("AcpiEcServiceIO: Write = %x at %x\n", WriteData, WritePort));
  201. AcpiEcLogAction (EcData,
  202. (WritePort == EcData->CommandPort) ?
  203. EC_ACTION_WRITE_CMD : EC_ACTION_WRITE_DATA,
  204. WriteData);
  205. WRITE_PORT_UCHAR (WritePort, WriteData);
  206. KeStallExecutionProcessor (1);
  207. StallAccumulator += 1;
  208. WritePort = NULL;
  209. NoWorkStall = 0; // work was done
  210. }
  211. //
  212. // If work was done, clear pending timeout condition if it exists to
  213. // continue servicing the device
  214. //
  215. if (NoWorkStall == 0 && Timeout) {
  216. Timeout = NULL;
  217. EcBusy = TRUE;
  218. }
  219. //
  220. // If NoWorkStall is non-zero, then no work was performed. Determine
  221. // if the type of delay to issue while waiting (spinning) for the device
  222. //
  223. if (NoWorkStall) {
  224. //
  225. // No work was done the last time around.
  226. // If its time to timeout, exit the service loop.
  227. //
  228. if (Timeout) {
  229. break;
  230. }
  231. //
  232. // If device is idle, setup as if a timeout is occuring. This
  233. // will acquire the device lock, clear the gpe sts bit and terminate
  234. // the service loop (or if the device is now busy, continue)
  235. //
  236. if (!EcBusy) {
  237. if (Status & EC_BURST) {
  238. //
  239. // Before exiting, clear burst mode for embedded controller.
  240. // Has no response, no need to wait for EC to read it.
  241. //
  242. EcPrint (EC_IO, ("AcpiEcServiceIO: Clear Burst mode - Write = %x at %x\n", EC_CANCEL_BURST, EcData->CommandPort));
  243. AcpiEcLogAction (EcData, EC_ACTION_WRITE_CMD, EC_CANCEL_BURST);
  244. WRITE_PORT_UCHAR (EcData->CommandPort, EC_CANCEL_BURST);
  245. Timeout = &EcData->BurstComplete;
  246. } else {
  247. Timeout = &i;
  248. }
  249. } else {
  250. //
  251. // Interject stalls while spinning on device
  252. //
  253. StallAccumulator += NoWorkStall;
  254. KeStallExecutionProcessor (NoWorkStall);
  255. //
  256. // If wait is over the limit, prepare for a timeout.
  257. //
  258. if (!(Status & EC_BURST)) {
  259. if (NoWorkStall >= EcData->MaxNonBurstStall) {
  260. Timeout = &EcData->NonBurstTimeout;
  261. }
  262. } else {
  263. if (NoWorkStall >= EcData->MaxBurstStall) {
  264. Timeout = &EcData->BurstTimeout;
  265. }
  266. }
  267. }
  268. if (Timeout) {
  269. //
  270. // Over time limit, clear the GPE status bit
  271. //
  272. AcpiEcLogAction (EcData, EC_ACTION_CLEAR_GPE, 0);
  273. AcpiInterfaces.GpeClearStatus (AcpiInterfaces.Context,
  274. EcData->GpeVectorObject);
  275. }
  276. }
  277. //
  278. // Increase stall time and indicate no work was done
  279. //
  280. NoWorkStall += 1;
  281. //
  282. // Check Status
  283. //
  284. Status = READ_PORT_UCHAR (EcData->StatusPort);
  285. AcpiEcLogAction (EcData, EC_ACTION_READ_STATUS, Status);
  286. EcPrint(EC_IO, ("AcpiEcServiceIO: Status Read = %x at %x\n", Status, EcData->StatusPort));
  287. //
  288. // Keep bursts dropped by the EC stat
  289. //
  290. if (BurstEnabled && !(Status & EC_BURST)) {
  291. EcData->BurstAborted += 1;
  292. BurstEnabled = FALSE;
  293. Status |= EC_BURST; // move one char
  294. }
  295. //
  296. // If Embedded controller has data for us, process it
  297. //
  298. if (Status & EC_OUTPUT_FULL) {
  299. Data = READ_PORT_UCHAR (EcData->DataPort);
  300. AcpiEcLogAction (EcData, EC_ACTION_READ_DATA, Data);
  301. EcPrint(EC_IO, ("AcpiEcServiceIO: Data Read = %x at %x\n", Data, EcData->DataPort));
  302. switch (EcData->IoState) {
  303. case EC_IO_READ_QUERY:
  304. //
  305. // Response to a read query. Get the query value and set it
  306. //
  307. EcPrint(EC_NOTE, ("AcpiEcServiceIO: Query %x\n", Data));
  308. if (Data) {
  309. //
  310. // If not set, set pending bit
  311. //
  312. KeAcquireSpinLock (&EcData->Lock, &OldIrql);
  313. i = Data / BITS_PER_ULONG;
  314. j = 1 << (Data % BITS_PER_ULONG);
  315. if (!(EcData->QuerySet[i] & j)) {
  316. EcData->QuerySet[i] |= j;
  317. //
  318. // Queue the query or vector operation
  319. //
  320. if (EcData->QueryType[i] & j) {
  321. //
  322. // This is a vector, put it in the vector pending list
  323. //
  324. Data = EcData->QueryMap[Data];
  325. EcData->VectorTable[Data].Next = EcData->VectorHead;
  326. EcData->VectorHead = Data;
  327. } else {
  328. //
  329. // This is a query, put in in the query pending list
  330. //
  331. EcData->QueryMap[Data] = EcData->QueryHead;
  332. EcData->QueryHead = Data;
  333. }
  334. }
  335. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  336. ProcessQuery = TRUE;
  337. }
  338. EcData->IoState = EC_IO_NONE;
  339. break;
  340. case EC_IO_READ_BYTE:
  341. //
  342. // Read transfer. Read the data byte
  343. //
  344. *EcData->IoBuffer = Data;
  345. EcData->IoState = EC_IO_NEXT_BYTE;
  346. break;
  347. case EC_IO_BURST_ACK:
  348. //
  349. // Burst ACK byte
  350. //
  351. EcData->IoState = EcData->IoBurstState;
  352. EcData->IoBurstState = EC_IO_UNKNOWN;
  353. EcData->TotalBursts += 1;
  354. BurstEnabled = TRUE;
  355. break;
  356. default:
  357. EcPrint(EC_ERROR,
  358. ("AcpiEcService: Spurious data received State = %x, Data = %x\n",
  359. EcData->IoState, Data)
  360. );
  361. AcpiEcLogError (EcData, ACPIEC_ERR_SPURIOUS_DATA);
  362. EcData->Errors += 1;
  363. break;
  364. }
  365. NoWorkStall = 0;
  366. continue;
  367. }
  368. if (Status & EC_INPUT_FULL) {
  369. //
  370. // The embedded controllers input buffer is full, wait
  371. //
  372. continue;
  373. }
  374. //
  375. // Embedded controller is ready to receive data, see if anything
  376. // is already being sent
  377. //
  378. switch (EcData->IoState) {
  379. case EC_IO_NEXT_BYTE:
  380. //
  381. // Data transfer.
  382. //
  383. if (EcData->IoRemain) {
  384. if (!(Status & EC_BURST)) {
  385. //
  386. // Not in burst mode. Write burst mode command
  387. //
  388. EcData->IoState = EC_IO_BURST_ACK;
  389. EcData->IoBurstState = EC_IO_NEXT_BYTE;
  390. WritePort = EcData->CommandPort;
  391. WriteData = EC_BURST_TRANSFER;
  392. } else {
  393. //
  394. // Send command to transfer next byte
  395. //
  396. EcData->IoBuffer += 1;
  397. EcData->IoAddress += 1;
  398. EcData->IoRemain -= 1;
  399. EcData->IoState = EC_IO_SEND_ADDRESS;
  400. WritePort = EcData->CommandPort;
  401. WriteData = EcData->IoTransferMode;
  402. }
  403. } else {
  404. //
  405. // Transfer complete
  406. //
  407. EcData->IoState = EC_IO_NONE;
  408. EcData->IoRemain = 0;
  409. Irp = EcData->DeviceObject->CurrentIrp;
  410. EcData->DeviceObject->CurrentIrp = NULL;
  411. Irp->IoStatus.Status = STATUS_SUCCESS;
  412. Irp->IoStatus.Information = EcData->IoLength;
  413. InsertTailList (&CompleteQueue, &Irp->Tail.Overlay.ListEntry);
  414. }
  415. break;
  416. case EC_IO_SEND_ADDRESS:
  417. //
  418. // Send address of transfer request
  419. //
  420. WritePort = EcData->DataPort;
  421. WriteData = EcData->IoAddress;
  422. //
  423. // Wait or send data byte next
  424. //
  425. if (EcData->IoTransferMode == EC_READ_BYTE) {
  426. EcData->IoState = EC_IO_READ_BYTE;
  427. } else {
  428. EcData->IoState = EC_IO_WRITE_BYTE;
  429. }
  430. break;
  431. case EC_IO_WRITE_BYTE:
  432. //
  433. // Write transfer - write the data byte
  434. //
  435. EcData->IoState = EC_IO_NEXT_BYTE;
  436. WritePort = EcData->DataPort;
  437. WriteData = *EcData->IoBuffer;
  438. break;
  439. }
  440. //
  441. // If something to write, loop and handle it
  442. //
  443. if (WritePort) {
  444. continue;
  445. }
  446. //
  447. // If state is NONE, then nothing is pending see what should be
  448. // initiated
  449. //
  450. if (EcData->IoState == EC_IO_NONE) {
  451. EcData->ConsecutiveFailures = 0;
  452. if (Status & EC_QEVT_PENDING) {
  453. //
  454. // Embedded Controller has some sort of event pending
  455. //
  456. EcPrint(EC_QUERY, ("AcpiEcServiceIO: Received Query Request.\n"));
  457. EcData->IoState = EC_IO_READ_QUERY;
  458. WritePort = EcData->CommandPort;
  459. WriteData = EC_QUERY_EVENT;
  460. //
  461. // Reset the watchdog timer
  462. //
  463. KeSetTimer (&EcData->WatchdogTimer,
  464. AcpiEcWatchdogTimeout,
  465. &EcData->WatchdogDpc);
  466. } else {
  467. //
  468. // Get next tranfer from IO queue
  469. //
  470. Link = ExInterlockedRemoveHeadList (&EcData->WorkQueue, &EcData->Lock);
  471. //
  472. // If there's a transfer initiate it
  473. //
  474. if (Link) {
  475. EcPrint(EC_HANDLER, ("AcpiEcServiceIO: Got next work item %x\n", Link));
  476. Irp = CONTAINING_RECORD (
  477. Link,
  478. IRP,
  479. Tail.Overlay.ListEntry
  480. );
  481. IrpSp = IoGetCurrentIrpStackLocation(Irp);
  482. EcData->DeviceObject->CurrentIrp = Irp;
  483. EcData->IoBuffer = Irp->AssociatedIrp.SystemBuffer;
  484. EcData->IoAddress = (UCHAR) IrpSp->Parameters.Read.ByteOffset.LowPart;
  485. EcData->IoLength = (UCHAR) IrpSp->Parameters.Read.Length;
  486. EcData->IoTransferMode =
  487. IrpSp->MajorFunction == IRP_MJ_READ ? EC_READ_BYTE : EC_WRITE_BYTE;
  488. //
  489. // Set it up via EC_IO_NEXT_BYTE and back up one byte
  490. //
  491. EcData->IoBuffer -= 1;
  492. EcData->IoAddress -= 1;
  493. EcData->IoRemain = EcData->IoLength;
  494. EcData->IoState = EC_IO_NEXT_BYTE;
  495. NoWorkStall = 0;
  496. //
  497. // Reset the watchdog timer
  498. //
  499. KeSetTimer (&EcData->WatchdogTimer,
  500. AcpiEcWatchdogTimeout,
  501. &EcData->WatchdogDpc);
  502. } else {
  503. //
  504. // Nothing but nothing to do.
  505. //
  506. EcBusy = FALSE;
  507. //
  508. // Clear the Watchdog timer
  509. //
  510. KeCancelTimer (&EcData->WatchdogTimer);
  511. }
  512. }
  513. }
  514. }
  515. //
  516. // Keep stat as to why service loop terminated
  517. //
  518. *Timeout += 1;
  519. //
  520. // Track maximum service loop stall accumulator
  521. //
  522. if (StallAccumulator > EcData->MaxServiceLoop) {
  523. EcData->MaxServiceLoop = StallAccumulator;
  524. }
  525. //
  526. // Complete processed io requests
  527. //
  528. while (!IsListEmpty(&CompleteQueue)) {
  529. Link = RemoveHeadList(&CompleteQueue);
  530. Irp = CONTAINING_RECORD (
  531. Link,
  532. IRP,
  533. Tail.Overlay.ListEntry
  534. );
  535. EcPrint(EC_IO, ("AcpiEcServiceIO: IOComplete: Irp=%Lx\n", Irp));
  536. #if DEBUG
  537. if (ECDebug && EC_TRANSACTION) {
  538. IrpSp = IoGetCurrentIrpStackLocation(Irp);
  539. if (IrpSp->MajorFunction == IRP_MJ_READ) {
  540. EcPrint (EC_TRANSACTION, ("AcpiEcServiceIO: Read ("));
  541. for (i=0; i < IrpSp->Parameters.Read.Length; i++) {
  542. EcPrint (EC_TRANSACTION, ("%02x ",
  543. ((PUCHAR)Irp->AssociatedIrp.SystemBuffer) [i]));
  544. }
  545. EcPrint (EC_TRANSACTION, (") from %02x length %02x\n",
  546. (UCHAR)IrpSp->Parameters.Read.ByteOffset.LowPart,
  547. (UCHAR)IrpSp->Parameters.Read.Length));
  548. }
  549. }
  550. #endif
  551. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  552. }
  553. //
  554. // If queries occured, dispatch them
  555. //
  556. if (ProcessQuery) {
  557. AcpiEcDispatchQueries (EcData);
  558. }
  559. }
  560. VOID
  561. AcpiEcWatchdogDpc(
  562. IN PKDPC Dpc,
  563. IN PECDATA EcData,
  564. IN PVOID SystemArgument1,
  565. IN PVOID SystemArgument2
  566. )
  567. /*++
  568. Routine Description:
  569. Gets called if EC doesn't respond within 5 seconds of request.
  570. Arguments:
  571. EcData - Pointer to embedded controller to service.
  572. Return Value:
  573. None.
  574. --*/
  575. {
  576. UCHAR ecStatus;
  577. PIRP Irp;
  578. KIRQL OldIrql;
  579. #if DEBUG
  580. UCHAR i;
  581. #endif
  582. ecStatus = READ_PORT_UCHAR (EcData->StatusPort);
  583. AcpiEcLogAction (EcData, EC_ACTION_READ_STATUS, ecStatus);
  584. if (EcData->ConsecutiveFailures < 255) {
  585. EcData->ConsecutiveFailures++;
  586. }
  587. if (EcData->ConsecutiveFailures <= 5) {
  588. //
  589. // Only log an error for the first 5 consecutive failures. After that just be quiet about it.
  590. //
  591. AcpiEcLogError(EcData, ACPIEC_ERR_WATCHDOG);
  592. }
  593. EcPrint (EC_ERROR, ("AcpiEcWatchdogDpc: EC error encountered. \nAction History:\n"
  594. " D time IoState Action Data\n"
  595. " *%3dns\n", (ULONG)(1000000000/EcData->PerformanceFrequency.QuadPart)));
  596. #if DEBUG
  597. i = EcData->LastAction;
  598. do {
  599. i++;
  600. i &= ACPIEC_ACTION_COUNT_MASK;
  601. if ((EcData->RecentActions[i].IoStateAction & EC_ACTION_MASK) == 0) {
  602. continue;
  603. }
  604. EcPrint (EC_ERROR, (" %04x %01x %s 0x%02x\n",
  605. EcData->RecentActions[i].Time,
  606. EcData->RecentActions[i].IoStateAction & ~EC_ACTION_MASK,
  607. (EcData->RecentActions[i].IoStateAction & EC_ACTION_MASK) < EC_ACTION_MAX ?
  608. AcpiEcActionDescription [(EcData->RecentActions[i].IoStateAction & EC_ACTION_MASK) >> 4] : "",
  609. EcData->RecentActions[i].Data));
  610. } while (i != EcData->LastAction);
  611. #endif
  612. KeAcquireSpinLock (&EcData->Lock, &OldIrql);
  613. if (EcData->InService) {
  614. //
  615. // This is not likely to happen.
  616. // If the service loop is running, this should exit.
  617. // Reset the Watchdog Timer. This may be set again or canceld by the service loop
  618. //
  619. KeSetTimer (&EcData->WatchdogTimer,
  620. AcpiEcWatchdogTimeout,
  621. &EcData->WatchdogDpc);
  622. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  623. return;
  624. }
  625. //
  626. // Hold Spinlock throughout so we can guatantee there won't be a conflict in the IO queue.
  627. //
  628. EcData->InService = TRUE;
  629. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  630. switch (EcData->IoState) {
  631. case EC_IO_NONE:
  632. //
  633. // This shouldn't happen. The watchdog should be shut off if the
  634. // driver isn't busy.
  635. //
  636. break;
  637. case EC_IO_READ_BYTE:
  638. case EC_IO_BURST_ACK:
  639. if (ecStatus & EC_OUTPUT_FULL) {
  640. //
  641. // EC appears to be ready. Log an error and continue.
  642. //
  643. } else {
  644. //
  645. // If the embedded controller is not ready yet, somthing went wrong.
  646. // Retry the transaction.
  647. //
  648. if (EcData->IoState == EC_IO_READ_BYTE) {
  649. EcData->IoBuffer -= 1;
  650. EcData->IoAddress -= 1;
  651. EcData->IoRemain += 1;
  652. }
  653. EcData->IoState = EC_IO_NEXT_BYTE;
  654. }
  655. break;
  656. case EC_IO_READ_QUERY:
  657. if (ecStatus & EC_OUTPUT_FULL) {
  658. //
  659. // EC appears to be ready. Log an error and continue.
  660. //
  661. } else {
  662. //
  663. // If the embedded controller is not ready yet, somthing went wrong.
  664. // This could mean that the query was lost. If the Query bit is still set,
  665. // the driver will retry automatically.
  666. //
  667. EcData->IoState = EC_IO_NONE;
  668. }
  669. break;
  670. case EC_IO_WRITE_BYTE:
  671. case EC_IO_SEND_ADDRESS:
  672. //
  673. // This is just waiting for IBF==0. If it took this long, chances are the
  674. // state was lost. Retry the transaction.
  675. //
  676. EcData->IoBuffer -= 1;
  677. EcData->IoAddress -= 1;
  678. EcData->IoRemain += 1;
  679. EcData->IoState = EC_IO_NEXT_BYTE;
  680. break;
  681. case EC_IO_NEXT_BYTE:
  682. //
  683. // This could happen if IBF is still set
  684. //
  685. if (ecStatus & EC_INPUT_FULL) {
  686. //
  687. // Try thwaking it to see if it will wake up.
  688. //
  689. EcPrint (EC_IO, ("AcpiEcWatchDog: Clear Burst mode - Write = %x at %x\n", EC_CANCEL_BURST, EcData->CommandPort));
  690. AcpiEcLogAction (EcData, EC_ACTION_WRITE_CMD, EC_CANCEL_BURST);
  691. WRITE_PORT_UCHAR (EcData->CommandPort, EC_CANCEL_BURST);
  692. }
  693. }
  694. KeAcquireSpinLock (&EcData->Lock, &OldIrql);
  695. EcData->InService = FALSE;
  696. KeReleaseSpinLock (&EcData->Lock, OldIrql);
  697. //
  698. // Set the timer. The ServiceIoLoop won't reset the timer, until some forward progress is made.
  699. //
  700. KeSetTimer (&EcData->WatchdogTimer,
  701. AcpiEcWatchdogTimeout,
  702. &EcData->WatchdogDpc);
  703. AcpiEcServiceDevice(EcData);
  704. }
  705. VOID
  706. AcpiEcLogAction (
  707. PECDATA EcData,
  708. UCHAR Action,
  709. UCHAR Data
  710. )
  711. {
  712. UCHAR i, j;
  713. LARGE_INTEGER time, temp;
  714. i = EcData->LastAction;
  715. j = (i-1)&ACPIEC_ACTION_COUNT_MASK;
  716. if ( ((EcData->RecentActions [i].IoStateAction & EC_ACTION_MASK) == EC_ACTION_REPEATED) &&
  717. (EcData->RecentActions [j].IoStateAction == (EcData->IoState | Action)) &&
  718. (EcData->RecentActions [j].Data == Data)) {
  719. //
  720. // If we already have a repeated action, increment the count on the repeated action
  721. // then update the time on the latest one. We only care about the time of the first and last one.
  722. //
  723. EcData->RecentActions [i].Data++;
  724. if (EcData->RecentActions [i].Data == 0) {
  725. //
  726. // If we've logged 255 repeats, don't roll over to 0.
  727. //
  728. EcData->RecentActions [i].Data = 0xff;
  729. }
  730. time = KeQueryPerformanceCounter (NULL);
  731. temp.QuadPart = time.QuadPart - AcpiEcLastActionTime.QuadPart;
  732. temp.QuadPart = temp.QuadPart + EcData->RecentActions[i].Time;
  733. if (temp.QuadPart > ((USHORT) -1)) {
  734. temp.QuadPart = (USHORT) -1;
  735. }
  736. EcData->RecentActions[i].Time = (USHORT) temp.LowPart;
  737. } else if ((EcData->RecentActions [i].IoStateAction == (EcData->IoState | Action)) &&
  738. (EcData->RecentActions [i].Data == Data)) {
  739. //
  740. // This is the same action as the last one. list as a repeated action
  741. //
  742. EcData->LastAction++;
  743. EcData->LastAction &= ACPIEC_ACTION_COUNT_MASK;
  744. EcData->RecentActions[EcData->LastAction].Data = 1;
  745. time = KeQueryPerformanceCounter (NULL);
  746. temp.QuadPart = time.QuadPart - AcpiEcLastActionTime.QuadPart;
  747. if (temp.QuadPart > ((USHORT) -1)) {
  748. temp.QuadPart = (USHORT) -1;
  749. }
  750. EcData->RecentActions[EcData->LastAction].Time = (USHORT) temp.LowPart;
  751. AcpiEcLastActionTime = time;
  752. // Set this last since it is the key to saying that an entry is complete.
  753. EcData->RecentActions[EcData->LastAction].IoStateAction = EC_ACTION_REPEATED | EcData->IoState;
  754. } else {
  755. EcData->LastAction++;
  756. EcData->LastAction &= ACPIEC_ACTION_COUNT_MASK;
  757. EcData->RecentActions[EcData->LastAction].Data = Data;
  758. time = KeQueryPerformanceCounter (NULL);
  759. temp.QuadPart = time.QuadPart - AcpiEcLastActionTime.QuadPart;
  760. if (temp.QuadPart > ((USHORT) -1)) {
  761. temp.QuadPart = (USHORT) -1;
  762. }
  763. EcData->RecentActions[EcData->LastAction].Time = (USHORT) temp.LowPart;
  764. AcpiEcLastActionTime = time;
  765. // Set this last since it is the key to saying that an entry is complete.
  766. EcData->RecentActions[EcData->LastAction].IoStateAction = Action | EcData->IoState;
  767. }
  768. }
  769. VOID
  770. AcpiEcLogError (
  771. PECDATA EcData,
  772. NTSTATUS ErrCode
  773. )
  774. {
  775. PIO_ERROR_LOG_PACKET logEntry = NULL;
  776. PACPIEC_ACTION action;
  777. ULONG size;
  778. UCHAR i;
  779. logEntry = IoAllocateErrorLogEntry(EcData->DeviceObject,
  780. ERROR_LOG_MAXIMUM_SIZE);
  781. if (!logEntry) {
  782. EcPrint (EC_ERROR, ("AcpiEcLogError: Couldn't write error to errorlog\n"));
  783. return;
  784. }
  785. RtlZeroMemory(logEntry, ERROR_LOG_MAXIMUM_SIZE);
  786. //
  787. // Fill out the packet
  788. //
  789. logEntry->DumpDataSize = (USHORT) ERROR_LOG_MAXIMUM_SIZE - sizeof(IO_ERROR_LOG_PACKET);
  790. logEntry->NumberOfStrings = 0;
  791. logEntry->ErrorCode = ErrCode;
  792. //
  793. // Fill in data
  794. //
  795. logEntry->DumpData[0] = EcData->PerformanceFrequency.LowPart;
  796. action = (PACPIEC_ACTION) (&logEntry->DumpData[1]);
  797. size = sizeof(IO_ERROR_LOG_PACKET) + sizeof(logEntry->DumpData[0]) + sizeof(ACPIEC_ACTION);
  798. i = EcData->LastAction;
  799. while (size <= ERROR_LOG_MAXIMUM_SIZE) {
  800. RtlCopyMemory (action, &EcData->RecentActions[i], sizeof(ACPIEC_ACTION));
  801. i--;
  802. i &= ACPIEC_ACTION_COUNT_MASK;
  803. if (i == EcData->LastAction) {
  804. break;
  805. }
  806. action++;
  807. size += sizeof(ACPIEC_ACTION);
  808. }
  809. //
  810. // Submit error log packet
  811. //
  812. IoWriteErrorLogEntry(logEntry);
  813. }