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.

1808 lines
44 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. thermal.c
  5. Abstract:
  6. Thermal Zone support
  7. A small discourse on the use and function of the THRM_WAIT_FOR_NOTIFY
  8. flag. This flag was added to ensure that at least one Notify() operation
  9. occured between each query of the temperature. In other words, we didn't
  10. want to loop forever asking and receiving the same temperature information.
  11. One of the side effects of this flag is that if we get a QUERY, then
  12. a SET (instead of another QUERY), then the set must clear the flag.
  13. Failure to do so will prevent the ThermalLoop() code from ever completing
  14. the IRP. And that means that the temperature mechanisms will stop working.
  15. Author:
  16. Stephane Plante (splante)
  17. Environment:
  18. NT Kernel Model Driver only
  19. Revision History:
  20. July 7, 1997 - Complete Rewrite
  21. --*/
  22. #include "pch.h"
  23. WMIGUIDREGINFO ACPIThermalGuidList =
  24. {
  25. &THERMAL_ZONE_GUID,
  26. 1,
  27. 0
  28. };
  29. //
  30. // Spinlock to protect the thermal list
  31. //
  32. KSPIN_LOCK AcpiThermalLock;
  33. //
  34. // List entry to store the thermal requests on
  35. //
  36. LIST_ENTRY AcpiThermalList;
  37. #ifdef ALLOC_PRAGMA
  38. #pragma alloc_text(PAGE, ACPIThermalStartDevice)
  39. #pragma alloc_text(PAGE, ACPIThermalWorker)
  40. #pragma alloc_text(PAGE, ACPIThermalQueryWmiRegInfo)
  41. #pragma alloc_text(PAGE, ACPIThermalQueryWmiDataBlock)
  42. #pragma alloc_text(PAGE, ACPIThermalWmi)
  43. #endif
  44. VOID
  45. ACPIThermalCalculateProcessorMask(
  46. IN PNSOBJ ProcessorObject,
  47. IN PTHRM_INFO Thrm
  48. )
  49. /*++
  50. Routine Description:
  51. This routine, which is only called from ACPIThermalWorker, has been
  52. created so that we don't have to worry about locking down the
  53. ACPIThermalWorker, takes a processor object from the namespace and
  54. sets the proper affinity bit in the thermal info
  55. Arguments:
  56. ProcessorObject - Pointer to Namespace Processor Object
  57. Thrm - Thermal Information Structure
  58. Return Value:
  59. None
  60. --*/
  61. {
  62. KIRQL OldIrql;
  63. PDEVICE_EXTENSION ProcessorExtension;
  64. //
  65. // Sanity check
  66. //
  67. if (ProcessorObject == NULL) {
  68. return;
  69. }
  70. //
  71. // We need the spinlock to deref the processor extension
  72. //
  73. KeAcquireSpinLock( &AcpiDeviceTreeLock, &OldIrql );
  74. //
  75. // The context pointer is the device extension
  76. //
  77. ProcessorExtension = (PDEVICE_EXTENSION) ProcessorObject->Context;
  78. if (ProcessorExtension) {
  79. //
  80. // We know what index it is within the processor list.
  81. // This should be a good enough guess for now
  82. //
  83. Thrm->Info.Processors |= (1 << ProcessorExtension->Processor.ProcessorIndex);
  84. }
  85. //
  86. // Done with the spinlock
  87. //
  88. KeReleaseSpinLock( &AcpiDeviceTreeLock, OldIrql );
  89. }
  90. VOID
  91. ACPIThermalCancelRequest (
  92. IN PDEVICE_OBJECT DeviceObject,
  93. IN PIRP Irp
  94. )
  95. /*++
  96. Routine Description:
  97. This routine cancels an outstanding thermal request
  98. Arguments
  99. DeviceObject - The device which has a request being cancelled
  100. Irp - The cancelling irp
  101. Return Value:
  102. None
  103. --*/
  104. {
  105. KIRQL oldIrql;
  106. PDEVICE_EXTENSION deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  107. #if DBG
  108. ULONGLONG currentTime = KeQueryInterruptTime();
  109. ACPIThermalPrint( (
  110. ACPI_PRINT_THERMAL,
  111. deviceExtension,
  112. currentTime,
  113. "ACPIThermalCancelRequest: Irp %08lx\n",
  114. Irp
  115. ) );
  116. #endif
  117. //
  118. // We no longer need the cancel lock
  119. //
  120. IoReleaseCancelSpinLock (Irp->CancelIrql);
  121. //
  122. // We do however need the thermal queue lock
  123. //
  124. KeAcquireSpinLock( &AcpiThermalLock, &oldIrql );
  125. //
  126. // Remove the irp from the list that it is on
  127. //
  128. RemoveEntryList( &(Irp->Tail.Overlay.ListEntry) );
  129. //
  130. // Done with the thermal lock now
  131. //
  132. KeReleaseSpinLock( &AcpiThermalLock, oldIrql );
  133. //
  134. // Complete the irp now
  135. //
  136. Irp->IoStatus.Status = STATUS_CANCELLED;
  137. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  138. }
  139. VOID
  140. EXPORT
  141. ACPIThermalComplete (
  142. IN PNSOBJ AcpiObject,
  143. IN NTSTATUS Status,
  144. IN POBJDATA Result OPTIONAL,
  145. IN PVOID DeviceExtension
  146. )
  147. /*++
  148. Routine Description:
  149. This routine is called when the interpreter has completed a request
  150. Arguments:
  151. AcpiObject - The request that was completed
  152. Status - The status of the request
  153. Result - What the result of the request was
  154. DevExt - The context of the request
  155. Return Value:
  156. NONE
  157. --*/
  158. {
  159. ACPIThermalLoop (DeviceExtension, THRM_BUSY);
  160. }
  161. BOOLEAN
  162. ACPIThermalCompletePendingIrps(
  163. IN PDEVICE_EXTENSION DeviceExtension,
  164. IN PTHRM_INFO Thermal
  165. )
  166. /*++
  167. Routine Description:
  168. This routine is called, with no spinlock being held. This routine
  169. completes any IOCTLs associated with the device object
  170. This routine will return TRUE if it completed any requests, false
  171. otherwise
  172. Arguments:
  173. DeviceExtension - The device extension whose requests we want to complete
  174. Thermal - Pointer to the thermal information for the extension
  175. Return Value:
  176. BOOLEAN
  177. --*/
  178. {
  179. BOOLEAN handledRequest = FALSE;
  180. KIRQL oldIrql;
  181. LIST_ENTRY doneList;
  182. PDEVICE_EXTENSION irpExtension;
  183. PDEVICE_OBJECT deviceObject;
  184. PIO_STACK_LOCATION irpSp;
  185. PIRP irp;
  186. PLIST_ENTRY listEntry;
  187. PTHERMAL_INFORMATION thermalInfo;
  188. //
  189. // Initialize the list that will hold the requets that we need to complete
  190. //
  191. InitializeListHead( &doneList );
  192. //
  193. // Acquire the thermal lock so that we can pend these requests
  194. //
  195. KeAcquireSpinLock( &AcpiThermalLock, &oldIrql );
  196. //
  197. // Walk the list of pending irps to see which ones match this extensions
  198. //
  199. listEntry = AcpiThermalList.Flink;
  200. while (listEntry != &AcpiThermalList) {
  201. //
  202. // Grab the irp from the list entry and update the next list entry
  203. // that we will look at
  204. //
  205. irp = CONTAINING_RECORD( listEntry, IRP, Tail.Overlay.ListEntry );
  206. listEntry = listEntry->Flink;
  207. //
  208. // We need the current irp stack location
  209. //
  210. irpSp = IoGetCurrentIrpStackLocation( irp );
  211. //
  212. // Grab the device object from the irp stack and turn that into a
  213. // device extension
  214. //
  215. irpExtension = ACPIInternalGetDeviceExtension( irpSp->DeviceObject );
  216. //
  217. // Is this an irp that we care about? IE: does the target match the
  218. // one specified in this function
  219. //
  220. if (irpExtension != DeviceExtension) {
  221. continue;
  222. }
  223. //
  224. // If this is a query information irp then, we must be able to set the
  225. // cancel routine to NULL to make sure that it cannot be cancelled on
  226. // us
  227. //
  228. if (irpSp->Parameters.DeviceIoControl.IoControlCode == IOCTL_THERMAL_QUERY_INFORMATION) {
  229. if (IoSetCancelRoutine(irp, NULL) == NULL) {
  230. //
  231. // Cancel routine is active, stop processing this irp and move on
  232. //
  233. continue;
  234. }
  235. //
  236. // Copy the data that we got back to the irp
  237. //
  238. DeviceExtension->Thermal.Flags |= THRM_WAIT_FOR_NOTIFY;
  239. thermalInfo = (PTHERMAL_INFORMATION) irp->AssociatedIrp.SystemBuffer;
  240. memcpy (thermalInfo, Thermal, sizeof (THERMAL_INFORMATION));
  241. //
  242. // Set the parameters that we will return
  243. //
  244. irp->IoStatus.Information = sizeof(THERMAL_INFORMATION);
  245. } else {
  246. //
  247. // Set the parameters that we will return
  248. //
  249. irp->IoStatus.Information = 0;
  250. }
  251. //
  252. // Always succeed these irps
  253. //
  254. irp->IoStatus.Status = STATUS_SUCCESS;
  255. //
  256. // Remove the entry from the list
  257. //
  258. RemoveEntryList( &(irp->Tail.Overlay.ListEntry) );
  259. //
  260. // Insert the list into the next queue so that we know to complete it
  261. // later on
  262. //
  263. InsertTailList( &doneList, &(irp->Tail.Overlay.ListEntry) );
  264. }
  265. //
  266. // At this point, drop our thermal lock
  267. //
  268. KeReleaseSpinLock( &AcpiThermalLock, oldIrql );
  269. //
  270. // Walk the list of irpts to be completed
  271. //
  272. listEntry = doneList.Flink;
  273. while (listEntry != &doneList) {
  274. //
  275. // Grab the irp from the list entry and update the next list entry
  276. // that we will look at
  277. //
  278. irp = CONTAINING_RECORD( listEntry, IRP, Tail.Overlay.ListEntry );
  279. listEntry = listEntry->Flink;
  280. RemoveEntryList( &(irp->Tail.Overlay.ListEntry) );
  281. ACPIThermalPrint( (
  282. ACPI_PRINT_THERMAL,
  283. DeviceExtension,
  284. KeQueryInterruptTime(),
  285. "Completing Irp 0x%x\n",
  286. irp
  287. ) );
  288. //
  289. // Now complete the irp
  290. //
  291. IoCompleteRequest( irp, IO_NO_INCREMENT );
  292. //
  293. // Remember that we handled a request
  294. //
  295. handledRequest = TRUE;
  296. }
  297. //
  298. // Return wether or not we handled a request
  299. return handledRequest;
  300. }
  301. NTSTATUS
  302. ACPIThermalDeviceControl (
  303. IN PDEVICE_OBJECT DeviceObject,
  304. IN PIRP Irp
  305. )
  306. /*++
  307. Routine Description:
  308. Fixed button device IOCTL handler
  309. Arguments:
  310. DeviceObject - fixed feature button device object
  311. Irp - the ioctl request
  312. Return Value:
  313. NTSTATUS
  314. --*/
  315. {
  316. KIRQL oldIrql;
  317. PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation(Irp);
  318. PDEVICE_EXTENSION deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  319. PTHERMAL_INFORMATION thermalInfo;
  320. PULONG Mode;
  321. PTHRM_INFO Thrm = deviceExtension->Thermal.Info;
  322. NTSTATUS Status = STATUS_PENDING;
  323. ULONG ThermalWork = 0;
  324. ULONGLONG currentTime;
  325. //
  326. // Do not allow user mode IRPs in this routine
  327. //
  328. if (Irp->RequestorMode != KernelMode) {
  329. return ACPIDispatchIrpInvalid( DeviceObject, Irp );
  330. }
  331. #if DBG
  332. currentTime = KeQueryInterruptTime();
  333. #endif
  334. switch (IrpSp->Parameters.DeviceIoControl.IoControlCode) {
  335. case IOCTL_THERMAL_QUERY_INFORMATION:
  336. //
  337. // If this irp's stamp does not match the known last stamp, then we
  338. // need a new temp now
  339. //
  340. thermalInfo = (PTHERMAL_INFORMATION) Irp->AssociatedIrp.SystemBuffer;
  341. if (thermalInfo->ThermalStamp != Thrm->Info.ThermalStamp) {
  342. ThermalWork = THRM_TEMP | THRM_WAIT_FOR_NOTIFY;
  343. }
  344. #if DBG
  345. ACPIThermalPrint( (
  346. ACPI_PRINT_THERMAL,
  347. deviceExtension,
  348. currentTime,
  349. "%08x - THERMAL_QUERY_INFORMATION: %x - %x\n",
  350. Irp,
  351. thermalInfo->ThermalStamp,
  352. Thrm->Info.ThermalStamp
  353. ) );
  354. #endif
  355. break;
  356. case IOCTL_THERMAL_SET_COOLING_POLICY:
  357. //
  358. // Set the thermal zone's policy mode
  359. //
  360. Thrm->Mode = *((PUCHAR) Irp->AssociatedIrp.SystemBuffer);
  361. ThermalWork = THRM_MODE | THRM_TRIP_POINTS | THRM_WAIT_FOR_NOTIFY;
  362. #if DBG
  363. ACPIThermalPrint( (
  364. ACPI_PRINT_THERMAL,
  365. deviceExtension,
  366. currentTime,
  367. "%08x - SET_COOLING_POLICY: %x\n",
  368. Irp,
  369. Thrm->Mode
  370. ) );
  371. #endif
  372. break;
  373. case IOCTL_RUN_ACTIVE_COOLING_METHOD:
  374. Thrm->CoolingLevel = *((PUCHAR) Irp->AssociatedIrp.SystemBuffer);
  375. ThermalWork = THRM_COOLING_LEVEL | THRM_WAIT_FOR_NOTIFY;
  376. #if DBG
  377. ACPIThermalPrint( (
  378. ACPI_PRINT_THERMAL,
  379. deviceExtension,
  380. currentTime,
  381. "%08x - ACTIVE_COOLING_LEVEL: %x\n",
  382. Irp,
  383. Thrm->CoolingLevel
  384. ) );
  385. #endif
  386. break;
  387. default:
  388. Irp->IoStatus.Information = 0;
  389. Irp->IoStatus.Status = STATUS_NOT_SUPPORTED;
  390. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  391. return STATUS_NOT_SUPPORTED;
  392. }
  393. //
  394. // Grab the thermal lock, queue the request to the proper place, and make
  395. // sure to set a cancel routine --- no that we will only allow a cancel
  396. // routine if this is a query irp
  397. //
  398. KeAcquireSpinLock( &AcpiThermalLock, &oldIrql );
  399. //
  400. // There is one fly in the ointment: What to do if the device is no longer
  401. // there. The only way to really handle that is to just fail the request.
  402. // Its important to note that this check is done while the ThermalLock
  403. // is being held because the code that builds a SurpriseRemoved extension
  404. // will attempt to call AcpiThermalCompletePendingIrps which also
  405. // acquires this lock.
  406. //
  407. if (deviceExtension->Flags & DEV_TYPE_SURPRISE_REMOVED) {
  408. KeReleaseSpinLock( &AcpiThermalLock, oldIrql );
  409. Irp->IoStatus.Status = STATUS_NO_SUCH_DEVICE;
  410. Irp->IoStatus.Information = 0;
  411. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  412. return STATUS_NO_SUCH_DEVICE;
  413. }
  414. if (IrpSp->Parameters.DeviceIoControl.IoControlCode == IOCTL_THERMAL_QUERY_INFORMATION) {
  415. IoSetCancelRoutine (Irp, ACPIThermalCancelRequest);
  416. if (Irp->Cancel && IoSetCancelRoutine( Irp, NULL ) ) {
  417. //
  418. // If we got here, that means that the irp has been cancelled and we
  419. // beat the IO manager to the ThermalLock. So release the irp, and
  420. // mark the irp as being cancelled
  421. //
  422. KeReleaseSpinLock( &AcpiThermalLock, oldIrql );
  423. Irp->IoStatus.Information = 0;
  424. Irp->IoStatus.Status = STATUS_CANCELLED;
  425. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  426. return STATUS_CANCELLED;
  427. }
  428. //
  429. // If we got to this point, we are going to queue the request and do some
  430. // work on it later
  431. //
  432. IoMarkIrpPending( Irp );
  433. }
  434. //
  435. // If we got here, we know we can queue the irp in the outstanding
  436. // work list entry
  437. //
  438. InsertTailList( &AcpiThermalList, &(Irp->Tail.Overlay.ListEntry) );
  439. //
  440. // Done with the lock at this point
  441. //
  442. KeReleaseSpinLock( &AcpiThermalLock, oldIrql );
  443. //
  444. // Fire off the workter thread
  445. //
  446. ACPIThermalLoop (deviceExtension, ThermalWork);
  447. return Status;
  448. }
  449. VOID
  450. ACPIThermalEvent (
  451. IN PDEVICE_OBJECT DeviceObject,
  452. IN ULONG EventData
  453. )
  454. /*++
  455. Routine Description:
  456. This routine handles thermal events
  457. Arguments:
  458. DeviceObject - The device that received the event
  459. EventData - The event that just happened
  460. Return Value:
  461. NONE
  462. --*/
  463. {
  464. PDEVICE_EXTENSION deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  465. ULONG clear;
  466. ACPIThermalPrint( (
  467. ACPI_PRINT_THERMAL,
  468. deviceExtension,
  469. KeQueryInterruptTime(),
  470. "ACPIThermalEvent - Notify(%x)\n",
  471. EventData
  472. ) );
  473. //
  474. // Handle event type
  475. //
  476. clear = 0;
  477. switch (EventData) {
  478. case 0x80:
  479. //
  480. // Tempature changed notification
  481. //
  482. clear = THRM_WAIT_FOR_NOTIFY | THRM_TEMP;
  483. break;
  484. case 0x81:
  485. //
  486. // Trips points changed notification
  487. //
  488. clear = THRM_WAIT_FOR_NOTIFY | THRM_TEMP | THRM_TRIP_POINTS;
  489. break;
  490. default:
  491. break;
  492. }
  493. ACPIThermalLoop (deviceExtension, clear);
  494. }
  495. NTSTATUS
  496. ACPIThermalFanStartDevice(
  497. IN PDEVICE_OBJECT DeviceObject,
  498. IN PIRP Irp
  499. )
  500. /*++
  501. Routine Description:
  502. This is the routine that processes the start device for the fans
  503. Arguments:
  504. DeviceObject - The fan device
  505. Irp - The start request
  506. Return Value:
  507. NTSTATUS
  508. --*/
  509. {
  510. PDEVICE_EXTENSION deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  511. PIO_STACK_LOCATION irpStack = IoGetCurrentIrpStackLocation( Irp );
  512. UCHAR minorFunction = irpStack->MinorFunction;
  513. //
  514. // There is nothing to do when starting a fan --- it is really being
  515. // controlled by the thermal zones
  516. //
  517. deviceExtension->DeviceState = Started;
  518. //
  519. // Complete the request
  520. //
  521. Irp->IoStatus.Status = STATUS_SUCCESS;
  522. Irp->IoStatus.Information = (ULONG_PTR) NULL;
  523. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  524. //
  525. // Let the world know
  526. //
  527. ACPIDevPrint( (
  528. ACPI_PRINT_THERMAL,
  529. deviceExtension,
  530. "(0x%08lx): %s = 0x%08lx\n",
  531. Irp,
  532. ACPIDebugGetIrpText(IRP_MJ_PNP, minorFunction),
  533. STATUS_SUCCESS
  534. ) );
  535. //
  536. // Done
  537. //
  538. return STATUS_SUCCESS;
  539. }
  540. VOID
  541. ACPIThermalLoop (
  542. IN PDEVICE_EXTENSION DeviceExtension,
  543. IN ULONG Clear
  544. )
  545. /*++
  546. Routine Description:
  547. This is the routine that processes all thermal events
  548. Arguments:
  549. DevExt - The device extension of the thermal zone
  550. Clear - Bits to clear
  551. Return Value:
  552. NTSTATUS
  553. --*/
  554. {
  555. BOOLEAN doneRequests;
  556. BOOLEAN lockHeld;
  557. KIRQL oldIrql;
  558. PTHRM_INFO thermal;
  559. NTSTATUS status;
  560. thermal = DeviceExtension->Thermal.Info;
  561. KeAcquireSpinLock (&DeviceExtension->Thermal.SpinLock, &oldIrql);
  562. lockHeld = TRUE;
  563. DeviceExtension->Thermal.Flags &= ~Clear;
  564. //
  565. // If we're not in the service loop, enter it now
  566. //
  567. if (!(DeviceExtension->Thermal.Flags & THRM_IN_SERVICE_LOOP)) {
  568. DeviceExtension->Thermal.Flags |= THRM_IN_SERVICE_LOOP;
  569. //
  570. // Loop while there's work to be done
  571. //
  572. for (; ;) {
  573. //
  574. // Synchronize the thermal zone
  575. //
  576. if (!lockHeld) {
  577. KeAcquireSpinLock(&DeviceExtension->Thermal.SpinLock, &oldIrql);
  578. lockHeld = TRUE;
  579. }
  580. //
  581. // If some work is pending, wait for it to complete
  582. //
  583. if (DeviceExtension->Thermal.Flags & THRM_BUSY) {
  584. break;
  585. }
  586. //
  587. // Make sure that the thermal zone is initialized. This must
  588. // be the first thing that we do in the loop!!!
  589. //
  590. if (!(DeviceExtension->Thermal.Flags & THRM_INITIALIZE) ) {
  591. DeviceExtension->Thermal.Flags |= THRM_BUSY | THRM_INITIALIZE;
  592. ACPISetDeviceWorker(
  593. DeviceExtension,
  594. THRM_COOLING_LEVEL | THRM_INITIALIZE
  595. );
  596. continue;
  597. }
  598. //
  599. // If the thermal zone mode needs updated, do it now
  600. //
  601. if (!(DeviceExtension->Thermal.Flags & THRM_MODE)) {
  602. DeviceExtension->Thermal.Flags |= THRM_BUSY | THRM_MODE;
  603. KeReleaseSpinLock (&DeviceExtension->Thermal.SpinLock, oldIrql);
  604. lockHeld = FALSE;
  605. status = ACPIGetNothingEvalIntegerAsync(
  606. DeviceExtension,
  607. PACKED_SCP,
  608. thermal->Mode,
  609. ACPIThermalComplete,
  610. DeviceExtension
  611. );
  612. if (status != STATUS_PENDING) {
  613. ACPIThermalComplete(
  614. NULL,
  615. status,
  616. NULL,
  617. DeviceExtension
  618. );
  619. }
  620. continue;
  621. }
  622. //
  623. // If the trip point infomation needs updated, get it. Note that
  624. // updating the trip points means that we also need to redo the
  625. // cooling level
  626. //
  627. if (!(DeviceExtension->Thermal.Flags & THRM_TRIP_POINTS)) {
  628. DeviceExtension->Thermal.Flags |= THRM_BUSY | THRM_TRIP_POINTS;
  629. ACPISetDeviceWorker( DeviceExtension, THRM_TRIP_POINTS );
  630. continue;
  631. }
  632. //
  633. // If the cooling level has changed,
  634. //
  635. if (!(DeviceExtension->Thermal.Flags & THRM_COOLING_LEVEL)) {
  636. DeviceExtension->Thermal.Flags |= THRM_BUSY | THRM_COOLING_LEVEL;
  637. ACPISetDeviceWorker (DeviceExtension, THRM_COOLING_LEVEL);
  638. continue;
  639. }
  640. //
  641. // Prevent the recursion that occurs when we complete an irp and
  642. // the completion routine is able to queue the IRP before we resume
  643. // the loop
  644. //
  645. if ( (DeviceExtension->Thermal.Flags & THRM_WAIT_FOR_NOTIFY) &&
  646. (DeviceExtension->Thermal.Flags & THRM_TEMP) ) {
  647. break;
  648. }
  649. //
  650. // If we don't have a temp, get it
  651. //
  652. if (!(DeviceExtension->Thermal.Flags & THRM_TEMP)) {
  653. //
  654. // Is the temp object not present?
  655. //
  656. #if DBG
  657. if (thermal->TempMethod == NULL) {
  658. ACPIInternalError( ACPI_THERMAL );
  659. }
  660. #endif
  661. thermal->Info.ThermalStamp += 1;
  662. DeviceExtension->Thermal.Flags |= THRM_BUSY | THRM_TEMP;
  663. KeReleaseSpinLock (&DeviceExtension->Thermal.SpinLock, oldIrql);
  664. lockHeld = FALSE;
  665. RtlZeroMemory (&thermal->Temp, sizeof(OBJDATA));
  666. thermal->Temp.dwDataType = OBJTYPE_UNKNOWN;
  667. status = AMLIAsyncEvalObject(
  668. thermal->TempMethod,
  669. &thermal->Temp,
  670. 0,
  671. NULL,
  672. ACPIThermalTempatureRead,
  673. DeviceExtension
  674. );
  675. if (status != STATUS_PENDING) {
  676. ACPIThermalTempatureRead(
  677. thermal->TempMethod,
  678. status,
  679. &thermal->Temp,
  680. DeviceExtension
  681. );
  682. }
  683. continue;
  684. }
  685. //
  686. // Everything is up to date. Check for a pending irp to see if
  687. // we can complete it.
  688. //
  689. //
  690. // Call into a child function to determine if we have completed
  691. // any requests
  692. //
  693. doneRequests = ACPIThermalCompletePendingIrps(
  694. DeviceExtension,
  695. thermal
  696. );
  697. if (doneRequests) {
  698. continue;
  699. }
  700. break;
  701. }
  702. //
  703. // No longer in the serivce loop
  704. //
  705. DeviceExtension->Thermal.Flags &= ~THRM_IN_SERVICE_LOOP;
  706. }
  707. KeReleaseSpinLock (&DeviceExtension->Thermal.SpinLock, oldIrql);
  708. return ;
  709. }
  710. VOID
  711. ACPIThermalPowerCallback (
  712. IN PDEVICE_EXTENSION DeviceExtension,
  713. IN PVOID Context,
  714. IN NTSTATUS Status
  715. )
  716. /*++
  717. Routine Description:
  718. This is the routine that is called after we have sent an internal
  719. power request to the device
  720. Arguments:
  721. DeviceExtension - the device that was set
  722. Context - Not used
  723. Status - Result
  724. Return Value:
  725. None
  726. --*/
  727. {
  728. if (!NT_SUCCESS(Status)) {
  729. ACPIDevPrint( (
  730. ACPI_PRINT_CRITICAL,
  731. DeviceExtension,
  732. "ACPIThermalPowerCallBack: failed power setting %x\n",
  733. Status
  734. ) );
  735. }
  736. }
  737. NTSTATUS
  738. ACPIThermalQueryWmiDataBlock(
  739. IN PDEVICE_OBJECT DeviceObject,
  740. IN PIRP Irp,
  741. IN ULONG GuidIndex,
  742. IN ULONG InstanceIndex,
  743. IN ULONG InstanceCount,
  744. IN OUT PULONG InstanceLengthArray,
  745. IN ULONG BufferAvail,
  746. OUT PUCHAR Buffer
  747. )
  748. /*++
  749. Routine Description:
  750. This routine is a callback into the driver to query for the contents of
  751. all instances of a data block. When the driver has finished filling the
  752. data block it must call WmiCompleteRequest to complete the irp. The
  753. driver can return STATUS_PENDING if the irp cannot be completed
  754. immediately.
  755. Arguments:
  756. DeviceObject is the device whose data block is being queried
  757. Irp is the Irp that makes this request
  758. GuidIndex is the index into the list of guids provided when the
  759. device registered
  760. InstanceCount is the number of instnaces expected to be returned for
  761. the data block.
  762. InstanceLengthArray is a pointer to an array of ULONG that returns the
  763. lengths of each instance of the data block. If this is NULL then
  764. there was not enough space in the output buffer to fufill the request
  765. so the irp should be completed with the buffer needed.
  766. BufferAvail on entry has the maximum size available to write the data
  767. blocks.
  768. Buffer on return is filled with the returned data blocks. Note that each
  769. instance of the data block must be aligned on a 8 byte boundry.
  770. Return Value:
  771. status
  772. --*/
  773. {
  774. NTSTATUS status;
  775. PDEVICE_EXTENSION deviceExtension;
  776. ULONG sizeNeeded;
  777. PTHRM_INFO info;
  778. PTHERMAL_INFORMATION thermalInfo;
  779. PTHERMAL_INFORMATION wmiThermalInfo;
  780. PAGED_CODE();
  781. deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  782. if (GuidIndex == 0) {
  783. //
  784. // ThermalZone temperature query
  785. //
  786. info = (PTHRM_INFO) deviceExtension->Thermal.Info;
  787. thermalInfo = &info->Info;
  788. wmiThermalInfo = (PTHERMAL_INFORMATION)Buffer;
  789. sizeNeeded = sizeof(THERMAL_INFORMATION);
  790. if (BufferAvail >= sizeNeeded) {
  791. // NOTE - Synchronize with thread getting this data
  792. *InstanceLengthArray = sizeNeeded;
  793. RtlCopyMemory(wmiThermalInfo, thermalInfo, sizeNeeded);
  794. status = STATUS_SUCCESS;
  795. } else {
  796. status = STATUS_BUFFER_TOO_SMALL;
  797. }
  798. } else {
  799. status = STATUS_WMI_GUID_NOT_FOUND;
  800. sizeNeeded = 0;
  801. }
  802. status = WmiCompleteRequest(
  803. DeviceObject,
  804. Irp,
  805. status,
  806. sizeNeeded,
  807. IO_NO_INCREMENT
  808. );
  809. return status;
  810. }
  811. NTSTATUS
  812. ACPIThermalQueryWmiRegInfo(
  813. IN PDEVICE_OBJECT DeviceObject,
  814. OUT ULONG *RegFlags,
  815. OUT PUNICODE_STRING InstanceName,
  816. OUT PUNICODE_STRING *RegistryPath,
  817. OUT PUNICODE_STRING MofResourceName,
  818. OUT PDEVICE_OBJECT *Pdo
  819. )
  820. /*++
  821. Routine Description:
  822. This routine is a callback into the driver to retrieve information about
  823. the guids being registered.
  824. Implementations of this routine may be in paged memory
  825. Arguments:
  826. DeviceObject is the device whose registration information is needed
  827. *RegFlags returns with a set of flags that describe all of the guids being
  828. registered for this device. If the device wants enable and disable
  829. collection callbacks before receiving queries for the registered
  830. guids then it should return the WMIREG_FLAG_EXPENSIVE flag. Also the
  831. returned flags may specify WMIREG_FLAG_INSTANCE_PDO in which case
  832. the instance name is determined from the PDO associated with the
  833. device object. Note that the PDO must have an associated devnode. If
  834. WMIREG_FLAG_INSTANCE_PDO is not set then Name must return a unique
  835. name for the device. These flags are ORed into the flags specified
  836. by the GUIDREGINFO for each guid.
  837. InstanceName returns with the instance name for the guids if
  838. WMIREG_FLAG_INSTANCE_PDO is not set in the returned *RegFlags. The
  839. caller will call ExFreePool with the buffer returned.
  840. *RegistryPath returns with the registry path of the driver. This is
  841. required
  842. MofResourceName returns with the name of the MOF resource attached to
  843. the binary file. If the driver does not have a mof resource attached
  844. then this can be returned unmodified. If a value is returned then
  845. it is NOT freed.
  846. *Pdo returns with the device object for the PDO associated with this
  847. device if the WMIREG_FLAG_INSTANCE_PDO flag is retured in
  848. *RegFlags.
  849. Return Value:
  850. status
  851. --*/
  852. {
  853. PAGED_CODE();
  854. if (AcpiRegistryPath.Buffer != NULL) {
  855. *RegistryPath = &AcpiRegistryPath;
  856. } else {
  857. *RegistryPath = NULL;
  858. }
  859. *RegFlags = WMIREG_FLAG_INSTANCE_PDO;
  860. *Pdo = DeviceObject;
  861. return STATUS_SUCCESS;
  862. }
  863. NTSTATUS
  864. ACPIThermalStartDevice (
  865. IN PDEVICE_OBJECT DeviceObject,
  866. IN PIRP Irp
  867. )
  868. /*++
  869. Routine Description:
  870. This routine is called to start the thermal zone
  871. Arguments:
  872. DeviceObject - The device that is starting up
  873. Irp - The request
  874. Return Value:
  875. NTSTATUS
  876. --*/
  877. {
  878. NTSTATUS status;
  879. PDEVICE_EXTENSION deviceExtension;
  880. PWMILIB_CONTEXT wmilibContext;
  881. deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  882. ACPIDevPrint( (
  883. ACPI_PRINT_IRP,
  884. deviceExtension,
  885. "(0x%08lx): IRP_MN_START_DEVICE\n",
  886. Irp
  887. ) );
  888. status = ACPIInternalSetDeviceInterface (
  889. DeviceObject,
  890. (LPGUID) &GUID_DEVICE_THERMAL_ZONE
  891. );
  892. if (!NT_SUCCESS(status)) {
  893. ACPIDevPrint( (
  894. ACPI_PRINT_FAILURE,
  895. deviceExtension,
  896. "ACPIThermalStartDevice -> SetDeviceInterface = 0x%08lx\n",
  897. status
  898. ) );
  899. goto ACPIThermalStartDeviceExit;
  900. }
  901. ACPIRegisterForDeviceNotifications(
  902. DeviceObject,
  903. (PDEVICE_NOTIFY_CALLBACK) ACPIThermalEvent,
  904. (PVOID) DeviceObject
  905. );
  906. //
  907. // Initialize device object for WMILIB
  908. //
  909. wmilibContext = ExAllocatePoolWithTag(
  910. PagedPool,
  911. sizeof(WMILIB_CONTEXT),
  912. ACPI_THERMAL_POOLTAG
  913. );
  914. if (wmilibContext == NULL) {
  915. status = STATUS_INSUFFICIENT_RESOURCES;
  916. goto ACPIThermalStartDeviceExit;
  917. }
  918. RtlZeroMemory(wmilibContext, sizeof(WMILIB_CONTEXT));
  919. wmilibContext->GuidCount = ACPIThermalGuidCount;
  920. wmilibContext->GuidList = &ACPIThermalGuidList;
  921. wmilibContext->QueryWmiRegInfo = ACPIThermalQueryWmiRegInfo;
  922. wmilibContext->QueryWmiDataBlock = ACPIThermalQueryWmiDataBlock;
  923. deviceExtension->Thermal.WmilibContext = wmilibContext;
  924. //
  925. // Register for WMI events
  926. //
  927. status = IoWMIRegistrationControl(
  928. DeviceObject,
  929. WMIREG_ACTION_REGISTER
  930. );
  931. if (!NT_SUCCESS(status)) {
  932. deviceExtension->Thermal.WmilibContext = NULL;
  933. ExFreePool(wmilibContext);
  934. goto ACPIThermalStartDeviceExit;
  935. }
  936. //
  937. // Mark the device as started
  938. //
  939. deviceExtension->DeviceState = Started;
  940. //
  941. // Request that the device go to the D0 state
  942. // Note: that we don't block on this call, since we assume that
  943. // we can process thermal events asynchronously from being in
  944. // the D0 state. However, there may be a future occasion where
  945. // this is not true, so this makes the code more ready to handle
  946. // that case
  947. //
  948. status = ACPIDeviceInternalDeviceRequest(
  949. deviceExtension,
  950. PowerDeviceD0,
  951. NULL,
  952. NULL,
  953. 0
  954. );
  955. if (status == STATUS_PENDING) {
  956. status = STATUS_SUCCESS;
  957. }
  958. //
  959. // Start the thermal engine
  960. //
  961. ACPIThermalLoop( deviceExtension, THRM_TRIP_POINTS | THRM_MODE);
  962. ACPIThermalStartDeviceExit:
  963. Irp->IoStatus.Status = status;
  964. IoCompleteRequest( Irp, IO_NO_INCREMENT );
  965. return status;
  966. }
  967. VOID
  968. EXPORT
  969. ACPIThermalTempatureRead (
  970. IN PNSOBJ AcpiObject,
  971. IN NTSTATUS Status,
  972. IN POBJDATA Result OPTIONAL,
  973. IN PVOID Context
  974. )
  975. /*++
  976. Routine Description:
  977. This routine is called to read the temperature. It is used a callback to
  978. an interpreter call
  979. Arguments:
  980. AcpiObject - The object that was executed
  981. Status - The status of the execution
  982. Result - The result of the execution
  983. Context - The device extension
  984. Return Value:
  985. NTSTATUS
  986. --*/
  987. {
  988. PTHRM_INFO Thrm;
  989. PDEVICE_EXTENSION deviceExtension;
  990. deviceExtension = Context;
  991. if (NT_SUCCESS(Status)) {
  992. ASSERT (Result->dwDataType == OBJTYPE_INTDATA);
  993. Thrm = deviceExtension->Thermal.Info;
  994. Thrm->Info.CurrentTemperature = (ULONG)Result->uipDataValue;
  995. AMLIFreeDataBuffs (Result, 1);
  996. ACPIThermalPrint( (
  997. ACPI_PRINT_THERMAL,
  998. deviceExtension,
  999. KeQueryInterruptTime(),
  1000. "Current Temperature is %d.%dK\n",
  1001. (Thrm->Info.CurrentTemperature / 10 ),
  1002. (Thrm->Info.CurrentTemperature % 10 )
  1003. ) );
  1004. }
  1005. ACPIThermalLoop (deviceExtension, THRM_BUSY);
  1006. }
  1007. VOID
  1008. ACPIThermalWorker (
  1009. IN PDEVICE_EXTENSION DeviceExtension,
  1010. IN ULONG Events
  1011. )
  1012. /*++
  1013. Routine Description:
  1014. Worker thread for thermal regions
  1015. Arguments:
  1016. DeviceExtension - The device extension that we are manipulating
  1017. Events - What just happened
  1018. Return Value:
  1019. None
  1020. --*/
  1021. {
  1022. BOOLEAN TurnOn;
  1023. PTHRM_INFO Thrm;
  1024. NTSTATUS Status;
  1025. PNSOBJ ThrmObj;
  1026. PNSOBJ ALobj;
  1027. OBJDATA ALPackage;
  1028. OBJDATA ALElement;
  1029. PNSOBJ ACDevObj;
  1030. ULONG Index;
  1031. ULONG Level;
  1032. ULONG PackageSize;
  1033. ULONGLONG currentTime;
  1034. PAGED_CODE();
  1035. #if DBG
  1036. currentTime = KeQueryInterruptTime();
  1037. #endif
  1038. Thrm = DeviceExtension->Thermal.Info;
  1039. ThrmObj = DeviceExtension->AcpiObject;
  1040. //
  1041. // Initialization code
  1042. //
  1043. if (Events & THRM_INITIALIZE) {
  1044. ULONG names[10] = {
  1045. PACKED_AL0,
  1046. PACKED_AL1,
  1047. PACKED_AL2,
  1048. PACKED_AL3,
  1049. PACKED_AL4,
  1050. PACKED_AL5,
  1051. PACKED_AL6,
  1052. PACKED_AL7,
  1053. PACKED_AL8,
  1054. PACKED_AL9,
  1055. };
  1056. //
  1057. // Start the system in PASSIVE mode
  1058. //
  1059. Thrm->Mode = 1;
  1060. //
  1061. // Fetch all of the objects associated with each cooling level
  1062. //
  1063. for (Level = 0; Level < 10; Level++) {
  1064. //
  1065. // Find this level's active list
  1066. //
  1067. ALobj = ACPIAmliGetNamedChild(
  1068. ThrmObj,
  1069. names[Level]
  1070. );
  1071. if (ALobj == NULL) {
  1072. break;
  1073. }
  1074. //
  1075. // Remember that we have this object
  1076. //
  1077. Thrm->ActiveList[Level] = ALobj;
  1078. }
  1079. }
  1080. //
  1081. // Do this before we update the trips points
  1082. //
  1083. if ( (Events & THRM_COOLING_LEVEL) ) {
  1084. RtlZeroMemory (&ALPackage, sizeof(OBJDATA));
  1085. RtlZeroMemory (&ALElement, sizeof(OBJDATA));
  1086. for (Level=0; Level < 10; Level++) {
  1087. //
  1088. // Is there a cooling object?
  1089. //
  1090. ALobj = Thrm->ActiveList[Level];
  1091. if (ALobj == NULL) {
  1092. break;
  1093. }
  1094. //
  1095. // Evalute the list to its package
  1096. //
  1097. Status = AMLIEvalNameSpaceObject(
  1098. ALobj,
  1099. &ALPackage,
  1100. 0,
  1101. NULL
  1102. );
  1103. if (!NT_SUCCESS(Status)) {
  1104. break;
  1105. }
  1106. //
  1107. // Remember how large the package is
  1108. //
  1109. PackageSize = ((PPACKAGEOBJ) ALPackage.pbDataBuff)->dwcElements;
  1110. //
  1111. // Walk the names in the package
  1112. //
  1113. for (Index = 0; Index < PackageSize; Index += 1) {
  1114. //
  1115. // Grab the object name
  1116. Status = AMLIEvalPkgDataElement(
  1117. &ALPackage,
  1118. Index,
  1119. &ALElement
  1120. );
  1121. if (!NT_SUCCESS(Status)) {
  1122. break;
  1123. }
  1124. //
  1125. // Determine if we are going to the device on or off
  1126. //
  1127. TurnOn = (Level >= Thrm->CoolingLevel);
  1128. //
  1129. // Tell the world
  1130. //
  1131. #if DBG
  1132. ACPIThermalPrint( (
  1133. ACPI_PRINT_THERMAL,
  1134. DeviceExtension,
  1135. currentTime,
  1136. "ACPIThermalWorker: Turn %s %s\n",
  1137. TurnOn ? "on " : "off",
  1138. ALElement.pbDataBuff
  1139. ) );
  1140. #endif
  1141. //
  1142. // Find this device of this name
  1143. //
  1144. Status = AMLIGetNameSpaceObject(
  1145. ALElement.pbDataBuff,
  1146. ThrmObj,
  1147. &ACDevObj,
  1148. 0
  1149. );
  1150. AMLIFreeDataBuffs (&ALElement, 1);
  1151. if (!NT_SUCCESS(Status) || !ACDevObj->Context) {
  1152. break;
  1153. }
  1154. //
  1155. // Turn it on/off
  1156. //
  1157. ACPIDeviceInternalDeviceRequest (
  1158. (PDEVICE_EXTENSION) ACDevObj->Context,
  1159. TurnOn ? PowerDeviceD0 : PowerDeviceD3,
  1160. ACPIThermalPowerCallback,
  1161. NULL,
  1162. 0
  1163. );
  1164. }
  1165. AMLIFreeDataBuffs (&ALPackage, 1);
  1166. }
  1167. }
  1168. //
  1169. // If the trip points need to be re-freshed, go read them
  1170. //
  1171. if (Events & THRM_TRIP_POINTS) {
  1172. ULONG names[10] = {
  1173. PACKED_AC0,
  1174. PACKED_AC1,
  1175. PACKED_AC2,
  1176. PACKED_AC3,
  1177. PACKED_AC4,
  1178. PACKED_AC5,
  1179. PACKED_AC6,
  1180. PACKED_AC7,
  1181. PACKED_AC8,
  1182. PACKED_AC9,
  1183. };
  1184. //
  1185. // Get the thermal constants, passive & critical values
  1186. //
  1187. ACPIGetIntegerSync(
  1188. DeviceExtension,
  1189. PACKED_TC1,
  1190. &Thrm->Info.ThermalConstant1,
  1191. NULL
  1192. );
  1193. #if DBG
  1194. ACPIThermalPrint( (
  1195. ACPI_PRINT_THERMAL,
  1196. DeviceExtension,
  1197. currentTime,
  1198. "ACPIThermalWorker - ThermalConstant1 = %x\n",
  1199. Thrm->Info.ThermalConstant1
  1200. ) );
  1201. #endif
  1202. ACPIGetIntegerSync(
  1203. DeviceExtension,
  1204. PACKED_TC2,
  1205. &Thrm->Info.ThermalConstant2,
  1206. NULL
  1207. );
  1208. #if DBG
  1209. ACPIThermalPrint( (
  1210. ACPI_PRINT_THERMAL,
  1211. DeviceExtension,
  1212. currentTime,
  1213. "ACPIThermalWorker - ThermalConstant2 = %x\n",
  1214. Thrm->Info.ThermalConstant2
  1215. ) );
  1216. #endif
  1217. ACPIGetIntegerSync(
  1218. DeviceExtension,
  1219. PACKED_PSV,
  1220. &Thrm->Info.PassiveTripPoint,
  1221. NULL
  1222. );
  1223. #if DBG
  1224. ACPIThermalPrint( (
  1225. ACPI_PRINT_THERMAL,
  1226. DeviceExtension,
  1227. currentTime,
  1228. "ACPIThermalWorker - PassiveTripPoint = %d.%dK\n",
  1229. (Thrm->Info.PassiveTripPoint / 10),
  1230. (Thrm->Info.PassiveTripPoint % 10)
  1231. ) );
  1232. #endif
  1233. ACPIGetIntegerSync(
  1234. DeviceExtension,
  1235. PACKED_CRT,
  1236. &Thrm->Info.CriticalTripPoint,
  1237. NULL
  1238. );
  1239. #if DBG
  1240. ACPIThermalPrint( (
  1241. ACPI_PRINT_THERMAL,
  1242. DeviceExtension,
  1243. currentTime,
  1244. "ACPIThermalWorker - CriticalTripPoint = %d.%dK\n",
  1245. (Thrm->Info.CriticalTripPoint / 10),
  1246. (Thrm->Info.CriticalTripPoint % 10)
  1247. ) );
  1248. #endif
  1249. ACPIGetIntegerSync(
  1250. DeviceExtension,
  1251. PACKED_TSP,
  1252. &Thrm->Info.SamplingPeriod,
  1253. NULL
  1254. );
  1255. #if DBG
  1256. ACPIThermalPrint( (
  1257. ACPI_PRINT_THERMAL,
  1258. DeviceExtension,
  1259. currentTime,
  1260. "ACPIThermalWorker - SamplingPeriod = %x\n",
  1261. Thrm->Info.SamplingPeriod
  1262. ) );
  1263. #endif
  1264. //
  1265. // Get the active cooling limits
  1266. //
  1267. for (Level=0; Level < 10; Level++) {
  1268. Status = ACPIGetIntegerSync(
  1269. DeviceExtension,
  1270. names[Level],
  1271. &Thrm->Info.ActiveTripPoint[Level],
  1272. NULL
  1273. );
  1274. if (!NT_SUCCESS(Status)) {
  1275. break;
  1276. }
  1277. #if DBG
  1278. ACPIThermalPrint( (
  1279. ACPI_PRINT_THERMAL,
  1280. DeviceExtension,
  1281. currentTime,
  1282. "ACPIThermalWorker - Active Cooling Level %x = %d.%dK\n",
  1283. Level,
  1284. (Thrm->Info.ActiveTripPoint[Level] / 10),
  1285. (Thrm->Info.ActiveTripPoint[Level] % 10)
  1286. ) );
  1287. #endif
  1288. }
  1289. Thrm->Info.ActiveTripPointCount = (UCHAR) Level;
  1290. //
  1291. // Clean these variables for reuse
  1292. //
  1293. RtlZeroMemory (&ALPackage, sizeof(OBJDATA));
  1294. RtlZeroMemory (&ALElement, sizeof(OBJDATA));
  1295. //
  1296. // Assume an affinity of 0
  1297. //
  1298. Thrm->Info.Processors = 0;
  1299. //
  1300. // Get the passive cooling affinity object
  1301. //
  1302. ALobj = ACPIAmliGetNamedChild(
  1303. ThrmObj,
  1304. PACKED_PSL
  1305. );
  1306. if (ALobj != NULL) {
  1307. //
  1308. // Evaluate the processor affinity object
  1309. //
  1310. Status = AMLIEvalNameSpaceObject(
  1311. ALobj,
  1312. &ALPackage,
  1313. 0,
  1314. NULL
  1315. );
  1316. if (!NT_SUCCESS(Status)) {
  1317. goto ACPIThermalWorkerExit;
  1318. }
  1319. //
  1320. // Remember how large the package is
  1321. //
  1322. PackageSize = ((PPACKAGEOBJ) ALPackage.pbDataBuff)->dwcElements;
  1323. //
  1324. // Walk the elements in the package
  1325. //
  1326. for (Index = 0; Index < PackageSize ;Index++) {
  1327. Status = AMLIEvalPkgDataElement(
  1328. &ALPackage,
  1329. Index,
  1330. &ALElement
  1331. );
  1332. if (!NT_SUCCESS(Status)) {
  1333. break;
  1334. }
  1335. //
  1336. // Find this device of this name
  1337. //
  1338. Status = AMLIGetNameSpaceObject(
  1339. ALElement.pbDataBuff,
  1340. NULL,
  1341. &ACDevObj,
  1342. 0
  1343. );
  1344. //
  1345. // No longer need this information
  1346. //
  1347. AMLIFreeDataBuffs (&ALElement, 1);
  1348. //
  1349. // Did we find what we wanted?
  1350. //
  1351. if (!NT_SUCCESS(Status) ) {
  1352. break;
  1353. }
  1354. //
  1355. // Get the correct affinity mask. We call another
  1356. // function since that one requires a spinlock which
  1357. // don't want to take in this worker function
  1358. //
  1359. ACPIThermalCalculateProcessorMask( ACDevObj, Thrm );
  1360. }
  1361. //
  1362. // We are done with the package
  1363. //
  1364. AMLIFreeDataBuffs (&ALPackage, 1);
  1365. }
  1366. }
  1367. ACPIThermalWorkerExit:
  1368. //
  1369. // done, check for next work
  1370. //
  1371. ACPIThermalLoop (DeviceExtension, THRM_TEMP | THRM_BUSY);
  1372. }
  1373. NTSTATUS
  1374. ACPIThermalWmi(
  1375. IN PDEVICE_OBJECT DeviceObject,
  1376. IN PIRP Irp
  1377. )
  1378. {
  1379. NTSTATUS status;
  1380. PDEVICE_EXTENSION deviceExtension = ACPIInternalGetDeviceExtension(DeviceObject);
  1381. PIO_STACK_LOCATION irpSp;
  1382. PWMILIB_CONTEXT wmilibContext;
  1383. SYSCTL_IRP_DISPOSITION disposition;
  1384. wmilibContext = deviceExtension->Thermal.WmilibContext;
  1385. irpSp = IoGetCurrentIrpStackLocation(Irp);
  1386. status = WmiSystemControl(
  1387. wmilibContext,
  1388. DeviceObject,
  1389. Irp,
  1390. &disposition
  1391. );
  1392. switch (disposition) {
  1393. case IrpProcessed:
  1394. break;
  1395. case IrpNotCompleted:
  1396. IoCompleteRequest(Irp, IO_NO_INCREMENT);
  1397. break;
  1398. case IrpNotWmi:
  1399. case IrpForward:
  1400. default:
  1401. status = ACPIDispatchForwardIrp(DeviceObject, Irp);
  1402. break;
  1403. }
  1404. return status;
  1405. }