The following implements an Advanced Configuration and Power Interface (ACPI) based battery manager kernel extension (kext/driver) for laptops. It should work correctly on any laptop that correctly implements the ACPI standard DSDT methods as defined in the Advanced Configuration and Power Interface Specification 4.0a. Reference documents used:
The driver has both generic and HP specific functionality that is configurable and it will probably work on all platforms that properly implement the _BST, _BIF and/or _BIX methods in their DSDT. Therefore, there are several installation methods available:
Generic platforms:
- ACPI 3.x _BIF method: Configure key "UseExtendedBatteryInformationMethod" to false in Info.plist and install kext.
- ACPI 4.x _BIX method: Configure key "UseExtendedBatteryInformationMethod" to true in Info.plist and install kext.
- Ensure your ACPI _BIF, _BIX, _BST methods return data according to the ACPI specification referenced above.
HP DVx platforms:
As the HP _BST method on DVx isn't (IMHO) implemented correctly, DVx owners will also have to implement DSDT edits so that the _BST method returns proper values to the driver in order to calculate the time remaining (to charge or discharge) value. In addition, I have rewritten the _BIF method, created a new _BIX and BBIX (Bigger or Better _BIX) to correctly talk to the Smart Battery Subsystem (SBS) via the System Management Bus (SMBus) via the Embedded Controller (EC). This allows the reading of information from the battery that the HP methods do not include like cycle count, temperature and date of manufacture.
- To support ACPI 3.x _BIF method: Configure key "UseExtendedBatteryInformationMethod" to false in Info.plist and install kext, Modify DSDT methods UPBS, UPBI and ITOS.
- To support ACPI 4.x _BIX method: Configure key "UseExtendedBatteryInformationMethod" to true in Info.plist and install kext, Modify DSDT methods UPBS, UPBI, ITOS and add _BIX, UPBX and IVBX methods.
- To support non-ACPI BBIX method: Configure key "UseExtraBatteryInformationMethod" to true in Info.plist and install kext, Modify DSDT methods UPBS, UPBI and ITOS and add BBIX, UPBG and IVBG methods.
The BBIX method, although non-standard, returns *all* of the battery information available from SBS including temperature and is basically an extension of the _BIX method.
This driver polls the battery state by default every 30 seconds. If you wish to override this value to make the system respond to battery events quicker, you can set "BatteryPollingPeriodOverride" in the Info.plist to the number of seconds between polls.
This was tested on an HP DV8 but will likely work on any DVx model that implements similar DSDT methods. Also, I have tested this on both 32-bit/64-bit kernels and the driver has been compiled to a 10.6 32/64-bit universal (FAT) binary. It will also work on 10.7 Lion provided your DSDT methods are performing 8-bit I/O access to the SMBus if accessed through the EC, See "Optional Step 2d: Lion Support for DVx" section for details.
Step 1: ACPI 3.x/4.x _BST DSDT edit (Common to all optional steps below):
In your DSDT.dsl, locate the BAT0 device and add/replace the following:
CODE
// Battery status storage
Name (PBST, Package (0x04)
{
Zero, // 0x00, Battery state
0xFFFFFFFF, // 0x01, Battery present
rate 0xFFFFFFFF, // 0x02, Battery remaining capacity
0x2710 // 0x03, Battery present voltage
})
// Get battery status into PBST
Method (UPBS, 0, NotSerialized)
{
Store (^^PCI0.LPCB.EC0.MBST, Index (PBST, 0x00)) // 0x00, Battery state
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0a, RefOf (Local0)) // Current()
Store (Local0, Index (PBST, 0x01)) // 0x01, Battery present rate
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0f, RefOf (Local1)) // RemainingCapacity()
Store (Local1, Index (PBST, 0x02)) // 0x02, Battery remaining capacity
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x09, RefOf (Local2)) // Voltage()
Store (Local2, Index (PBST, 0x03)) // 0x03, Battery present voltage
}
// InValid Battery Status
// Store placeholder battery status in PBST when no battery installed
Method (IVBS, 0, NotSerialized)
{
Store (Zero, Index (PBST, Zero)) // 0x00, Battery state
Store (0xFFFFFFFF, Index (PBST, One)) // 0x01, Battery present rate
Store (0xFFFFFFFF, Index (PBST, 0x02)) // 0x02, Battery remaining capacity
Store (0x2710, Index (PBST, 0x03)) // 0x03, Battery present voltage (10000)
}
There is a bug in the HP ITOS() Method used by the optional methods below so locate ITOS() under the Scope(_SB) namespace and replace it with:
CODE
// Function: Integer to ASCII/OS String
//
// Arg0 = Integer (DWord) to convert
// Return = Buffer of ASCII representation (Length = 0x5)
Method (ITOS, 1, NotSerialized)
{
// Buffer to store converted string
Store (Buffer (0x06)
{
0x20, 0x20, 0x20, 0x20, 0x20, 0x00
}, Local0)
// Lookup table for ASCII digit
Store (Buffer (0x11)
{
"0123456789ABCDEF"
}, Local7)
Store (0x05, Local1) // Counter
Store (Zero, Local2) // Index into Local0 (String)
While (Local1)
{
Decrement (Local1)
And (ShiftRight (Arg0, ShiftLeft (Local1, 0x02)), 0x0F, Local4) // Get next digit to convert
GBFE (Local7, Local4, RefOf (Local5)) // Get ACSII version from Local7 lookup table
PBFE (Local0, Local2, Local5) // Put digit in string buffer
Increment (Local2) // Index++
}
Store(Zero, Index(Local0, Local2)) // Ensure string ends with '\0'
Return (Local0)
}
Optional Step 2a: ACPI 3.x _BIF with SBS information DSDT edit
In your DSDT.dsl, locate the BAT0 device and add/replace the following:
CODE
// Battery information storage
Name (PBIF, Package (0x0D)
{
0x00000001, // 0x00, Power
Unit 0xFFFFFFFF, // 0x01, Design Capacity
0xFFFFFFFF, // 0x02, Last Full Charge Capacity
0x00000001, // 0x03, Battery technology
0xFFFFFFFF, // 0x04, Design voltage
0x000000FA, // 0x05, Design capacity of warning
0x00000096, // 0x06, Design capacity of low
0x0000000A, // 0x07, Battery capacity gradularity 1
0x00000019, // 0x08, Battery capacity gradularity 2
" ", // 0x09, Model number
" ", // 0x0a, Serial number
" ", // 0x0b, Battery type
" " // 0x0c, OEM Information
})
// Get and store battery information in PBIF
Method (UPBI, 0, NotSerialized)
{
Store (0x01, Index (PBIF, 0x00)) // 0x00, Power Unit
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x18, RefOf (Local0)) // DesignCapacity() - WORD - unsigned int
Store (Local0, Index (PBIF, 0x01)) // 0x01, Design Capacity
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x10, RefOf (Local1)) // FullChargeCapacity() - WORD - unsigned int
Store (Local1, Index (PBIF, 0x02)) // 0x02, Last Full Charge Capacity
Store (0x01, Index (PBIF, 0x03)) // 0x03, Battery technology
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x19, RefOf (Local2)) // DesignVoltage() - WORD - unsigned int
Store (Local2, Index (PBIF, 0x04)) // 0x04, Design voltage
Store (0xFA, Index (PBIF, 0x05)) // 0x05, Design capacity of warning
Store (0x96, Index (PBIF, 0x06)) // 0x06, Design capacity of low
Store (0x0A, Index (PBIF, 0x07)) // 0x07, Battery capacity gradularity 1
Store (0x19, Index (PBIF, 0x08)) // 0x08, Battery capacity gradularity 2
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x21, RefOf (Local3)) // DeviceName() - BLOCK - string
Store (Local3, Index (PBIF, 0x09)) // 0x09, Model number
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x1c, RefOf (Local4)) // SerialNumber() - WORD - unsigned int
Store (ITOS (ToBCD (Local4)), Index (PBIF, 0x0a)) // 0x0a, Serial number
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x22, RefOf (Local5)) // DeviceChemistry() - BLOCK - string
Store (Local5, Index (PBIF, 0x0b)) // 0x0b, Battery type
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x20, RefOf (Local6)) // ManufacturerName() - BLOCK - string
Store (Local6, Index (PBIF, 0x0c)) // 0x0c, OEM Information
}
// InValid Battery Information
// Store placeholder battery information in PBIF when no battery installed
Method (IVBI, 0, NotSerialized)
{
Store (0x00000001, Index (PBIF, Zero)) // 0x00, Power Unit
Store (0xFFFFFFFF, Index (PBIF, One)) // 0x01, Design Capacity
Store (0xFFFFFFFF, Index (PBIF, 0x02)) // 0x02, Last Full Charge Capacity
Store (0x00000001, Index (PBIF, 0x03)) // 0x03, Battery technology
Store (0xFFFFFFFF, Index (PBIF, 0x04)) // 0x04, Design voltage
Store (0x000000FA, Index (PBIF, 0x05)) // 0x05, Design capacity of warning
Store (0x00000096, Index (PBIF, 0x06)) // 0x06, Design capacity of low
Store (0x0000000A, Index (PBIF, 0x07)) // 0x07, Battery capacity gradularity 1
Store (0x00000019, Index (PBIF, 0x08)) // 0x08, Battery capacity gradularity 2
Store ("Bad", Index (PBIF, 0x09)) // 0x09, Model number
Store ("Bad", Index (PBIF, 0x0A)) // 0x0a, Serial number
Store ("Bad", Index (PBIF, 0x0B)) // 0x0b, Battery type
Store ("Bad", Index (PBIF, 0x0C)) // 0x0c, OEM Information
}
Optional Step 2b: ACPI 4.x _BIX with SBS information DSDT edit
In your DSDT.dsl, locate the BAT0 device and add/replace the following:
CODE
// Battery information extended (ACPI 4.0)
Name (PBIX, Package (0x14)
{
0x00000001, // 0x00, Revision //Integer
0x00000001, // 0x01, Power Unit //Integer (DWORD)
0xFFFFFFFF, // 0x02, Design Capacity //Integer (DWORD)
0xFFFFFFFF, // 0x03, Last Full Charge Capacity //Integer (DWORD)
0x00000001, // 0x04, Battery Technology //Integer (DWORD)
0xFFFFFFFF, // 0x05, Design Voltage //Integer (DWORD)
0x000000FA, // 0x06, Design Capacity of Warning //Integer (DWORD)
0x00000096, // 0x07, Design Capacity of Low //Integer (DWORD)
0x00000000, // 0x08, Cycle Count //Integer (DWORD)
0x00100000, // 0x09, Measurement Accuracy //Integer (DWORD)
0xFFFFFFFF, // 0x0a, Max Sampling Time //Integer (DWORD)
0xFFFFFFFF, // 0x0b, Min Sampling Time //Integer (DWORD)
0xFFFFFFFF, // 0x0c, Max Averaging Interval //Integer (DWORD)
0xFFFFFFFF, // 0x0d, Min Averaging Interval //Integer (DWORD)
0x0000000A, // 0x0e, Battery Capacity Granularity 1 //Integer (DWORD)
0x00000019, // 0x0f, Battery Capacity Granularity 2 //Integer (DWORD)
" ", // 0x10, Model Number //String (ASCIIZ)
" ", // 0x11, Serial Number //String (ASCIIZ)
" ", // 0x12, Battery Type //String (ASCIIZ)
" " // 0x13, OEM Information //String (ASCIIZ)
})
// Return battery extended information in PBIX
Method (_BIX, 0, NotSerialized)
{
If (ECON)
{
If (^^PCI0.LPCB.EC0.MBTS)
{
UPBX () // Return battery information
}
Else
{
IVBX () // No battery, return placeholder info
}
}
Else
{
IVBX () // No battery, return placeholder info
}
Return (PBIX)
}
// Get and store battery extended information in PBIX
Method (UPBX, 0, NotSerialized)
{
Store (0x00, Index (PBIX, 0x00)) // 0x01, Revision
Store (0x01, Index (PBIX, 0x01)) // 0x01, Power Unit
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x18, RefOf (Local0)) // DesignCapacity() - WORD - unsigned int
Store (Local0, Index (PBIX, 0x02)) // 0x02, Design Capacity
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x10, RefOf (Local1)) // FullChargeCapacity() - WORD - unsigned int
Store (Local1, Index (PBIX, 0x03)) // 0x03, Last Full Charge Capacity
Store (0x01, Index (PBIX, 0x04)) // 0x04, Battery technology
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x19, RefOf (Local2)) // DesignVoltage() - WORD - unsigned int
Store (Local2, Index (PBIX, 0x05)) // 0x05, Design voltage
Store (0xFA, Index (PBIX, 0x06)) // 0x06, Design capacity of warning
Store (0x96, Index (PBIX, 0x07)) // 0x07, Design capacity of low
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x17, RefOf (Local0)) // CycleCount() - WORD - unsigned int
Store (Local0, Index (PBIX, 0x08)) // 0x08, Cycle Count
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0c, RefOf (Local0)) // MaxError() - WORD - unsigned int
Multiply (Local0, 0x1000, Index (PBIX, 0x09)) // 0x09, Measurement Accuracy
Store (0xFFFFFFFF, Index (PBIX, 0x0a)) // 0x0a, Max Sampling Time
Store (0xFFFFFFFF, Index (PBIX, 0x0b)) // 0x0b, Min Sampling Time
Store (0x0000EA60, Index (PBIX, 0x0c)) // 0x0c, Max Averaging Interval
Store (0x0000EA60, Index (PBIX, 0x0d)) // 0x0d, Min Averaging Interval
Store (0x0A, Index (PBIX, 0x0e)) // 0x0e, Battery capacity gradularity 1
Store (0x19, Index (PBIX, 0x0f)) // 0x0f, Battery capacity gradularity 2
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x21, RefOf (Local3)) // DeviceName() - BLOCK - string
Store (Local3, Index (PBIX, 0x10)) // 0x10, Model number
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x1c, RefOf (Local4)) // SerialNumber() - WORD - unsigned int
Store (ITOS (ToBCD (Local4)), Index (PBIX, 0x11)) // 0x11, Serial number
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x22, RefOf (Local5)) // DeviceChemistry() - BLOCK - string
Store (Local5, Index (PBIX, 0x12)) // 0x12, Battery type
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x20, RefOf (Local6)) // ManufacturerName() - BLOCK - string
Store (Local6, Index (PBIX, 0x13)) // 0x0c, OEM Information
}
// InValid Battery eXtended Information
// Store placeholder battery status in PBIX when no battery installed
Method (IVBX, 0, NotSerialized)
{
Store (0x00000000, Index (PBIX, 0x00)) // 0x00, Revision
Store (0x00000001, Index (PBIX, 0x01)) // 0x01, Power Unit
Store (0xFFFFFFFF, Index (PBIX, 0x02)) // 0x02, Design Capacity
Store (0xFFFFFFFF, Index (PBIX, 0x03)) // 0x03, Last Full Charge Capacity
Store (0x00000001, Index (PBIX, 0x04)) // 0x04, Battery technology
Store (0xFFFFFFFF, Index (PBIX, 0x05)) // 0x05, Design voltage
Store (0x000000FA, Index (PBIX, 0x06)) // 0x06, Design capacity of warning
Store (0x00000096, Index (PBIX, 0x07)) // 0x07, Design capacity of low
Store (0x00000000, Index (PBIX, 0x08)) // 0x08, Cycle Count
Store (0x00100000, Index (PBIX, 0x09)) // 0x09, Measurement Accuracy
Store (0xFFFFFFFF, Index (PBIX, 0x0a)) // 0x0a, Max Sampling Time
Store (0xFFFFFFFF, Index (PBIX, 0x0b)) // 0x0b, Min Sampling Time
Store (0x0000EA60, Index (PBIX, 0x0c)) // 0x0c, Max Averaging Interval
Store (0x0000EA60, Index (PBIX, 0x0d)) // 0x0d, Min Averaging Interval
Store (0x0000000A, Index (PBIX, 0x0e)) // 0x0e, Battery capacity gradularity 1
Store (0x00000019, Index (PBIX, 0x0f)) // 0x0f, Battery capacity gradularity 2
Store ("Bad", Index (PBIX, 0x10)) // 0x10, Model number
Store ("Bad", Index (PBIX, 0x11)) // 0x11, Serial number
Store ("Bad", Index (PBIX, 0x12)) // 0x12, Battery type
Store ("Bad", Index (PBIX, 0x13)) // 0x0c, OEM Information
}
Optional Step 2c: BBIX with all SBS information DSDT edit
In your DSDT.dsl, locate the BAT0 device and add/replace the following:
CODE
// Battery information - Non-standard stuff OSX will use
Name (PBIG, Package (0x10)
{
0x00000000, // 0x00, ManufacturerAccess() - WORD - ?
0x00000000, // 0x01, BatteryMode() - WORD - unsigned int
0xFFFFFFFF, // 0x02, AtRateTimeToFull() - WORD - unsigned int (min)
0xFFFFFFFF, // 0x03, AtRateTimeToEmpty() - WORD - unsigned int (min)
0x00000000, // 0x04, Temperature() - WORD - unsigned int (0.1K)
0x00000000, // 0x05, Voltage() - WORD - unsigned int (mV)
0x00000000, // 0x06, Current() - WORD - signed int (mA)
0x00000000, // 0x07, AverageCurrent() - WORD - signed int (mA)
0x00000000, // 0x08, RelativeStateOfCharge() - WORD - unsigned int (%)
0x00000000, // 0x09, AbsoluteStateOfCharge() - WORD - unsigned int (%)
0x00000000, // 0x0a, RemaingingCapacity() - WORD - unsigned int (mAh or 10mWh)
0xFFFFFFFF, // 0x0b, RunTimeToEmpty() - WORD - unsigned int (min)
0xFFFFFFFF, // 0x0c, AverageTimeToEmpty() - WORD - unsigned int (min)
0xFFFFFFFF, // 0x0d, AverageTimeToFull() - WORD - unsigned int (min)
0x00000000, // 0x0e, ManufactureDate() - WORD - unsigned int (packed date)
" " // 0x0f, ManufacturerData() - BLOCK - Unknown
})
// Return battery extra information in PBIG
Method (BBIX, 0, NotSerialized)
{
If (ECON)
{
If (^^PCI0.LPCB.EC0.MBTS)
{
UPBG () // Return battery information
}
Else
{
IVBG () // No battery, return placeholder info
}
}
Else
{
IVBG () // No battery, return placeholder info
}
Return (PBIG)
}
// Get and store battery extra information in PBIG
Method (UPBG, 0, NotSerialized)
{
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x00, RefOf (Local0)) // ManufacturerAccess() - WORD - ?
Store (Local0, Index (PBIG, 0x00))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x03, RefOf (Local0)) // BatteryMode() - WORD - unsigned int
Store (Local0, Index (PBIG, 0x01))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x05, RefOf (Local0)) // AtRateTimeToFull() - WORD - unsigned int (min)
Store (Local0, Index (PBIG, 0x02))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x06, RefOf (Local0)) // AtRateTimeToEmpty() - WORD - unsigned int (min)
Store (Local0, Index (PBIG, 0x03))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x08, RefOf (Local0)) // Temperature() - WORD - unsigned int (0.1K)
Store (Local0, Index (PBIG, 0x04))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x09, RefOf (Local0)) // Voltage() - WORD - unsigned int (mV)
Store (Local0, Index (PBIG, 0x05))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0a, RefOf (Local0)) // Current() - WORD - signed int (mA)
Store (Local0, Index (PBIG, 0x06))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0b, RefOf (Local0)) // AverageCurrent() - WORD - signed int (mA)
Store (Local0, Index (PBIG, 0x07))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0d, RefOf (Local0)) // RelativeStateOfCharge() - WORD - unsigned int (%)
Store (Local0, Index (PBIG, 0x08))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0e, RefOf (Local0)) // AbsoluteStateOfCharge() - WORD - unsigned int (%)
Store (Local0, Index (PBIG, 0x09))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x0f, RefOf (Local0)) // RemaingingCapacity() - WORD - unsigned int (mAh or 10mWh)
Store (Local0, Index (PBIG, 0x0A))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x11, RefOf (Local0)) // RunTimeToEmpty() - WORD - unsigned int (min)
Store (Local0, Index (PBIG, 0x0B))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x12, RefOf (Local0)) // AverageTimeToEmpty() - WORD - unsigned int (min)
Store (Local0, Index (PBIG, 0x0C))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x13, RefOf (Local0)) // AverageTimeToFull() - WORD - unsigned int (min)
Store (Local0, Index (PBIG, 0x0D))
^^PCI0.LPCB.EC0.SMRD (0x09, 0x16, 0x1b, RefOf (Local0)) // ManufactureDate() - WORD - unsigned int (packed date)
Store (Local0, Index (PBIG, 0x0E))
^^PCI0.LPCB.EC0.SMRD (0x0B, 0x16, 0x23, RefOf (Local0)) // ManufacturerData() - BLOCK - Unknown
Store (Local0, Index (PBIG, 0x0F))
}
// InValid battery extra information
// Store placeholder battery extra information in PBIG when no battery is installed
Method (IVBG, 0, NotSerialized)
{
Store (0x00000000, Index (PBIG, 0x00)) // 0x00, ManufacturerAccess() - WORD - ?
Store (0x00000000, Index (PBIG, 0x01)) // 0x01, BatteryMode() - WORD - unsigned int
Store (0xFFFFFFFF, Index (PBIG, 0x02)) // 0x02, AtRateTimeToFull() - WORD - unsigned int (min)
Store (0xFFFFFFFF, Index (PBIG, 0x03)) // 0x03, AtRateTimeToEmpty() - WORD - unsigned int (min)
Store (0x00000000, Index (PBIG, 0x04)) // 0x04, Temperature() - WORD - unsigned int (0.1K)
Store (0x00000000, Index (PBIG, 0x05)) // 0x05, Voltage() - WORD - unsigned int (mV)
Store (0x00000000, Index (PBIG, 0x06)) // 0x06, Current() - WORD - signed int (mA)
Store (0x00000000, Index (PBIG, 0x07)) // 0x07, AverageCurrent() - WORD - signed int (mA)
Store (0x00000000, Index (PBIG, 0x08)) // 0x08, RelativeStateOfCharge() - WORD - unsigned int (%)
Store (0x00000000, Index (PBIG, 0x09)) // 0x09, AbsoluteStateOfCharge() - WORD - unsigned int (%)
Store (0x00000000, Index (PBIG, 0x0a)) // 0x0a, RemaingingCapacity() - WORD - unsigned int (mAh or 10mWh)
Store (0xFFFFFFFF, Index (PBIG, 0x0b)) // 0x0b, RunTimeToEmpty() - WORD - unsigned int (min)
Store (0xFFFFFFFF, Index (PBIG, 0x0c)) // 0x0c, AverageTimeToEmpty() - WORD - unsigned int (min)
Store (0xFFFFFFFF, Index (PBIG, 0x0d)) // 0x0d, AverageTimeToFull() - WORD - unsigned int (min)
Store (0x00000000, Index (PBIG, 0x0e)) // 0x0e, ManufactureDate() - WORD - unsigned int (packed date)
Store (" ", Index (PBIG, 0x0f)) // 0x0f, ManufacturerData() - BLOCK - Unknown
}
Optional Step 2d: Lion Support for DVx
Due to stricter ACPI Machine Language (AML) parsing in the AppleACPIPlatform.kext in 10.7 Lion, it appears that any registers in the EC operation region (address space) must be accessed with 8-bit (byte) references. On my DV8, the SMBus that talks to the SBS is wired into the EC and therefore we must access these registers with 8-bit references even though the SMBus standard does provide for 16-bit (Word) access, etc.
The HP DSDT uses two methods under the EC device to read and write the SMBus, SMRD() and SMWR(). I re-wrote the parts of these methods that did Word and Block (string) I/O to do the same thing via bytes so that the AppleACPIPlatform AML parser would not throw an exception. The first thing we need is a new big-old-ugly field definition for accessing the 32 bytes (256 bits) of SMBus data registers. We need to know the offset of these registers, so looking at the Field definition for EmbeddedController operation region we see:
CODE
// Embedded controller I/O Registers
// SMBus mapped at offset 0x00
OperationRegion (ERAM, EmbeddedControl, Zero, 0xFF)
Field (ERAM, ByteAcc, Lock, Preserve)
{
SMPR, 8, // SMBus Protocol Register
SMST, 8, // SMBus Status Register
SMAD, 8, // SMBus Address Register
SMCM, 8, // SMBus Command Register
SMD0, 256, // SMBus Data Register[0-31] ( 32 byte buffer)
BCNT, 8, // SMBus Block Count Register
SMAA, 8, // SMBus Alarm Address Register
SAD0, 8, // SMBus Alarm Data Register[0]
SAD1, 8, // SMBus Alarm Data Register[1]
...
So we can see the SMBus Data Register starts at offset 0x04, so under the EC device add:
CODE
// Define 8-bit fields for accessing SMBus data registers [0-31]
Field (ERAM, ByteAcc, Lock, Preserve)
{
Offset (0x04),
SM00, 8,
SM01, 8,
SM02, 8,
SM03, 8,
SM04, 8,
SM05, 8,
SM06, 8,
SM07, 8,
SM08, 8,
SM09, 8,
SM10, 8,
SM11, 8,
SM12, 8,
SM13, 8,
SM14, 8,
SM15, 8,
SM16, 8,
SM17, 8,
SM18, 8,
SM19, 8,
SM20, 8,
SM21, 8,
SM22, 8,
SM23, 8,
SM24, 8,
SM25, 8,
SM26, 8,
SM27, 8,
SM28, 8,
SM29, 8,
SM30, 8,
SM31, 8
}
Ya, its ugly, but it works. I'm sure this can be done using an index into the field but I need to research this more. Also under the EC device, replace the SMRD and SMWR methods with these:
CODE
// SMBus Read
// See ACPI Spec, section "12.9 SMBus Host Controller Interface via Embedded Controller"
//
// Arg0 = Protocol to use
// Arg1 = Slave address to read from
// Arg2 = Command to send to slave
// Arg3 = Return data pointer
//
// Returns SMBus status (although no caller is checking the status for other than kIOSMBusStatusOK :(
Method (SMRD, 4, NotSerialized)
{
If (LNot (ECON))
{
Return (0x17) // No embedded controller! (kIOSMBusStatusDeviceAccessDenied)
}
If (LNotEqual (Arg0, 0x07)) // kIOSMBusProtocolReadByte
{
If (LNotEqual (Arg0, 0x09)) // kIOSMBusProtocolReadWord
{
If (LNotEqual (Arg0, 0x0B)) // kIOSMBusProtocolReadBlock
{
Return (0x19) // kIOSMBusStatusHostUnsupportedProtocol
}
}
}
Acquire (MUT0, 0xFFFF) // Lock SMBus
Store (0x04, Local0) // Try 3 times
While (LGreater (Local0, One))
{
And (SMST, 0x40, SMST) // Clear status (Don't clear Alarm bit!)
Store (Arg2, SMCM) // Command
Store (Arg1, SMAD) // Address
Store (Arg0, SMPR) // Protocol
Store (Zero, Local3) // Timeout counter
While (LNot (And (SMST, 0xBF, Local1))) // Is command done? (not Alarm!)
{
Sleep (0x02) // nope, nap
Increment (Local3) // count nap
If (LEqual (Local3, 0x32)) // count = 50?
{
And (SMST, 0x40, SMST) // Send command again!
Store (Arg2, SMCM)
Store (Arg1, SMAD)
Store (Arg0, SMPR)
Store (Zero, Local3) // reset counter
}
}
If (LEqual (Local1, 0x80)) // Is command DONE?
{
Store (Zero, Local0) // Yup, exit retry loop
}
Else
{
Decrement (Local0) // Nope, another round please!
}
}
If (Local0) // After 3 attempts, something is FUBAR
{
Store (And (Local1, 0x1F), Local0) // Return SMBus status error
}
Else // Command worked, return data
{
If (LEqual (Arg0, 0x07)) // kIOSMBusProtocolReadByte
{
Store (SM00, Arg3) // Byte Output
}
If (LEqual (Arg0, 0x09)) // kIOSMBusProtocolReadWord
{
Store (SM00, Local4) // Low byte (Word)
Store (SM01, Local5) // High byte (Word)
ShiftLeft (Local5, 0x08, Local5) // Move high byte into place
Or (Local4, Local5, Local6) // Add low byte
Store (Local6, Arg3) // Return word
}
If (LEqual (Arg0, 0x0B)) // kIOSMBusProtocolReadBlock
{
Store (And (BCNT, 0x1F) , Local3) // Get block count of data (bits0-4 or 01xf)
Increment (Local3) // Add one to data count (for '\0')
Store (Buffer (Local3) { "" }, Local4) // Create an empty buffer of data count + 1
Decrement (Local3) // Size of data
Store (Zero, Local5) // Index into buffer
While (LGreater (Local3, Local5)) // More data to copy?
{
If (LEqual (Local5, 0x00)) { Store (SM00, Local6) } // Get byte of data
If (LEqual (Local5, 0x01)) { Store (SM01, Local6) }
If (LEqual (Local5, 0x02)) { Store (SM02, Local6) }
If (LEqual (Local5, 0x03)) { Store (SM03, Local6) }
If (LEqual (Local5, 0x04)) { Store (SM04, Local6) }
If (LEqual (Local5, 0x05)) { Store (SM05, Local6) }
If (LEqual (Local5, 0x06)) { Store (SM06, Local6) }
If (LEqual (Local5, 0x07)) { Store (SM07, Local6) }
If (LEqual (Local5, 0x08)) { Store (SM08, Local6) }
If (LEqual (Local5, 0x09)) { Store (SM09, Local6) }
If (LEqual (Local5, 0x0A)) { Store (SM10, Local6) }
If (LEqual (Local5, 0x0B)) { Store (SM11, Local6) }
If (LEqual (Local5, 0x0C)) { Store (SM12, Local6) }
If (LEqual (Local5, 0x0D)) { Store (SM13, Local6) }
If (LEqual (Local5, 0x0E)) { Store (SM14, Local6) }
If (LEqual (Local5, 0x0F)) { Store (SM15, Local6) }
If (LEqual (Local5, 0x10)) { Store (SM16, Local6) }
If (LEqual (Local5, 0x11)) { Store (SM17, Local6) }
If (LEqual (Local5, 0x12)) { Store (SM18, Local6) }
If (LEqual (Local5, 0x13)) { Store (SM19, Local6) }
If (LEqual (Local5, 0x14)) { Store (SM20, Local6) }
If (LEqual (Local5, 0x15)) { Store (SM21, Local6) }
If (LEqual (Local5, 0x16)) { Store (SM22, Local6) }
If (LEqual (Local5, 0x17)) { Store (SM23, Local6) }
If (LEqual (Local5, 0x18)) { Store (SM24, Local6) }
If (LEqual (Local5, 0x19)) { Store (SM25, Local6) }
If (LEqual (Local5, 0x1A)) { Store (SM26, Local6) }
If (LEqual (Local5, 0x1B)) { Store (SM27, Local6) }
If (LEqual (Local5, 0x1C)) { Store (SM28, Local6) }
If (LEqual (Local5, 0x1D)) { Store (SM29, Local6) }
If (LEqual (Local5, 0x1E)) { Store (SM30, Local6) }
If (LEqual (Local5, 0x1F)) { Store (SM31, Local6) }
PBFE (Local4, Local5, Local6) // Store byte in buffer
Increment (Local5) // Next!
}
PBFE (Local4, Local5, Zero) // Finish buffer with '\0' or 0x00
Store (Local4, Arg3) // Return buffer/block
}
}
Release (MUT0) // Release SMBus lock
Return (Local0) // Return 0 (OK)
}
// SMBus Write
// See ACPI Spec, section "12.9 SMBus Host Controller Interface via Embedded Controller"
//
// Arg0 = Protocol to use
// Arg1 = Slave address to write to
// Arg2 = Command to send to slave
// Arg3 = Pointer to data to write
Method (SMWR, 4, NotSerialized)
{
If (LNot (ECON))
{
Return (0x17) // No embedded controller! (kIOSMBusStatusDeviceAccessDenied)
}
If (LNotEqual (Arg0, 0x06)) // kIOSMBusProtocolWriteByte
{
If (LNotEqual (Arg0, 0x08)) // kIOSMBusProtocolWriteWord
{
If (LNotEqual (Arg0, 0x0A)) // kIOSMBusProtocolWriteBlock
{
Return (0x19) // kIOSMBusStatusHostUnsupportedProtocol
}
}
}
Acquire (MUT0, 0xFFFF) // Lock SMBus
Store (0x04, Local0) // 3 tries
While (LGreater (Local0, One))
{
If (LEqual (Arg0, 0x06))
{
Store (Arg3, SM00) // Byte to write
}
If (LEqual (Arg0, 0x08))
{
Store (Arg3, Local6) // Word to write
And (Local6, 0xFF, Local4) // Low byte
ShiftRight (Local6, 0x08, Local5) // High byte
Store (Local4, SM00) // Low byte (Word)
Store (Local5, SM01) // High byte (Word)
}
If (LEqual (Arg0, 0x0A))
{
Store (SizeOf (Arg3), Local3) // Get buffer/block/string size
And (Local3, 0x1F, Local3) // Limit to 32 bytes!
Store (Arg3, Local4) // Buffer/block/string pointer
Store (Zero, Local5) // Index into buffer
While (LGreater (Local3, Local5)) // More data to copy?
{
GBFE (Local4, Local5, RefOf (Local6)) // Get byte from buffer
If (LEqual (Local5, 0x00)) { Store (Local6, SM00) } // Write byte of data
If (LEqual (Local5, 0x01)) { Store (Local6, SM01) }
If (LEqual (Local5, 0x02)) { Store (Local6, SM02) }
If (LEqual (Local5, 0x03)) { Store (Local6, SM03) }
If (LEqual (Local5, 0x04)) { Store (Local6, SM04) }
If (LEqual (Local5, 0x05)) { Store (Local6, SM05) }
If (LEqual (Local5, 0x06)) { Store (Local6, SM06) }
If (LEqual (Local5, 0x07)) { Store (Local6, SM07) }
If (LEqual (Local5, 0x08)) { Store (Local6, SM08) }
If (LEqual (Local5, 0x09)) { Store (Local6, SM09) }
If (LEqual (Local5, 0x0A)) { Store (Local6, SM10) }
If (LEqual (Local5, 0x0B)) { Store (Local6, SM11) }
If (LEqual (Local5, 0x0C)) { Store (Local6, SM12) }
If (LEqual (Local5, 0x0D)) { Store (Local6, SM13) }
If (LEqual (Local5, 0x0E)) { Store (Local6, SM14) }
If (LEqual (Local5, 0x0F)) { Store (Local6, SM15) }
If (LEqual (Local5, 0x10)) { Store (Local6, SM16) }
If (LEqual (Local5, 0x11)) { Store (Local6, SM17) }
If (LEqual (Local5, 0x12)) { Store (Local6, SM18) }
If (LEqual (Local5, 0x13)) { Store (Local6, SM19) }
If (LEqual (Local5, 0x14)) { Store (Local6, SM20) }
If (LEqual (Local5, 0x15)) { Store (Local6, SM21) }
If (LEqual (Local5, 0x16)) { Store (Local6, SM22) }
If (LEqual (Local5, 0x17)) { Store (Local6, SM23) }
If (LEqual (Local5, 0x18)) { Store (Local6, SM24) }
If (LEqual (Local5, 0x19)) { Store (Local6, SM25) }
If (LEqual (Local5, 0x1A)) { Store (Local6, SM26) }
If (LEqual (Local5, 0x1B)) { Store (Local6, SM27) }
If (LEqual (Local5, 0x1C)) { Store (Local6, SM28) }
If (LEqual (Local5, 0x1D)) { Store (Local6, SM29) }
If (LEqual (Local5, 0x1E)) { Store (Local6, SM30) }
If (LEqual (Local5, 0x1F)) { Store (Local6, SM31) }
Increment (Local5) // Next!
}
Store (And (Local3, 0x1F), BCNT) // Store block count (limit to 32!)
}
And (SMST, 0x40, SMST) // Don't clear Alarm bit!
Store (Arg2, SMCM) // Command
Store (Arg1, SMAD) // Address
Store (Arg0, SMPR) // Protocol
Store (Zero, Local3) // timeout counter
While (LNot (And (SMST, 0xBF, Local1))) // Is command done? (not Alarm!)
{
Sleep (0x02) // nap
Increment (Local3) // count
If (LEqual (Local3, 0x32)) // count = 50?
{
And (SMST, 0x40, SMST) // Send command again!
Store (Arg2, SMCM)
Store (Arg1, SMAD)
Store (Arg0, SMPR)
Store (Zero, Local3)
}
}
If (LEqual (Local1, 0x80)) // Command DONE?
{
Store (Zero, Local0) // Yup, exit retry loop
}
Else
{
Decrement (Local0) // Another round!
}
}
If (Local0) // If we tried 3 times, FUBAR!
{
Store (And (Local1, 0x1F), Local0) // Return error
}
Release (MUT0) // Release SMBus lock
Return (Local0) // Return 0 (OK)
}
Compile the DSDT and install as usual.
Step 3: Install Driver
Install the kext in /Extra/Extensions, set the permissions and options in Info.plust, rebuild your kext cache. Reboot.
There are two zip files below, a debug and release version. The debug version outputs messages to kernel.log and the release version does not, otherwise they are identical.
AppleACPIBatteryManager_20110802_release.zip ( 36.85K ) Number of downloads: 138
AppleACPIBatteryManager_20110802_debug.zip ( 44.63K ) Number of downloads: 27
The following two drivers are identical to the above versions except the driver classes and kext have been renamed to the same as Apple's Smart Battery Manager and therefore allows some 3rd party application such as iStat Menus and coconutBattery to work.
AppleSmartBatteryManager_20110802_release.zip ( 36.83K ) Number of downloads: 180
AppleSmartBatteryManager_20110802_debug.zip ( 44.62K ) Number of downloads: 51
Change Log
2011-09-25
- Added reference document links
- Reordered this change log so latest changes are at the top
2011-09-21
- Added "Optional Step 2d: Lion Support for DVx" section.
2011-08-05
- Renamed drivers classes to allow compatibility with some 3rd party battery information application.
- Added new driver binaries for 2011.0802 release.
2011-08-04
- Updated DSDT code in posting with stock EC0 device name.
2011-08-02
- Added ACPI 4.x support.
- Added non-standard support for temperature.
- Compiled with XCode 4.
- Lots of debugging.