diff options
author | Vic Yang <victoryang@google.com> | 2015-01-22 14:31:43 -0800 |
---|---|---|
committer | ChromeOS Commit Bot <chromeos-commit-bot@chromium.org> | 2015-01-29 23:32:13 +0000 |
commit | 920d701647c4c7d0d75aa793a9ea0a8e035f2206 (patch) | |
tree | c3315ec37510bd1ba27577e1037b2bcec83d10ea | |
parent | 2033b8a2d6b407d8bb660faaddc389c33077abbc (diff) | |
download | chrome-ec-920d701647c4c7d0d75aa793a9ea0a8e035f2206.tar.gz |
ryu: Implement charging profile
This allows us to charge at a faster rate.
BRANCH=Ryu
BUG=None
TEST=Charge Ryu P3 in S5 and check charging current.
Change-Id: If8fbaa42d34df45f3d4db35ba50031c511b0c7af
Signed-off-by: Vic Yang <victoryang@google.com>
Reviewed-on: https://chromium-review.googlesource.com/242666
Tested-by: Vic Yang <victoryang@chromium.org>
Reviewed-by: Alec Berg <alecaberg@chromium.org>
Commit-Queue: Vic Yang <victoryang@chromium.org>
-rw-r--r-- | board/ryu/battery.c | 38 | ||||
-rw-r--r-- | board/ryu/board.h | 3 | ||||
-rw-r--r-- | board/ryu/build.mk | 2 | ||||
-rw-r--r-- | driver/battery/ryu.c | 215 | ||||
-rw-r--r-- | driver/build.mk | 1 |
5 files changed, 219 insertions, 40 deletions
diff --git a/board/ryu/battery.c b/board/ryu/battery.c deleted file mode 100644 index dc65128719..0000000000 --- a/board/ryu/battery.c +++ /dev/null @@ -1,38 +0,0 @@ -/* Copyright 2014 The Chromium OS Authors. All rights reserved. - * Use of this source code is governed by a BSD-style license that can be - * found in the LICENSE file. - * - * Battery pack vendor provided charging profile - */ - -#include "battery.h" -#include "common.h" -#include "i2c.h" - -/* Battery temperature ranges in degrees C */ -static const struct battery_info info = { - /* Design voltage */ - .voltage_max = 4350, - .voltage_normal = 3800, - .voltage_min = 2800, - /* Pre-charge current: I <= 0.01C */ - .precharge_current = 64, /* mA */ - /* Operational temperature range */ - .start_charging_min_c = 0, - .start_charging_max_c = 45, - .charging_min_c = 0, - .charging_max_c = 50, - .discharging_min_c = -20, - .discharging_max_c = 60, -}; - -const struct battery_info *battery_get_info(void) -{ - return &info; -} - -int board_cut_off_battery(void) -{ - /* Write SET_SHUTDOWN(0x13) to CTRL(0x00) */ - return i2c_write16(I2C_PORT_BATTERY, 0xaa, 0x0, 0x13); -} diff --git a/board/ryu/board.h b/board/ryu/board.h index 0d54abdb99..66b991a09a 100644 --- a/board/ryu/board.h +++ b/board/ryu/board.h @@ -45,7 +45,7 @@ #define CONFIG_UART_RX_DMA_CH STM32_DMAC_USART2_RX /* Charging/Power configuration */ -#undef CONFIG_BATTERY_RYU /* TODO implement */ +#define CONFIG_BATTERY_RYU #define CONFIG_BATTERY_BQ27541 #define CONFIG_BATTERY_CUT_OFF #define CONFIG_BATTERY_REQUESTS_NIL_WHEN_DEAD @@ -54,6 +54,7 @@ #define CONFIG_CHARGER_V2 #define CONFIG_CHARGER_BQ24773 #define CONFIG_CHARGER_ILIM_PIN_DISABLED +#define CONFIG_CHARGER_PROFILE_OVERRIDE #define CONFIG_CHARGER_SENSE_RESISTOR 5 #define CONFIG_CHARGER_SENSE_RESISTOR_AC 10 #define CONFIG_CHARGER_INPUT_CURRENT 512 diff --git a/board/ryu/build.mk b/board/ryu/build.mk index d498972118..4671ea0c7c 100644 --- a/board/ryu/build.mk +++ b/board/ryu/build.mk @@ -9,5 +9,5 @@ CHIP:=stm32 CHIP_FAMILY:=stm32f3 CHIP_VARIANT:=stm32f373 -board-y=board.o battery.o +board-y=board.o board-$(CONFIG_USB_POWER_DELIVERY)+=usb_pd_policy.o diff --git a/driver/battery/ryu.c b/driver/battery/ryu.c new file mode 100644 index 0000000000..7bfcc949b8 --- /dev/null +++ b/driver/battery/ryu.c @@ -0,0 +1,215 @@ +/* Copyright 2015 The Chromium OS Authors. All rights reserved. + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + * + * Battery pack vendor provided charging profile + */ + +#include "battery.h" +#include "charge_state.h" +#include "console.h" +#include "ec_commands.h" +#include "i2c.h" +#include "util.h" + +/* Battery temperature ranges in degrees C */ +static const struct battery_info info = { + /* Design voltage */ + .voltage_max = 4350, + .voltage_normal = 3800, + .voltage_min = 2800, + /* Pre-charge current: I <= 0.01C */ + .precharge_current = 64, /* mA */ + /* Operational temperature range */ + .start_charging_min_c = 0, + .start_charging_max_c = 45, + .charging_min_c = 0, + .charging_max_c = 50, + .discharging_min_c = -20, + .discharging_max_c = 60, +}; + +const struct battery_info *battery_get_info(void) +{ + return &info; +} + +int board_cut_off_battery(void) +{ + /* Write SET_SHUTDOWN(0x13) to CTRL(0x00) */ + return i2c_write16(I2C_PORT_BATTERY, 0xaa, 0x0, 0x13); +} + +#ifdef CONFIG_CHARGER_PROFILE_OVERRIDE + +static int fast_charging_allowed = 1; + +/* + * This can override the smart battery's charging profile. To make a change, + * modify one or more of requested_voltage, requested_current, or state. + * Leave everything else unchanged. + * + * Return the next poll period in usec, or zero to use the default (which is + * state dependent). + */ +int charger_profile_override(struct charge_state_data *curr) +{ + /* temp in 0.1 deg C */ + int temp_c = curr->batt.temperature - 2731; + /* keep track of last temperature range for hysteresis */ + static enum { + TEMP_RANGE_1, + TEMP_RANGE_2, + TEMP_RANGE_3, + TEMP_RANGE_4, + TEMP_RANGE_5, + } temp_range = TEMP_RANGE_3; + /* keep track of last voltage range for hysteresis */ + static enum { + VOLTAGE_RANGE_LOW, + VOLTAGE_RANGE_HIGH, + } voltage_range = VOLTAGE_RANGE_LOW; + + /* Current and previous battery voltage */ + int batt_voltage; + static int prev_batt_voltage; + + /* + * Determine temperature range. The five ranges are: + * < 10C + * 10-15C + * 15-23C + * 23-45C + * > 45C + * + * Add 0.2 degrees of hysteresis. + * If temp reading was bad, use last range. + */ + if (!(curr->batt.flags & BATT_FLAG_BAD_TEMPERATURE)) { + if (temp_c < 99) + temp_range = TEMP_RANGE_1; + else if (temp_c > 101 && temp_c < 149) + temp_range = TEMP_RANGE_2; + else if (temp_c > 151 && temp_c < 229) + temp_range = TEMP_RANGE_3; + else if (temp_c > 231 && temp_c < 449) + temp_range = TEMP_RANGE_4; + else if (temp_c > 451) + temp_range = TEMP_RANGE_5; + } + + /* + * If battery voltage reading is bad, use the last reading. Otherwise, + * determine voltage range with 20mV * hysteresis. + */ + if (curr->batt.flags & BATT_FLAG_BAD_VOLTAGE) { + batt_voltage = prev_batt_voltage; + } else { + batt_voltage = prev_batt_voltage = curr->batt.voltage; + if (batt_voltage < 4130) + voltage_range = VOLTAGE_RANGE_LOW; + else if (batt_voltage > 4150) + voltage_range = VOLTAGE_RANGE_HIGH; + } + + /* + * If we are not charging or we aren't using fast charging profiles, + * then do not override desired current and voltage. + */ + if (curr->state != ST_CHARGE || !fast_charging_allowed) + return 0; + + /* + * Okay, impose our custom will: + * When battery is 0-10C: + * CC at 900mA @ 4.35V + * CV at 4.35V until current drops to 450mA + * + * When battery is <15C: + * CC at 2700mA @ 4.35V + * CV at 4.35V until current drops to 450mA + * + * When battery is <23C: + * CC at 6300mA until 4.15V @ 4.35V + * CC at 4500mA @ 4.35V + * CV at 4.35V until current drops to 450mA + * + * When battery is <45C: + * CC at 9000mA until 4.15V @ 4.35V + * CC at 4500mA @ 4.35V + * CV at 4.35V until current drops to 450mA + * + * When battery is >45C: + * CC at 4500mA @ 4.15V + * CV at 4.15V (when battery is hot we don't go to fully charged) + */ + switch (temp_range) { + case TEMP_RANGE_1: + curr->requested_current = 900; + curr->requested_voltage = 4350; + break; + case TEMP_RANGE_2: + curr->requested_current = 2700; + curr->requested_voltage = 4350; + break; + case TEMP_RANGE_3: + curr->requested_voltage = 4350; + if (voltage_range == VOLTAGE_RANGE_HIGH) + curr->requested_current = 4500; + else + curr->requested_current = 6300; + break; + case TEMP_RANGE_4: + curr->requested_voltage = 4350; + if (voltage_range == VOLTAGE_RANGE_HIGH) + curr->requested_current = 4500; + else + curr->requested_current = 9000; + break; + case TEMP_RANGE_5: + curr->requested_current = 4500; + curr->requested_voltage = 4150; + break; + } + + return 0; +} + +/* Customs options controllable by host command. */ +#define PARAM_FASTCHARGE (CS_PARAM_CUSTOM_PROFILE_MIN + 0) + +enum ec_status charger_profile_override_get_param(uint32_t param, + uint32_t *value) +{ + if (param == PARAM_FASTCHARGE) { + *value = fast_charging_allowed; + return EC_RES_SUCCESS; + } + return EC_RES_INVALID_PARAM; +} + +enum ec_status charger_profile_override_set_param(uint32_t param, + uint32_t value) +{ + if (param == PARAM_FASTCHARGE) { + fast_charging_allowed = value; + return EC_RES_SUCCESS; + } + return EC_RES_INVALID_PARAM; +} + +static int command_fastcharge(int argc, char **argv) +{ + if (argc > 1 && !parse_bool(argv[1], &fast_charging_allowed)) + return EC_ERROR_PARAM1; + + ccprintf("fastcharge %s\n", fast_charging_allowed ? "on" : "off"); + + return EC_SUCCESS; +} +DECLARE_CONSOLE_COMMAND(fastcharge, command_fastcharge, + "[on|off]", + "Get or set fast charging profile", + NULL); + +#endif /* CONFIG_CHARGER_PROFILE_OVERRIDE */ diff --git a/driver/build.mk b/driver/build.mk index 26430e2eec..98394eda3f 100644 --- a/driver/build.mk +++ b/driver/build.mk @@ -18,6 +18,7 @@ driver-$(CONFIG_BATTERY_BQ20Z453)+=battery/bq20z453.o driver-$(CONFIG_BATTERY_BQ27541)+=battery/bq27541.o driver-$(CONFIG_BATTERY_BQ27621)+=battery/bq27621_g1.o driver-$(CONFIG_BATTERY_LINK)+=battery/link.o +driver-$(CONFIG_BATTERY_RYU)+=battery/ryu.o driver-$(CONFIG_BATTERY_SAMUS)+=battery/samus.o driver-$(CONFIG_BATTERY_SMART)+=battery/smart.o |