xref: /reactos/drivers/bus/acpi/cmbatt/cmexec.c (revision 5c4fcd99)
1 /*
2  * PROJECT:         ReactOS ACPI-Compliant Control Method Battery
3  * LICENSE:         BSD - See COPYING.ARM in the top level directory
4  * FILE:            boot/drivers/bus/acpi/cmbatt/cmexec.c
5  * PURPOSE:         ACPI Method Execution/Evaluation Glue
6  * PROGRAMMERS:     ReactOS Portable Systems Group
7  */
8 
9 /* INCLUDES *******************************************************************/
10 
11 #include "cmbatt.h"
12 
13 #include <acpiioct.h>
14 #include <debug.h>
15 
16 typedef struct
17 {
18     LPSTR Name;
19     BOOLEAN IsString;
20     PVOID Data;
21 } ACPI_PACKAGE_FIELD, *PACPI_PACKAGE_FIELD;
22 
23 /* FUNCTIONS ******************************************************************/
24 
25 NTSTATUS
26 NTAPI
GetDwordElement(IN PACPI_METHOD_ARGUMENT Argument,OUT PULONG Value)27 GetDwordElement(IN PACPI_METHOD_ARGUMENT Argument,
28                 OUT PULONG Value)
29 {
30     NTSTATUS Status;
31 
32     /* Must have an integer */
33     if (Argument->Type != ACPI_METHOD_ARGUMENT_INTEGER)
34     {
35         /* Not an integer, fail */
36         Status = STATUS_ACPI_INVALID_DATA;
37         if (CmBattDebug & 0x4C)
38             DbgPrint("GetDwordElement: Object contained wrong data type - %d\n",
39                      Argument->Type);
40     }
41     else
42     {
43         /* Read the integer value */
44         *Value = Argument->Argument;
45         Status = STATUS_SUCCESS;
46     }
47 
48     /* Return status */
49     return Status;
50 }
51 
52 NTSTATUS
53 NTAPI
GetStringElement(IN PACPI_METHOD_ARGUMENT Argument,OUT PCHAR Value)54 GetStringElement(IN PACPI_METHOD_ARGUMENT Argument,
55                  OUT PCHAR Value)
56 {
57     NTSTATUS Status;
58 
59     /* Must have a string of buffer */
60     if ((Argument->Type == ACPI_METHOD_ARGUMENT_STRING) ||
61         (Argument->Type == ACPI_METHOD_ARGUMENT_BUFFER))
62     {
63         /* String must be less than 256 characters */
64         if (Argument->DataLength < 256)
65         {
66             /* Copy the buffer */
67             RtlCopyMemory(Value, Argument->Data, Argument->DataLength);
68             Status = STATUS_SUCCESS;
69         }
70         else
71         {
72             /* The buffer is too small (the string is too large) */
73             Status = STATUS_BUFFER_TOO_SMALL;
74             if (CmBattDebug & 0x4C)
75                 DbgPrint("GetStringElement: return buffer not big enough - %d\n", Argument->DataLength);
76         }
77     }
78     else
79     {
80         /* Not valid string data */
81         Status = STATUS_ACPI_INVALID_DATA;
82         if (CmBattDebug & 0x4C)
83             DbgPrint("GetStringElement: Object contained wrong data type - %d\n", Argument->Type);
84     }
85 
86     /* Return the status */
87     return Status;
88 }
89 
90 NTSTATUS
91 NTAPI
CmBattSendDownStreamIrp(IN PDEVICE_OBJECT DeviceObject,IN ULONG IoControlCode,IN PVOID InputBuffer,IN ULONG InputBufferLength,IN PACPI_EVAL_OUTPUT_BUFFER OutputBuffer,IN ULONG OutputBufferLength)92 CmBattSendDownStreamIrp(IN PDEVICE_OBJECT DeviceObject,
93                         IN ULONG IoControlCode,
94                         IN PVOID InputBuffer,
95                         IN ULONG InputBufferLength,
96                         IN PACPI_EVAL_OUTPUT_BUFFER OutputBuffer,
97                         IN ULONG OutputBufferLength)
98 {
99     PIRP Irp;
100     NTSTATUS Status;
101     KEVENT Event;
102     IO_STATUS_BLOCK IoStatusBlock;
103     PAGED_CODE();
104 
105     /* Initialize our wait event */
106     KeInitializeEvent(&Event, SynchronizationEvent, 0);
107 
108     /* Allocate the IRP */
109     Irp = IoBuildDeviceIoControlRequest(IoControlCode,
110                                         DeviceObject,
111                                         InputBuffer,
112                                         InputBufferLength,
113                                         OutputBuffer,
114                                         OutputBufferLength,
115                                         0,
116                                         &Event,
117                                         &IoStatusBlock);
118     if (!Irp)
119     {
120         /* No IRP, fail */
121         if (CmBattDebug & 0x4C)
122             DbgPrint("CmBattSendDownStreamIrp: Failed to allocate Irp\n");
123         return STATUS_INSUFFICIENT_RESOURCES;
124     }
125 
126     /* Call ACPI */
127     if (CmBattDebug & 0x40)
128         DbgPrint("CmBattSendDownStreamIrp: Irp %x [Tid] %x\n",
129                  Irp, KeGetCurrentThread());
130     Status = IoCallDriver(DeviceObject, Irp);
131     if (Status == STATUS_PENDING)
132     {
133         /* Wait for completion */
134         KeWaitForSingleObject(&Event,
135                               Executive,
136                               KernelMode,
137                               FALSE,
138                               NULL);
139         Status = Irp->IoStatus.Status;
140     }
141 
142     /* Check if caller wanted output */
143     if (OutputBuffer)
144     {
145         /* Make sure it's valid ACPI output buffer */
146         if ((OutputBuffer->Signature != ACPI_EVAL_OUTPUT_BUFFER_SIGNATURE) ||
147             !(OutputBuffer->Count))
148         {
149             /* It isn't, so set failure code */
150             Status = STATUS_ACPI_INVALID_DATA;
151         }
152     }
153 
154     /* Return status */
155     if (CmBattDebug & 0x40)
156         DbgPrint("CmBattSendDownStreamIrp: Irp %x completed %x! [Tid] %x\n",
157                  Irp, Status, KeGetCurrentThread());
158     return Status;
159 }
160 
161 static
162 NTSTATUS
CmBattCallAcpiPackage(_In_ LPCSTR FunctionName,_In_ PCMBATT_DEVICE_EXTENSION DeviceExtension,_In_ ULONG PackageName,_In_ ULONG OutputBufferSize,_In_ PACPI_PACKAGE_FIELD PackageFields,_In_ ULONG PackageFieldCount)163 CmBattCallAcpiPackage(
164     _In_ LPCSTR FunctionName,
165     _In_ PCMBATT_DEVICE_EXTENSION DeviceExtension,
166     _In_ ULONG PackageName,
167     _In_ ULONG OutputBufferSize,
168     _In_ PACPI_PACKAGE_FIELD PackageFields,
169     _In_ ULONG PackageFieldCount)
170 {
171     NTSTATUS Status;
172     PACPI_EVAL_OUTPUT_BUFFER OutputBuffer;
173     ACPI_EVAL_INPUT_BUFFER InputBuffer;
174     PACPI_METHOD_ARGUMENT Argument;
175     ULONG i;
176     PAGED_CODE();
177 
178     OutputBuffer = ExAllocatePoolWithTag(PagedPool,
179                                          OutputBufferSize,
180                                          'MtaB');
181     if (!OutputBuffer)
182     {
183         if (CmBattDebug & (CMBATT_ACPI_ENTRY_EXIT | CMBATT_ACPI_WARNING | CMBATT_GENERIC_WARNING))
184             DbgPrint("%s: Failed to allocate Buffer\n", FunctionName);
185         return STATUS_INSUFFICIENT_RESOURCES;
186     }
187 
188     /* Initialize to zero */
189     RtlZeroMemory(OutputBuffer, OutputBufferSize);
190 
191     /* Request the ACPI method */
192     *(PULONG)InputBuffer.MethodName = PackageName;
193     InputBuffer.Signature = ACPI_EVAL_INPUT_BUFFER_SIGNATURE;
194 
195     /* Send it to ACPI */
196     Status = CmBattSendDownStreamIrp(DeviceExtension->AttachedDevice,
197                                      IOCTL_ACPI_EVAL_METHOD,
198                                      &InputBuffer,
199                                      sizeof(InputBuffer),
200                                      OutputBuffer,
201                                      OutputBufferSize);
202     if (!NT_SUCCESS(Status))
203     {
204         if (CmBattDebug & (CMBATT_ACPI_ASSERT | CMBATT_ACPI_ENTRY_EXIT | CMBATT_ACPI_WARNING | CMBATT_GENERIC_WARNING))
205             DbgPrint("%s: Failed 0x%08x method on device %x - Status (0x%x)\n",
206                      FunctionName, PackageName, DeviceExtension->DeviceId, Status);
207         ExFreePoolWithTag(OutputBuffer, 'MtaB');
208         return Status;
209     }
210 
211     /* Check if we got the right number of elements */
212     if (OutputBuffer->Count != PackageFieldCount)
213     {
214         if (CmBattDebug & (CMBATT_ACPI_ASSERT | CMBATT_ACPI_ENTRY_EXIT | CMBATT_ACPI_WARNING | CMBATT_GENERIC_WARNING))
215             DbgPrint("%s: 0x%08x method returned %d elements (requires %d)\n",
216                      FunctionName, PackageName, OutputBuffer->Count, PackageFieldCount);
217         ExFreePoolWithTag(OutputBuffer, 'MtaB');
218         return STATUS_ACPI_INVALID_DATA;
219     }
220 
221     Argument = OutputBuffer->Argument;
222     for (i = 0; i < PackageFieldCount && NT_SUCCESS(Status); i++)
223     {
224         if (PackageFields[i].IsString)
225             Status = GetStringElement(Argument, PackageFields[i].Data);
226         else
227             Status = GetDwordElement(Argument, PackageFields[i].Data);
228         if (!NT_SUCCESS(Status))
229         {
230             if (CmBattDebug & (CMBATT_ACPI_ASSERT | CMBATT_ACPI_ENTRY_EXIT | CMBATT_ACPI_WARNING | CMBATT_GENERIC_WARNING))
231                 DbgPrint("%s: Failed to get %s\n", FunctionName, PackageFields[i].Name);
232             break;
233         }
234         Argument = ACPI_METHOD_NEXT_ARGUMENT(Argument);
235     }
236 
237     ExFreePoolWithTag(OutputBuffer, 'MtaB');
238     return Status;
239 }
240 
241 NTSTATUS
242 NTAPI
CmBattGetPsrData(IN PDEVICE_OBJECT DeviceObject,OUT PULONG PsrData)243 CmBattGetPsrData(IN PDEVICE_OBJECT DeviceObject,
244                  OUT PULONG PsrData)
245 {
246     NTSTATUS Status;
247     ACPI_EVAL_OUTPUT_BUFFER OutputBuffer;
248     ACPI_EVAL_INPUT_BUFFER InputBuffer;
249     PAGED_CODE();
250     if (CmBattDebug & 0x40)
251         DbgPrint("CmBattGetPsrData: Entered with Pdo %x Tid %x\n",
252                  DeviceObject, KeGetCurrentThread());
253 
254     /* Initialize to zero */
255     ASSERT(PsrData != NULL);
256     *PsrData = 0;
257 
258     /* Request the _PSR method */
259     *(PULONG)InputBuffer.MethodName = 'RSP_';
260     InputBuffer.Signature = ACPI_EVAL_INPUT_BUFFER_SIGNATURE;
261 
262     /* Send it to ACPI */
263     Status = CmBattSendDownStreamIrp(DeviceObject,
264                                      IOCTL_ACPI_EVAL_METHOD,
265                                      &InputBuffer,
266                                      sizeof(InputBuffer),
267                                      &OutputBuffer,
268                                      sizeof(OutputBuffer));
269     if (NT_SUCCESS(Status))
270     {
271         /* Read the result */
272         Status = GetDwordElement(OutputBuffer.Argument, PsrData);
273         if (CmBattDebug & 0x440)
274             DbgPrint("CmBattGetPsrData: _PSR method returned %x \n", *PsrData);
275     }
276     else if (CmBattDebug & 0x44C)
277     {
278         /* Failure */
279         DbgPrint("CmBattGetPsrData: Failed _PSR method - Status (0x%x)\n", Status);
280     }
281 
282     /* Return status */
283     return Status;
284 }
285 
286 NTSTATUS
287 NTAPI
CmBattGetStaData(IN PDEVICE_OBJECT DeviceObject,OUT PULONG StaData)288 CmBattGetStaData(IN PDEVICE_OBJECT DeviceObject,
289                  OUT PULONG StaData)
290 {
291     NTSTATUS Status;
292     ACPI_EVAL_OUTPUT_BUFFER OutputBuffer;
293     ACPI_EVAL_INPUT_BUFFER InputBuffer;
294     PAGED_CODE();
295     if (CmBattDebug & 0x40)
296         DbgPrint("CmBattGetStaData: Entered with Pdo %x Tid %x\n",
297                  DeviceObject, KeGetCurrentThread());
298 
299     /* Initialize to zero */
300     ASSERT(StaData != NULL);
301     *StaData = 0;
302 
303     /* Request the _STA method */
304     *(PULONG)InputBuffer.MethodName = 'ATS_';
305     InputBuffer.Signature = ACPI_EVAL_INPUT_BUFFER_SIGNATURE;
306 
307     /* Send it to ACPI */
308     Status = CmBattSendDownStreamIrp(DeviceObject,
309                                      IOCTL_ACPI_EVAL_METHOD,
310                                      &InputBuffer,
311                                      sizeof(InputBuffer),
312                                      &OutputBuffer,
313                                      sizeof(OutputBuffer));
314     if (NT_SUCCESS(Status))
315     {
316         /* Read the result */
317         Status = GetDwordElement(OutputBuffer.Argument, StaData);
318         if (CmBattDebug & 0x440)
319             DbgPrint("CmBattGetStaData: _STA method returned %x \n", *StaData);
320     }
321     else if (CmBattDebug & 0x44C)
322     {
323         /* Failure */
324         DbgPrint("CmBattGetStaData: Failed _STA method - Status (0x%x)\n", Status);
325         Status = STATUS_NO_SUCH_DEVICE;
326     }
327 
328     /* Return status */
329     return Status;
330 }
331 
332 NTSTATUS
333 NTAPI
CmBattGetUniqueId(IN PDEVICE_OBJECT DeviceObject,OUT PULONG UniqueId)334 CmBattGetUniqueId(IN PDEVICE_OBJECT DeviceObject,
335                   OUT PULONG UniqueId)
336 {
337     NTSTATUS Status;
338     ACPI_EVAL_OUTPUT_BUFFER OutputBuffer;
339     ACPI_EVAL_INPUT_BUFFER InputBuffer;
340     PAGED_CODE();
341     if (CmBattDebug & 0x40)
342         DbgPrint("CmBattGetUniqueId: Entered with Pdo %x Tid %x\n",
343                  DeviceObject, KeGetCurrentThread());
344 
345     /* Initialize to zero */
346     ASSERT(UniqueId != NULL);
347     *UniqueId = 0;
348 
349     /* Request the _UID method */
350     *(PULONG)InputBuffer.MethodName = 'DIU_';
351     InputBuffer.Signature = ACPI_EVAL_INPUT_BUFFER_SIGNATURE;
352 
353     /* Send it to ACPI */
354     Status = CmBattSendDownStreamIrp(DeviceObject,
355                                      IOCTL_ACPI_EVAL_METHOD,
356                                      &InputBuffer,
357                                      sizeof(InputBuffer),
358                                      &OutputBuffer,
359                                      sizeof(OutputBuffer));
360     if (NT_SUCCESS(Status))
361     {
362         /* Read the result */
363         Status = GetDwordElement(OutputBuffer.Argument, UniqueId);
364         if (CmBattDebug & 0x440)
365             DbgPrint("CmBattGetUniqueId: _UID method returned %x \n", *UniqueId);
366     }
367     else if (CmBattDebug & 0x44C)
368     {
369         /* Failure */
370         DbgPrint("CmBattGetUniqueId: Failed _UID method - Status (0x%x)\n", Status);
371         Status = STATUS_NO_SUCH_DEVICE;
372     }
373 
374     /* Return status */
375     return Status;
376 }
377 
378 NTSTATUS
379 NTAPI
CmBattSetTripPpoint(IN PCMBATT_DEVICE_EXTENSION DeviceExtension,IN ULONG AlarmValue)380 CmBattSetTripPpoint(IN PCMBATT_DEVICE_EXTENSION DeviceExtension,
381                     IN ULONG AlarmValue)
382 {
383     NTSTATUS Status;
384     ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER InputBuffer;
385     PAGED_CODE();
386     if (CmBattDebug & 0x440)
387         DbgPrint("CmBattSetTripPpoint: _BTP Alarm Value %x Device %x Tid %x\n",
388                  AlarmValue, DeviceExtension->DeviceId, KeGetCurrentThread);
389 
390     /* Request the _BTP method */
391     *(PULONG)InputBuffer.MethodName = 'PTB_';
392     InputBuffer.Signature = ACPI_EVAL_INPUT_BUFFER_SIMPLE_INTEGER_SIGNATURE;
393     InputBuffer.IntegerArgument = AlarmValue;
394 
395     /* Send it to ACPI */
396     Status = CmBattSendDownStreamIrp(DeviceExtension->AttachedDevice,
397                                      IOCTL_ACPI_EVAL_METHOD,
398                                      &InputBuffer,
399                                      sizeof(InputBuffer),
400                                      NULL,
401                                      0);
402     if (!(NT_SUCCESS(Status)) && (CmBattDebug & 0x440))
403         DbgPrint("CmBattSetTripPpoint: Failed _BTP method on device %x - Status (0x%x)\n",
404                  DeviceExtension->DeviceId, Status);
405 
406     /* Return status */
407     return Status;
408 }
409 
410 NTSTATUS
411 NTAPI
CmBattGetBifData(PCMBATT_DEVICE_EXTENSION DeviceExtension,PACPI_BIF_DATA BifData)412 CmBattGetBifData(PCMBATT_DEVICE_EXTENSION DeviceExtension,
413                  PACPI_BIF_DATA BifData)
414 {
415     ACPI_PACKAGE_FIELD BifFields[] = {
416         { "PowerUnit", FALSE, &BifData->PowerUnit },
417         { "DesignCapacity", FALSE, &BifData->DesignCapacity },
418         { "LastFullCapacity", FALSE, &BifData->LastFullCapacity },
419         { "BatteryTechnology", FALSE, &BifData->BatteryTechnology },
420         { "DesignVoltage", FALSE, &BifData->DesignVoltage },
421         { "DesignCapacityWarning", FALSE, &BifData->DesignCapacityWarning },
422         { "DesignCapacityLow", FALSE, &BifData->DesignCapacityLow },
423         { "BatteryCapacityGranularity1", FALSE, &BifData->BatteryCapacityGranularity1 },
424         { "BatteryCapacityGranularity2", FALSE, &BifData->BatteryCapacityGranularity2 },
425         { "ModelNumber", TRUE, &BifData->ModelNumber },
426         { "SerialNumber", TRUE, &BifData->SerialNumber },
427         { "BatteryType", TRUE, &BifData->BatteryType },
428         { "OemInfo", TRUE, &BifData->OemInfo },
429     };
430     PAGED_CODE();
431 
432     if (CmBattDebug & CMBATT_ACPI_ENTRY_EXIT)
433         DbgPrint("CmBattGetBifData: Buffer (0x%x) Device %x Tid %x\n",
434                  BifData, DeviceExtension->DeviceId, KeGetCurrentThread());
435 
436     /* Request the _BIF method */
437     /* Note that _BIF method is deprecated since ACPI 4.0, and replaced by _BIX method.
438      * However, VirtualBox 7.0 only support _BIF method.
439      */
440     return CmBattCallAcpiPackage("CmBattGetBifData",
441                                  DeviceExtension,
442                                  'FIB_',
443                                  512,
444                                  BifFields,
445                                  RTL_NUMBER_OF(BifFields));
446 }
447 
448 /**
449  * @brief
450  * Retrieves the eXtended static battery information from the
451  * ACPI _BIX method.
452  *
453  * @param[in] DeviceExtension
454  * A pointer to a Control Method (CM) battery device extension.
455  * It is used to send the ACPI method evaluation operation
456  * to the ACPI driver of which it is attached to this CM battery.
457  *
458  * @param[out] BixData
459  * A pointer to a structure that contains the _BIX data fields,
460  * returned to caller.
461  *
462  * @return
463  * Returns STATUS_SUCCESS if the operation has succeeded successfully,
464  * otherwise a failure NTSTATUS code is returned.
465  */
466 NTSTATUS
467 NTAPI
CmBattGetBixData(_In_ PCMBATT_DEVICE_EXTENSION DeviceExtension,_Out_ PACPI_BIX_DATA BixData)468 CmBattGetBixData(
469     _In_ PCMBATT_DEVICE_EXTENSION DeviceExtension,
470     _Out_ PACPI_BIX_DATA BixData)
471 {
472     ACPI_PACKAGE_FIELD BixFields[] = {
473         { "Revision", FALSE, &BixData->Revision },
474         { "PowerUnit", FALSE, &BixData->PowerUnit },
475         { "DesignCapacity", FALSE, &BixData->DesignCapacity },
476         { "LastFullCapacity", FALSE, &BixData->LastFullCapacity },
477         { "BatteryTechnology", FALSE, &BixData->BatteryTechnology },
478         { "DesignVoltage", FALSE, &BixData->DesignVoltage },
479         { "DesignCapacityWarning", FALSE, &BixData->DesignCapacityWarning },
480         { "DesignCapacityLow", FALSE, &BixData->DesignCapacityLow },
481         { "CycleCount", FALSE, &BixData->CycleCount },
482         { "Accuracy", FALSE, &BixData->Accuracy },
483         { "MaxSampleTime", FALSE, &BixData->MaxSampleTime },
484         { "MinSampleTime", FALSE, &BixData->MinSampleTime },
485         { "MaxAverageInterval", FALSE, &BixData->MaxAverageInterval },
486         { "MinAverageInterval", FALSE, &BixData->MinAverageInterval },
487         { "BatteryCapacityGranularity1", FALSE, &BixData->BatteryCapacityGranularity1 },
488         { "BatteryCapacityGranularity2", FALSE, &BixData->BatteryCapacityGranularity2 },
489         { "ModelNumber", TRUE, &BixData->ModelNumber },
490         { "SerialNumber", TRUE, &BixData->SerialNumber },
491         { "BatteryType", TRUE, &BixData->BatteryType },
492         { "OemInfo", TRUE, &BixData->OemInfo },
493         { "SwapCapability", FALSE, &BixData->SwapCapability },
494     };
495     PAGED_CODE();
496 
497     if (CmBattDebug & CMBATT_ACPI_ENTRY_EXIT)
498     {
499         DbgPrint("CmBattGetBixData: Buffer (0x%x) Device %x Tid %x\n",
500                  BixData, DeviceExtension->DeviceId, KeGetCurrentThread());
501     }
502 
503     /* Request the ACPI driver to get the _BIX data for us */
504     return CmBattCallAcpiPackage("CmBattGetBifData",
505                                  DeviceExtension,
506                                  'XIB_',
507                                  512,
508                                  BixFields,
509                                  RTL_NUMBER_OF(BixFields));
510 }
511 
512 NTSTATUS
513 NTAPI
CmBattGetBstData(PCMBATT_DEVICE_EXTENSION DeviceExtension,PACPI_BST_DATA BstData)514 CmBattGetBstData(PCMBATT_DEVICE_EXTENSION DeviceExtension,
515                  PACPI_BST_DATA BstData)
516 {
517     ACPI_PACKAGE_FIELD BstFields[] = {
518         { "State", FALSE, &BstData->State },
519         { "PresentRate", FALSE, &BstData->PresentRate },
520         { "RemainingCapacity", FALSE, &BstData->RemainingCapacity },
521         { "PresentVoltage", FALSE, &BstData->PresentVoltage },
522     };
523     PAGED_CODE();
524 
525     if (CmBattDebug & CMBATT_ACPI_ENTRY_EXIT)
526         DbgPrint("CmBattGetBstData: Buffer (0x%x) Device %x Tid %x\n",
527                  BstData, DeviceExtension->DeviceId, KeGetCurrentThread());
528 
529 
530     return CmBattCallAcpiPackage("CmBattGetBstData",
531                                  DeviceExtension,
532                                  'TSB_',
533                                  512,
534                                  BstFields,
535                                  RTL_NUMBER_OF(BstFields));
536 }
537 
538 /* EOF */
539