From: 许盛飞 Date: Tue, 6 May 2014 03:27:03 +0000 (+0800) Subject: RK818: add rk818-battery driver X-Git-Tag: firefly_0821_release~5366 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=f575a0f312121901f9ec9645e8087fd9743e33db;p=firefly-linux-kernel-4.4.55.git RK818: add rk818-battery driver --- diff --git a/arch/arm/boot/dts/rk3288-tb.dts b/arch/arm/boot/dts/rk3288-tb.dts index ee39c6afd430..6f6643562980 100755 --- a/arch/arm/boot/dts/rk3288-tb.dts +++ b/arch/arm/boot/dts/rk3288-tb.dts @@ -1146,6 +1146,15 @@ rockchip,power_type = ; }; }; }; + battery { + ocv_table = <3350 3677 3693 3719 3752 3770 3775 3778 3785 3796 3812 3839 3881 3907 3933 3958 3978 4033 4087 4123 4174>; + design_capacity = <2100>; + design_qmax = <2200>; + max_overcharge = <100>; + max_charge_currentma = <1500>; + max_charge_voltagemv = <4260>; + max_bat_voltagemv = <4200>; + }; }; &lcdc_vdd_domain { diff --git a/drivers/mfd/rk818.c b/drivers/mfd/rk818.c index 4b73da29ff96..13ebe3fec977 100755 --- a/drivers/mfd/rk818.c +++ b/drivers/mfd/rk818.c @@ -49,11 +49,11 @@ static struct mfd_cell rk818s[] = { { .name = "rk818-rtc", }, - /* + { .name = "rk818-battery", }, - { + /* { .name = "rk818-power", }, */ diff --git a/drivers/power/Kconfig b/drivers/power/Kconfig index a59488ab56be..f93f742e962a 100755 --- a/drivers/power/Kconfig +++ b/drivers/power/Kconfig @@ -385,6 +385,14 @@ config BATTERY_GOLDFISH Say Y to enable support for the battery and AC power in the Goldfish emulator. +config BATTERY_RK818 + bool "RK818 Battery driver" + depends on MFD_RK818 + default n + help + Support for RK818 Battery driver. + This driver can give support for Rk818 Battery Charge Interface. + config CHARGER_DISPLAY bool "Support charger display" diff --git a/drivers/power/Makefile b/drivers/power/Makefile index 4dc7f07376bc..0030e120ba70 100755 --- a/drivers/power/Makefile +++ b/drivers/power/Makefile @@ -59,3 +59,4 @@ obj-$(CONFIG_CW2015_BATTERY) += cw2015_battery.o obj-$(CONFIG_BATTERY_RICOH619) += ricoh619-battery.o obj-$(CONFIG_CHARGER_DISPLAY) += rk29_charger_display.o obj-$(CONFIG_POWER_RESET) += reset/ +obj-$(CONFIG_BATTERY_RK818) += rk818_battery.o diff --git a/drivers/power/rk818_battery.c b/drivers/power/rk818_battery.c new file mode 100644 index 000000000000..713afa0d605c --- /dev/null +++ b/drivers/power/rk818_battery.c @@ -0,0 +1,2102 @@ +/* + * rk818 battery driver + * + * This package is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +//#include +#include +#include +#include + +static int dbg_enable =0; +module_param_named(dbg_level, dbg_enable, int, 0644); +#define DBG( args...) \ + do { \ + if (dbg_enable) { \ + pr_info(args); \ + } \ + } while (0) + +#define VB_MOD_REG 0x21 + +#define CHRG_COMP_REG1 0x99 +#define CHRG_COMP_REG2 0x9A +#define SUP_STS_REG 0xA0 +#define USB_CTRL_REG 0xA1 +#define CHRG_CTRL_REG1 0xA3 +#define CHRG_CTRL_REG2 0xA4 +#define CHRG_CTRL_REG3 0xA5 +#define BAT_CTRL_REG 0xA6 +#define BAT_HTS_TS1_REG 0xA8 +#define BAT_LTS_TS1_REG 0xA9 +#define BAT_HTS_TS2_REG 0xAA +#define BAT_LTS_TS2_REG 0xAB + + +#define TS_CTRL_REG 0xAC +#define ADC_CTRL_REG 0xAD + +#define ON_SOURCE 0xAE +#define OFF_SOURCE 0xAF + +#define GGCON 0xB0 +#define GGSTS 0xB1 +#define FRAME_SMP_INTERV_REG 0xB2 +#define AUTO_SLP_CUR_THR_REG 0xB3 + +#define GASCNT_CAL_REG3 0xB4 +#define GASCNT_CAL_REG2 0xB5 +#define GASCNT_CAL_REG1 0xB6 +#define GASCNT_CAL_REG0 0xB7 +#define GASCNT3 0xB8 +#define GASCNT2 0xB9 +#define GASCNT1 0xBA +#define GASCNT0 0xBB + +#define BAT_CUR_AVG_REGH 0xBC +#define BAT_CUR_AVG_REGL 0xBD + + +#define TS1_ADC_REGH 0xBE +#define TS1_ADC_REGL 0xBF +#define TS2_ADC_REGH 0xC0 +#define TS2_ADC_REGL 0xC1 + +#define BAT_OCV_REGH 0xC2 +#define BAT_OCV_REGL 0xC3 +#define BAT_VOL_REGH 0xC4 +#define BAT_VOL_REGL 0xC5 + +#define RELAX_ENTRY_THRES_REGH 0xC6 +#define RELAX_ENTRY_THRES_REGL 0xC7 +#define RELAX_EXIT_THRES_REGH 0xC8 +#define RELAX_EXIT_THRES_REGL 0xC9 + +#define RELAX_VOL1_REGH 0xCA +#define RELAX_VOL1_REGL 0xCB +#define RELAX_VOL2_REGH 0xCC +#define RELAX_VOL2_REGL 0xCD + +#define BAT_CUR_R_CALC_REGH 0xCE +#define BAT_CUR_R_CALC_REGL 0xCF +#define BAT_VOL_R_CALC_REGH 0xD0 +#define BAT_VOL_R_CALC_REGL 0xD1 + +#define CAL_OFFSET_REGH 0xD2 +#define CAL_OFFSET_REGL 0xD3 + +#define NON_ACT_TIMER_CNT_REGL 0xD4 + +#define VCALIB0_REGH 0xD5 +#define VCALIB0_REGL 0xD6 +#define VCALIB1_REGH 0xD7 +#define VCALIB1_REGL 0xD8 + +#define IOFFSET_REGH 0xDD +#define IOFFSET_REGL 0xDE + + +/*0xE0 ~0xF2 data register,*/ +#define SOC_REG 0xE0 + +#define REMAIN_CAP_REG3 0xE1 +#define REMAIN_CAP_REG2 0xE2 +#define REMAIN_CAP_REG1 0xE3 +#define REMAIN_CAP_REG0 0xE4 + + + +#define FCC_REGL 0xE1 +#define FCC_REGH 0xE2 + +#define GG_EN 1<<7 // gasgauge module enable bit 0: disable 1:enabsle TS_CTRL_REG 0xAC +//ADC_CTRL_REG +#define ADC_VOL_EN 1<<7 //if GG_EN = 0 , then the ADC of BAT voltage controlled by the bit 0:diabsle 1:enable +#define ADC_CUR_EN 1<<6 //if GG_EN = 0, then the ADC of BAT current controlled by the bit 0: disable 1: enable +#define ADC_TS1_EN 1<<5 //the ADC of TS1 controlled by the bit 0:disabsle 1:enable +#define ADC_TS2_EN 1<<4 //the ADC of TS2 controlled by the bit 0:disabsle 1:enable +#define ADC_PHASE 1<<3 //ADC colock phase 0:normal 1:inverted +#define ADC_CLK_SEL 7 +/******************************************************************* +#define ADC_CLK_SEL_2M 0x000 +#define ADC_CLK_SEL_1M 0x001 +#define ADC_CLK_SEL_500K 0x002 +#define ADC_CLK_SEL_250K 0x003 +#define ADC_CLK_SEL_125K 0x004 +**********************************************************************/ +//GGCON +#define CUR_SAMPL_CON_TIMES 3<<6 // ADC bat current continue sample times 00:8 01:16 10:32 11:64 +#define ADC_OFF_CAL_INTERV 3<<4 //ADC offset calibreation interval time 00:8min 01:16min 10:32min 11:48min +#define OCV_SAMPL_INTERV 3<<2 //OCV sampling interval time 00:8min 01:16min 10:32min :11:48min + +//???????? +#define ADC_CUR_VOL_MODE 1<<1 //ADC working in current voltage collection mode +#define ADC_RES_MODE 1 //ADC working in resistor calculation mode 0:disable 1:enable + +//GGSTS +#define RES_CUR_AVG_SEL 3<<5 //average current filter times 00:1/2 01:1/4 10:1/8 11:1/16 +#define BAT_CON 1<<4 //battery first connection,edge trigger 0:NOT 1:YES +#define RELAX_VOL1_UPD 1<<3 //battery voltage1 update in relax status 0: NOT 1:YE +#define RELAX_VOL2_UPD 1<<2 //battery voltage2 update in relax status 0: NOT 1:YE +#define RELAX_STS 1<<1 //battery coming into relax status 0: NOT 1:YE +#define IV_AVG_UPD_STS 1<<0 //battery average voltage and current updated status 0: NOT 1:YES + +//FRAME_SMP_INTERV_REG +#define AUTO_SLP_EN 1<<5 // auto sleep mode 0:disable 1:enable +#define FRAME_SMP_INTERV_TIME 0x1F // + +#define PLUG_IN_STS 1<<6 + +//SUP_STS_REG +#define BAT_EXS (1<<7) +#define CHARGE_OFF (0x00<<4) +#define DEAD_CHARGE (0x01<<4) +#define TRICKLE_CHARGE (0x02<<4) +#define CC_OR_CV (0x03<<4) +#define CHARGE_FINISH (0x04<<4) +#define USB_OVER_VOL (0x05<<4) +#define BAT_TMP_ERR (0x06<<4) +#define TIMER_ERR (0x07<<4) +#define USB_EXIST (1<<1)// usb is exists +#define USB_EFF (1<<0)// usb is effective + +//USB_CTRL_REG +#define CHRG_CT_EN (1<<7) +// USB_VLIM_SEL +#define VLIM_4000MV (0x00<<4) +#define VLIM_4100MV (0x01<<4) +#define VLIM_4200MV (0x02<<4) +#define VLIM_4300MV (0x03<<4) +#define VLIM_4400MV (0x04<<4) +#define VLIM_4500MV (0x05<<4) +#define VLIM_4600MV (0x06<<4) +#define VLIM_4700MV (0x07<<4) +//USB_ILIM_SEL +#define ILIM_45MA (0x00) +#define ILIM_300MA (0x01) +#define ILIM_80MA (0x02) +#define ILIM_820MA (0x03) +#define ILIM_1000MA (0x04) +#define ILIM_1200MA (0x05) +#define ILIM_1400MA (0x06) +#define ILIM_1600MA (0x07) +#define ILIM_1800MA (0x08) +#define ILIM_2000MA (0x09) +#define ILIM_2200MA (0x0A) +#define ILIM_2400MA (0x0B) +#define ILIM_2600MA (0x0C) +#define ILIM_2800MA (0x0D) +#define ILIM_3000MA (0x0E) + +//CHRG_CTRL_REG +#define CHRG_EN (0x01<<7) +// CHRG_VOL_SEL + +#define CHRG_VOL4050 (0x00<<4) +#define CHRG_VOL4100 (0x01<<4) +#define CHRG_VOL4150 (0x02<<4) +#define CHRG_VOL4200 (0x03<<4) +#define CHRG_VOL4300 (0x04<<4) +#define CHRG_VOL4350 (0x05<<4) + +//CHRG_CUR_SEL +#define CHRG_CUR1000mA (0x00) +#define CHRG_CUR1200mA (0x01) +#define CHRG_CUR1400mA (0x02) +#define CHRG_CUR1600mA (0x03) +#define CHRG_CUR1800mA (0x04) +#define CHRG_CUR2000mA (0x05) +#define CHRG_CUR2200mA (0x06) +#define CHRG_CUR2400mA (0x07) +#define CHRG_CUR2600mA (0x08) +#define CHRG_CUR2800mA (0x09) +#define CHRG_CUR3000mA (0x0A) + + +#define DRIVER_VERSION "1.0.0" +#define ROLEX_SPEED 100 * 1000 + +#define CHARGING 0x01 +#define DISCHARGING 0x00 + +#define TIMER_MS_COUNTS 1000 +#define MAX_CHAR 0x7F +#define MAX_UNSIGNED_CHAR 0xFF +#define MAX_INT 0x7FFFFFFF +#define MAX_UNSIGNED_INT 0xFFFF +#define MAX_INT8 0x7F +#define MAX_UINT8 0xFF + +/* Voltage and Current buffers */ +#define AV_SIZE 5 + +static int16_t av_v[AV_SIZE]; +static int16_t av_c[AV_SIZE]; + +static uint16_t av_v_index; +static uint16_t av_c_index; + +#define INTERPOLATE_MAX 1000 +//#define OCV_TABLE_SIZE + +struct battery_info{ + struct device *dev; + struct cell_state cell; + struct power_supply bat; + struct power_supply ac; + struct power_supply usb; + struct delayed_work work; +// struct i2c_client *client; + + struct rk818 *rk818; + struct battery_platform_data *platform_data; + struct notifier_block battery_nb; + struct workqueue_struct *wq; + struct delayed_work battery_monitor_work; + struct delayed_work charge_check_work; + + int ac_online; + int usb_online; + int health; + int tempreture; + int present; + int status; + + int bat_current; + int current_avg; + int current_offset; + + int voltage; + int voltage_avg; + int voltage_offset; + int voltage_ocv; + + int poweroff_voltage; + int warnning_voltage; + int poweron_voltage; + + int design_capacity; + int fcc; + int new_fcc; + u32 qmax; + int remain_capacity; + int warnning_capacity; + int nac; + int temp_nac; + + int real_soc; + int display_soc; + int temp_soc; + + int soc_counter; + + int dod0; + int dod0_capacity; + int dod1; + int dod1_capacity; + + int temperature; + + int time2empty; + int time2full; + + int *ocv_table; + int ocv_size; + int *res_table; + + int current_k;//(ICALIB0,ICALIB1) + int current_b; + + int voltage_k;//VCALIB0 VCALIB1 + int voltage_b; + + int relax_entry_thres; + int relax_exit_thres; + + int relax_vol1; + int relax_vol2; + + u8 sleep_cur; + u8 sleep_smp_time; + u8 check_count; +// u32 status; + struct timeval soc_timer; + struct timeval change_timer; + + bool resume; + int charge_otg; + +}; + struct battery_info *data; + + +u32 interpolate(int value, u32 *table, int size) +{ + uint8_t i; + uint16_t d; + + for (i = 0; i < size; i++){ + if (value < table[i]) + break; + } + + if ((i > 0) && (i < size)) { + d = (value - table[i-1]) * (INTERPOLATE_MAX/(size-1)); + d /= table[i] - table[i-1]; + d = d + (i-1) * (INTERPOLATE_MAX/(size-1)); + } else { + d = i * ((INTERPOLATE_MAX+size/2)/size); + } + + if (d > 1000) + d = 1000; + + return d; +} +/* Returns (a * b) / c */ +int32_t ab_div_c(u32 a, u32 b, u32 c) +{ + bool sign; + u32 ans = MAX_INT; + int32_t tmp; + + sign = ((((a^b)^c) & 0x80000000) != 0); + + if (c != 0) { + if (sign) + c = -c; + + tmp = ((int32_t) a*b + (c>>1)) / c; + + if (tmp < MAX_INT) + ans = tmp; + } + + if (sign) + ans = -ans; + + return ans; +} + +static int32_t abs_int(int32_t x) +{ + return (x > 0) ? x : -x; +} + +/* Returns diviation between 'size' array members */ +uint16_t diff_array(int16_t *arr, uint8_t size) +{ + uint8_t i; + uint32_t diff = 0; + + for (i = 0; i < size-1; i++) + diff += abs_int(arr[i] - arr[i+1]); + + if (diff > MAX_UNSIGNED_INT) + diff = MAX_UNSIGNED_INT; + + return (uint16_t) diff; +} + + +static enum power_supply_property rk818_battery_props[] = { + + POWER_SUPPLY_PROP_STATUS, + POWER_SUPPLY_PROP_CURRENT_NOW, + POWER_SUPPLY_PROP_VOLTAGE_NOW, + POWER_SUPPLY_PROP_PRESENT, + POWER_SUPPLY_PROP_PRESENT, + POWER_SUPPLY_PROP_CAPACITY, +#if 0 + POWER_SUPPLY_PROP_STATUS, + POWER_SUPPLY_PROP_PRESENT, + POWER_SUPPLY_PROP_VOLTAGE_NOW, + POWER_SUPPLY_PROP_CURRENT_NOW, + POWER_SUPPLY_PROP_CAPACITY, + POWER_SUPPLY_PROP_TEMP, + POWER_SUPPLY_PROP_TECHNOLOGY, + POWER_SUPPLY_PROP_HEALTH, + //POWER_SUPPLY_PROP_TIME_TO_EMPTY_NOW, + //POWER_SUPPLY_PROP_TIME_TO_EMPTY_AVG, + //POWER_SUPPLY_PROP_TIME_TO_FULL_NOW, +#endif + +}; + +static enum power_supply_property rk818_battery_ac_props[] = { + POWER_SUPPLY_PROP_ONLINE, +}; +static enum power_supply_property rk818_battery_usb_props[] = { + POWER_SUPPLY_PROP_ONLINE, +}; + + +static int battery_read(struct rk818 *rk818, u8 reg, u8 buf[], unsigned len) +{ + int ret; + ret = rk818_i2c_read(rk818, reg, len,buf); + return ret; +} + +static int battery_write(struct rk818 *rk818, u8 reg, u8 const buf[], unsigned len) +{ + int ret; + ret = rk818_i2c_write(rk818, reg,(int)len, *buf); + return ret; +} +static void dump_gauge_register(struct battery_info *di) +{ + int i = 0; + char buf; + DBG("%s dump charger register start: \n",__FUNCTION__); + for(i = 0xAC;i < 0xDE; i ++){ + battery_read(di ->rk818, i, &buf,1); + DBG(" the register is 0x%02x, the value is 0x%02x\n ", i, buf); + } + DBG("demp end!\n"); + +} + +static void dump_charger_register(struct battery_info *di) +{ + + int i = 0; + char buf; + DBG("%s dump the register start: \n",__FUNCTION__); + for(i = 0x99;i < 0xAB; i ++){ + battery_read(di ->rk818, i, &buf,1); + DBG(" the register is 0x%02x, the value is 0x%02x\n ", i, buf); + } + DBG("demp end!\n"); + +} +#if 0 +//POWER_SUPPLY_PROP_STATUS +static int rk818_battery_status(struct battery_info *di) +{ + return di->status; +} +//POWER_SUPPLY_PROP_PRESENT, +static int rk818_battery_present(struct rk818_battery_info *di) +{ + return 1; +} +#endif +/* OCV Lookup table + * Open Circuit Voltage (OCV) correction routine. This function estimates SOC, + * based on the voltage. + */ +static int _voltage_to_capacity(struct battery_info * di, int voltage) +{ + u32 *ocv_table; + int ocv_size; + u32 tmp; + + ocv_table = di->platform_data->battery_ocv; + ocv_size = di->platform_data->ocv_size; + // ocv_table = di->ocv_table; + // ocv_size = di->ocv_size; + tmp = interpolate(voltage, ocv_table, ocv_size); + di->temp_soc = ab_div_c(tmp, MAX_PERCENTAGE, INTERPOLATE_MAX); + di->temp_nac= ab_div_c(tmp, di->fcc, INTERPOLATE_MAX); + DBG("temp = %d real-soc =%d nac= %d, fcc = %d\n", tmp, di->temp_soc, di->temp_nac,di->fcc); + return 0; +} +//POWER_SUPPLY_PROP_CURRENT_NOW, +static int _get_average_current(struct battery_info *di) +{ + u8 buf;//[2]; + int ret; + int current_now; + int temp; + + ret = battery_read(di->rk818,BAT_CUR_AVG_REGL, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading BAT_CUR_AVG_REGL"); + return ret; + } + current_now = buf; + ret = battery_read(di->rk818,BAT_CUR_AVG_REGH, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading BAT_CUR_AVG_REGH"); + return ret; + } + current_now |= (buf<<8); + + if(current_now &0x800) + current_now -= 4096; + +// temp = current_now*1000*90/14/4096*500/521; + temp = current_now*1506/1000;//1000*90/14/4096*500/521; + + if(ret < 0){ + dev_err(di->dev, "error reading BAT_CUR_AVG_REGH"); + return ret; + } + + DBG("%s, average current current_now = %d current = %d\n",__FUNCTION__, current_now, temp); + return temp; + +} + +#define to_device_info(x) container_of((x), \ + struct battery_info, bat); + +static int rk818_battery_get_property(struct power_supply *psy, + enum power_supply_property psp, + union power_supply_propval *val) +{ + int ret = 0; + struct battery_info *di = to_device_info(psy); + + switch (psp) { + case POWER_SUPPLY_PROP_CURRENT_NOW: + val->intval = di->current_avg; + break; + + case POWER_SUPPLY_PROP_VOLTAGE_NOW: + case POWER_SUPPLY_PROP_PRESENT: + val->intval = di->voltage;// rk818_battery_voltage(di); + if (psp == POWER_SUPPLY_PROP_PRESENT) + val->intval = val->intval <= 0 ? 0 : 1; + break; + + case POWER_SUPPLY_PROP_CAPACITY: + if(di->real_soc < 0) + di->real_soc = 0; + if(di->real_soc > 100) + di->real_soc = 100; + val->intval =di->real_soc; + //DBG("POWER_SUPPLY_PROP_CAPACITY = %d, val->intval =%d\n", di->real_soc, val->intval); + break; + case POWER_SUPPLY_PROP_HEALTH: + val->intval = POWER_SUPPLY_HEALTH_GOOD;//rk818_battery_health(di); + break; + + case POWER_SUPPLY_PROP_STATUS: + val->intval = di->status; + //DBG("gBatStatus=%d\n",val->intval); + break; + + default: + return -EINVAL; + } + + return ret; +} + +#define to_ac_device_info(x) container_of((x), \ + struct battery_info, ac); + +static int rk818_battery_ac_get_property(struct power_supply *psy, + enum power_supply_property psp, + union power_supply_propval *val) +{ + //DBG("%s:%d psp = %d\n",__FUNCTION__,__LINE__,psp); + int ret = 0; + struct battery_info *di = to_ac_device_info(psy); + + switch (psp) { + case POWER_SUPPLY_PROP_ONLINE: + val->intval = di->ac_online; /*discharging*/ + //DBG("%s:%d val->intval = %d di->status = %d\n",__FUNCTION__,__LINE__,val->intval, di->status); + break; + + default: + ret = -EINVAL; + break; + } + return ret; +} + +#define to_usb_device_info(x) container_of((x), \ + struct battery_info, usb); + +static int rk818_battery_usb_get_property(struct power_supply *psy, + enum power_supply_property psp, + union power_supply_propval *val) +{ + int ret = 0; + struct battery_info *di = to_usb_device_info(psy); + + switch (psp) { + case POWER_SUPPLY_PROP_ONLINE: + val->intval = di->usb_online; /*discharging*/ + //DBG("%s:%d val->intval = %d\n",__FUNCTION__,__LINE__,val->intval); + break; + + default: + ret = -EINVAL; + break; + } + return ret; +} + + +static void battery_powersupply_init(struct battery_info *di) +{ + di->bat.name = "BATTERY"; + di->bat.type = POWER_SUPPLY_TYPE_BATTERY; + di->bat.properties = rk818_battery_props; + di->bat.num_properties = ARRAY_SIZE(rk818_battery_props); + di->bat.get_property = rk818_battery_get_property; + + di->ac.name = "AC"; + di->ac.type = POWER_SUPPLY_TYPE_MAINS; + di->ac.properties = rk818_battery_ac_props; + di->ac.num_properties = ARRAY_SIZE(rk818_battery_ac_props); + di->ac.get_property = rk818_battery_ac_get_property; + + di->usb.name = "USB"; + di->usb.type = POWER_SUPPLY_TYPE_USB; + di->usb.properties = rk818_battery_usb_props; + di->usb.num_properties = ARRAY_SIZE(rk818_battery_usb_props); + di->usb.get_property = rk818_battery_usb_get_property; +} + +//enabsle GG_EN +static int _gauge_enable(struct battery_info *di) +{ + int ret; + u8 buf; + DBG("%s start \n", __FUNCTION__); + ret = battery_read(di->rk818,TS_CTRL_REG, &buf, 1); + DBG("_gauge_enable read-%d\n", buf); + + if(ret < 0){ + dev_err(di->dev, "error reading TS_CTRL_REG"); + return ret; + } + if(!(buf & GG_EN)){ + buf |= GG_EN; + ret = battery_write(di->rk818, TS_CTRL_REG, &buf, 1); //enable + ret = battery_read(di->rk818,TS_CTRL_REG, &buf, 1); + return 0; + } + + DBG("%s,%d\n",__FUNCTION__, buf); + return 0; + +} + +#if 0 + +static int _gauge_disable(struct battery_info *di) +{ + int ret; + u8 buf; + + ret = battery_read(di->rk818,TS_CTRL_REG, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading TS_CTRL_REG"); + return ret; + } + if((buf & GG_EN)){ + buf &= (~0x80);//GG_EN + ret = battery_write(di->rk818, TS_CTRL_REG, &buf, 1); //enable + return 0; + } + return 0; +} + +static int _set_auto_sleep_cur(struct battery_info *di, u8 value) +{ + int ret; + u8 buf; + buf = value; + ret = battery_write(di->rk818, AUTO_SLP_CUR_THR_REG, &buf, 1); //enable + return 0; +} +static int _set_sleep_smp_time(struct battery_info *di, u8 value) +{ + + int ret; + u8 temp; + u8 buf; + + ret = battery_read(di->rk818,FRAME_SMP_INTERV_REG, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading FRAME_SMP_INTERV_REG"); + return ret; + } + + temp = (buf&(AUTO_SLP_EN))|value; + ret = battery_write(di->rk818, FRAME_SMP_INTERV_REG, &temp, 1); //enable + + return 0; +} + +static int _autosleep_enable(struct battery_info *di) +{ + int ret; + u8 buf; + + ret = battery_read(di->rk818,FRAME_SMP_INTERV_REG, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading FRAME_SMP_INTERV_REG"); + return ret; + } + if(!(buf & AUTO_SLP_EN)){ + buf |= AUTO_SLP_EN; + ret = battery_write(di->rk818, FRAME_SMP_INTERV_REG, &buf, 1); //enable + return 0; + } + + _set_auto_sleep_cur(di, di->sleep_cur); // sleep_cur , into sleep-mode + _set_sleep_smp_time(di, di->sleep_smp_time); // time of adc work , sleep-mode + + + return 0; + + +} + +static int _autosleep_disable(struct battery_info *di) +{ + int ret; + u8 buf; + + ret = battery_read(di->rk818,FRAME_SMP_INTERV_REG, &buf, 1); + if(ret < 0){ + dev_err(di->dev, "error reading FRAME_SMP_INTERV_REG"); + return ret; + } + if((buf & AUTO_SLP_EN)){ + buf &= (~AUTO_SLP_EN); + ret = battery_write(di->rk818, FRAME_SMP_INTERV_REG, &buf, 1); //enable + return 0; + } + return 0; + + +} + +#endif +static void _set_relax_thres(struct battery_info *di) +{ + u8 buf; + int enter_thres,exit_thres; + struct cell_state *cell = &di->cell; + + enter_thres = (cell->config->ocv->sleep_enter_current)*1000/1506; + exit_thres = (cell->config->ocv->sleep_exit_current)*1000/1506; + + buf = enter_thres&0xff; + battery_write(di->rk818, RELAX_ENTRY_THRES_REGL, &buf,1); + buf = (enter_thres>>8)&0xff; + battery_write(di->rk818, RELAX_ENTRY_THRES_REGH, &buf,1); + + buf = exit_thres&0xff; + battery_write(di->rk818, RELAX_EXIT_THRES_REGL, &buf,1); + buf = (exit_thres>>8)&0xff; + battery_write(di->rk818, RELAX_EXIT_THRES_REGH, &buf,1); + + //set sample time + battery_read(di->rk818,GGCON, &buf, 1); + buf &= ~(3<<2); + battery_write(di->rk818, GGCON, &buf,1); + } + +static int rk818_battery_voltage(struct battery_info *di) +{ + int ret; + int voltage_now = 0; + u8 buf; + int temp; +#if 1 + ret = battery_read(di->rk818,BAT_VOL_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,BAT_VOL_REGH,&buf, 1); + temp |= buf<<8; +#endif + + //ret = battery_read(di->rk818,BAT_VOL_REGH, buf, 2); + if(ret < 0){ + dev_err(di->dev, "error reading BAT_VOL_REGH"); + return ret; + } + + //voltage_now = temp;//(buf[0]<<8)|buf[1]; + voltage_now = di ->voltage_k*temp + di->voltage_b; + + DBG("the rea-time voltage is %d\n",voltage_now); + return voltage_now; +} + +static int _get_OCV_voltage(struct battery_info *di) +{ + int ret; + int voltage_now = 0; + u8 buf; + int temp; +#if 1 + ret = battery_read(di->rk818,BAT_OCV_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,BAT_OCV_REGH, &buf, 1); + temp |= buf<<8; +#endif + + //ret = battery_read(di->rk818,BAT_OCV_REGH, &buf, 2); + if(ret < 0){ + dev_err(di->dev, "error reading BAT_OCV_REGH"); + return ret; + } + + //voltage_now = temp;//(buf[0]<<8)|buf[1]; + voltage_now = di ->voltage_k*temp + di->voltage_b; + DBG("the OCV voltage is %d\n", voltage_now); + + return voltage_now; +} +#if 0 +static int _get_ts1_adc(struct battery_info *di) +{ + int ret; + int temp = 0; + u8 buf; + + ret = battery_read(di->rk818,TS1_ADC_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,TS1_ADC_REGH, &buf, 1); + temp = (buf<<8); + + return temp; +} + +static int _get_ts2_adc(struct battery_info *di) +{ + int ret; + int temp = 0; + u8 buf; +#if 1 + ret = battery_read(di->rk818,TS2_ADC_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,TS2_ADC_REGH, &buf, 1); + temp |= buf<<8; +#endif + + return temp; +} +#endif +static void _capacity_init(struct battery_info *di, u32 capacity) +{ + + u8 buf; + u32 capacity_ma; + + capacity_ma = capacity*2201;//36*14/900*4096/521*500; + DBG("%s WRITE GANCNT_CAL_REG %d\n", __FUNCTION__, capacity_ma); + do{ + buf = (capacity_ma>>24)&0xff; + battery_write(di->rk818, GASCNT_CAL_REG3, &buf,1); + buf = (capacity_ma>>16)&0xff; + battery_write(di->rk818, GASCNT_CAL_REG2, &buf,1); + buf = (capacity_ma>>8)&0xff; + battery_write(di->rk818, GASCNT_CAL_REG1, &buf,1); + buf = (capacity_ma&0xff)|0x01; + battery_write(di->rk818, GASCNT_CAL_REG0, &buf,1); + battery_read(di->rk818,GASCNT_CAL_REG0, &buf, 1); + + }while(buf == 0); + return; + +} + +static void _save_remain_capacity(struct battery_info *di, u32 capacity) +{ + + u8 buf; + u32 capacity_ma; + + if(capacity >= di ->qmax){ + capacity = di ->qmax; + } + capacity_ma = capacity; +// DBG("%s WRITE GANCNT_CAL_REG %d\n", __FUNCTION__, capacity_ma); + buf = (capacity_ma>>24)&0xff; + battery_write(di->rk818, REMAIN_CAP_REG3, &buf,1); + buf = (capacity_ma>>16)&0xff; + battery_write(di->rk818, REMAIN_CAP_REG2, &buf,1); + buf = (capacity_ma>>8)&0xff; + battery_write(di->rk818, REMAIN_CAP_REG1, &buf,1); + buf = (capacity_ma&0xff)|0x01; + battery_write(di->rk818, REMAIN_CAP_REG0, &buf,1); + + return; + +} + +static int _get_remain_capacity(struct battery_info *di) +{ + int ret; + int temp = 0; + u8 buf; + u32 capacity; + + ret = battery_read(di->rk818,REMAIN_CAP_REG3, &buf, 1); + temp = buf<<24; + ret = battery_read(di->rk818,REMAIN_CAP_REG2, &buf, 1); + temp |= buf<<16; + ret = battery_read(di->rk818,REMAIN_CAP_REG1, &buf, 1); + temp |= buf<<8; + ret = battery_read(di->rk818,REMAIN_CAP_REG0, &buf, 1); + temp |= buf; + + capacity = temp;///4096*900/14/36*500/521; + DBG("%s GASCNT_CAL_REG %d capacity =%d \n",__FUNCTION__, temp, capacity); + return capacity; + +} + + +static int _get_capacity(struct battery_info *di) +{ + int ret; + int temp = 0; + u8 buf; + u32 capacity; + + ret = battery_read(di->rk818,GASCNT_CAL_REG3, &buf, 1); + temp = buf<<24; + ret = battery_read(di->rk818,GASCNT_CAL_REG2, &buf, 1); + temp |= buf<<16; + ret = battery_read(di->rk818,GASCNT_CAL_REG1, &buf, 1); + temp |= buf<<8; + ret = battery_read(di->rk818,GASCNT_CAL_REG0, &buf, 1); + temp |= buf; + + capacity = temp/2201;///4096*900/14/36*500/521; + //DBG("%s GASCNT_CAL_REG %d capacity =%d \n",__FUNCTION__, temp, capacity); + return capacity; + +} + +static int _get_realtime_capacity(struct battery_info *di) +{ + + int ret; + int temp = 0; + u8 buf; + u32 capacity; + + ret = battery_read(di->rk818,GASCNT3, &buf, 1); + temp = buf<<24; + ret = battery_read(di->rk818,GASCNT2, &buf, 1); + temp |= buf<<16; + ret = battery_read(di->rk818,GASCNT1, &buf, 1); + temp |= buf<<8; + ret = battery_read(di->rk818,GASCNT0, &buf, 1); + temp |= buf; +// ret = battery_read(di->rk818,GASCNT_CAL_REG3, &buf, 4); +// temp = buf[0] << 24 | buf[1] << 24 | buf[2] << 24 |buf[3] ; + capacity = temp/2201;///4096*900/14/36*500/521; + //DBG("%s GASCNT = 0x%4x capacity =%d \n",__FUNCTION__, temp,capacity); + return capacity; + + +} + +static int _get_relax_vol1(struct battery_info *di) +{ + int ret; + int temp = 0,voltage_now; + u8 buf; + + ret = battery_read(di->rk818,RELAX_VOL1_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,RELAX_VOL1_REGH, &buf, 1); + temp |= buf<<8; +// ret = battery_read(di->rk818,RELAX_VOL1_REGH, &buf, 2); +// temp = (buf[0]<<8)|buf[1]; + voltage_now = di ->voltage_k*temp + di->voltage_b; + + return voltage_now; +} + +static int _get_relax_vol2(struct battery_info *di) +{ + int ret; + int temp = 0,voltage_now; + u8 buf; + + ret = battery_read(di->rk818,RELAX_VOL2_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,RELAX_VOL2_REGH, &buf, 1); + temp |= buf<<8; +// ret = battery_read(di->rk818,RELAX_VOL2_REGH, &buf, 2); +// temp = (buf[0]<<8)|buf[1]; + voltage_now = di ->voltage_k*temp + di->voltage_b; + + return temp; +} + +static bool _is_relax_mode(struct battery_info *di) +{ + int ret; + u8 status; + int relax_vol1,relax_vol2; + + struct cell_state *cell = &di->cell; + + ret = battery_read(di->rk818,GGSTS, &status, 1); + DBG(" GGSTS the value is %2x the realsoc = %d \n", status, di->real_soc); + + if(!(status&RELAX_STS)) + return false; + if((!(status&RELAX_VOL1_UPD))||(!(status&RELAX_VOL2_UPD))) + return false; + else{ + if ((di->real_soc>= cell->config->ocv->flat_zone_low) + && (di->real_soc <= cell->config->ocv->flat_zone_high)) + return false; + relax_vol1 = _get_relax_vol1(di); + relax_vol2 = _get_relax_vol2(di); + DBG("relax_vol1 = %d relax_vol2 =%d \n", relax_vol1,relax_vol2); + if((abs_int((relax_vol2 - relax_vol1)))/8/60 > 4 ) + return false; + } + + return true; +} + + +static int relax_soc(struct battery_info *di) +{ + //int relax_soc; + _voltage_to_capacity( di, di->voltage); + return di->temp_soc; +} + +static int _get_vcalib0(struct battery_info *di) +{ + + int ret; + int temp = 0; + u8 buf; +#if 1 + ret = battery_read(di->rk818,VCALIB0_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,VCALIB0_REGH, &buf, 1); + temp |= buf<<8; +#endif + //ret = battery_read(di->rk818,VCALIB0_REGH, &buf,2); + //temp = (buf[0]<<8)|buf[1]; + + DBG("%s voltage0 offset vale is %d\n",__FUNCTION__, temp); + return temp; +} + +static int _get_vcalib1(struct battery_info *di) +{ + + int ret; + int temp = 0; + u8 buf; + #if 1 + ret = battery_read(di->rk818,VCALIB1_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,VCALIB1_REGH, &buf, 1); + temp |= buf<<8; + #endif + //ret = battery_read(di->rk818,VCALIB1_REGH, &buf, 2); + //temp = (buf[0]<<8)|buf[1]; + DBG("%s voltage1 offset vale is %d\n",__FUNCTION__, temp); + return temp; +} + +static void _get_voltage_offset_value(struct battery_info *di) +{ + int vcalib0,vcalib1; + + vcalib0 = _get_vcalib0(di); + vcalib1 = _get_vcalib1(di); + + di->voltage_k = (4200 - 3000)/(vcalib1 - vcalib0); + di->voltage_b = 4200 - di->voltage_k*vcalib1; + + return; +} + +static int _get_ioffset(struct battery_info *di) +{ + + int ret; + int temp = 0; + u8 buf; + + ret = battery_read(di->rk818,IOFFSET_REGL, &buf, 1); + temp = buf; + ret = battery_read(di->rk818,IOFFSET_REGH, &buf, 1); + temp |= buf<<8; + + //ret = battery_read(di->rk818,IOFFSET_REGH, &buf, 2); + //temp = (buf[0]<<8)|buf[1]; + + DBG("%s IOFFSET value is %d\n", __FUNCTION__, temp); + return temp; +} + +static int _set_cal_offset(struct battery_info *di, u32 value) +{ + int ret; + int temp = 0; + u8 buf; + DBG("%s\n",__FUNCTION__); + buf = value&0xff; + ret = battery_write(di->rk818, CAL_OFFSET_REGL, &buf, 1); //enable + buf = (value >> 8)&0xff; + ret = battery_write(di->rk818, CAL_OFFSET_REGH, &buf, 1); //enable + DBG("%s set CAL_OFFSET_REG %d\n",__FUNCTION__, temp); + + return 0; +} + +static bool _is_first_poweron(struct battery_info * di) +{ + u8 buf; + u8 temp; + u8 ret; + + ret = battery_read(di->rk818,GGSTS, &buf, 1); + DBG("%s GGSTS value is %2x \n", __FUNCTION__, buf ); + if( buf&BAT_CON){ + buf &=~(BAT_CON); + do{ + battery_write(di->rk818,GGSTS, &buf, 1); + battery_read(di->rk818,GGSTS, &temp, 1); + }while(temp&BAT_CON); + return true; + } + return false; +} + + +#if 0 +static bool fg_check_relaxed(struct battery_info * di)//(struct cell_state *cell) +{ + struct cell_state *cell = &di->cell; + + struct timeval now; + + if (!cell->sleep) { + if (abs_int(di->current_avg) <= + cell->config->ocv->sleep_enter_current) { + if (cell->sleep_samples < MAX_UINT8) + cell->sleep_samples++; + + if (cell->sleep_samples >= + cell->config->ocv->sleep_enter_samples) { + /* Entering sleep mode */ + do_gettimeofday(&cell->sleep_timer); + do_gettimeofday(&cell->el_sleep_timer); + cell->sleep = true; + cell->calibrate = true; + } + } else { + cell->sleep_samples = 0; + } + } else { + /* The battery cell is Sleeping, checking if need to exit + sleep mode count number of seconds that cell spent in + sleep */ + do_gettimeofday(&now); + cell->cumulative_sleep += + now.tv_sec + cell->el_sleep_timer.tv_sec; + do_gettimeofday(&cell->el_sleep_timer); + + /* Check if we need to reset Sleep */ + if (abs_int(di->current_avg) > + cell->config->ocv->sleep_exit_current) { + + if (abs_int(di->current_avg) > + cell->config->ocv->sleep_exit_current) { + + if (cell->sleep_samples < MAX_UINT8) + cell->sleep_samples++; + + } else { + cell->sleep_samples = 0; + } + + /* Check if we need to reset a Sleep timer */ + if (cell->sleep_samples > + cell->config->ocv->sleep_exit_samples) { + /* Exit sleep mode */ + + cell->sleep_timer.tv_sec = 0; + cell->sleep = false; + cell->relax = false; + } + } else { + cell->sleep_samples = 0; + + if (!cell->relax) { + + if (now.tv_sec-cell->sleep_timer.tv_sec > + cell->config->ocv->relax_period) { + cell->relax = true; + cell->calibrate = true; + } + } + } + } + + return cell->relax; +} + +/* Checks for right conditions for OCV correction */ +static bool fg_can_ocv(struct battery_info * di)//(struct cell_state *cell) +{ + struct cell_state *cell = &di->cell; +#if 1 + /* Voltage should be stable */ + if (cell->config->ocv->voltage_diff <= diff_array(av_v, AV_SIZE)) + return false; + + /* Current should be stable */ + if (cell->config->ocv->current_diff <= diff_array(av_c, AV_SIZE)) + return false; +#endif + /* SOC should be out of Flat Zone */ + if ((di->real_soc>= cell->config->ocv->flat_zone_low) + && (di->real_soc <= cell->config->ocv->flat_zone_high)) + return false; + + /* Current should be less then SleepEnterCurrent */ + if (abs_int(di->current_avg) >= cell->config->ocv->sleep_enter_current) + return false; + + /* Don't allow OCV below EDV1, unless OCVbelowEDV1 is set */ + //if (cell->edv1 && !cell->config->ocv_below_edv1) + // return false; + + return true; +} + +#endif + +/* Sets the battery Voltage, and recalculates the average voltage */ +void fg_set_voltage(int16_t voltage) +{ + int16_t i; + int32_t tmp = 0; + + /* put voltage reading int the buffer and update average */ + av_v_index++; + av_v_index %= AV_SIZE; + av_v[av_v_index] = voltage; + for (i = 0; i < AV_SIZE; i++) + tmp += av_v[i]; +} + + +/* Sets the battery Current, and recalculates the average current */ +void fg_set_current( int16_t cur) +{ + int16_t i; + int32_t tmp = 0; + + /* put current reading int the buffer and update average */ + av_c_index++; + av_c_index %= AV_SIZE; + av_c[av_c_index] = cur; + for (i = 0; i < AV_SIZE; i++) + tmp += av_c[i]; + +} + +static int _copy_soc(struct battery_info * di, u8 save_soc) +{ + u8 soc; + + soc = save_soc; + //soc = 85; + battery_write(di->rk818, SOC_REG, &soc, 1); + battery_read(di->rk818, SOC_REG, &soc, 1); + DBG(" the save soc-reg = %d \n", soc); + + return 0; +} +static void _save_rsoc_nac(struct battery_info * di) +{ + u8 buf; + + buf = di->real_soc; + + battery_write(di->rk818, SOC_REG, &buf, 1); + +} + +static int _rsoc_init(struct battery_info * di) +{ + int vol; + u8 temp; + u32 remain_capacity; + + vol = di->voltage_ocv; //_get_OCV_voltage(di); + DBG("OCV voltage = %d\n" , di->voltage_ocv); + if(_is_first_poweron(di)){ + + DBG(" %s this is first poweron\n", __FUNCTION__); + _voltage_to_capacity(di, di->voltage_ocv); + di->real_soc = di->temp_soc; + di->nac = di->temp_nac; + }else{ + DBG(" %s this is not not not first poweron\n", __FUNCTION__); + battery_read(di->rk818,SOC_REG, &temp, 1); + remain_capacity = _get_remain_capacity(di); + if(remain_capacity >= di->qmax) + remain_capacity = di->qmax; + DBG("saved SOC_REG = 0x%8x\n", temp); + DBG("saved remain_capacity = %d\n", remain_capacity); + + + di->real_soc = temp; + //di->nac = di->fcc*temp/100; + di->nac = remain_capacity; + } + return 0; +} + +static int _get_soc(struct battery_info *di) +{ + + return di->remain_capacity * 100 / di->fcc; +} + +static u8 get_charge_status(struct battery_info * di) +{ + u8 status; + u8 ret =0; + + battery_read(di->rk818, SUP_STS_REG, &status, 1); + DBG("%s ----- SUP_STS_REG(0xA0) = 0x%02x\n", __FUNCTION__, status); + status &= ~(0x07<<4); + switch(status){ + case CHARGE_OFF: + ret = CHARGE_OFF; + break; + case DEAD_CHARGE: + ret = DEAD_CHARGE; + break; + case TRICKLE_CHARGE:// (0x02<<4) + ret = DEAD_CHARGE; + break; + case CC_OR_CV: // (0x03<<4) + ret = CC_OR_CV; + break; + case CHARGE_FINISH:// (0x04<<4) + ret = CHARGE_FINISH; + break; + + case USB_OVER_VOL:// (0x05<<4) + ret = USB_OVER_VOL; + break; + + case BAT_TMP_ERR:// (0x06<<4) + ret = BAT_TMP_ERR; + break; + + case TIMER_ERR:// (0x07<<4) + ret = TIMER_ERR; + break; + + case USB_EXIST:// (1<<1)// usb is exists + ret = USB_EXIST; + break; + + case USB_EFF:// (1<<0)// usb is effective + ret = USB_EFF; + break; + default: + return -EINVAL; + } + + return ret; + +} + +static void rk818_battery_charger_init(struct battery_info *di) +{ + u8 chrg_ctrl_reg1,usb_ctrl_reg;// chrg_ctrl_reg2; + u8 sup_sts_reg; + + + DBG("%s start\n",__FUNCTION__); + + battery_read(di->rk818, USB_CTRL_REG, &usb_ctrl_reg, 1); + battery_read(di->rk818, CHRG_CTRL_REG1, &chrg_ctrl_reg1, 1); +// battery_read(di->rk818, CHRG_CTRL_REG2, &chrg_ctrl_reg2, 1); + battery_read(di->rk818, SUP_STS_REG, &sup_sts_reg, 1); + + DBG("old usb_ctrl_reg =0x%2x,CHRG_CTRL_REG1=0x%2x\n ",usb_ctrl_reg, chrg_ctrl_reg1); + //usb_ctrl_reg &= (0x01<<7); + usb_ctrl_reg |= (VLIM_4400MV | ILIM_1200MA)|(0x01<<7); + + chrg_ctrl_reg1 &= (0x00); + chrg_ctrl_reg1 |=(0x01<<7)| (CHRG_VOL4200| CHRG_CUR1400mA); + + sup_sts_reg &= ~(0x01<<3); + sup_sts_reg |= (0x01<<2); + + battery_write(di->rk818, USB_CTRL_REG, &usb_ctrl_reg, 1); + + battery_write(di->rk818, CHRG_CTRL_REG1, &chrg_ctrl_reg1, 1); + //battery_write(di->rk818, CHRG_CTRL_REG2, &chrg_ctrl_reg2, 1); + battery_write(di->rk818, SUP_STS_REG, &sup_sts_reg, 1); + + + battery_read(di->rk818, CHRG_CTRL_REG1, &chrg_ctrl_reg1, 1); +// battery_read(di->rk818, CHRG_CTRL_REG2, &chrg_ctrl_reg2, 1); + battery_read(di->rk818, SUP_STS_REG, &sup_sts_reg, 1); + battery_read(di->rk818, USB_CTRL_REG, &usb_ctrl_reg, 1); + DBG(" new usb_ctrl_reg =0x%2x,CHRG_CTRL_REG1=0x%2x, SUP_STS_REG=0x%2x\n ", + usb_ctrl_reg, chrg_ctrl_reg1,sup_sts_reg); + + DBG("%s end\n",__FUNCTION__); + +} + +extern int rk818_set_bits(struct rk818 *rk818, u8 reg, u8 mask, u8 val); + +void charge_disable_open_otg(struct battery_info *di, int value ) +{ +// u8 chrg_ctrl_reg1,dcdc_en_reg; + if(value == 1){ + DBG("1 ---- charge disable \n"); + rk818_set_bits(di->rk818, CHRG_CTRL_REG1, 1 << 7, 0<< 7); //ldo9 + rk818_set_bits(di->rk818, 0x23, 1 << 7, 1 << 7); //ldo9 + } + if(value == 0){ + DBG("1 ---- charge disable \n"); + rk818_set_bits(di->rk818, 0x23, 1 << 7, 0 << 7); //ldo9 + rk818_set_bits(di->rk818, CHRG_CTRL_REG1, 1 << 7, 1 << 7); //ldo9 + } + +} + +static void fg_init(struct battery_info *di) +{ + DBG("%s start\n",__FUNCTION__); + _gauge_enable(di); + _get_voltage_offset_value(di); //get the volatege offset +// _autosleep_enable(di); + rk818_battery_charger_init(di); +// _set_relax_thres(di); + di->current_offset = _get_ioffset(di); //get the current offset , the value write to the CAL_OFFSET + _set_cal_offset(di,di->current_offset+42); + + di->voltage = rk818_battery_voltage(di); + di->voltage_ocv = _get_OCV_voltage(di); + _rsoc_init( di); + _capacity_init(di, di->nac); +// _get_realtime_capacity( di); + di->remain_capacity = _get_capacity(di); + // _get_realtime_capacity( di); + do_gettimeofday(&di->soc_timer); + di->change_timer = di->soc_timer; +#if 0 + for (i = 0; i < AV_SIZE; i++) { + av_v[i] = di->voltage; + av_c[i] = 0; + } + av_v_index = 0; + av_c_index = 0; +#endif + dump_gauge_register(di); + dump_charger_register(di); + DBG("nac =%d , remain_capacity = %d \n" + " OCV_voltage =%d, voltage =%d \n", + di->nac, di->remain_capacity, + di->voltage_ocv, di->voltage); +} + +#if 0 +static int capacity_changed(struct battery_info *di) +{ + s32 acc_value, samples = 0; + int ret; + int acc_q; + +// fg_set_voltage(&di->cell, di->voltage_mV); + //fg_set_current(&di->cell, (int16_t)(di->current_uA/1000)); + + + return 0; +} + +static void rk818_battery_info(struct battery_info *di) +{ + //di->status = rk818_battery_status(di); + //di->voltage = rk818_battery_voltage(di); + di->present = rk818_battery_present(di); + di->bat_current = _get_average_current(di); + di->temp_soc= rk818_battery_soc(di); + di->tempreture =rk818_battery_temperature(di); + di->health = rk818_battery_health(di); +} +#endif + +static void rk818_battery_display_smooth(struct battery_info *di) +{ + int status; + u8 charge_status; +// int relaxmode_soc; +// int coulomp_soc, soc; + + status = di->status; + if(status == POWER_SUPPLY_STATUS_CHARGING){ + //DBG("charging smooth ... \n"); + if(1){ + //DBG(" BATTERY NOT RELAX MODE \n"); + DBG("di->remain_capacity =%d, di->fcc = %d\n", di->remain_capacity,di->fcc); + di->temp_soc = _get_soc(di); + charge_status = get_charge_status( di); + if(di->temp_soc >= 100){ + di->temp_soc = 100; + //di->status = POWER_SUPPLY_STATUS_FULL; + } + + do_gettimeofday(&di->soc_timer); + + if(di->temp_soc!= di->real_soc){ + di->change_timer = di->soc_timer; + if(di->real_soc < di->temp_soc) + di->real_soc++; + else + di->real_soc =di->temp_soc; + } + + // DBG("charge_status =0x%x\n", charge_status); + if((charge_status ==CHARGE_FINISH) && (di->real_soc < 100)){ + DBG("CHARGE_FINISH di->real_soc < 100 \n "); + if((di->soc_counter < 10)){ + di->soc_counter ++; + }else{ + di->soc_counter = 0; + if(di->real_soc < 100){ + di->real_soc ++; + // _save_rsoc_nac( di); + } + } + } + + } + if(di->real_soc <= 0) + di->real_soc = 0; + if(di->real_soc >= 100){ + di->real_soc = 100; + di->status = POWER_SUPPLY_STATUS_FULL; + } + + } + if(status == POWER_SUPPLY_STATUS_DISCHARGING){ + //DBG("discharging smooth ... \n"); + di->temp_soc = _get_soc(di); + do_gettimeofday(&di->soc_timer); + if(di->temp_soc!= di->real_soc){ + di->change_timer = di->soc_timer; + di->real_soc = di->temp_soc; + // _save_rsoc_nac( di); + } + if(di->real_soc <= 0) + di->real_soc = 0; + if(di->real_soc >= 100){ + di->real_soc = 100; + } +#if 0 + if(!_is_relax_mode( di)){ + DBG(" BATTERY NOT RELAX MODE \n"); + di->temp_soc = _get_soc(di); + do_gettimeofday(&di->soc_timer); + if(di->temp_soc!= di->real_soc){ + di->change_timer = di->soc_timer; + di->real_soc = di->temp_soc; + _save_rsoc_nac( di); + } + + }else{ + DBG("BATTERY RELAX MODE\n "); + //relaxmode_soc = relax_soc(di); + coulomp_soc = _get_soc(di); + soc =coulomp_soc;// (coulomp_soc*20 + relaxmode_soc*80)/100; + + if((soc > di->real_soc)&&(di->soc_counter < 10)){ + di->soc_counter ++; + + }else{ + di->soc_counter = 0; + if(di->real_soc < 100){ + di->real_soc --; + _save_rsoc_nac( di); + } + } + DBG(" remaxmode_soc = %d , coulomp-soc =%d soc = %d\n",relaxmode_soc, coulomp_soc, soc); + } +#endif + + } + //DBG("%s exit \n", __FUNCTION__); +} + +static void rk818_battery_update_status(struct battery_info *di) +{ + + di->voltage = rk818_battery_voltage( di); + di->current_avg = _get_average_current(di); + di->remain_capacity = _get_realtime_capacity( di); + _get_capacity(di); + + rk818_battery_display_smooth(di); + + DBG("%s\n" + "voltage = %d, current-avg = %d\n" + "fcc = %d ,remain_capacity =%d\n" + "real_soc = %d\n", + __FUNCTION__, + di->voltage, di->current_avg, + di->fcc, di->remain_capacity, + di->real_soc + ); +} +extern int dwc_vbus_status(void); +extern int get_gadget_connect_flag(void); + + //state of charge ----running +static int get_charging_status(struct battery_info *di) +{ + +//////////////////////////////////////////// +#if 0 + u8 usb_ctrl_reg;// chrg_ctrl_reg2; + + + + battery_read(di->rk818, USB_CTRL_REG, &usb_ctrl_reg, 1); +// battery_read(di->rk818, CHRG_CTRL_REG2, &chrg_ctrl_reg2, 1); + + DBG("old usb_ctrl_reg =0x%2x,CHRG_CTRL_REG1=0x%2x\n ",usb_ctrl_reg, chrg_ctrl_reg1); + usb_ctrl_reg &= (0x01<<7); + usb_ctrl_reg |= ( ILIM_300MA); +#endif +///////////////////////////////////////// + +// struct rk30_adc_battery_platform_data *pdata = bat->pdata; + int usb_status = 0; // 0--dischage ,1 ---usb charge, 2 ---ac charge + int vbus_status = dwc_vbus_status(); + if (1 == vbus_status) { + if (0 == get_gadget_connect_flag()){ + if (++di->check_count >= 5){ + + di->ac_online = 1; + di->usb_online = 0; + }else{ + di->ac_online =0; + di->usb_online = 1; + + } + }else{ + + di->ac_online =0; + di->usb_online = 1; + } + + }else{ + if (2 == vbus_status) { + + di->ac_online = 1; + di->usb_online = 0; + }else{ + + di->ac_online = 0; + di->usb_online = 0; + } + di->check_count=0; + + } + return usb_status; + +} + +static void get_battery_status(struct battery_info *di) +{ + + u8 buf; + int ret; + ret = battery_read(di->rk818,VB_MOD_REG, &buf, 1); + //int vbus_status = dwc_vbus_status(); + + if(buf&PLUG_IN_STS){ + //if(vbus_status != 0){ + get_charging_status(di); + di->status = POWER_SUPPLY_STATUS_CHARGING; + // di->ac_online = 1; + if(di->real_soc == 100) + di->status = POWER_SUPPLY_STATUS_FULL; + } + else{ + di->status = POWER_SUPPLY_STATUS_DISCHARGING; + di->ac_online =0; + di->usb_online =0; + + } + //DBG("%s ,di->status = %d\n",__FUNCTION__, di->status); +} + +static void rk818_battery_work(struct work_struct *work) +{ + u8 buf; + struct battery_info *di = container_of(work, + struct battery_info, battery_monitor_work.work); + int vbus_status ; + get_battery_status(di); + battery_read(di->rk818,0x00, &buf, 1); + DBG("RTC =0x%2x\n ", buf); + battery_read(di->rk818,VB_MOD_REG, &buf, 1); + //DBG("VB_MOD_REG =%2x, the value is %2x\n ", VB_MOD_REG,buf); + battery_read(di->rk818,SUP_STS_REG, &buf, 1); +// DBG("SUP_STS_REG =%2x, the value is %2x\n ", SUP_STS_REG,buf); + vbus_status = dwc_vbus_status(); +// DBG("vbus_status =%2x\n ", vbus_status); + + rk818_battery_update_status(di); + + if(di ->resume){ + di ->resume = false; + di->real_soc = _get_soc(di); + if(di->real_soc <= 0) + di->real_soc = 0; + if(di->real_soc >= 100) + di->real_soc = 100; + } + if ((di->ac_online == 0 )&&( di->usb_online ==0)&&(di->remain_capacity > di->qmax +10)){ + _capacity_init(di, di->qmax); + di->remain_capacity = _get_realtime_capacity( di); + } + + //DBG("soc = %d", di->real_soc); + _copy_soc(di, di->real_soc); + _save_remain_capacity(di, di->remain_capacity); + power_supply_changed(&di->bat); +// power_supply_changed(&di->usb); + power_supply_changed(&di->ac); + queue_delayed_work(di->wq, &di->battery_monitor_work, msecs_to_jiffies(TIMER_MS_COUNTS*5)); + +} + +static void rk818_battery_charge_check_work(struct work_struct *work) +{ + struct battery_info *di = container_of(work, + struct battery_info, charge_check_work.work); + charge_disable_open_otg(di,di->charge_otg); +} + +static BLOCKING_NOTIFIER_HEAD(battery_chain_head); + +int register_battery_notifier(struct notifier_block *nb) +{ + return blocking_notifier_chain_register(&battery_chain_head, nb); +} +EXPORT_SYMBOL_GPL(register_battery_notifier); + +int unregister_battery_notifier(struct notifier_block *nb) +{ + return blocking_notifier_chain_unregister(&battery_chain_head, nb); +} +EXPORT_SYMBOL_GPL(unregister_battery_notifier); + +int battery_notifier_call_chain(unsigned long val) +{ + return (blocking_notifier_call_chain(&battery_chain_head, val, NULL) + == NOTIFY_BAD) ? -EINVAL : 0; +} +EXPORT_SYMBOL_GPL(battery_notifier_call_chain); +static int battery_notifier_call(struct notifier_block *nb, + unsigned long event, void *data) +{ + struct battery_info *di= + container_of(nb, struct battery_info, battery_nb); + + switch (event) { + case 0: + DBG(" CHARGE enable \n"); + di ->charge_otg = 0; + queue_delayed_work(di->wq, &di->charge_check_work, msecs_to_jiffies(50)); + break; + + case 1: + di ->charge_otg = 1; + queue_delayed_work(di->wq, &di->charge_check_work, msecs_to_jiffies(50)); + + DBG("charge disable OTG enable \n"); + break; + default: + return NOTIFY_OK; + } + return NOTIFY_OK; +} +#ifdef CONFIG_OF +static int rk818_battery_parse_dt(struct rk818 *rk818) +{ + struct device_node *regs,*rk818_pmic_np; + struct battery_platform_data *data; + struct cell_config *cell_cfg; + struct property *prop; + u32 out_value; + int i, length, ret; + + rk818_pmic_np = of_node_get(rk818->dev->of_node); + if (!rk818_pmic_np) { + printk("could not find pmic sub-node\n"); + return -EINVAL; + } + + regs = of_find_node_by_name(rk818_pmic_np, "battery"); + if (!regs){ + printk("could not find battery sub-node\n"); + return -EINVAL; + } + + data = devm_kzalloc(rk818->dev, sizeof(*data), GFP_KERNEL); + memset(data, 0, sizeof(*data)); + + cell_cfg = devm_kzalloc(rk818->dev, sizeof(*cell_cfg), GFP_KERNEL); + /* determine the number of brightness levels */ + prop = of_find_property(regs, "ocv_table", &length); + if (!prop) + return -EINVAL; + data->ocv_size= length / sizeof(u32); + /* read brightness levels from DT property */ + if (data->ocv_size > 0) { + size_t size = sizeof(*data->battery_ocv) * data->ocv_size; + data->battery_ocv= devm_kzalloc(rk818->dev, size, GFP_KERNEL); + if (!data->battery_ocv) + return -ENOMEM; + ret = of_property_read_u32_array(regs, "ocv_table", data->battery_ocv, data->ocv_size); + DBG("the battery OCV TABLE : "); + for(i =0; i< data->ocv_size; i++ ) + DBG("%d ", data->battery_ocv[i]); + DBG("\n"); + if (ret < 0) + return ret; + } + ret = of_property_read_u32(regs, "max_charge_currentmA", &out_value); + if (ret < 0) + return ret; + data->max_charger_currentmA= out_value; + ret = of_property_read_u32(regs, "max_charge_voltagemV", &out_value); + if (ret < 0) + return ret; + data->max_charger_voltagemV= out_value; + ret = of_property_read_u32(regs, "design_capacity", &out_value); + if (ret < 0) + return ret; + cell_cfg->design_capacity = out_value; + ret = of_property_read_u32(regs, "design_qmax", &out_value); + if (ret < 0) + return ret; + cell_cfg->design_qmax =out_value; + data->cell_cfg =cell_cfg; + rk818->battery_data = data; + DBG("max_charge_currentmA :%d\n", data->max_charger_currentmA); + DBG("max_charge_voltagemV :%d\n", data->max_charger_voltagemV); + DBG("design_capacity :%d\n", cell_cfg->design_capacity); + DBG("design_qmax :%d\n", cell_cfg->design_qmax); + + return 0; +} + +static struct of_device_id rk818_battery_of_match[] = { + { .compatible = "rk818_battery" }, + { } +}; + +MODULE_DEVICE_TABLE(of, rk818_battery_of_match); +#else +static int rk818_battery_parse_dt(struct device *dev) +{ + return -ENODEV; +} +#endif + +static int battery_probe(struct platform_device *pdev) +{ + struct rk818 *chip = dev_get_drvdata(pdev->dev.parent); + struct rk818_platform_data *rk818_platform_data = chip->dev->platform_data; +// struct battery_platform_data *pdata ;//= rk818_platform_data->battery_data; +// struct battery_platform_data defdata ;//= rk818_platform_data->battery_data; + struct battery_info *di; + struct ocv_config *ocv; + struct edv_config *edv; + int ret; + + DBG("%s is the battery driver version %s\n",__FUNCTION__,DRIVER_VERSION); + rk818_battery_parse_dt(chip); + + di = devm_kzalloc(&pdev->dev, sizeof(*di), GFP_KERNEL); + if (!di) { + dev_err(&pdev->dev, "no memory for state\n"); + ret = -ENOMEM; + return ret; + } + ocv = devm_kzalloc(&pdev->dev, sizeof(*ocv), GFP_KERNEL); + if (!ocv) { + dev_err(&pdev->dev, "ocv no memory for state\n"); + ret = -ENOMEM; + return ret; + } + edv = devm_kzalloc(&pdev->dev, sizeof(*edv), GFP_KERNEL); + if (!edv) { + dev_err(&pdev->dev, "edv no memory for state\n"); + ret = -ENOMEM; + return ret; + } + + di->rk818 = chip; +#if 0 + di->platform_data = kmemdup(pdata, sizeof(*pdata), GFP_KERNEL); + if (!di->platform_data) { + kfree(di); + return -ENOMEM; + } +#endif +// data = di; + platform_set_drvdata(pdev, di); + /*apply battery cell configuration*/ + //di->cell.config = di->platform_data->cell_cfg; + di->platform_data = chip->battery_data; + di->platform_data->cell_cfg = chip->battery_data->cell_cfg; + di->platform_data->cell_cfg->ocv = ocv; + di->platform_data->cell_cfg->edv = edv; + di->design_capacity = chip->battery_data->cell_cfg->design_capacity; + di->qmax = chip->battery_data->cell_cfg->design_qmax; + di->fcc = di->design_capacity; + di->status = POWER_SUPPLY_STATUS_DISCHARGING; + + battery_powersupply_init(di); + fg_init(di); + ret = power_supply_register(&pdev->dev, &di->bat); + if (ret) { + dev_dbg(&pdev->dev, "failed to register main battery\n"); + goto batt_failed; + } + ret = power_supply_register(&pdev->dev, &di->usb); + if (ret) { + dev_dbg(&pdev->dev, "failed to register usb power supply\n"); + goto usb_failed; + } + ret = power_supply_register(&pdev->dev, &di->ac); + if (ret) { + dev_dbg(&pdev->dev, "failed to register ac power supply\n"); + goto ac_failed; + } + + di->wq = create_singlethread_workqueue("battery-work"); + INIT_DELAYED_WORK(&di->battery_monitor_work,rk818_battery_work); + queue_delayed_work(di->wq, &di->battery_monitor_work, msecs_to_jiffies(TIMER_MS_COUNTS*5)); + //queue_delayed_work(di->wq, &di->charge_check_work, msecs_to_jiffies(TIMER_MS_COUNTS*5)); + INIT_DELAYED_WORK(&di->charge_check_work,rk818_battery_charge_check_work); + + di->battery_nb.notifier_call = battery_notifier_call; + register_battery_notifier(&di->battery_nb); + printk("battery probe ok... \n"); + return ret; + +ac_failed: + power_supply_unregister(&di->ac); +usb_failed: + power_supply_unregister(&di->usb); +batt_failed: + power_supply_unregister(&di->bat); + return ret; +} + +static int battery_remove(struct platform_device *dev) +{ + return 0; +} +#if 1 +static int battery_suspend(struct platform_device *dev,pm_message_t state) +{ + int irq; + struct battery_info *di = platform_get_drvdata(dev); + DBG("%s--------------------\n",__FUNCTION__); + if(di == NULL) + printk("battery NULL di\n"); + cancel_delayed_work(&di ->battery_monitor_work); + DBG("%s---------end--------\n",__FUNCTION__); + + return 0; +} + +static int battery_resume(struct platform_device *dev) +{ + int irq; + + u8 buf; + int ret; + struct battery_info *di = platform_get_drvdata(dev); + + ret = battery_read(di->rk818,VB_MOD_REG, &buf, 1); + +// struct battery_info *di = platform_get_drvdata(dev); + DBG("%s--------------------\n",__FUNCTION__); + queue_delayed_work(di->wq, &di->battery_monitor_work, msecs_to_jiffies(TIMER_MS_COUNTS)); + di ->resume = true; + DBG("charge--status 0x%02x--------------------buf = 0x%02x\n", get_charge_status( di),buf); + + return 0; +} +#endif +static struct platform_driver battery_driver = { + .probe = battery_probe, + .remove = battery_remove, + .suspend = battery_suspend, + .resume = battery_resume, + + .driver = { + .name = "rk818-battery", + //.pm = &pm_ops, + // .of_match_table = of_match_ptr(rk818_battery_parse_dt), + }, +}; + +static int __init battery_init(void) +{ + return platform_driver_register(&battery_driver); +} +fs_initcall_sync(battery_init); +static void __exit battery_exit(void) +{ + platform_driver_unregister(&battery_driver); +} +module_exit(battery_exit); + +MODULE_LICENSE("GPL"); +MODULE_ALIAS("platform:rk818-battery"); +MODULE_AUTHOR("ROCKCHIP"); + + + + + + + + diff --git a/include/linux/mfd/rk818.h b/include/linux/mfd/rk818.h index f7795aaa2d5c..02022044d2bc 100755 --- a/include/linux/mfd/rk818.h +++ b/include/linux/mfd/rk818.h @@ -16,7 +16,7 @@ #include #include #include -//#include +#include //#define RK818_START 30 @@ -129,41 +129,6 @@ struct rk818; #define EN_VBAT_LOW_IRQ (0x1 <<4 ) #define VBAT_LOW_ACT_MASK (0x1 << 4) - -struct rk818_battery_info { - struct i2c_client *client; - struct rk818 *chip; - struct power_supply battery; - struct power_supply ac; - struct power_supply usb; - - struct delayed_work monitor_work; - struct wake_lock monitor_wake_lock; - struct wake_lock low_battery_wake_lock; - struct wake_lock status_wake_lock; - struct wake_lock test_wake_lock; - struct workqueue_struct *workqueue; - struct work_struct usb_irq_work; - - int usb_cnt; - - struct mutex var_lock; - - u16 vcell; - /* battery current */ - s16 curr; - u16 soc; - s16 ext_temp; - u8 online; - u8 status; - u8 internal_status; - u8 health; - u8 present; - unsigned ac_online:1; - unsigned usb_online:1; - -}; - struct rk818_board { int irq; int irq_base; @@ -200,7 +165,7 @@ struct rk818 { struct irq_domain *irq_domain; int (*read)(struct rk818 *rk818, u8 reg, int size, void *dest); int (*write)(struct rk818 *rk818, u8 reg, int size, void *src); - struct rk818_battery_info *battery_info; + struct battery_platform_data *battery_data; int pmic_sleep_gpio; /* */ unsigned int dcdc_slp_voltage[3]; /* buckx_voltage in uV */ bool pmic_sleep; @@ -217,7 +182,7 @@ struct rk818_platform_data { }; int rk818_irq_init(struct rk818 *rk818, int irq,struct rk818_board *pdata); - int rk818_i2c_read(struct rk818 *rk818, char reg, int count,u8 *dest); +int rk818_i2c_read(struct rk818 *rk818, char reg, int count,u8 *dest); //int rk818_i2c_read(struct i2c_client *i2c, char reg, int count,u16 *dest); // int rk818_i2c_read(struct rk818 *rk818 , u8 reg, int bytes,void *dest); int rk818_i2c_write(struct rk818 *rk818, char reg, int count, const u8 src); diff --git a/include/linux/power/rk818_battery.h b/include/linux/power/rk818_battery.h new file mode 100644 index 000000000000..f9b00e32e352 --- /dev/null +++ b/include/linux/power/rk818_battery.h @@ -0,0 +1,275 @@ +/* + *rk818-battery.h - Battery fuel gauge driver structures + * + */ + +#include +/* Gas Gauge Constatnts */ +#define TEMP_0C 2732 +#define MAX_CAPACITY 0x7fff +#define MAX_SOC 100 +#define MAX_PERCENTAGE 100 + +/* Num, cycles with no Learning, after this many cycles, the gauge + start adjusting FCC, based on Estimated Cell Degradation */ +#define NO_LEARNING_CYCLES 25 +/* Size of the OCV Lookup table */ +#define OCV_TABLE_SIZE 21 +/* + * OCV Config + */ +struct ocv_config { +/*voltage_diff, current_diff: Maximal allowed deviation of the voltage and the current from +one reading to the next that allows the fuel gauge to apply an OCV correction. The main purpose + of these thresholds is to filter current and voltage spikes. Recommended value: these value are highly + depend on the load nature. if the load creates a lot of current spikes .the value may need to be increase*/ + uint8_t voltage_diff; + uint8_t current_diff; +/* sleep_enter_current: if the current remains under this threshold for [sleep_enter_samples] + consecutive samples. the gauge enters the SLEEP MODE*/ + uint8_t sleep_enter_current; +/*sleep_enter_samples: the number of samples that satis fy asleep enter or exit condition in order +to actually enter of exit SLEEP mode*/ + uint8_t sleep_enter_samples; +/*sleep_exit_samples: to exit SLEEP mode , average current should pass this threshold first. then +current should remain above this threshold for [sleep_exit_samples] consecutive samples*/ + uint8_t sleep_exit_current; +/*sleep_exit_samples: to exit SLEEP mode, average current should pass this threshold first, then current +should remain above this threshold for [sleep_exit_samples] consecutive samples.*/ + uint8_t sleep_exit_samples; +/*relax_period: defines the number of seconds the fuel gauge should spend in the SLEEP mode +before entering the OCV mode, this setting makes the gauge wait for a cell voltage recovery after +a charge or discharge operation*/ + uint16_t relax_period; +/* flat_zone_low : flat_zone_high :if soc falls into the flat zone low% - flat zone high %.the fuel gauge +wait for a cell voltage recovery after a charge or discharge operation.*/ + uint8_t flat_zone_low; + uint8_t flat_zone_high; +/*FCC leaning is disqualified if the discharge capacity in the OCV mode is greater than this threshold*/ + uint16_t max_ocv_discharge; +/*the 21-point OCV table*/ + uint16_t table[OCV_TABLE_SIZE]; + //uint16_t *table; +}; + +/* EDV Point */ +struct edv_point { + int16_t voltage; + uint8_t percent; +}; + +/* EDV Point tracking data */ +struct edv_state { + int16_t voltage; + uint8_t percent; + int16_t min_capacity; + uint8_t edv_cmp; +}; + +/* EDV Configuration */ +struct edv_config { +/*avieraging: True = evokes averaging on voltage reading to detect an EDV condition. + False = no averaging of voltage readings to detect an EDV conditation.*/ + bool averaging; +/*sequential_edv: the sequential_edv setting defines how many times in a row the battery should +pass the EDV threshold to detect an EDV condition. this setting is intended to fiter short voltage spikes +cause by current spikes*/ + uint8_t sequential_edv; +/*filter_light: difine the calculated EDV voltage recovery IIR filter strength + light-lsetting : for light load (below Qmax/5) + heavy setting : for ligh load (above Qmax/5) + the filter is applied only if the load is greater than Qmax/3 + if average = True. then the Qmax/5 threshold is compared to averge current. + otherwise it is compared to current. + Recommended value: 15-255. 255---disabsle the filter + */ + uint8_t filter_light; + uint8_t filter_heavy; +/*overload_current: the current level above which an EDV condition will not be detected and +capacity not reconciled*/ + int16_t overload_current; + + struct edv_point edv[3]; //xsf +/*edv: the end-of-discharge voltage-to-capactiy correlation points.*/ + //struct edv_point *edv; +}; + +/* General Battery Cell Gauging Configuration */ +struct cell_config { + bool cc_polarity; //´ý¶¨To Be Determined + bool cc_out; + /*ocv_below_edv1: if set (True), OCV correction allowed bellow EDV1 point*/ + bool ocv_below_edv1; + /*cc_voltage: the charge complete voltage threshold(e.g. 4.2v) of the battery. + charge cannot be considered complete if the battery voltage is below this threshold*/ + int16_t cc_voltage; + /*cc_current:the charge complete current threshold(e.g. c/20). charge cannot be considered complete + when charge current and average current are greater than this threshold*/ + int16_t cc_current; + /*design_capacity: design capacity of the battery. the battery datasheet should provide this value*/ + uint16_t design_capacity; + /*design_qmax: the calculated discharge capacity of the OCV discharge curve*/ + int16_t design_qmax; + /*r_sense: the resistance of the current sence element. the sense resistor needs to be slelected to + ensure accurate current measuremen and integration at currents >OFF consumption*/ + uint8_t r_sense; + /*qmax_adjust: the value decremented from QMAX every cycle for aging compensation.*/ + uint8_t qmax_adjust; + /*fcc_adjust: the value decremented from the FCC when no learning happen for 25 cycles in a row*/ + uint8_t fcc_adjust; + /*max_overcharge: the fuel gauge tracks the capacity that goes into the battery after a termination + condition is detected. this improve gauging accuracy if the charger's charge termination condition does't + match to the fuel gauge charge termination condition.*/ + uint16_t max_overcharge; + /*electronics_load: the current that the system consumes int the OFF mode(MPU low power, screen OFF)*/ + uint16_t electronics_load; + /*max_increment: the maximum increment of FCC if the learned capacity is much greater than the exiting + FCC. recommentded value 150mAh*/ + int16_t max_increment; + /*max_decrement: the maximum increment of FCC if the learned capacity is much lower + than the exiting FCC*/ + int16_t max_decrement; + /*low_temp: the correlation between voltage and remaining capacity is considered inaccurate below + this temperature. any leaning will be disqualified, if the battery temperature is below this threshold*/ + uint8_t low_temp; + /*deep_dsg_voltage:in order to qualify capacity learning on the discharge, the battery voltage should + be within EDV-deep-dsg_voltage and EDV.*/ + uint16_t deep_dsg_voltage; + /*max_dsg_voltage:limits the amount of the estimated discharge when learning is in progress.\ + if the amount of the capacity estimation get greater than this threshold ,the learning gets disqualified*/ + uint16_t max_dsg_estimate; + /*light_load: FCC learning on discharge disqualifies if the load is below this threshold when the + when EDV2 is reached.*/ + uint8_t light_load; + /*near_full: this defines a capacity zone from FCC to FCC - near_full. A discharge cycles start + from this capacity zone qualifies for FCC larning.*/ + uint16_t near_full; + /*cycle_threshold: the amount of capacity that should be dicharged from the battery to increment + the cycle count by 1.cycle counting happens on the discharge only.*/ + uint16_t cycle_threshold; + /*recharge: the voltage of recharge.*/ + uint16_t recharge; + /*mode_swtich_capacity: this defines how much capacity should pass through the coulomb counter + to cause a cycle count start condition (either charge or discharge). the gauge support 2 cycle typeds. + charge and discharge. a cycle starts when mode_switch_capacity passes through the coulomb counter + the cycle get canceled and switches to the opposite direciton if mode_switch_capacity passes though + the coulomb counter in oppositer direciton.*/ + uint8_t mode_switch_capacity; + /*call_period: approximate time between fuel gauge calls.*/ + uint8_t call_period; + + struct ocv_config *ocv; + struct edv_config *edv; + //struct ocv_config ocv; + //struct edv_config edv; + +}; + +/* Cell State */ +/* +light-load: ( < C/40) + +*/ +struct cell_state { +/*SOC : state-of-charge of the battery in %,it represents the % full of the battery from the + system empty voltage. + SOC = NAC/FCC, SOC = 1 -DOD +*/ + int16_t soc; +/* nac :nominal avaiable charge of the battery in mAh. it represents the present +remain capacity of the battery to the system empty voltage under nominal conditions*/ + int16_t nac; +/*fcc: full battery capacity .this represents the discharge capacity of the battery from +the defined full condition to the system empty voltage(EDV0) under nominal conditions. + the value is learned by the algorithm on qualified charge and discharge cycleds*/ + int16_t fcc; +/* qmax: the battery capacity(mAh) at the OCV curve discharge rate*/ + int16_t qmax; + + int16_t voltage; + int16_t av_voltage; + int16_t cur; + int16_t av_current; + + int16_t temperature; +/*cycle_count: it represents how many charge or discharge cycles a battery has experience. +this is used to estimate the change of impedance of the battery due to "aging"*/ + int16_t cycle_count; +/*sleep : in this mode ,the battery fuel gauge is counting discharge with the coulomb +counter and checking for the battery relaxed condition, if a relaxed battery is +destected the fuel gauge enters OCV mode*/ + bool sleep; + bool relax; + + bool chg; + bool dsg; + + bool edv0; + bool edv1; + bool edv2; + bool ocv; + bool cc; + bool full; + + bool eocl; + bool vcq; + bool vdq; + bool init; + + struct timeval sleep_timer; + struct timeval el_sleep_timer; + uint16_t cumulative_sleep; + + int16_t prev_soc; + int16_t learn_q; + uint16_t dod_eoc; + int16_t learn_offset; + uint16_t learned_cycle; + int16_t new_fcc; + int16_t ocv_total_q; + int16_t ocv_enter_q; + int16_t negative_q; + int16_t overcharge_q; + int16_t charge_cycle_q; + int16_t discharge_cycle_q; + int16_t cycle_q; + uint8_t sequential_cc; + uint8_t sleep_samples; + uint8_t sequential_edvs; + + uint16_t electronics_load; + uint16_t cycle_dsg_estimate; + + struct edv_state edv; + + bool updated; + bool calibrate; + + struct cell_config *config; +}; + +struct battery_platform_data { + int *battery_tmp_tbl; + unsigned int tblsize; + u32 * battery_ocv ; + unsigned int ocv_size; + + unsigned int monitoring_interval; + + unsigned int max_charger_currentmA; + unsigned int max_charger_voltagemV; + unsigned int termination_currentmA; + + unsigned int max_bat_voltagemV; + unsigned int low_bat_voltagemV; + + unsigned int sense_resistor_mohm; + + /* twl6032 */ + unsigned long features; + unsigned long errata; + + struct cell_config *cell_cfg; +}; + +