/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "healthd-flounder" #include <healthd.h> #include <sys/stat.h> #include <fcntl.h> #include <cutils/klog.h> #include <sys/types.h> #include <sys/sysinfo.h> /* Nominal voltage for ENERGY_COUNTER computation */ #define VOLTAGE_NOMINAL 3.7 #define POWER_SUPPLY_SUBSYSTEM "power_supply" #define POWER_SUPPLY_SYSFS_PATH "/sys/class/" POWER_SUPPLY_SUBSYSTEM #define MAX17050_PATH POWER_SUPPLY_SYSFS_PATH "/battery" #define CHARGE_COUNTER_EXT_PATH MAX17050_PATH "/charge_counter_ext" #define PALMAS_VOLTAGE_MONITOR_PATH POWER_SUPPLY_SYSFS_PATH "/palmas_voltage_monitor" #define VOLTAGE_MONITOR_PATH PALMAS_VOLTAGE_MONITOR_PATH "/device/voltage_monitor" #define BATTERY_FULL 100 #define BATTERY_LOW 15 #define BATTERY_CRITICAL_LOW_MV (3000) #define BATTERY_DEAD_MV (2800) #define NORMAL_MAX_SOC_DEC (2) #define CRITICAL_LOW_FORCE_SOC_DROP (6) #define UPDATE_PERIOD_MINIMUM_S (55) using namespace android; static bool first_update_done; static int lasttime_soc; static unsigned int flounder_monitor_voltage; static long last_update_time; static bool force_decrease; static int read_sysfs(const char *path, char *buf, size_t size) { char *cp = NULL; int fd = open(path, O_RDONLY, 0); if (fd == -1) { KLOG_ERROR(LOG_TAG, "Could not open '%s'\n", path); return -1; } ssize_t count = TEMP_FAILURE_RETRY(read(fd, buf, size)); if (count > 0) cp = (char *)memrchr(buf, '\n', count); if (cp) *cp = '\0'; else buf[0] = '\0'; close(fd); return count; } static void write_sysfs(const char *path, char *s) { char buf[80]; int len; int fd = open(path, O_WRONLY); if (fd < 0) { strerror_r(errno, buf, sizeof(buf)); KLOG_ERROR(LOG_TAG, "Error opening %s: %s\n", path, buf); return; } len = write(fd, s, strlen(s)); if (len < 0) { strerror_r(errno, buf, sizeof(buf)); KLOG_ERROR(LOG_TAG, "Error writing to %s: %s\n", path, buf); } close(fd); } static int64_t get_int64_field(const char *path) { const int SIZE = 21; char buf[SIZE]; int64_t value = 0; if (read_sysfs(path, buf, SIZE) > 0) { value = strtoll(buf, NULL, 0); } return value; } static void flounder_status_check(struct BatteryProperties *props) { if (props->batteryStatus == BATTERY_STATUS_UNKNOWN) props->batteryStatus = BATTERY_STATUS_DISCHARGING; else if (props->batteryStatus == BATTERY_STATUS_FULL && props->batteryLevel < BATTERY_FULL) props->batteryStatus = BATTERY_STATUS_CHARGING; } static void flounder_health_check(struct BatteryProperties *props) { if (props->batteryLevel >= BATTERY_FULL) props->batteryHealth = BATTERY_HEALTH_GOOD; else if (props->batteryLevel < BATTERY_LOW) props->batteryHealth = BATTERY_HEALTH_DEAD; else props->batteryHealth = BATTERY_HEALTH_GOOD; } static void flounder_voltage_monitor_check(struct BatteryProperties *props) { unsigned int monitor_voltage = 0; int vcell_mv; char voltage[10]; if (props->batteryStatus != BATTERY_STATUS_CHARGING && props->batteryStatus != BATTERY_STATUS_FULL && props->batteryLevel > 0) { vcell_mv = props->batteryVoltage; if (vcell_mv > BATTERY_CRITICAL_LOW_MV) monitor_voltage = BATTERY_CRITICAL_LOW_MV; else if (vcell_mv > BATTERY_DEAD_MV) monitor_voltage = BATTERY_DEAD_MV; } if (monitor_voltage != flounder_monitor_voltage) { snprintf(voltage, sizeof(voltage), "%d", monitor_voltage); write_sysfs(VOLTAGE_MONITOR_PATH, voltage); flounder_monitor_voltage = monitor_voltage; } } static void flounder_soc_adjust(struct BatteryProperties *props) { int soc_decrease; int soc, vcell_mv; struct sysinfo info; long uptime = 0; int ret; ret = sysinfo(&info); if (ret) { KLOG_ERROR(LOG_TAG, "Fail to get sysinfo!!\n"); uptime = last_update_time; } else uptime = info.uptime; if (!first_update_done) { if (props->batteryLevel >= BATTERY_FULL) { props->batteryLevel = BATTERY_FULL - 1; lasttime_soc = BATTERY_FULL - 1; } else { lasttime_soc = props->batteryLevel; } last_update_time = uptime; first_update_done = true; } if (props->batteryStatus == BATTERY_STATUS_FULL) soc = BATTERY_FULL; else if (props->batteryLevel >= BATTERY_FULL && lasttime_soc < BATTERY_FULL) soc = BATTERY_FULL - 1; else soc = props->batteryLevel; if (props->batteryLevel > BATTERY_FULL) props->batteryLevel = BATTERY_FULL; else if (props->batteryLevel < 0) props->batteryLevel = 0; vcell_mv = props->batteryVoltage; if (props->batteryStatus == BATTERY_STATUS_DISCHARGING || props->batteryStatus == BATTERY_STATUS_NOT_CHARGING || props->batteryStatus == BATTERY_STATUS_UNKNOWN) { if (vcell_mv >= BATTERY_CRITICAL_LOW_MV) { force_decrease = false; soc_decrease = lasttime_soc - soc; if (soc_decrease < 0) { soc = lasttime_soc; goto done; } if (uptime > last_update_time && uptime - last_update_time <= UPDATE_PERIOD_MINIMUM_S) { soc = lasttime_soc; goto done; } if (soc_decrease < 0) soc_decrease = 0; else if (soc_decrease > NORMAL_MAX_SOC_DEC) soc_decrease = NORMAL_MAX_SOC_DEC; soc = lasttime_soc - soc_decrease; } else if (vcell_mv < BATTERY_DEAD_MV) { soc = 0; } else { if (force_decrease && uptime > last_update_time && uptime - last_update_time <= UPDATE_PERIOD_MINIMUM_S) { soc = lasttime_soc; goto done; } soc_decrease = CRITICAL_LOW_FORCE_SOC_DROP; if (lasttime_soc <= soc_decrease) soc = 0; else soc = lasttime_soc - soc_decrease; force_decrease = true; } } else { force_decrease = false; if (soc > lasttime_soc) soc = lasttime_soc + 1; } last_update_time = uptime; done: props->batteryLevel = lasttime_soc = soc; } static void flounder_bat_monitor(struct BatteryProperties *props) { flounder_soc_adjust(props); flounder_health_check(props); flounder_status_check(props); flounder_voltage_monitor_check(props); } int healthd_board_battery_update(struct BatteryProperties *props) { flounder_bat_monitor(props); // return 0 to log periodic polled battery status to kernel log return 0; } static int flounder_energy_counter(int64_t *energy) { *energy = get_int64_field(CHARGE_COUNTER_EXT_PATH) * VOLTAGE_NOMINAL; return 0; } void healthd_board_init(struct healthd_config *config) { config->energyCounter = flounder_energy_counter; }