2010-03-25 14:26:04 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Composite Battery Driver
|
|
|
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
|
|
|
* FILE: boot/drivers/bus/acpi/compbatt/compbatt.c
|
|
|
|
* PURPOSE: Main Initialization Code and IRP Handling
|
|
|
|
* PROGRAMMERS: ReactOS Portable Systems Group
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
|
|
|
|
#include "compbatt.h"
|
|
|
|
|
2014-01-27 23:11:48 +00:00
|
|
|
#include <debug.h>
|
|
|
|
|
2010-03-25 14:26:04 +00:00
|
|
|
/* GLOBALS ********************************************************************/
|
|
|
|
|
|
|
|
ULONG CompBattDebug;
|
|
|
|
|
|
|
|
/* FUNCTIONS ******************************************************************/
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattOpenClose(IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp)
|
|
|
|
{
|
2010-03-27 21:34:06 +00:00
|
|
|
PAGED_CODE();
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: ENTERING OpenClose\n");
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Complete the IRP with success */
|
|
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
|
|
Irp->IoStatus.Information = 0;
|
2010-03-29 00:02:05 +00:00
|
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Return success */
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: Exiting OpenClose\n");
|
|
|
|
return STATUS_SUCCESS;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattSystemControl(IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp)
|
|
|
|
{
|
2010-03-27 21:34:06 +00:00
|
|
|
PCOMPBATT_DEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
|
|
NTSTATUS Status;
|
|
|
|
PAGED_CODE();
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: ENTERING System Control\n");
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Are we attached yet? */
|
|
|
|
if (DeviceExtension->AttachedDevice)
|
|
|
|
{
|
|
|
|
/* Send it up the stack */
|
|
|
|
IoSkipCurrentIrpStackLocation(Irp);
|
|
|
|
Status = IoCallDriver(DeviceExtension->AttachedDevice, Irp);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* We don't support WMI */
|
|
|
|
Status = STATUS_NOT_SUPPORTED;
|
|
|
|
Irp->IoStatus.Status = STATUS_NOT_SUPPORTED;
|
2010-03-29 00:02:05 +00:00
|
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
2010-03-27 21:34:06 +00:00
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Return status */
|
|
|
|
return Status;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattMonitorIrpComplete(IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp,
|
|
|
|
IN PKEVENT Event)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2010-03-29 00:02:05 +00:00
|
|
|
CompBattMonitorIrpCompleteWorker(IN PCOMPBATT_BATTERY_DATA BatteryData)
|
2010-03-25 14:26:04 +00:00
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
VOID
|
2010-03-25 14:26:04 +00:00
|
|
|
NTAPI
|
|
|
|
CompBattRecalculateTag(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension)
|
|
|
|
{
|
2010-03-29 02:02:07 +00:00
|
|
|
PCOMPBATT_BATTERY_DATA BatteryData;
|
|
|
|
ULONG Tag;
|
|
|
|
PLIST_ENTRY ListHead, NextEntry;
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: ENTERING CompBattRecalculateTag\n");
|
|
|
|
|
|
|
|
/* Loop the battery list */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
ListHead = &DeviceExtension->BatteryList;
|
|
|
|
NextEntry = ListHead->Flink;
|
|
|
|
while (NextEntry != ListHead)
|
|
|
|
{
|
|
|
|
/* Get the battery information and check if it has a tag */
|
|
|
|
BatteryData = CONTAINING_RECORD(NextEntry, COMPBATT_BATTERY_DATA, BatteryLink);
|
|
|
|
if (BatteryData->Flags & COMPBATT_TAG_ASSIGNED)
|
|
|
|
{
|
|
|
|
/* Generate the next tag and exit */
|
|
|
|
Tag = DeviceExtension->NextTag;
|
|
|
|
DeviceExtension->Flags |= COMPBATT_TAG_ASSIGNED;
|
|
|
|
DeviceExtension->Tag = Tag;
|
|
|
|
DeviceExtension->NextTag = Tag + 1;
|
|
|
|
break;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* No tag for this device extension, clear it */
|
|
|
|
DeviceExtension->Tag = 0;
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
|
|
|
/* We're done */
|
2010-03-29 02:02:07 +00:00
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: EXITING CompBattRecalculateTag\n");
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattIoctl(IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp)
|
|
|
|
{
|
2010-03-27 21:34:06 +00:00
|
|
|
PCOMPBATT_DEVICE_EXTENSION DeviceExtension = DeviceObject->DeviceExtension;
|
|
|
|
NTSTATUS Status;
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: ENTERING Ioctl\n");
|
|
|
|
|
|
|
|
/* Let the class driver handle it */
|
|
|
|
Status = BatteryClassIoctl(DeviceExtension->ClassData, Irp);
|
|
|
|
if (Status == STATUS_NOT_SUPPORTED)
|
|
|
|
{
|
|
|
|
/* It failed, try the next driver up the stack */
|
|
|
|
Irp->IoStatus.Status = Status;
|
|
|
|
IoSkipCurrentIrpStackLocation(Irp);
|
|
|
|
Status = IoCallDriver(DeviceExtension->AttachedDevice, Irp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return status */
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: EXITING Ioctl\n");
|
|
|
|
return Status;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattQueryTag(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension,
|
|
|
|
OUT PULONG Tag)
|
|
|
|
{
|
2010-03-29 02:02:07 +00:00
|
|
|
NTSTATUS Status;
|
|
|
|
PAGED_CODE();
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: ENTERING QueryTag\n");
|
|
|
|
|
|
|
|
/* Was a tag assigned? */
|
|
|
|
if (!(DeviceExtension->Flags & COMPBATT_TAG_ASSIGNED))
|
|
|
|
{
|
|
|
|
/* Assign one */
|
|
|
|
CompBattRecalculateTag(DeviceExtension);
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Do we have a tag now? */
|
|
|
|
if ((DeviceExtension->Flags & COMPBATT_TAG_ASSIGNED) && (DeviceExtension->Tag))
|
|
|
|
{
|
|
|
|
/* Return the tag */
|
|
|
|
*Tag = DeviceExtension->Tag;
|
|
|
|
Status = STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* No tag */
|
|
|
|
*Tag = 0;
|
|
|
|
Status = STATUS_NO_SUCH_DEVICE;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Return status */
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: EXITING QueryTag\n");
|
|
|
|
return Status;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattDisableStatusNotify(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension)
|
|
|
|
{
|
2010-03-29 02:02:07 +00:00
|
|
|
PCOMPBATT_BATTERY_DATA BatteryData;
|
|
|
|
PLIST_ENTRY ListHead, NextEntry;
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: ENTERING DisableStatusNotify\n");
|
|
|
|
|
|
|
|
/* Loop the battery list */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
ListHead = &DeviceExtension->BatteryList;
|
|
|
|
NextEntry = ListHead->Flink;
|
|
|
|
while (NextEntry != ListHead)
|
|
|
|
{
|
|
|
|
/* Get the battery information and clear capacity data */
|
|
|
|
BatteryData = CONTAINING_RECORD(NextEntry, COMPBATT_BATTERY_DATA, BatteryLink);
|
|
|
|
BatteryData->WaitStatus.LowCapacity = 0;
|
|
|
|
BatteryData->WaitStatus.HighCapacity = 0x7FFFFFFF;
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Done */
|
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
|
|
|
if (CompBattDebug & 0x100) DbgPrint("CompBatt: EXITING DisableStatusNotify\n");
|
|
|
|
return STATUS_SUCCESS;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattSetStatusNotify(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension,
|
|
|
|
IN ULONG BatteryTag,
|
|
|
|
IN PBATTERY_NOTIFY BatteryNotify)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattQueryStatus(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension,
|
|
|
|
IN ULONG Tag,
|
|
|
|
IN PBATTERY_STATUS BatteryStatus)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2010-03-29 02:02:07 +00:00
|
|
|
CompBattGetBatteryInformation(OUT PBATTERY_INFORMATION BatteryInfo,
|
2010-03-25 14:26:04 +00:00
|
|
|
IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension)
|
|
|
|
{
|
2010-03-29 02:02:07 +00:00
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
BATTERY_QUERY_INFORMATION InputBuffer;
|
|
|
|
PCOMPBATT_BATTERY_DATA BatteryData;
|
|
|
|
PLIST_ENTRY ListHead, NextEntry;
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: ENTERING GetBatteryInformation\n");
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Set defaults */
|
|
|
|
BatteryInfo->DefaultAlert1 = 0;
|
|
|
|
BatteryInfo->DefaultAlert2 = 0;
|
|
|
|
BatteryInfo->CriticalBias = 0;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Loop the battery list */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
ListHead = &DeviceExtension->BatteryList;
|
|
|
|
NextEntry = ListHead->Flink;
|
|
|
|
while (NextEntry != ListHead)
|
|
|
|
{
|
|
|
|
/* Try to acquire the remove lock */
|
|
|
|
BatteryData = CONTAINING_RECORD(NextEntry, COMPBATT_BATTERY_DATA, BatteryLink);
|
|
|
|
if (NT_SUCCESS(IoAcquireRemoveLock(&BatteryData->RemoveLock, 0)))
|
|
|
|
{
|
|
|
|
/* Now release the device lock since the battery can't go away */
|
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Build the query */
|
|
|
|
InputBuffer.BatteryTag = BatteryData->Tag;
|
|
|
|
InputBuffer.InformationLevel = BatteryInformation;
|
|
|
|
InputBuffer.AtRate = 0;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Make sure the battery has a tag */
|
|
|
|
if (BatteryData->Tag)
|
|
|
|
{
|
|
|
|
/* Do we already have the data? */
|
|
|
|
if (!(BatteryData->Flags & COMPBATT_BATTERY_INFORMATION_PRESENT))
|
|
|
|
{
|
|
|
|
/* Send the IOCTL to query the information */
|
|
|
|
RtlZeroMemory(&BatteryData->BatteryInformation,
|
|
|
|
sizeof(BatteryData->BatteryInformation));
|
|
|
|
Status = BatteryIoctl(IOCTL_BATTERY_QUERY_INFORMATION,
|
|
|
|
BatteryData->DeviceObject,
|
|
|
|
&InputBuffer,
|
|
|
|
sizeof(InputBuffer),
|
|
|
|
&BatteryData->BatteryInformation,
|
|
|
|
sizeof(BatteryData->BatteryInformation),
|
|
|
|
0);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Fail if the query had a problem */
|
|
|
|
if (Status == STATUS_DEVICE_REMOVED) Status = STATUS_NO_SUCH_DEVICE;
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
IoReleaseRemoveLock(&BatteryData->RemoveLock, 0);
|
|
|
|
break;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Next time we can use the static copy */
|
|
|
|
BatteryData->Flags |= COMPBATT_BATTERY_INFORMATION_PRESENT;
|
|
|
|
if (CompBattDebug & 2)
|
|
|
|
DbgPrint("CompBattGetBatteryInformation: Read individual BATTERY_INFORMATION\n"
|
|
|
|
"-------- Capabilities = %x\n-------- Technology = %x\n"
|
|
|
|
"-------- Chemistry[4] = %x\n-------- DesignedCapacity = %x\n"
|
|
|
|
"-------- FullChargedCapacity = %x\n-------- DefaultAlert1 = %x\n"
|
|
|
|
"-------- DefaultAlert2 = %x\n-------- CriticalBias = %x\n"
|
|
|
|
"-------- CycleCount = %x\n",
|
|
|
|
BatteryData->BatteryInformation.Capabilities,
|
|
|
|
BatteryData->BatteryInformation.Technology,
|
|
|
|
BatteryData->BatteryInformation.Chemistry,
|
|
|
|
BatteryData->BatteryInformation.DesignedCapacity,
|
|
|
|
BatteryData->BatteryInformation.FullChargedCapacity,
|
|
|
|
BatteryData->BatteryInformation.DefaultAlert1,
|
|
|
|
BatteryData->BatteryInformation.DefaultAlert2,
|
|
|
|
BatteryData->BatteryInformation.CriticalBias,
|
|
|
|
BatteryData->BatteryInformation.CycleCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Combine capabilities */
|
|
|
|
BatteryInfo->Capabilities |= BatteryData->BatteryInformation.Capabilities;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Add-on capacity */
|
|
|
|
if (BatteryData->BatteryInformation.DesignedCapacity != BATTERY_UNKNOWN_CAPACITY)
|
|
|
|
{
|
|
|
|
BatteryInfo->DesignedCapacity += BatteryData->BatteryInformation.DesignedCapacity;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add on fully charged capacity */
|
|
|
|
if (BatteryData->BatteryInformation.FullChargedCapacity != BATTERY_UNKNOWN_CAPACITY)
|
|
|
|
{
|
|
|
|
BatteryInfo->FullChargedCapacity += BatteryData->BatteryInformation.FullChargedCapacity;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Choose the highest alert */
|
|
|
|
BatteryInfo->DefaultAlert1 = max(BatteryInfo->DefaultAlert1,
|
|
|
|
BatteryData->BatteryInformation.DefaultAlert1);
|
|
|
|
|
|
|
|
/* Choose the highest alert */
|
|
|
|
BatteryInfo->DefaultAlert2 = max(BatteryInfo->DefaultAlert2,
|
|
|
|
BatteryData->BatteryInformation.DefaultAlert2);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Choose the highest critical bias */
|
|
|
|
BatteryInfo->CriticalBias = max(BatteryInfo->CriticalBias,
|
|
|
|
BatteryData->BatteryInformation.CriticalBias);
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Re-acquire the device extension lock and release the remove lock */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
IoReleaseRemoveLock(&BatteryData->RemoveLock, 0);
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Next entry */
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
|
|
|
/* We are done with the list, check if the information was queried okay */
|
2010-03-29 02:02:07 +00:00
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* If there's no fully charged capacity, use the design capacity */
|
|
|
|
if (!BatteryInfo->FullChargedCapacity)
|
|
|
|
{
|
|
|
|
BatteryInfo->FullChargedCapacity = BatteryInfo->DesignedCapacity;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Print out final combined data */
|
|
|
|
if (CompBattDebug & 2)
|
|
|
|
DbgPrint("CompBattGetBatteryInformation: Returning BATTERY_INFORMATION\n"
|
|
|
|
"-------- Capabilities = %x\n-------- Technology = %x\n"
|
|
|
|
"-------- Chemistry[4] = %x\n-------- DesignedCapacity = %x\n"
|
|
|
|
"-------- FullChargedCapacity = %x\n-------- DefaultAlert1 = %x\n"
|
|
|
|
"-------- DefaultAlert2 = %x\n-------- CriticalBias = %x\n"
|
|
|
|
"-------- CycleCount = %x\n",
|
|
|
|
BatteryInfo->Capabilities,
|
|
|
|
BatteryInfo->Technology,
|
|
|
|
BatteryInfo->Chemistry,
|
|
|
|
BatteryInfo->DesignedCapacity,
|
|
|
|
BatteryInfo->FullChargedCapacity,
|
|
|
|
BatteryInfo->DefaultAlert1,
|
|
|
|
BatteryInfo->DefaultAlert2,
|
|
|
|
BatteryInfo->CriticalBias,
|
|
|
|
BatteryInfo->CycleCount);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* Copy the data into the device extension */
|
|
|
|
RtlCopyMemory(&DeviceExtension->BatteryInformation,
|
|
|
|
BatteryInfo,
|
|
|
|
sizeof(DeviceExtension->BatteryInformation));
|
|
|
|
DeviceExtension->Flags |= COMPBATT_BATTERY_INFORMATION_PRESENT;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:02:07 +00:00
|
|
|
/* We are done */
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: EXITING GetBatteryInformation\n");
|
|
|
|
return Status;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattGetBatteryGranularity(OUT PBATTERY_REPORTING_SCALE ReportingScale,
|
|
|
|
IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension)
|
|
|
|
{
|
2010-03-29 02:27:41 +00:00
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
BATTERY_QUERY_INFORMATION InputBuffer;
|
|
|
|
PCOMPBATT_BATTERY_DATA BatteryData;
|
|
|
|
BATTERY_REPORTING_SCALE BatteryScale[4];
|
|
|
|
PLIST_ENTRY ListHead, NextEntry;
|
|
|
|
ULONG i;
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: ENTERING GetBatteryGranularity\n");
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Set defaults */
|
|
|
|
ReportingScale[0].Granularity = -1;
|
|
|
|
ReportingScale[1].Granularity = -1;
|
|
|
|
ReportingScale[2].Granularity = -1;
|
|
|
|
ReportingScale[3].Granularity = -1;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Loop the battery list */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
ListHead = &DeviceExtension->BatteryList;
|
|
|
|
NextEntry = ListHead->Flink;
|
|
|
|
while (NextEntry != ListHead)
|
|
|
|
{
|
|
|
|
/* Try to acquire the remove lock */
|
|
|
|
BatteryData = CONTAINING_RECORD(NextEntry, COMPBATT_BATTERY_DATA, BatteryLink);
|
|
|
|
if (NT_SUCCESS(IoAcquireRemoveLock(&BatteryData->RemoveLock, 0)))
|
|
|
|
{
|
|
|
|
/* Now release the device lock since the battery can't go away */
|
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Build the query */
|
|
|
|
InputBuffer.BatteryTag = BatteryData->Tag;
|
|
|
|
InputBuffer.InformationLevel = BatteryGranularityInformation;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Make sure the battery has a tag */
|
|
|
|
if (BatteryData->Tag)
|
|
|
|
{
|
|
|
|
/* Send the IOCTL to query the information */
|
|
|
|
RtlZeroMemory(&BatteryData->BatteryInformation,
|
|
|
|
sizeof(BatteryData->BatteryInformation));
|
|
|
|
Status = BatteryIoctl(IOCTL_BATTERY_QUERY_INFORMATION,
|
|
|
|
BatteryData->DeviceObject,
|
|
|
|
&InputBuffer,
|
|
|
|
sizeof(InputBuffer),
|
|
|
|
&BatteryScale,
|
|
|
|
sizeof(BatteryScale),
|
|
|
|
0);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Fail if the query had a problem */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
IoReleaseRemoveLock(&BatteryData->RemoveLock, 0);
|
|
|
|
break;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Loop all 4 scales */
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
/* Check for valid granularity */
|
|
|
|
if (BatteryScale[i].Granularity)
|
|
|
|
{
|
|
|
|
/* If it's smaller, use it instead */
|
|
|
|
ReportingScale[i].Granularity = min(BatteryScale[i].Granularity,
|
|
|
|
ReportingScale[i].Granularity);
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Re-acquire the device extension lock and release the remove lock */
|
|
|
|
ExAcquireFastMutex(&DeviceExtension->Lock);
|
|
|
|
IoReleaseRemoveLock(&BatteryData->RemoveLock, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Next entry */
|
|
|
|
NextEntry = NextEntry->Flink;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* All done */
|
|
|
|
ExReleaseFastMutex(&DeviceExtension->Lock);
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: EXITING GetBatteryGranularity\n");
|
|
|
|
return STATUS_SUCCESS;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
CompBattGetEstimatedTime(OUT PULONG Time,
|
|
|
|
IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-25 14:26:04 +00:00
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
2010-03-29 02:27:41 +00:00
|
|
|
CompBattQueryInformation(IN PCOMPBATT_DEVICE_EXTENSION DeviceExtension,
|
2010-03-25 14:26:04 +00:00
|
|
|
IN ULONG Tag,
|
|
|
|
IN BATTERY_QUERY_INFORMATION_LEVEL InfoLevel,
|
|
|
|
IN OPTIONAL LONG AtRate,
|
|
|
|
IN PVOID Buffer,
|
|
|
|
IN ULONG BufferLength,
|
|
|
|
OUT PULONG ReturnedLength)
|
|
|
|
{
|
2010-03-29 02:27:41 +00:00
|
|
|
BATTERY_INFORMATION BatteryInfo;
|
|
|
|
BATTERY_REPORTING_SCALE BatteryGranularity[4];
|
|
|
|
PWCHAR BatteryName = L"Composite Battery";
|
2011-09-11 00:40:20 +00:00
|
|
|
//BATTERY_MANUFACTURE_DATE Date;
|
2010-03-29 02:27:41 +00:00
|
|
|
ULONG Dummy, Time;
|
|
|
|
PVOID QueryData = NULL;
|
|
|
|
ULONG QueryLength = 0;
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
PAGED_CODE();
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: ENTERING QueryInformation\n");
|
|
|
|
|
|
|
|
/* Check for valid/correct tag */
|
|
|
|
if ((Tag != DeviceExtension->Tag) ||
|
|
|
|
(!(DeviceExtension->Flags & COMPBATT_TAG_ASSIGNED)))
|
|
|
|
{
|
|
|
|
/* Not right, so fail */
|
|
|
|
return STATUS_NO_SUCH_DEVICE;
|
|
|
|
}
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Check what caller wants */
|
|
|
|
switch (InfoLevel)
|
|
|
|
{
|
|
|
|
case BatteryInformation:
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Query combined battery information */
|
|
|
|
RtlZeroMemory(&BatteryInfo, sizeof(BatteryInfo));
|
|
|
|
Status = CompBattGetBatteryInformation(&BatteryInfo, DeviceExtension);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Return the data if successful */
|
|
|
|
QueryData = &BatteryInfo;
|
|
|
|
QueryLength = sizeof(BatteryInfo);
|
|
|
|
}
|
|
|
|
break;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
case BatteryGranularityInformation:
|
|
|
|
|
|
|
|
/* Query combined granularity information */
|
|
|
|
RtlZeroMemory(&BatteryGranularity, sizeof(BatteryGranularity));
|
|
|
|
Status = CompBattGetBatteryGranularity(BatteryGranularity, DeviceExtension);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Return the data if successful */
|
|
|
|
QueryLength = sizeof(BatteryGranularity);
|
|
|
|
QueryData = &BatteryGranularity;
|
|
|
|
}
|
|
|
|
break;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
case BatteryEstimatedTime:
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Query combined time estimate information */
|
|
|
|
RtlZeroMemory(&Time, sizeof(Time));
|
|
|
|
Status = CompBattGetEstimatedTime(&Time, DeviceExtension);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Return the data if successful */
|
|
|
|
QueryLength = sizeof(Time);
|
2021-06-11 12:29:21 +00:00
|
|
|
QueryData = &Time;
|
2010-03-29 02:27:41 +00:00
|
|
|
}
|
|
|
|
break;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
case BatteryManufactureName:
|
|
|
|
case BatteryDeviceName:
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Return the static buffer */
|
|
|
|
QueryData = BatteryName;
|
|
|
|
QueryLength = sizeof(L"Composite Battery");
|
|
|
|
break;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
case BatteryManufactureDate:
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Static data */
|
2011-09-11 00:40:20 +00:00
|
|
|
//Date.Day = 26;
|
|
|
|
//Date.Month = 06;
|
|
|
|
//Date.Year = 1997;
|
2010-03-29 02:27:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BatteryTemperature:
|
|
|
|
case BatteryUniqueID:
|
|
|
|
|
|
|
|
/* Return zero */
|
|
|
|
Dummy = 0;
|
|
|
|
QueryData = &Dummy;
|
|
|
|
QueryLength = sizeof(Dummy);
|
|
|
|
break;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
default:
|
|
|
|
/* Everything else is unknown */
|
|
|
|
Status = STATUS_INVALID_PARAMETER;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return the required length and check if the caller supplied enough */
|
|
|
|
*ReturnedLength = QueryLength;
|
|
|
|
if (BufferLength < QueryLength) Status = STATUS_BUFFER_TOO_SMALL;
|
|
|
|
|
|
|
|
/* Copy the data if there's enough space and it exists */
|
|
|
|
if ((NT_SUCCESS(Status)) && (QueryData)) RtlCopyMemory(Buffer, QueryData, QueryLength);
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-29 02:27:41 +00:00
|
|
|
/* Return function result */
|
|
|
|
if (CompBattDebug & 1) DbgPrint("CompBatt: EXITING QueryInformation\n");
|
|
|
|
return Status;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
DriverEntry(IN PDRIVER_OBJECT DriverObject,
|
|
|
|
IN PUNICODE_STRING RegistryPath)
|
|
|
|
{
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Register add device routine */
|
|
|
|
DriverObject->DriverExtension->AddDevice = CompBattAddDevice;
|
2021-06-11 12:29:21 +00:00
|
|
|
|
2010-03-27 21:34:06 +00:00
|
|
|
/* Register other handlers */
|
2010-03-29 02:02:07 +00:00
|
|
|
DriverObject->MajorFunction[IRP_MJ_CREATE] = CompBattOpenClose;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_CLOSE] = CompBattOpenClose;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = CompBattIoctl;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_POWER] = CompBattPowerDispatch;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_SYSTEM_CONTROL] = CompBattSystemControl;
|
|
|
|
DriverObject->MajorFunction[IRP_MJ_PNP] = CompBattPnpDispatch;
|
2010-03-27 21:34:06 +00:00
|
|
|
return STATUS_SUCCESS;
|
2010-03-25 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|