450 lines
13 KiB
C
450 lines
13 KiB
C
/*
|
|
* Copyright (C) 2007-2008 Apple Inc. All rights reserved.
|
|
* Copyright (C) 2006 Apple Computer, Inc. All rights reserved.
|
|
*
|
|
* This document is the property of Apple Inc.
|
|
* It is considered confidential and proprietary.
|
|
*
|
|
* This document may not be reproduced or transmitted in any form,
|
|
* in whole or in part, without the express written permission of
|
|
* Apple Inc.
|
|
*/
|
|
#include <debug.h>
|
|
#include <drivers/iic.h>
|
|
#include <drivers/charger.h>
|
|
#include <drivers/power.h>
|
|
#include <platform.h>
|
|
#include <platform/gpiodef.h>
|
|
#include <sys.h>
|
|
#include <sys/boot.h>
|
|
#include <sys/menu.h>
|
|
#include <sys/task.h>
|
|
#include <target/powerconfig.h>
|
|
|
|
#define SN2400_ADDR_R 0xEA
|
|
#define SN2400_ADDR_W 0xEB
|
|
|
|
static int charger_reset_battery_protection(void);
|
|
|
|
enum {
|
|
FAULT_LOG = 0x00,
|
|
FAULT_LOG_BAT_SHORT = (1 << 0),
|
|
|
|
IRQ_1_MASK = 0x01,
|
|
IRQ_2_MASK = 0x02,
|
|
IRQ_3_MASK = 0x03,
|
|
|
|
EVENT_count = 3,
|
|
|
|
EVENT_1 = 0x04,
|
|
EVENT_1_VBUS_OV = (1 << 7),
|
|
EVENT_1_VBUS_UV = (1 << 6),
|
|
EVENT_1_VBUS_OC = (1 << 5),
|
|
EVENT_1_OVER_TEMP = (1 << 4),
|
|
EVENT_1_TEMP_WARNING = (1 << 3),
|
|
EVENT_1_BAT_SHORT = (1 << 2),
|
|
EVENT_1_MAIN_SHORT = (1 << 1),
|
|
EVENT_1_RECOVERY_TIMEOUT = (1 << 0),
|
|
|
|
EVENT_2 = 0x05,
|
|
EVENT_2_LDO_OCP = (1 << 7),
|
|
EVENT_2_BATT_ALERT = (1 << 6),
|
|
EVENT_2_MISSING_BATTERY = (1 << 5),
|
|
EVENT_2_HDQ_HOST_ALERT = (1 << 4),
|
|
EVENT_2_VBUS_VLIM_ACTIVE = (1 << 3),
|
|
EVENT_2_ACCUM_OVERFLOW = (1 << 2),
|
|
EVENT_2_ADC_DONE = (1 << 1),
|
|
EVENT_2_HDQ_INT = (1 << 0),
|
|
|
|
EVENT_3 = 0x06,
|
|
EVENT_3_MAX_DUTY = (1 << 2),
|
|
EVENT_3_ZERO_IBAT = (1 << 1),
|
|
EVENT_3_VBUS_DET = (1 << 0),
|
|
|
|
SYSTEM_STATE = 0x07,
|
|
SYSTEM_STATE_BUCK_EN_STAT = (1 << 7),
|
|
SYSTEM_STATE_EXT_OVP_STAT = (1 << 6),
|
|
SYSTEM_STATE_VBUS_DET_STAT = (1 << 5),
|
|
SYSTEM_STATE_SYS_ALIVE_STAT = (1 << 4),
|
|
SYSTEM_STATE_DISCHARGE_BAT_STAT = (1 << 3),
|
|
SYSTEM_STATE_IBAT_ZERO = (1 << 2),
|
|
SYSTEM_STATE_RECOVERY_STAT = (1 << 1),
|
|
SYSTEM_STATE_CHARGE_EN = (1 << 0),
|
|
|
|
DC_LIMITER = 0x08,
|
|
DC_LIMITER_I_REDUCED = (1 << 6),
|
|
DC_LIMITER_BUCK_EN = (1 << 5),
|
|
DC_LIMITER_VBUS_LIM = (1 << 4),
|
|
DC_LIMITER_ICHG_LIM = (1 << 3),
|
|
DC_LIMITER_VBAT_LIM = (1 << 2),
|
|
DC_LIMITER_VMAIN_LIM = (1 << 1),
|
|
DC_LIMITER_VBUS_VLIM = (1 << 0),
|
|
|
|
INPUT_CURRENT_LIMIT = 0x09,
|
|
INPUT_CURRENT_LIMIT_base = 90,
|
|
INPUT_CURRENT_LIMIT_step = 10,
|
|
INPUT_CURRENT_LIMIT_100mA = 0x01,
|
|
INPUT_CURRENT_LIMIT_500mA = 0x29,
|
|
INPUT_CURRENT_LIMIT_1000mA = 0x5B,
|
|
INPUT_CURRENT_LIMIT_1500mA = 0x8D,
|
|
INPUT_CURRENT_LIMIT_2000mA = 0xBF,
|
|
INPUT_CURRENT_LIMIT_max = INPUT_CURRENT_LIMIT_2000mA,
|
|
|
|
POR_INPUT_CURRENT_LIMIT = 0x0A,
|
|
ACTUAL_INPUT_CURRENT_LIMIT = 0x0B,
|
|
|
|
TARGET_CHARGE_CURRENT = 0x0C,
|
|
CHARGE_CURRENT_base = 0,
|
|
CHARGE_CURRENT_step = 10,
|
|
CHARGE_CURRENT_disabled = 0x00,
|
|
CHARGE_CURRENT_10mA = 0x01,
|
|
CHARGE_CURRENT_max = 0xFF,
|
|
|
|
HOST_CONTROLLED_CHARGE_CURRENT = 0x0D,
|
|
ACTUAL_CHARGE_CURRENT = 0x0E,
|
|
|
|
TARGET_VOLTAGE_REGULATION = 0x0F,
|
|
VOLTAGE_REGULATION_base = 3000,
|
|
VOLTAGE_REGULATION_step = 10,
|
|
VOLTAGE_REGULATION_3000mV = 0x00,
|
|
VOLTAGE_REGULATION_4200mV = 0x78,
|
|
VOLTAGE_REGULATION_4300mV = 0x82,
|
|
VOLTAGE_REGULATION_4500mV = 0x96,
|
|
VOLTAGE_REGULATION_max = VOLTAGE_REGULATION_4500mV,
|
|
|
|
CONTROL = 0x10,
|
|
CONTROL_EN_ADAPTER_HS = (1 << 4),
|
|
CONTROL_USB_SUSP = (1 << 3),
|
|
CONTROL_VBUS_IMIL_SEL = (1 << 2),
|
|
CONTROL_BUCK_FREQ_SHIFT_mask = (3 << 0),
|
|
CONTROL_BUCK_FREQ_SHIFT_normal = (0 << 0),
|
|
CONTROL_BUCK_FREQ_SHIFT_plus_150 = (1 << 0),
|
|
CONTROL_BUCK_FREQ_SHIFT_minus_150 = (2 << 0),
|
|
CONTROL_BUCK_FREQ_SHIFT_spread = (3 << 0),
|
|
|
|
ADC_CONTROL = 0x11,
|
|
ADC_CONTROL_MANUAL_START = (1 << 4),
|
|
ADC_CONTROL_CHANNEL_mask = (7 << 1),
|
|
ADC_CONTROL_CHANNEL_vbus = (0 << 1),
|
|
ADC_CONTROL_CHANNEL_ibus = (1 << 1),
|
|
ADC_CONTROL_CHANNEL_ibat = (2 << 1),
|
|
ADC_CONTROL_CHANNEL_vdd_main = (3 << 1),
|
|
ADC_CONTROL_CHANNEL_die_temp = (4 << 1),
|
|
ADC_CONTROL_CHANNEL_vbat = (5 << 1),
|
|
ADC_CONTROL_ACCUM_RESET = (1 << 0),
|
|
|
|
ADC_OUTPUT = 0x12,
|
|
IBUS_AUTO = 0x13,
|
|
VBUS_AUTO = 0x14,
|
|
|
|
ADC_IBUS_ACCUMULATION_3 = 0x15,
|
|
ADC_IBUS_ACCUMULATION_2 = 0x16,
|
|
ADC_IBUS_ACCUMULATION_1 = 0x17,
|
|
ADC_VBUS_ACCUMULATION_3 = 0x18,
|
|
ADC_VBUS_ACCUMULATION_2 = 0x19,
|
|
ADC_VBUS_ACCUMULATION_1 = 0x1A,
|
|
SAMPLE_COUNT_2 = 0x1B,
|
|
SAMPLE_COUNT_1 = 0x1C,
|
|
|
|
HDQ_INTERFACE = 0x1D,
|
|
HDQ_INTERFACE_HDQ_MSTR_mask = (3 << 6),
|
|
HDQ_INTERFACE_HDQ_MSTR_none = (3 << 6),
|
|
HDQ_INTERFACE_HDQ_MSTR_charger = (3 << 6),
|
|
HDQ_INTERFACE_HDQ_MSTR_host = (3 << 6),
|
|
HDQ_INTERFACE_HDQ_MSTR_bmu = (3 << 6),
|
|
HDQ_INTERFACE_HDQ_HOST_EN = (1 << 5),
|
|
HDQ_INTERFACE_MISSING_BATTERY = (1 << 4),
|
|
HDQ_INTERFACE_MSTR_REQ_CHGR = (1 << 3),
|
|
HDQ_INTERFACE_MSTR_REQ_HOST = (1 << 2),
|
|
HDQ_INTERFACE_MSTR_REQ_BMU = (1 << 1),
|
|
HDQ_INTERFACE_MSTR_FORCE_HOST = (1 << 0),
|
|
|
|
TEST_MODE_ENABLE = 0x20,
|
|
TEST_MODE_ENABLE_off = 0x00,
|
|
TEST_MODE_ENABLE_on = 0xA5,
|
|
|
|
TEST_MODE_CONTROL = 0x21,
|
|
TEST_MODE_CONTROL_ENB_AUTOCONV = (1 << 5),
|
|
TEST_MODE_CONTROL_EN_FORCEMODE = (1 << 4),
|
|
TEST_MODE_CONTROL_FORCE_MODE_mask = (3 << 2),
|
|
TEST_MODE_CONTROL_FORCE_MODE_stdby = (0 << 2),
|
|
TEST_MODE_CONTROL_FORCE_MODE_chgdis = (1 << 2),
|
|
TEST_MODE_CONTROL_FORCE_MODE_recov = (2 << 2),
|
|
TEST_MODE_CONTROL_FORCE_MODE_normal = (3 << 2),
|
|
TEST_MODE_CONTROL_GG_DIS = (1 << 1),
|
|
TEST_MODE_CONTROL_SOFT_RESET = (1 << 0),
|
|
};
|
|
|
|
static int charger_write(uint8_t reg, uint8_t value)
|
|
{
|
|
uint8_t data[2] = { reg, value };
|
|
return iic_write(CHARGER_IIC_BUS, SN2400_ADDR_W, data, sizeof(data));
|
|
}
|
|
|
|
static int charger_read(uint8_t reg, uint8_t *data)
|
|
{
|
|
return iic_read(CHARGER_IIC_BUS, SN2400_ADDR_R, ®, sizeof(reg), data, sizeof(uint8_t), IIC_NORMAL);
|
|
}
|
|
|
|
static int charger_readn(uint8_t reg, uint8_t *data, unsigned int count)
|
|
{
|
|
return iic_read(CHARGER_IIC_BUS, SN2400_ADDR_R, ®, sizeof(reg), data, count, IIC_NORMAL);
|
|
}
|
|
|
|
static int charger_read_events(uint8_t data[EVENT_count])
|
|
{
|
|
uint8_t reg = EVENT_1;
|
|
return iic_read(CHARGER_IIC_BUS, SN2400_ADDR_R, ®, sizeof(reg), data, EVENT_count, IIC_NORMAL);
|
|
}
|
|
|
|
static int charger_read_adc(uint8_t channel, uint8_t *result)
|
|
{
|
|
uint8_t data = ADC_CONTROL_MANUAL_START | (channel & ADC_CONTROL_CHANNEL_mask);
|
|
if (charger_write(ADC_CONTROL, data) != 0) return -1;
|
|
|
|
do {
|
|
task_sleep(5 * 1000); // data sheet lists 3.7 ms as typ. average time
|
|
if (charger_read(ADC_CONTROL, &data) != 0) return -1;
|
|
} while (data & ADC_CONTROL_MANUAL_START);
|
|
|
|
return charger_read(ADC_OUTPUT, result);
|
|
}
|
|
|
|
void charger_early_init(void)
|
|
{
|
|
// clear any events
|
|
uint8_t events[EVENT_count];
|
|
charger_read_events(events);
|
|
|
|
#if TARGET_POWER_NEEDS_BATTERY_PROTECTION_RESET && PRODUCT_LLB
|
|
// check status for recovery mode
|
|
uint8_t data;
|
|
if (charger_read(SYSTEM_STATE, &data) == 0) {
|
|
dprintf(DEBUG_INFO, "sn2400 : 0x07=%02x\n", data);
|
|
if ((data & SYSTEM_STATE_RECOVERY_STAT) != 0) {
|
|
charger_reset_battery_protection();
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void charger_clear_alternate_usb_current_limit(void)
|
|
{
|
|
uint8_t data;
|
|
charger_read(CONTROL, &data);
|
|
data |= CONTROL_VBUS_IMIL_SEL;
|
|
charger_write(CONTROL, data);
|
|
}
|
|
|
|
bool charger_has_usb(int dock)
|
|
{
|
|
uint8_t data;
|
|
charger_read(SYSTEM_STATE, &data);
|
|
return data & SYSTEM_STATE_VBUS_DET_STAT;
|
|
}
|
|
|
|
bool charger_has_firewire(int dock)
|
|
{
|
|
return false; // FW not supported
|
|
}
|
|
|
|
bool charger_has_external(int dock)
|
|
{
|
|
return false; // Ext. not supported
|
|
}
|
|
|
|
bool charger_check_usb_change(int dock, bool expected)
|
|
{
|
|
uint8_t events[EVENT_count];
|
|
if (charger_read_events(events) != 0) return false;
|
|
|
|
if (events[0] & (EVENT_1_VBUS_UV | EVENT_1_VBUS_OV)) return true;
|
|
if (events[2] & (EVENT_3_VBUS_DET)) return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
// CBAT, maybe next...
|
|
static uint8_t sn2400_adjust_iset(uint32_t iset)
|
|
{
|
|
if ( iset==0 ) {
|
|
// leave it
|
|
} else if ( iset<3 ) {
|
|
iset=1;
|
|
} else if ( iset<INPUT_CURRENT_LIMIT_500mA ) {
|
|
iset-=2; // 20mA
|
|
} else if ( iset<INPUT_CURRENT_LIMIT_1000mA ) {
|
|
iset-=5; // 50mA
|
|
} else if ( iset<INPUT_CURRENT_LIMIT_1500mA ) {
|
|
iset-=8; // 80mA
|
|
} else {
|
|
iset-=10; // 100ma
|
|
}
|
|
|
|
return iset;
|
|
}
|
|
|
|
void charger_set_charging(int dock, uint32_t input_current_ma, uint32_t *charge_current_ma)
|
|
{
|
|
uint8_t charge_current;
|
|
if (*charge_current_ma > (CHARGE_CURRENT_base + (CHARGE_CURRENT_max * CHARGE_CURRENT_step))) {
|
|
charge_current = CHARGE_CURRENT_max;
|
|
} else {
|
|
charge_current = (*charge_current_ma - CHARGE_CURRENT_base) / CHARGE_CURRENT_step;
|
|
}
|
|
|
|
*charge_current_ma -= CHARGE_CURRENT_base + (charge_current * CHARGE_CURRENT_step);
|
|
charger_write(HOST_CONTROLLED_CHARGE_CURRENT, charge_current ) ;
|
|
|
|
if (input_current_ma < 100) {
|
|
charger_write(INPUT_CURRENT_LIMIT, INPUT_CURRENT_LIMIT_100mA);
|
|
|
|
uint8_t data;
|
|
if (charger_read(CONTROL, &data) != 0) return;
|
|
data |= CONTROL_USB_SUSP;
|
|
charger_write(CONTROL, data);
|
|
} else {
|
|
uint8_t data;
|
|
if (charger_read(CONTROL, &data) != 0) return;
|
|
if (data & CONTROL_USB_SUSP) {
|
|
data &= ~CONTROL_USB_SUSP;
|
|
charger_write(CONTROL, data);
|
|
|
|
task_sleep(100 * 1000);
|
|
}
|
|
|
|
if (charger_read(ACTUAL_INPUT_CURRENT_LIMIT, &data) != 0) return;
|
|
|
|
uint8_t input_current_limit;
|
|
if (input_current_ma > (INPUT_CURRENT_LIMIT_base + (INPUT_CURRENT_LIMIT_max * INPUT_CURRENT_LIMIT_step))) {
|
|
input_current_limit = INPUT_CURRENT_LIMIT_max;
|
|
} else {
|
|
input_current_limit = (input_current_ma - INPUT_CURRENT_LIMIT_base) / INPUT_CURRENT_LIMIT_step;
|
|
}
|
|
|
|
while (data < input_current_limit) {
|
|
data += (100 / INPUT_CURRENT_LIMIT_step);
|
|
if (data > input_current_limit) data = input_current_limit;
|
|
charger_write(INPUT_CURRENT_LIMIT, sn2400_adjust_iset(data) );
|
|
task_sleep(10 * 1000);
|
|
}
|
|
}
|
|
}
|
|
|
|
void charger_clear_usb_state(void)
|
|
{
|
|
// empty
|
|
}
|
|
|
|
bool charger_charge_done(int dock)
|
|
{
|
|
uint8_t data;
|
|
if (charger_read(SYSTEM_STATE, &data) != 0) return false;
|
|
return !(data & SYSTEM_STATE_CHARGE_EN);
|
|
}
|
|
|
|
bool charger_has_batterypack(int dock)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
uint32_t charger_get_max_charge_current(int dock)
|
|
{
|
|
return CHARGE_CURRENT_base + (CHARGE_CURRENT_max * CHARGE_CURRENT_step);
|
|
}
|
|
|
|
int charger_read_battery_level(uint32_t *milliVolts)
|
|
{
|
|
uint8_t adc;
|
|
if (charger_read_adc(ADC_CONTROL_CHANNEL_vbat, &adc) != 0) return -1;
|
|
*milliVolts = 2500 + (10 * adc);
|
|
return 0;
|
|
}
|
|
|
|
int charger_set_gasgauge_interface(bool enabled)
|
|
{
|
|
if (enabled) {
|
|
if (charger_write(HDQ_INTERFACE, HDQ_INTERFACE_MSTR_REQ_HOST) != 0) return -1;
|
|
|
|
utime_t timeout = system_time() + 1*1000*1000;
|
|
for (;;) {
|
|
uint8_t data;
|
|
if (charger_read(HDQ_INTERFACE, &data) != 0) {
|
|
charger_write(HDQ_INTERFACE, 0);
|
|
return -1;
|
|
}
|
|
if (data & HDQ_INTERFACE_HDQ_HOST_EN) break;
|
|
if (system_time() > timeout) {
|
|
printf("charger HDQ interface request read timed out\n");
|
|
return (-1);
|
|
}
|
|
task_sleep(10 * 1000);
|
|
}
|
|
} else {
|
|
charger_write(HDQ_INTERFACE, 0);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int charger_reset_battery_protection(void)
|
|
{
|
|
// enter test mode and force charging at 4.3V VBATREG and 10mA
|
|
// charging current for 20mS
|
|
int ret, ret2;
|
|
|
|
dprintf(DEBUG_CRITICAL, "starting battery protection reset...\n");
|
|
// check all PMU temp sensors for "ambient temperature" check
|
|
for (int idx = 1; idx <= 4; idx++) {
|
|
int temp_cC;
|
|
ret = pmu_read_system_temperature(idx, &temp_cC);
|
|
if (ret != 0) {
|
|
dprintf(DEBUG_CRITICAL, "temp failure\n");
|
|
return -1;
|
|
}
|
|
if (temp_cC <= 500 || temp_cC >= 4000) {
|
|
dprintf(DEBUG_CRITICAL, "system temp %d out of range: %d.%02dC\n", idx, temp_cC / 100, temp_cC % 100);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
ret = charger_write(HOST_CONTROLLED_CHARGE_CURRENT, CHARGE_CURRENT_10mA);
|
|
if (ret == 0) ret = charger_write(TEST_MODE_ENABLE, TEST_MODE_ENABLE_on);
|
|
if (ret == 0) ret = charger_write(TEST_MODE_CONTROL, TEST_MODE_CONTROL_GG_DIS);
|
|
if (ret == 0) ret = charger_write(TARGET_CHARGE_CURRENT, CHARGE_CURRENT_disabled);
|
|
if (ret == 0) ret = charger_write(TARGET_VOLTAGE_REGULATION, VOLTAGE_REGULATION_4300mV);
|
|
if (ret == 0) ret = charger_write(TARGET_CHARGE_CURRENT, CHARGE_CURRENT_10mA);
|
|
|
|
task_sleep(20 * 1000);
|
|
|
|
if (ret == 0) ret = charger_write(TARGET_CHARGE_CURRENT, CHARGE_CURRENT_disabled);
|
|
if (ret == 0) ret = charger_write(TARGET_VOLTAGE_REGULATION, VOLTAGE_REGULATION_4200mV);
|
|
if (ret == 0) ret = charger_write(TEST_MODE_CONTROL, 0x00);
|
|
// always attempt to take out of test mode, even on failure
|
|
ret2 = charger_write(TEST_MODE_ENABLE, TEST_MODE_ENABLE_off);
|
|
if (ret == 0) ret = ret2;
|
|
if (ret == 0) ret = charger_write(HOST_CONTROLLED_CHARGE_CURRENT, CHARGE_CURRENT_max);
|
|
|
|
dprintf(DEBUG_CRITICAL, "battery protection reset completed: %s\n", (ret == 0) ? "success" : "error");
|
|
|
|
return ret;
|
|
}
|
|
|
|
void charger_print_status(void)
|
|
{
|
|
int rc;
|
|
uint8_t data[7];
|
|
|
|
rc=charger_readn(SYSTEM_STATE, data, 2);
|
|
if ( rc!=0 ) {
|
|
dprintf(DEBUG_CRITICAL, "cannot read %x (%d)\n", SYSTEM_STATE, rc);
|
|
} else {
|
|
rc=charger_readn(ACTUAL_INPUT_CURRENT_LIMIT, &data[2], 5);
|
|
if ( rc!=0 ) {
|
|
dprintf(DEBUG_CRITICAL, "cannot read %x (%d)\n", ACTUAL_INPUT_CURRENT_LIMIT, rc);
|
|
} else {
|
|
dprintf(DEBUG_CRITICAL, "sn2400 : 0x07=%02x 0x08=%02x 0x0b=%02x 0x0c=%02x 0x0d=%02x 0x0e=%02x 0x0f=%02x\n",
|
|
data[0], data[1], data[2], data[3], data[4], data[5], data[6]);
|
|
}
|
|
}
|
|
}
|