Use this new function to make code more comprehensible, since we are reinitialzing the completion, not initializing. [akpm@linux-foundation.org: linux-next resyncs] Signed-off-by: Wolfram Sang <wsa@the-dreams.de> Acked-by: Linus Walleij <linus.walleij@linaro.org> (personally at LCE13) Cc: Ingo Molnar <mingo@kernel.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			3303 lines
		
	
	
	
		
			84 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			3303 lines
		
	
	
	
		
			84 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) ST-Ericsson AB 2012
 | 
						|
 *
 | 
						|
 * Main and Back-up battery management driver.
 | 
						|
 *
 | 
						|
 * Note: Backup battery management is required in case of Li-Ion battery and not
 | 
						|
 * for capacitive battery. HREF boards have capacitive battery and hence backup
 | 
						|
 * battery management is not used and the supported code is available in this
 | 
						|
 * driver.
 | 
						|
 *
 | 
						|
 * License Terms: GNU General Public License v2
 | 
						|
 * Author:
 | 
						|
 *	Johan Palsson <johan.palsson@stericsson.com>
 | 
						|
 *	Karl Komierowski <karl.komierowski@stericsson.com>
 | 
						|
 *	Arun R Murthy <arun.murthy@stericsson.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/init.h>
 | 
						|
#include <linux/module.h>
 | 
						|
#include <linux/device.h>
 | 
						|
#include <linux/interrupt.h>
 | 
						|
#include <linux/platform_device.h>
 | 
						|
#include <linux/power_supply.h>
 | 
						|
#include <linux/kobject.h>
 | 
						|
#include <linux/slab.h>
 | 
						|
#include <linux/delay.h>
 | 
						|
#include <linux/time.h>
 | 
						|
#include <linux/of.h>
 | 
						|
#include <linux/completion.h>
 | 
						|
#include <linux/mfd/core.h>
 | 
						|
#include <linux/mfd/abx500.h>
 | 
						|
#include <linux/mfd/abx500/ab8500.h>
 | 
						|
#include <linux/mfd/abx500/ab8500-bm.h>
 | 
						|
#include <linux/mfd/abx500/ab8500-gpadc.h>
 | 
						|
#include <linux/kernel.h>
 | 
						|
 | 
						|
#define MILLI_TO_MICRO			1000
 | 
						|
#define FG_LSB_IN_MA			1627
 | 
						|
#define QLSB_NANO_AMP_HOURS_X10		1071
 | 
						|
#define INS_CURR_TIMEOUT		(3 * HZ)
 | 
						|
 | 
						|
#define SEC_TO_SAMPLE(S)		(S * 4)
 | 
						|
 | 
						|
#define NBR_AVG_SAMPLES			20
 | 
						|
 | 
						|
#define LOW_BAT_CHECK_INTERVAL		(HZ / 16) /* 62.5 ms */
 | 
						|
 | 
						|
#define VALID_CAPACITY_SEC		(45 * 60) /* 45 minutes */
 | 
						|
#define BATT_OK_MIN			2360 /* mV */
 | 
						|
#define BATT_OK_INCREMENT		50 /* mV */
 | 
						|
#define BATT_OK_MAX_NR_INCREMENTS	0xE
 | 
						|
 | 
						|
/* FG constants */
 | 
						|
#define BATT_OVV			0x01
 | 
						|
 | 
						|
#define interpolate(x, x1, y1, x2, y2) \
 | 
						|
	((y1) + ((((y2) - (y1)) * ((x) - (x1))) / ((x2) - (x1))));
 | 
						|
 | 
						|
#define to_ab8500_fg_device_info(x) container_of((x), \
 | 
						|
	struct ab8500_fg, fg_psy);
 | 
						|
 | 
						|
/**
 | 
						|
 * struct ab8500_fg_interrupts - ab8500 fg interupts
 | 
						|
 * @name:	name of the interrupt
 | 
						|
 * @isr		function pointer to the isr
 | 
						|
 */
 | 
						|
struct ab8500_fg_interrupts {
 | 
						|
	char *name;
 | 
						|
	irqreturn_t (*isr)(int irq, void *data);
 | 
						|
};
 | 
						|
 | 
						|
enum ab8500_fg_discharge_state {
 | 
						|
	AB8500_FG_DISCHARGE_INIT,
 | 
						|
	AB8500_FG_DISCHARGE_INITMEASURING,
 | 
						|
	AB8500_FG_DISCHARGE_INIT_RECOVERY,
 | 
						|
	AB8500_FG_DISCHARGE_RECOVERY,
 | 
						|
	AB8500_FG_DISCHARGE_READOUT_INIT,
 | 
						|
	AB8500_FG_DISCHARGE_READOUT,
 | 
						|
	AB8500_FG_DISCHARGE_WAKEUP,
 | 
						|
};
 | 
						|
 | 
						|
static char *discharge_state[] = {
 | 
						|
	"DISCHARGE_INIT",
 | 
						|
	"DISCHARGE_INITMEASURING",
 | 
						|
	"DISCHARGE_INIT_RECOVERY",
 | 
						|
	"DISCHARGE_RECOVERY",
 | 
						|
	"DISCHARGE_READOUT_INIT",
 | 
						|
	"DISCHARGE_READOUT",
 | 
						|
	"DISCHARGE_WAKEUP",
 | 
						|
};
 | 
						|
 | 
						|
enum ab8500_fg_charge_state {
 | 
						|
	AB8500_FG_CHARGE_INIT,
 | 
						|
	AB8500_FG_CHARGE_READOUT,
 | 
						|
};
 | 
						|
 | 
						|
static char *charge_state[] = {
 | 
						|
	"CHARGE_INIT",
 | 
						|
	"CHARGE_READOUT",
 | 
						|
};
 | 
						|
 | 
						|
enum ab8500_fg_calibration_state {
 | 
						|
	AB8500_FG_CALIB_INIT,
 | 
						|
	AB8500_FG_CALIB_WAIT,
 | 
						|
	AB8500_FG_CALIB_END,
 | 
						|
};
 | 
						|
 | 
						|
struct ab8500_fg_avg_cap {
 | 
						|
	int avg;
 | 
						|
	int samples[NBR_AVG_SAMPLES];
 | 
						|
	__kernel_time_t time_stamps[NBR_AVG_SAMPLES];
 | 
						|
	int pos;
 | 
						|
	int nbr_samples;
 | 
						|
	int sum;
 | 
						|
};
 | 
						|
 | 
						|
struct ab8500_fg_cap_scaling {
 | 
						|
	bool enable;
 | 
						|
	int cap_to_scale[2];
 | 
						|
	int disable_cap_level;
 | 
						|
	int scaled_cap;
 | 
						|
};
 | 
						|
 | 
						|
struct ab8500_fg_battery_capacity {
 | 
						|
	int max_mah_design;
 | 
						|
	int max_mah;
 | 
						|
	int mah;
 | 
						|
	int permille;
 | 
						|
	int level;
 | 
						|
	int prev_mah;
 | 
						|
	int prev_percent;
 | 
						|
	int prev_level;
 | 
						|
	int user_mah;
 | 
						|
	struct ab8500_fg_cap_scaling cap_scale;
 | 
						|
};
 | 
						|
 | 
						|
struct ab8500_fg_flags {
 | 
						|
	bool fg_enabled;
 | 
						|
	bool conv_done;
 | 
						|
	bool charging;
 | 
						|
	bool fully_charged;
 | 
						|
	bool force_full;
 | 
						|
	bool low_bat_delay;
 | 
						|
	bool low_bat;
 | 
						|
	bool bat_ovv;
 | 
						|
	bool batt_unknown;
 | 
						|
	bool calibrate;
 | 
						|
	bool user_cap;
 | 
						|
	bool batt_id_received;
 | 
						|
};
 | 
						|
 | 
						|
struct inst_curr_result_list {
 | 
						|
	struct list_head list;
 | 
						|
	int *result;
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * struct ab8500_fg - ab8500 FG device information
 | 
						|
 * @dev:		Pointer to the structure device
 | 
						|
 * @node:		a list of AB8500 FGs, hence prepared for reentrance
 | 
						|
 * @irq			holds the CCEOC interrupt number
 | 
						|
 * @vbat:		Battery voltage in mV
 | 
						|
 * @vbat_nom:		Nominal battery voltage in mV
 | 
						|
 * @inst_curr:		Instantenous battery current in mA
 | 
						|
 * @avg_curr:		Average battery current in mA
 | 
						|
 * @bat_temp		battery temperature
 | 
						|
 * @fg_samples:		Number of samples used in the FG accumulation
 | 
						|
 * @accu_charge:	Accumulated charge from the last conversion
 | 
						|
 * @recovery_cnt:	Counter for recovery mode
 | 
						|
 * @high_curr_cnt:	Counter for high current mode
 | 
						|
 * @init_cnt:		Counter for init mode
 | 
						|
 * @low_bat_cnt		Counter for number of consecutive low battery measures
 | 
						|
 * @nbr_cceoc_irq_cnt	Counter for number of CCEOC irqs received since enabled
 | 
						|
 * @recovery_needed:	Indicate if recovery is needed
 | 
						|
 * @high_curr_mode:	Indicate if we're in high current mode
 | 
						|
 * @init_capacity:	Indicate if initial capacity measuring should be done
 | 
						|
 * @turn_off_fg:	True if fg was off before current measurement
 | 
						|
 * @calib_state		State during offset calibration
 | 
						|
 * @discharge_state:	Current discharge state
 | 
						|
 * @charge_state:	Current charge state
 | 
						|
 * @ab8500_fg_started	Completion struct used for the instant current start
 | 
						|
 * @ab8500_fg_complete	Completion struct used for the instant current reading
 | 
						|
 * @flags:		Structure for information about events triggered
 | 
						|
 * @bat_cap:		Structure for battery capacity specific parameters
 | 
						|
 * @avg_cap:		Average capacity filter
 | 
						|
 * @parent:		Pointer to the struct ab8500
 | 
						|
 * @gpadc:		Pointer to the struct gpadc
 | 
						|
 * @bm:           	Platform specific battery management information
 | 
						|
 * @fg_psy:		Structure that holds the FG specific battery properties
 | 
						|
 * @fg_wq:		Work queue for running the FG algorithm
 | 
						|
 * @fg_periodic_work:	Work to run the FG algorithm periodically
 | 
						|
 * @fg_low_bat_work:	Work to check low bat condition
 | 
						|
 * @fg_reinit_work	Work used to reset and reinitialise the FG algorithm
 | 
						|
 * @fg_work:		Work to run the FG algorithm instantly
 | 
						|
 * @fg_acc_cur_work:	Work to read the FG accumulator
 | 
						|
 * @fg_check_hw_failure_work:	Work for checking HW state
 | 
						|
 * @cc_lock:		Mutex for locking the CC
 | 
						|
 * @fg_kobject:		Structure of type kobject
 | 
						|
 */
 | 
						|
struct ab8500_fg {
 | 
						|
	struct device *dev;
 | 
						|
	struct list_head node;
 | 
						|
	int irq;
 | 
						|
	int vbat;
 | 
						|
	int vbat_nom;
 | 
						|
	int inst_curr;
 | 
						|
	int avg_curr;
 | 
						|
	int bat_temp;
 | 
						|
	int fg_samples;
 | 
						|
	int accu_charge;
 | 
						|
	int recovery_cnt;
 | 
						|
	int high_curr_cnt;
 | 
						|
	int init_cnt;
 | 
						|
	int low_bat_cnt;
 | 
						|
	int nbr_cceoc_irq_cnt;
 | 
						|
	bool recovery_needed;
 | 
						|
	bool high_curr_mode;
 | 
						|
	bool init_capacity;
 | 
						|
	bool turn_off_fg;
 | 
						|
	enum ab8500_fg_calibration_state calib_state;
 | 
						|
	enum ab8500_fg_discharge_state discharge_state;
 | 
						|
	enum ab8500_fg_charge_state charge_state;
 | 
						|
	struct completion ab8500_fg_started;
 | 
						|
	struct completion ab8500_fg_complete;
 | 
						|
	struct ab8500_fg_flags flags;
 | 
						|
	struct ab8500_fg_battery_capacity bat_cap;
 | 
						|
	struct ab8500_fg_avg_cap avg_cap;
 | 
						|
	struct ab8500 *parent;
 | 
						|
	struct ab8500_gpadc *gpadc;
 | 
						|
	struct abx500_bm_data *bm;
 | 
						|
	struct power_supply fg_psy;
 | 
						|
	struct workqueue_struct *fg_wq;
 | 
						|
	struct delayed_work fg_periodic_work;
 | 
						|
	struct delayed_work fg_low_bat_work;
 | 
						|
	struct delayed_work fg_reinit_work;
 | 
						|
	struct work_struct fg_work;
 | 
						|
	struct work_struct fg_acc_cur_work;
 | 
						|
	struct delayed_work fg_check_hw_failure_work;
 | 
						|
	struct mutex cc_lock;
 | 
						|
	struct kobject fg_kobject;
 | 
						|
};
 | 
						|
static LIST_HEAD(ab8500_fg_list);
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_get() - returns a reference to the primary AB8500 fuel gauge
 | 
						|
 * (i.e. the first fuel gauge in the instance list)
 | 
						|
 */
 | 
						|
struct ab8500_fg *ab8500_fg_get(void)
 | 
						|
{
 | 
						|
	struct ab8500_fg *fg;
 | 
						|
 | 
						|
	if (list_empty(&ab8500_fg_list))
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	fg = list_first_entry(&ab8500_fg_list, struct ab8500_fg, node);
 | 
						|
	return fg;
 | 
						|
}
 | 
						|
 | 
						|
/* Main battery properties */
 | 
						|
static enum power_supply_property ab8500_fg_props[] = {
 | 
						|
	POWER_SUPPLY_PROP_VOLTAGE_NOW,
 | 
						|
	POWER_SUPPLY_PROP_CURRENT_NOW,
 | 
						|
	POWER_SUPPLY_PROP_CURRENT_AVG,
 | 
						|
	POWER_SUPPLY_PROP_ENERGY_FULL_DESIGN,
 | 
						|
	POWER_SUPPLY_PROP_ENERGY_FULL,
 | 
						|
	POWER_SUPPLY_PROP_ENERGY_NOW,
 | 
						|
	POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN,
 | 
						|
	POWER_SUPPLY_PROP_CHARGE_FULL,
 | 
						|
	POWER_SUPPLY_PROP_CHARGE_NOW,
 | 
						|
	POWER_SUPPLY_PROP_CAPACITY,
 | 
						|
	POWER_SUPPLY_PROP_CAPACITY_LEVEL,
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * This array maps the raw hex value to lowbat voltage used by the AB8500
 | 
						|
 * Values taken from the UM0836
 | 
						|
 */
 | 
						|
static int ab8500_fg_lowbat_voltage_map[] = {
 | 
						|
	2300 ,
 | 
						|
	2325 ,
 | 
						|
	2350 ,
 | 
						|
	2375 ,
 | 
						|
	2400 ,
 | 
						|
	2425 ,
 | 
						|
	2450 ,
 | 
						|
	2475 ,
 | 
						|
	2500 ,
 | 
						|
	2525 ,
 | 
						|
	2550 ,
 | 
						|
	2575 ,
 | 
						|
	2600 ,
 | 
						|
	2625 ,
 | 
						|
	2650 ,
 | 
						|
	2675 ,
 | 
						|
	2700 ,
 | 
						|
	2725 ,
 | 
						|
	2750 ,
 | 
						|
	2775 ,
 | 
						|
	2800 ,
 | 
						|
	2825 ,
 | 
						|
	2850 ,
 | 
						|
	2875 ,
 | 
						|
	2900 ,
 | 
						|
	2925 ,
 | 
						|
	2950 ,
 | 
						|
	2975 ,
 | 
						|
	3000 ,
 | 
						|
	3025 ,
 | 
						|
	3050 ,
 | 
						|
	3075 ,
 | 
						|
	3100 ,
 | 
						|
	3125 ,
 | 
						|
	3150 ,
 | 
						|
	3175 ,
 | 
						|
	3200 ,
 | 
						|
	3225 ,
 | 
						|
	3250 ,
 | 
						|
	3275 ,
 | 
						|
	3300 ,
 | 
						|
	3325 ,
 | 
						|
	3350 ,
 | 
						|
	3375 ,
 | 
						|
	3400 ,
 | 
						|
	3425 ,
 | 
						|
	3450 ,
 | 
						|
	3475 ,
 | 
						|
	3500 ,
 | 
						|
	3525 ,
 | 
						|
	3550 ,
 | 
						|
	3575 ,
 | 
						|
	3600 ,
 | 
						|
	3625 ,
 | 
						|
	3650 ,
 | 
						|
	3675 ,
 | 
						|
	3700 ,
 | 
						|
	3725 ,
 | 
						|
	3750 ,
 | 
						|
	3775 ,
 | 
						|
	3800 ,
 | 
						|
	3825 ,
 | 
						|
	3850 ,
 | 
						|
	3850 ,
 | 
						|
};
 | 
						|
 | 
						|
static u8 ab8500_volt_to_regval(int voltage)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	if (voltage < ab8500_fg_lowbat_voltage_map[0])
 | 
						|
		return 0;
 | 
						|
 | 
						|
	for (i = 0; i < ARRAY_SIZE(ab8500_fg_lowbat_voltage_map); i++) {
 | 
						|
		if (voltage < ab8500_fg_lowbat_voltage_map[i])
 | 
						|
			return (u8) i - 1;
 | 
						|
	}
 | 
						|
 | 
						|
	/* If not captured above, return index of last element */
 | 
						|
	return (u8) ARRAY_SIZE(ab8500_fg_lowbat_voltage_map) - 1;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_is_low_curr() - Low or high current mode
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @curr:	the current to base or our decision on
 | 
						|
 *
 | 
						|
 * Low current mode if the current consumption is below a certain threshold
 | 
						|
 */
 | 
						|
static int ab8500_fg_is_low_curr(struct ab8500_fg *di, int curr)
 | 
						|
{
 | 
						|
	/*
 | 
						|
	 * We want to know if we're in low current mode
 | 
						|
	 */
 | 
						|
	if (curr > -di->bm->fg_params->high_curr_threshold)
 | 
						|
		return true;
 | 
						|
	else
 | 
						|
		return false;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_add_cap_sample() - Add capacity to average filter
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @sample:	the capacity in mAh to add to the filter
 | 
						|
 *
 | 
						|
 * A capacity is added to the filter and a new mean capacity is calculated and
 | 
						|
 * returned
 | 
						|
 */
 | 
						|
static int ab8500_fg_add_cap_sample(struct ab8500_fg *di, int sample)
 | 
						|
{
 | 
						|
	struct timespec ts;
 | 
						|
	struct ab8500_fg_avg_cap *avg = &di->avg_cap;
 | 
						|
 | 
						|
	getnstimeofday(&ts);
 | 
						|
 | 
						|
	do {
 | 
						|
		avg->sum += sample - avg->samples[avg->pos];
 | 
						|
		avg->samples[avg->pos] = sample;
 | 
						|
		avg->time_stamps[avg->pos] = ts.tv_sec;
 | 
						|
		avg->pos++;
 | 
						|
 | 
						|
		if (avg->pos == NBR_AVG_SAMPLES)
 | 
						|
			avg->pos = 0;
 | 
						|
 | 
						|
		if (avg->nbr_samples < NBR_AVG_SAMPLES)
 | 
						|
			avg->nbr_samples++;
 | 
						|
 | 
						|
		/*
 | 
						|
		 * Check the time stamp for each sample. If too old,
 | 
						|
		 * replace with latest sample
 | 
						|
		 */
 | 
						|
	} while (ts.tv_sec - VALID_CAPACITY_SEC > avg->time_stamps[avg->pos]);
 | 
						|
 | 
						|
	avg->avg = avg->sum / avg->nbr_samples;
 | 
						|
 | 
						|
	return avg->avg;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_clear_cap_samples() - Clear average filter
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * The capacity filter is is reset to zero.
 | 
						|
 */
 | 
						|
static void ab8500_fg_clear_cap_samples(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
	struct ab8500_fg_avg_cap *avg = &di->avg_cap;
 | 
						|
 | 
						|
	avg->pos = 0;
 | 
						|
	avg->nbr_samples = 0;
 | 
						|
	avg->sum = 0;
 | 
						|
	avg->avg = 0;
 | 
						|
 | 
						|
	for (i = 0; i < NBR_AVG_SAMPLES; i++) {
 | 
						|
		avg->samples[i] = 0;
 | 
						|
		avg->time_stamps[i] = 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_fill_cap_sample() - Fill average filter
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @sample:	the capacity in mAh to fill the filter with
 | 
						|
 *
 | 
						|
 * The capacity filter is filled with a capacity in mAh
 | 
						|
 */
 | 
						|
static void ab8500_fg_fill_cap_sample(struct ab8500_fg *di, int sample)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
	struct timespec ts;
 | 
						|
	struct ab8500_fg_avg_cap *avg = &di->avg_cap;
 | 
						|
 | 
						|
	getnstimeofday(&ts);
 | 
						|
 | 
						|
	for (i = 0; i < NBR_AVG_SAMPLES; i++) {
 | 
						|
		avg->samples[i] = sample;
 | 
						|
		avg->time_stamps[i] = ts.tv_sec;
 | 
						|
	}
 | 
						|
 | 
						|
	avg->pos = 0;
 | 
						|
	avg->nbr_samples = NBR_AVG_SAMPLES;
 | 
						|
	avg->sum = sample * NBR_AVG_SAMPLES;
 | 
						|
	avg->avg = sample;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_coulomb_counter() - enable coulomb counter
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @enable:	enable/disable
 | 
						|
 *
 | 
						|
 * Enable/Disable coulomb counter.
 | 
						|
 * On failure returns negative value.
 | 
						|
 */
 | 
						|
static int ab8500_fg_coulomb_counter(struct ab8500_fg *di, bool enable)
 | 
						|
{
 | 
						|
	int ret = 0;
 | 
						|
	mutex_lock(&di->cc_lock);
 | 
						|
	if (enable) {
 | 
						|
		/* To be able to reprogram the number of samples, we have to
 | 
						|
		 * first stop the CC and then enable it again */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8500_RTC_CC_CONF_REG, 0x00);
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		/* Program the samples */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_NCOV_ACCU,
 | 
						|
			di->fg_samples);
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		/* Start the CC */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8500_RTC_CC_CONF_REG,
 | 
						|
			(CC_DEEP_SLEEP_ENA | CC_PWR_UP_ENA));
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		di->flags.fg_enabled = true;
 | 
						|
	} else {
 | 
						|
		/* Clear any pending read requests */
 | 
						|
		ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG,
 | 
						|
			(RESET_ACCU | READ_REQ), 0);
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		ret = abx500_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_NCOV_ACCU_CTRL, 0);
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		/* Stop the CC */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8500_RTC_CC_CONF_REG, 0);
 | 
						|
		if (ret)
 | 
						|
			goto cc_err;
 | 
						|
 | 
						|
		di->flags.fg_enabled = false;
 | 
						|
 | 
						|
	}
 | 
						|
	dev_dbg(di->dev, " CC enabled: %d Samples: %d\n",
 | 
						|
		enable, di->fg_samples);
 | 
						|
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
 | 
						|
	return ret;
 | 
						|
cc_err:
 | 
						|
	dev_err(di->dev, "%s Enabling coulomb counter failed\n", __func__);
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_inst_curr_start() - start battery instantaneous current
 | 
						|
 * @di:         pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns 0 or error code
 | 
						|
 * Note: This is part "one" and has to be called before
 | 
						|
 * ab8500_fg_inst_curr_finalize()
 | 
						|
 */
 | 
						|
int ab8500_fg_inst_curr_start(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	u8 reg_val;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	mutex_lock(&di->cc_lock);
 | 
						|
 | 
						|
	di->nbr_cceoc_irq_cnt = 0;
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8500_RTC_CC_CONF_REG, ®_val);
 | 
						|
	if (ret < 0)
 | 
						|
		goto fail;
 | 
						|
 | 
						|
	if (!(reg_val & CC_PWR_UP_ENA)) {
 | 
						|
		dev_dbg(di->dev, "%s Enable FG\n", __func__);
 | 
						|
		di->turn_off_fg = true;
 | 
						|
 | 
						|
		/* Program the samples */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_NCOV_ACCU,
 | 
						|
			SEC_TO_SAMPLE(10));
 | 
						|
		if (ret)
 | 
						|
			goto fail;
 | 
						|
 | 
						|
		/* Start the CC */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8500_RTC_CC_CONF_REG,
 | 
						|
			(CC_DEEP_SLEEP_ENA | CC_PWR_UP_ENA));
 | 
						|
		if (ret)
 | 
						|
			goto fail;
 | 
						|
	} else {
 | 
						|
		di->turn_off_fg = false;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Return and WFI */
 | 
						|
	reinit_completion(&di->ab8500_fg_started);
 | 
						|
	reinit_completion(&di->ab8500_fg_complete);
 | 
						|
	enable_irq(di->irq);
 | 
						|
 | 
						|
	/* Note: cc_lock is still locked */
 | 
						|
	return 0;
 | 
						|
fail:
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_inst_curr_started() - check if fg conversion has started
 | 
						|
 * @di:         pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns 1 if conversion started, 0 if still waiting
 | 
						|
 */
 | 
						|
int ab8500_fg_inst_curr_started(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	return completion_done(&di->ab8500_fg_started);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_inst_curr_done() - check if fg conversion is done
 | 
						|
 * @di:         pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns 1 if conversion done, 0 if still waiting
 | 
						|
 */
 | 
						|
int ab8500_fg_inst_curr_done(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	return completion_done(&di->ab8500_fg_complete);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_inst_curr_finalize() - battery instantaneous current
 | 
						|
 * @di:         pointer to the ab8500_fg structure
 | 
						|
 * @res:	battery instantenous current(on success)
 | 
						|
 *
 | 
						|
 * Returns 0 or an error code
 | 
						|
 * Note: This is part "two" and has to be called at earliest 250 ms
 | 
						|
 * after ab8500_fg_inst_curr_start()
 | 
						|
 */
 | 
						|
int ab8500_fg_inst_curr_finalize(struct ab8500_fg *di, int *res)
 | 
						|
{
 | 
						|
	u8 low, high;
 | 
						|
	int val;
 | 
						|
	int ret;
 | 
						|
	int timeout;
 | 
						|
 | 
						|
	if (!completion_done(&di->ab8500_fg_complete)) {
 | 
						|
		timeout = wait_for_completion_timeout(
 | 
						|
			&di->ab8500_fg_complete,
 | 
						|
			INS_CURR_TIMEOUT);
 | 
						|
		dev_dbg(di->dev, "Finalize time: %d ms\n",
 | 
						|
			((INS_CURR_TIMEOUT - timeout) * 1000) / HZ);
 | 
						|
		if (!timeout) {
 | 
						|
			ret = -ETIME;
 | 
						|
			disable_irq(di->irq);
 | 
						|
			di->nbr_cceoc_irq_cnt = 0;
 | 
						|
			dev_err(di->dev, "completion timed out [%d]\n",
 | 
						|
				__LINE__);
 | 
						|
			goto fail;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	disable_irq(di->irq);
 | 
						|
	di->nbr_cceoc_irq_cnt = 0;
 | 
						|
 | 
						|
	ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG,
 | 
						|
			READ_REQ, READ_REQ);
 | 
						|
 | 
						|
	/* 100uS between read request and read is needed */
 | 
						|
	usleep_range(100, 100);
 | 
						|
 | 
						|
	/* Read CC Sample conversion value Low and high */
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_SMPL_CNVL_REG,  &low);
 | 
						|
	if (ret < 0)
 | 
						|
		goto fail;
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_SMPL_CNVH_REG,  &high);
 | 
						|
	if (ret < 0)
 | 
						|
		goto fail;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * negative value for Discharging
 | 
						|
	 * convert 2's compliment into decimal
 | 
						|
	 */
 | 
						|
	if (high & 0x10)
 | 
						|
		val = (low | (high << 8) | 0xFFFFE000);
 | 
						|
	else
 | 
						|
		val = (low | (high << 8));
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Convert to unit value in mA
 | 
						|
	 * Full scale input voltage is
 | 
						|
	 * 63.160mV => LSB = 63.160mV/(4096*res) = 1.542mA
 | 
						|
	 * Given a 250ms conversion cycle time the LSB corresponds
 | 
						|
	 * to 107.1 nAh. Convert to current by dividing by the conversion
 | 
						|
	 * time in hours (250ms = 1 / (3600 * 4)h)
 | 
						|
	 * 107.1nAh assumes 10mOhm, but fg_res is in 0.1mOhm
 | 
						|
	 */
 | 
						|
	val = (val * QLSB_NANO_AMP_HOURS_X10 * 36 * 4) /
 | 
						|
		(1000 * di->bm->fg_res);
 | 
						|
 | 
						|
	if (di->turn_off_fg) {
 | 
						|
		dev_dbg(di->dev, "%s Disable FG\n", __func__);
 | 
						|
 | 
						|
		/* Clear any pending read requests */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG, 0);
 | 
						|
		if (ret)
 | 
						|
			goto fail;
 | 
						|
 | 
						|
		/* Stop the CC */
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8500_RTC_CC_CONF_REG, 0);
 | 
						|
		if (ret)
 | 
						|
			goto fail;
 | 
						|
	}
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	(*res) = val;
 | 
						|
 | 
						|
	return 0;
 | 
						|
fail:
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_inst_curr_blocking() - battery instantaneous current
 | 
						|
 * @di:         pointer to the ab8500_fg structure
 | 
						|
 * @res:	battery instantenous current(on success)
 | 
						|
 *
 | 
						|
 * Returns 0 else error code
 | 
						|
 */
 | 
						|
int ab8500_fg_inst_curr_blocking(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	int timeout;
 | 
						|
	int res = 0;
 | 
						|
 | 
						|
	ret = ab8500_fg_inst_curr_start(di);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "Failed to initialize fg_inst\n");
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Wait for CC to actually start */
 | 
						|
	if (!completion_done(&di->ab8500_fg_started)) {
 | 
						|
		timeout = wait_for_completion_timeout(
 | 
						|
			&di->ab8500_fg_started,
 | 
						|
			INS_CURR_TIMEOUT);
 | 
						|
		dev_dbg(di->dev, "Start time: %d ms\n",
 | 
						|
			((INS_CURR_TIMEOUT - timeout) * 1000) / HZ);
 | 
						|
		if (!timeout) {
 | 
						|
			ret = -ETIME;
 | 
						|
			dev_err(di->dev, "completion timed out [%d]\n",
 | 
						|
				__LINE__);
 | 
						|
			goto fail;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	ret = ab8500_fg_inst_curr_finalize(di, &res);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "Failed to finalize fg_inst\n");
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	dev_dbg(di->dev, "%s instant current: %d", __func__, res);
 | 
						|
	return res;
 | 
						|
fail:
 | 
						|
	disable_irq(di->irq);
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_acc_cur_work() - average battery current
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Updated the average battery current obtained from the
 | 
						|
 * coulomb counter.
 | 
						|
 */
 | 
						|
static void ab8500_fg_acc_cur_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	int val;
 | 
						|
	int ret;
 | 
						|
	u8 low, med, high;
 | 
						|
 | 
						|
	struct ab8500_fg *di = container_of(work,
 | 
						|
		struct ab8500_fg, fg_acc_cur_work);
 | 
						|
 | 
						|
	mutex_lock(&di->cc_lock);
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_NCOV_ACCU_CTRL, RD_NCONV_ACCU_REQ);
 | 
						|
	if (ret)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_NCOV_ACCU_LOW,  &low);
 | 
						|
	if (ret < 0)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_NCOV_ACCU_MED,  &med);
 | 
						|
	if (ret < 0)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_GAS_GAUGE,
 | 
						|
		AB8500_GASG_CC_NCOV_ACCU_HIGH, &high);
 | 
						|
	if (ret < 0)
 | 
						|
		goto exit;
 | 
						|
 | 
						|
	/* Check for sign bit in case of negative value, 2's compliment */
 | 
						|
	if (high & 0x10)
 | 
						|
		val = (low | (med << 8) | (high << 16) | 0xFFE00000);
 | 
						|
	else
 | 
						|
		val = (low | (med << 8) | (high << 16));
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Convert to uAh
 | 
						|
	 * Given a 250ms conversion cycle time the LSB corresponds
 | 
						|
	 * to 112.9 nAh.
 | 
						|
	 * 112.9nAh assumes 10mOhm, but fg_res is in 0.1mOhm
 | 
						|
	 */
 | 
						|
	di->accu_charge = (val * QLSB_NANO_AMP_HOURS_X10) /
 | 
						|
		(100 * di->bm->fg_res);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Convert to unit value in mA
 | 
						|
	 * by dividing by the conversion
 | 
						|
	 * time in hours (= samples / (3600 * 4)h)
 | 
						|
	 * and multiply with 1000
 | 
						|
	 */
 | 
						|
	di->avg_curr = (val * QLSB_NANO_AMP_HOURS_X10 * 36) /
 | 
						|
		(1000 * di->bm->fg_res * (di->fg_samples / 4));
 | 
						|
 | 
						|
	di->flags.conv_done = true;
 | 
						|
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
 | 
						|
	queue_work(di->fg_wq, &di->fg_work);
 | 
						|
 | 
						|
	dev_dbg(di->dev, "fg_res: %d, fg_samples: %d, gasg: %d, accu_charge: %d \n",
 | 
						|
				di->bm->fg_res, di->fg_samples, val, di->accu_charge);
 | 
						|
	return;
 | 
						|
exit:
 | 
						|
	dev_err(di->dev,
 | 
						|
		"Failed to read or write gas gauge registers\n");
 | 
						|
	mutex_unlock(&di->cc_lock);
 | 
						|
	queue_work(di->fg_wq, &di->fg_work);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_bat_voltage() - get battery voltage
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns battery voltage(on success) else error code
 | 
						|
 */
 | 
						|
static int ab8500_fg_bat_voltage(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int vbat;
 | 
						|
	static int prev;
 | 
						|
 | 
						|
	vbat = ab8500_gpadc_convert(di->gpadc, MAIN_BAT_V);
 | 
						|
	if (vbat < 0) {
 | 
						|
		dev_err(di->dev,
 | 
						|
			"%s gpadc conversion failed, using previous value\n",
 | 
						|
			__func__);
 | 
						|
		return prev;
 | 
						|
	}
 | 
						|
 | 
						|
	prev = vbat;
 | 
						|
	return vbat;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_volt_to_capacity() - Voltage based capacity
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @voltage:	The voltage to convert to a capacity
 | 
						|
 *
 | 
						|
 * Returns battery capacity in per mille based on voltage
 | 
						|
 */
 | 
						|
static int ab8500_fg_volt_to_capacity(struct ab8500_fg *di, int voltage)
 | 
						|
{
 | 
						|
	int i, tbl_size;
 | 
						|
	const struct abx500_v_to_cap *tbl;
 | 
						|
	int cap = 0;
 | 
						|
 | 
						|
	tbl = di->bm->bat_type[di->bm->batt_id].v_to_cap_tbl,
 | 
						|
	tbl_size = di->bm->bat_type[di->bm->batt_id].n_v_cap_tbl_elements;
 | 
						|
 | 
						|
	for (i = 0; i < tbl_size; ++i) {
 | 
						|
		if (voltage > tbl[i].voltage)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	if ((i > 0) && (i < tbl_size)) {
 | 
						|
		cap = interpolate(voltage,
 | 
						|
			tbl[i].voltage,
 | 
						|
			tbl[i].capacity * 10,
 | 
						|
			tbl[i-1].voltage,
 | 
						|
			tbl[i-1].capacity * 10);
 | 
						|
	} else if (i == 0) {
 | 
						|
		cap = 1000;
 | 
						|
	} else {
 | 
						|
		cap = 0;
 | 
						|
	}
 | 
						|
 | 
						|
	dev_dbg(di->dev, "%s Vbat: %d, Cap: %d per mille",
 | 
						|
		__func__, voltage, cap);
 | 
						|
 | 
						|
	return cap;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_uncomp_volt_to_capacity() - Uncompensated voltage based capacity
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns battery capacity based on battery voltage that is not compensated
 | 
						|
 * for the voltage drop due to the load
 | 
						|
 */
 | 
						|
static int ab8500_fg_uncomp_volt_to_capacity(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	di->vbat = ab8500_fg_bat_voltage(di);
 | 
						|
	return ab8500_fg_volt_to_capacity(di, di->vbat);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_battery_resistance() - Returns the battery inner resistance
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns battery inner resistance added with the fuel gauge resistor value
 | 
						|
 * to get the total resistance in the whole link from gnd to bat+ node.
 | 
						|
 */
 | 
						|
static int ab8500_fg_battery_resistance(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int i, tbl_size;
 | 
						|
	const struct batres_vs_temp *tbl;
 | 
						|
	int resist = 0;
 | 
						|
 | 
						|
	tbl = di->bm->bat_type[di->bm->batt_id].batres_tbl;
 | 
						|
	tbl_size = di->bm->bat_type[di->bm->batt_id].n_batres_tbl_elements;
 | 
						|
 | 
						|
	for (i = 0; i < tbl_size; ++i) {
 | 
						|
		if (di->bat_temp / 10 > tbl[i].temp)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	if ((i > 0) && (i < tbl_size)) {
 | 
						|
		resist = interpolate(di->bat_temp / 10,
 | 
						|
			tbl[i].temp,
 | 
						|
			tbl[i].resist,
 | 
						|
			tbl[i-1].temp,
 | 
						|
			tbl[i-1].resist);
 | 
						|
	} else if (i == 0) {
 | 
						|
		resist = tbl[0].resist;
 | 
						|
	} else {
 | 
						|
		resist = tbl[tbl_size - 1].resist;
 | 
						|
	}
 | 
						|
 | 
						|
	dev_dbg(di->dev, "%s Temp: %d battery internal resistance: %d"
 | 
						|
	    " fg resistance %d, total: %d (mOhm)\n",
 | 
						|
		__func__, di->bat_temp, resist, di->bm->fg_res / 10,
 | 
						|
		(di->bm->fg_res / 10) + resist);
 | 
						|
 | 
						|
	/* fg_res variable is in 0.1mOhm */
 | 
						|
	resist += di->bm->fg_res / 10;
 | 
						|
 | 
						|
	return resist;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_load_comp_volt_to_capacity() - Load compensated voltage based capacity
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns battery capacity based on battery voltage that is load compensated
 | 
						|
 * for the voltage drop
 | 
						|
 */
 | 
						|
static int ab8500_fg_load_comp_volt_to_capacity(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int vbat_comp, res;
 | 
						|
	int i = 0;
 | 
						|
	int vbat = 0;
 | 
						|
 | 
						|
	ab8500_fg_inst_curr_start(di);
 | 
						|
 | 
						|
	do {
 | 
						|
		vbat += ab8500_fg_bat_voltage(di);
 | 
						|
		i++;
 | 
						|
		usleep_range(5000, 6000);
 | 
						|
	} while (!ab8500_fg_inst_curr_done(di));
 | 
						|
 | 
						|
	ab8500_fg_inst_curr_finalize(di, &di->inst_curr);
 | 
						|
 | 
						|
	di->vbat = vbat / i;
 | 
						|
	res = ab8500_fg_battery_resistance(di);
 | 
						|
 | 
						|
	/* Use Ohms law to get the load compensated voltage */
 | 
						|
	vbat_comp = di->vbat - (di->inst_curr * res) / 1000;
 | 
						|
 | 
						|
	dev_dbg(di->dev, "%s Measured Vbat: %dmV,Compensated Vbat %dmV, "
 | 
						|
		"R: %dmOhm, Current: %dmA Vbat Samples: %d\n",
 | 
						|
		__func__, di->vbat, vbat_comp, res, di->inst_curr, i);
 | 
						|
 | 
						|
	return ab8500_fg_volt_to_capacity(di, vbat_comp);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_convert_mah_to_permille() - Capacity in mAh to permille
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @cap_mah:	capacity in mAh
 | 
						|
 *
 | 
						|
 * Converts capacity in mAh to capacity in permille
 | 
						|
 */
 | 
						|
static int ab8500_fg_convert_mah_to_permille(struct ab8500_fg *di, int cap_mah)
 | 
						|
{
 | 
						|
	return (cap_mah * 1000) / di->bat_cap.max_mah_design;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_convert_permille_to_mah() - Capacity in permille to mAh
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @cap_pm:	capacity in permille
 | 
						|
 *
 | 
						|
 * Converts capacity in permille to capacity in mAh
 | 
						|
 */
 | 
						|
static int ab8500_fg_convert_permille_to_mah(struct ab8500_fg *di, int cap_pm)
 | 
						|
{
 | 
						|
	return cap_pm * di->bat_cap.max_mah_design / 1000;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_convert_mah_to_uwh() - Capacity in mAh to uWh
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @cap_mah:	capacity in mAh
 | 
						|
 *
 | 
						|
 * Converts capacity in mAh to capacity in uWh
 | 
						|
 */
 | 
						|
static int ab8500_fg_convert_mah_to_uwh(struct ab8500_fg *di, int cap_mah)
 | 
						|
{
 | 
						|
	u64 div_res;
 | 
						|
	u32 div_rem;
 | 
						|
 | 
						|
	div_res = ((u64) cap_mah) * ((u64) di->vbat_nom);
 | 
						|
	div_rem = do_div(div_res, 1000);
 | 
						|
 | 
						|
	/* Make sure to round upwards if necessary */
 | 
						|
	if (div_rem >= 1000 / 2)
 | 
						|
		div_res++;
 | 
						|
 | 
						|
	return (int) div_res;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_calc_cap_charging() - Calculate remaining capacity while charging
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Return the capacity in mAh based on previous calculated capcity and the FG
 | 
						|
 * accumulator register value. The filter is filled with this capacity
 | 
						|
 */
 | 
						|
static int ab8500_fg_calc_cap_charging(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	dev_dbg(di->dev, "%s cap_mah %d accu_charge %d\n",
 | 
						|
		__func__,
 | 
						|
		di->bat_cap.mah,
 | 
						|
		di->accu_charge);
 | 
						|
 | 
						|
	/* Capacity should not be less than 0 */
 | 
						|
	if (di->bat_cap.mah + di->accu_charge > 0)
 | 
						|
		di->bat_cap.mah += di->accu_charge;
 | 
						|
	else
 | 
						|
		di->bat_cap.mah = 0;
 | 
						|
	/*
 | 
						|
	 * We force capacity to 100% once when the algorithm
 | 
						|
	 * reports that it's full.
 | 
						|
	 */
 | 
						|
	if (di->bat_cap.mah >= di->bat_cap.max_mah_design ||
 | 
						|
		di->flags.force_full) {
 | 
						|
		di->bat_cap.mah = di->bat_cap.max_mah_design;
 | 
						|
	}
 | 
						|
 | 
						|
	ab8500_fg_fill_cap_sample(di, di->bat_cap.mah);
 | 
						|
	di->bat_cap.permille =
 | 
						|
		ab8500_fg_convert_mah_to_permille(di, di->bat_cap.mah);
 | 
						|
 | 
						|
	/* We need to update battery voltage and inst current when charging */
 | 
						|
	di->vbat = ab8500_fg_bat_voltage(di);
 | 
						|
	di->inst_curr = ab8500_fg_inst_curr_blocking(di);
 | 
						|
 | 
						|
	return di->bat_cap.mah;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_calc_cap_discharge_voltage() - Capacity in discharge with voltage
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @comp:	if voltage should be load compensated before capacity calc
 | 
						|
 *
 | 
						|
 * Return the capacity in mAh based on the battery voltage. The voltage can
 | 
						|
 * either be load compensated or not. This value is added to the filter and a
 | 
						|
 * new mean value is calculated and returned.
 | 
						|
 */
 | 
						|
static int ab8500_fg_calc_cap_discharge_voltage(struct ab8500_fg *di, bool comp)
 | 
						|
{
 | 
						|
	int permille, mah;
 | 
						|
 | 
						|
	if (comp)
 | 
						|
		permille = ab8500_fg_load_comp_volt_to_capacity(di);
 | 
						|
	else
 | 
						|
		permille = ab8500_fg_uncomp_volt_to_capacity(di);
 | 
						|
 | 
						|
	mah = ab8500_fg_convert_permille_to_mah(di, permille);
 | 
						|
 | 
						|
	di->bat_cap.mah = ab8500_fg_add_cap_sample(di, mah);
 | 
						|
	di->bat_cap.permille =
 | 
						|
		ab8500_fg_convert_mah_to_permille(di, di->bat_cap.mah);
 | 
						|
 | 
						|
	return di->bat_cap.mah;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_calc_cap_discharge_fg() - Capacity in discharge with FG
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Return the capacity in mAh based on previous calculated capcity and the FG
 | 
						|
 * accumulator register value. This value is added to the filter and a
 | 
						|
 * new mean value is calculated and returned.
 | 
						|
 */
 | 
						|
static int ab8500_fg_calc_cap_discharge_fg(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int permille_volt, permille;
 | 
						|
 | 
						|
	dev_dbg(di->dev, "%s cap_mah %d accu_charge %d\n",
 | 
						|
		__func__,
 | 
						|
		di->bat_cap.mah,
 | 
						|
		di->accu_charge);
 | 
						|
 | 
						|
	/* Capacity should not be less than 0 */
 | 
						|
	if (di->bat_cap.mah + di->accu_charge > 0)
 | 
						|
		di->bat_cap.mah += di->accu_charge;
 | 
						|
	else
 | 
						|
		di->bat_cap.mah = 0;
 | 
						|
 | 
						|
	if (di->bat_cap.mah >= di->bat_cap.max_mah_design)
 | 
						|
		di->bat_cap.mah = di->bat_cap.max_mah_design;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Check against voltage based capacity. It can not be lower
 | 
						|
	 * than what the uncompensated voltage says
 | 
						|
	 */
 | 
						|
	permille = ab8500_fg_convert_mah_to_permille(di, di->bat_cap.mah);
 | 
						|
	permille_volt = ab8500_fg_uncomp_volt_to_capacity(di);
 | 
						|
 | 
						|
	if (permille < permille_volt) {
 | 
						|
		di->bat_cap.permille = permille_volt;
 | 
						|
		di->bat_cap.mah = ab8500_fg_convert_permille_to_mah(di,
 | 
						|
			di->bat_cap.permille);
 | 
						|
 | 
						|
		dev_dbg(di->dev, "%s voltage based: perm %d perm_volt %d\n",
 | 
						|
			__func__,
 | 
						|
			permille,
 | 
						|
			permille_volt);
 | 
						|
 | 
						|
		ab8500_fg_fill_cap_sample(di, di->bat_cap.mah);
 | 
						|
	} else {
 | 
						|
		ab8500_fg_fill_cap_sample(di, di->bat_cap.mah);
 | 
						|
		di->bat_cap.permille =
 | 
						|
			ab8500_fg_convert_mah_to_permille(di, di->bat_cap.mah);
 | 
						|
	}
 | 
						|
 | 
						|
	return di->bat_cap.mah;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_capacity_level() - Get the battery capacity level
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Get the battery capacity level based on the capacity in percent
 | 
						|
 */
 | 
						|
static int ab8500_fg_capacity_level(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int ret, percent;
 | 
						|
 | 
						|
	percent = DIV_ROUND_CLOSEST(di->bat_cap.permille, 10);
 | 
						|
 | 
						|
	if (percent <= di->bm->cap_levels->critical ||
 | 
						|
		di->flags.low_bat)
 | 
						|
		ret = POWER_SUPPLY_CAPACITY_LEVEL_CRITICAL;
 | 
						|
	else if (percent <= di->bm->cap_levels->low)
 | 
						|
		ret = POWER_SUPPLY_CAPACITY_LEVEL_LOW;
 | 
						|
	else if (percent <= di->bm->cap_levels->normal)
 | 
						|
		ret = POWER_SUPPLY_CAPACITY_LEVEL_NORMAL;
 | 
						|
	else if (percent <= di->bm->cap_levels->high)
 | 
						|
		ret = POWER_SUPPLY_CAPACITY_LEVEL_HIGH;
 | 
						|
	else
 | 
						|
		ret = POWER_SUPPLY_CAPACITY_LEVEL_FULL;
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_calculate_scaled_capacity() - Capacity scaling
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Calculates the capacity to be shown to upper layers. Scales the capacity
 | 
						|
 * to have 100% as a reference from the actual capacity upon removal of charger
 | 
						|
 * when charging is in maintenance mode.
 | 
						|
 */
 | 
						|
static int ab8500_fg_calculate_scaled_capacity(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	struct ab8500_fg_cap_scaling *cs = &di->bat_cap.cap_scale;
 | 
						|
	int capacity = di->bat_cap.prev_percent;
 | 
						|
 | 
						|
	if (!cs->enable)
 | 
						|
		return capacity;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * As long as we are in fully charge mode scale the capacity
 | 
						|
	 * to show 100%.
 | 
						|
	 */
 | 
						|
	if (di->flags.fully_charged) {
 | 
						|
		cs->cap_to_scale[0] = 100;
 | 
						|
		cs->cap_to_scale[1] =
 | 
						|
			max(capacity, di->bm->fg_params->maint_thres);
 | 
						|
		dev_dbg(di->dev, "Scale cap with %d/%d\n",
 | 
						|
			 cs->cap_to_scale[0], cs->cap_to_scale[1]);
 | 
						|
	}
 | 
						|
 | 
						|
	/* Calculates the scaled capacity. */
 | 
						|
	if ((cs->cap_to_scale[0] != cs->cap_to_scale[1])
 | 
						|
					&& (cs->cap_to_scale[1] > 0))
 | 
						|
		capacity = min(100,
 | 
						|
				 DIV_ROUND_CLOSEST(di->bat_cap.prev_percent *
 | 
						|
						 cs->cap_to_scale[0],
 | 
						|
						 cs->cap_to_scale[1]));
 | 
						|
 | 
						|
	if (di->flags.charging) {
 | 
						|
		if (capacity < cs->disable_cap_level) {
 | 
						|
			cs->disable_cap_level = capacity;
 | 
						|
			dev_dbg(di->dev, "Cap to stop scale lowered %d%%\n",
 | 
						|
				cs->disable_cap_level);
 | 
						|
		} else if (!di->flags.fully_charged) {
 | 
						|
			if (di->bat_cap.prev_percent >=
 | 
						|
			    cs->disable_cap_level) {
 | 
						|
				dev_dbg(di->dev, "Disabling scaled capacity\n");
 | 
						|
				cs->enable = false;
 | 
						|
				capacity = di->bat_cap.prev_percent;
 | 
						|
			} else {
 | 
						|
				dev_dbg(di->dev,
 | 
						|
					"Waiting in cap to level %d%%\n",
 | 
						|
					cs->disable_cap_level);
 | 
						|
				capacity = cs->disable_cap_level;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return capacity;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_update_cap_scalers() - Capacity scaling
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * To be called when state change from charge<->discharge to update
 | 
						|
 * the capacity scalers.
 | 
						|
 */
 | 
						|
static void ab8500_fg_update_cap_scalers(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	struct ab8500_fg_cap_scaling *cs = &di->bat_cap.cap_scale;
 | 
						|
 | 
						|
	if (!cs->enable)
 | 
						|
		return;
 | 
						|
	if (di->flags.charging) {
 | 
						|
		di->bat_cap.cap_scale.disable_cap_level =
 | 
						|
			di->bat_cap.cap_scale.scaled_cap;
 | 
						|
		dev_dbg(di->dev, "Cap to stop scale at charge %d%%\n",
 | 
						|
				di->bat_cap.cap_scale.disable_cap_level);
 | 
						|
	} else {
 | 
						|
		if (cs->scaled_cap != 100) {
 | 
						|
			cs->cap_to_scale[0] = cs->scaled_cap;
 | 
						|
			cs->cap_to_scale[1] = di->bat_cap.prev_percent;
 | 
						|
		} else {
 | 
						|
			cs->cap_to_scale[0] = 100;
 | 
						|
			cs->cap_to_scale[1] =
 | 
						|
				max(di->bat_cap.prev_percent,
 | 
						|
				    di->bm->fg_params->maint_thres);
 | 
						|
		}
 | 
						|
 | 
						|
		dev_dbg(di->dev, "Cap to scale at discharge %d/%d\n",
 | 
						|
				cs->cap_to_scale[0], cs->cap_to_scale[1]);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_check_capacity_limits() - Check if capacity has changed
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 * @init:	capacity is allowed to go up in init mode
 | 
						|
 *
 | 
						|
 * Check if capacity or capacity limit has changed and notify the system
 | 
						|
 * about it using the power_supply framework
 | 
						|
 */
 | 
						|
static void ab8500_fg_check_capacity_limits(struct ab8500_fg *di, bool init)
 | 
						|
{
 | 
						|
	bool changed = false;
 | 
						|
	int percent = DIV_ROUND_CLOSEST(di->bat_cap.permille, 10);
 | 
						|
 | 
						|
	di->bat_cap.level = ab8500_fg_capacity_level(di);
 | 
						|
 | 
						|
	if (di->bat_cap.level != di->bat_cap.prev_level) {
 | 
						|
		/*
 | 
						|
		 * We do not allow reported capacity level to go up
 | 
						|
		 * unless we're charging or if we're in init
 | 
						|
		 */
 | 
						|
		if (!(!di->flags.charging && di->bat_cap.level >
 | 
						|
			di->bat_cap.prev_level) || init) {
 | 
						|
			dev_dbg(di->dev, "level changed from %d to %d\n",
 | 
						|
				di->bat_cap.prev_level,
 | 
						|
				di->bat_cap.level);
 | 
						|
			di->bat_cap.prev_level = di->bat_cap.level;
 | 
						|
			changed = true;
 | 
						|
		} else {
 | 
						|
			dev_dbg(di->dev, "level not allowed to go up "
 | 
						|
				"since no charger is connected: %d to %d\n",
 | 
						|
				di->bat_cap.prev_level,
 | 
						|
				di->bat_cap.level);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If we have received the LOW_BAT IRQ, set capacity to 0 to initiate
 | 
						|
	 * shutdown
 | 
						|
	 */
 | 
						|
	if (di->flags.low_bat) {
 | 
						|
		dev_dbg(di->dev, "Battery low, set capacity to 0\n");
 | 
						|
		di->bat_cap.prev_percent = 0;
 | 
						|
		di->bat_cap.permille = 0;
 | 
						|
		percent = 0;
 | 
						|
		di->bat_cap.prev_mah = 0;
 | 
						|
		di->bat_cap.mah = 0;
 | 
						|
		changed = true;
 | 
						|
	} else if (di->flags.fully_charged) {
 | 
						|
		/*
 | 
						|
		 * We report 100% if algorithm reported fully charged
 | 
						|
		 * and show 100% during maintenance charging (scaling).
 | 
						|
		 */
 | 
						|
		if (di->flags.force_full) {
 | 
						|
			di->bat_cap.prev_percent = percent;
 | 
						|
			di->bat_cap.prev_mah = di->bat_cap.mah;
 | 
						|
 | 
						|
			changed = true;
 | 
						|
 | 
						|
			if (!di->bat_cap.cap_scale.enable &&
 | 
						|
						di->bm->capacity_scaling) {
 | 
						|
				di->bat_cap.cap_scale.enable = true;
 | 
						|
				di->bat_cap.cap_scale.cap_to_scale[0] = 100;
 | 
						|
				di->bat_cap.cap_scale.cap_to_scale[1] =
 | 
						|
						di->bat_cap.prev_percent;
 | 
						|
				di->bat_cap.cap_scale.disable_cap_level = 100;
 | 
						|
			}
 | 
						|
		} else if (di->bat_cap.prev_percent != percent) {
 | 
						|
			dev_dbg(di->dev,
 | 
						|
				"battery reported full "
 | 
						|
				"but capacity dropping: %d\n",
 | 
						|
				percent);
 | 
						|
			di->bat_cap.prev_percent = percent;
 | 
						|
			di->bat_cap.prev_mah = di->bat_cap.mah;
 | 
						|
 | 
						|
			changed = true;
 | 
						|
		}
 | 
						|
	} else if (di->bat_cap.prev_percent != percent) {
 | 
						|
		if (percent == 0) {
 | 
						|
			/*
 | 
						|
			 * We will not report 0% unless we've got
 | 
						|
			 * the LOW_BAT IRQ, no matter what the FG
 | 
						|
			 * algorithm says.
 | 
						|
			 */
 | 
						|
			di->bat_cap.prev_percent = 1;
 | 
						|
			percent = 1;
 | 
						|
 | 
						|
			changed = true;
 | 
						|
		} else if (!(!di->flags.charging &&
 | 
						|
			percent > di->bat_cap.prev_percent) || init) {
 | 
						|
			/*
 | 
						|
			 * We do not allow reported capacity to go up
 | 
						|
			 * unless we're charging or if we're in init
 | 
						|
			 */
 | 
						|
			dev_dbg(di->dev,
 | 
						|
				"capacity changed from %d to %d (%d)\n",
 | 
						|
				di->bat_cap.prev_percent,
 | 
						|
				percent,
 | 
						|
				di->bat_cap.permille);
 | 
						|
			di->bat_cap.prev_percent = percent;
 | 
						|
			di->bat_cap.prev_mah = di->bat_cap.mah;
 | 
						|
 | 
						|
			changed = true;
 | 
						|
		} else {
 | 
						|
			dev_dbg(di->dev, "capacity not allowed to go up since "
 | 
						|
				"no charger is connected: %d to %d (%d)\n",
 | 
						|
				di->bat_cap.prev_percent,
 | 
						|
				percent,
 | 
						|
				di->bat_cap.permille);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (changed) {
 | 
						|
		if (di->bm->capacity_scaling) {
 | 
						|
			di->bat_cap.cap_scale.scaled_cap =
 | 
						|
				ab8500_fg_calculate_scaled_capacity(di);
 | 
						|
 | 
						|
			dev_info(di->dev, "capacity=%d (%d)\n",
 | 
						|
				di->bat_cap.prev_percent,
 | 
						|
				di->bat_cap.cap_scale.scaled_cap);
 | 
						|
		}
 | 
						|
		power_supply_changed(&di->fg_psy);
 | 
						|
		if (di->flags.fully_charged && di->flags.force_full) {
 | 
						|
			dev_dbg(di->dev, "Battery full, notifying.\n");
 | 
						|
			di->flags.force_full = false;
 | 
						|
			sysfs_notify(&di->fg_kobject, NULL, "charge_full");
 | 
						|
		}
 | 
						|
		sysfs_notify(&di->fg_kobject, NULL, "charge_now");
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void ab8500_fg_charge_state_to(struct ab8500_fg *di,
 | 
						|
	enum ab8500_fg_charge_state new_state)
 | 
						|
{
 | 
						|
	dev_dbg(di->dev, "Charge state from %d [%s] to %d [%s]\n",
 | 
						|
		di->charge_state,
 | 
						|
		charge_state[di->charge_state],
 | 
						|
		new_state,
 | 
						|
		charge_state[new_state]);
 | 
						|
 | 
						|
	di->charge_state = new_state;
 | 
						|
}
 | 
						|
 | 
						|
static void ab8500_fg_discharge_state_to(struct ab8500_fg *di,
 | 
						|
	enum ab8500_fg_discharge_state new_state)
 | 
						|
{
 | 
						|
	dev_dbg(di->dev, "Disharge state from %d [%s] to %d [%s]\n",
 | 
						|
		di->discharge_state,
 | 
						|
		discharge_state[di->discharge_state],
 | 
						|
		new_state,
 | 
						|
		discharge_state[new_state]);
 | 
						|
 | 
						|
	di->discharge_state = new_state;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_algorithm_charging() - FG algorithm for when charging
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Battery capacity calculation state machine for when we're charging
 | 
						|
 */
 | 
						|
static void ab8500_fg_algorithm_charging(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	/*
 | 
						|
	 * If we change to discharge mode
 | 
						|
	 * we should start with recovery
 | 
						|
	 */
 | 
						|
	if (di->discharge_state != AB8500_FG_DISCHARGE_INIT_RECOVERY)
 | 
						|
		ab8500_fg_discharge_state_to(di,
 | 
						|
			AB8500_FG_DISCHARGE_INIT_RECOVERY);
 | 
						|
 | 
						|
	switch (di->charge_state) {
 | 
						|
	case AB8500_FG_CHARGE_INIT:
 | 
						|
		di->fg_samples = SEC_TO_SAMPLE(
 | 
						|
			di->bm->fg_params->accu_charging);
 | 
						|
 | 
						|
		ab8500_fg_coulomb_counter(di, true);
 | 
						|
		ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_READOUT);
 | 
						|
 | 
						|
		break;
 | 
						|
 | 
						|
	case AB8500_FG_CHARGE_READOUT:
 | 
						|
		/*
 | 
						|
		 * Read the FG and calculate the new capacity
 | 
						|
		 */
 | 
						|
		mutex_lock(&di->cc_lock);
 | 
						|
		if (!di->flags.conv_done && !di->flags.force_full) {
 | 
						|
			/* Wasn't the CC IRQ that got us here */
 | 
						|
			mutex_unlock(&di->cc_lock);
 | 
						|
			dev_dbg(di->dev, "%s CC conv not done\n",
 | 
						|
				__func__);
 | 
						|
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		di->flags.conv_done = false;
 | 
						|
		mutex_unlock(&di->cc_lock);
 | 
						|
 | 
						|
		ab8500_fg_calc_cap_charging(di);
 | 
						|
 | 
						|
		break;
 | 
						|
 | 
						|
	default:
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Check capacity limits */
 | 
						|
	ab8500_fg_check_capacity_limits(di, false);
 | 
						|
}
 | 
						|
 | 
						|
static void force_capacity(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int cap;
 | 
						|
 | 
						|
	ab8500_fg_clear_cap_samples(di);
 | 
						|
	cap = di->bat_cap.user_mah;
 | 
						|
	if (cap > di->bat_cap.max_mah_design) {
 | 
						|
		dev_dbg(di->dev, "Remaining cap %d can't be bigger than total"
 | 
						|
			" %d\n", cap, di->bat_cap.max_mah_design);
 | 
						|
		cap = di->bat_cap.max_mah_design;
 | 
						|
	}
 | 
						|
	ab8500_fg_fill_cap_sample(di, di->bat_cap.user_mah);
 | 
						|
	di->bat_cap.permille = ab8500_fg_convert_mah_to_permille(di, cap);
 | 
						|
	di->bat_cap.mah = cap;
 | 
						|
	ab8500_fg_check_capacity_limits(di, true);
 | 
						|
}
 | 
						|
 | 
						|
static bool check_sysfs_capacity(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int cap, lower, upper;
 | 
						|
	int cap_permille;
 | 
						|
 | 
						|
	cap = di->bat_cap.user_mah;
 | 
						|
 | 
						|
	cap_permille = ab8500_fg_convert_mah_to_permille(di,
 | 
						|
		di->bat_cap.user_mah);
 | 
						|
 | 
						|
	lower = di->bat_cap.permille - di->bm->fg_params->user_cap_limit * 10;
 | 
						|
	upper = di->bat_cap.permille + di->bm->fg_params->user_cap_limit * 10;
 | 
						|
 | 
						|
	if (lower < 0)
 | 
						|
		lower = 0;
 | 
						|
	/* 1000 is permille, -> 100 percent */
 | 
						|
	if (upper > 1000)
 | 
						|
		upper = 1000;
 | 
						|
 | 
						|
	dev_dbg(di->dev, "Capacity limits:"
 | 
						|
		" (Lower: %d User: %d Upper: %d) [user: %d, was: %d]\n",
 | 
						|
		lower, cap_permille, upper, cap, di->bat_cap.mah);
 | 
						|
 | 
						|
	/* If within limits, use the saved capacity and exit estimation...*/
 | 
						|
	if (cap_permille > lower && cap_permille < upper) {
 | 
						|
		dev_dbg(di->dev, "OK! Using users cap %d uAh now\n", cap);
 | 
						|
		force_capacity(di);
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
	dev_dbg(di->dev, "Capacity from user out of limits, ignoring");
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_algorithm_discharging() - FG algorithm for when discharging
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Battery capacity calculation state machine for when we're discharging
 | 
						|
 */
 | 
						|
static void ab8500_fg_algorithm_discharging(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int sleep_time;
 | 
						|
 | 
						|
	/* If we change to charge mode we should start with init */
 | 
						|
	if (di->charge_state != AB8500_FG_CHARGE_INIT)
 | 
						|
		ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_INIT);
 | 
						|
 | 
						|
	switch (di->discharge_state) {
 | 
						|
	case AB8500_FG_DISCHARGE_INIT:
 | 
						|
		/* We use the FG IRQ to work on */
 | 
						|
		di->init_cnt = 0;
 | 
						|
		di->fg_samples = SEC_TO_SAMPLE(di->bm->fg_params->init_timer);
 | 
						|
		ab8500_fg_coulomb_counter(di, true);
 | 
						|
		ab8500_fg_discharge_state_to(di,
 | 
						|
			AB8500_FG_DISCHARGE_INITMEASURING);
 | 
						|
 | 
						|
		/* Intentional fallthrough */
 | 
						|
	case AB8500_FG_DISCHARGE_INITMEASURING:
 | 
						|
		/*
 | 
						|
		 * Discard a number of samples during startup.
 | 
						|
		 * After that, use compensated voltage for a few
 | 
						|
		 * samples to get an initial capacity.
 | 
						|
		 * Then go to READOUT
 | 
						|
		 */
 | 
						|
		sleep_time = di->bm->fg_params->init_timer;
 | 
						|
 | 
						|
		/* Discard the first [x] seconds */
 | 
						|
		if (di->init_cnt > di->bm->fg_params->init_discard_time) {
 | 
						|
			ab8500_fg_calc_cap_discharge_voltage(di, true);
 | 
						|
 | 
						|
			ab8500_fg_check_capacity_limits(di, true);
 | 
						|
		}
 | 
						|
 | 
						|
		di->init_cnt += sleep_time;
 | 
						|
		if (di->init_cnt > di->bm->fg_params->init_total_time)
 | 
						|
			ab8500_fg_discharge_state_to(di,
 | 
						|
				AB8500_FG_DISCHARGE_READOUT_INIT);
 | 
						|
 | 
						|
		break;
 | 
						|
 | 
						|
	case AB8500_FG_DISCHARGE_INIT_RECOVERY:
 | 
						|
		di->recovery_cnt = 0;
 | 
						|
		di->recovery_needed = true;
 | 
						|
		ab8500_fg_discharge_state_to(di,
 | 
						|
			AB8500_FG_DISCHARGE_RECOVERY);
 | 
						|
 | 
						|
		/* Intentional fallthrough */
 | 
						|
 | 
						|
	case AB8500_FG_DISCHARGE_RECOVERY:
 | 
						|
		sleep_time = di->bm->fg_params->recovery_sleep_timer;
 | 
						|
 | 
						|
		/*
 | 
						|
		 * We should check the power consumption
 | 
						|
		 * If low, go to READOUT (after x min) or
 | 
						|
		 * RECOVERY_SLEEP if time left.
 | 
						|
		 * If high, go to READOUT
 | 
						|
		 */
 | 
						|
		di->inst_curr = ab8500_fg_inst_curr_blocking(di);
 | 
						|
 | 
						|
		if (ab8500_fg_is_low_curr(di, di->inst_curr)) {
 | 
						|
			if (di->recovery_cnt >
 | 
						|
				di->bm->fg_params->recovery_total_time) {
 | 
						|
				di->fg_samples = SEC_TO_SAMPLE(
 | 
						|
					di->bm->fg_params->accu_high_curr);
 | 
						|
				ab8500_fg_coulomb_counter(di, true);
 | 
						|
				ab8500_fg_discharge_state_to(di,
 | 
						|
					AB8500_FG_DISCHARGE_READOUT);
 | 
						|
				di->recovery_needed = false;
 | 
						|
			} else {
 | 
						|
				queue_delayed_work(di->fg_wq,
 | 
						|
					&di->fg_periodic_work,
 | 
						|
					sleep_time * HZ);
 | 
						|
			}
 | 
						|
			di->recovery_cnt += sleep_time;
 | 
						|
		} else {
 | 
						|
			di->fg_samples = SEC_TO_SAMPLE(
 | 
						|
				di->bm->fg_params->accu_high_curr);
 | 
						|
			ab8500_fg_coulomb_counter(di, true);
 | 
						|
			ab8500_fg_discharge_state_to(di,
 | 
						|
				AB8500_FG_DISCHARGE_READOUT);
 | 
						|
		}
 | 
						|
		break;
 | 
						|
 | 
						|
	case AB8500_FG_DISCHARGE_READOUT_INIT:
 | 
						|
		di->fg_samples = SEC_TO_SAMPLE(
 | 
						|
			di->bm->fg_params->accu_high_curr);
 | 
						|
		ab8500_fg_coulomb_counter(di, true);
 | 
						|
		ab8500_fg_discharge_state_to(di,
 | 
						|
				AB8500_FG_DISCHARGE_READOUT);
 | 
						|
		break;
 | 
						|
 | 
						|
	case AB8500_FG_DISCHARGE_READOUT:
 | 
						|
		di->inst_curr = ab8500_fg_inst_curr_blocking(di);
 | 
						|
 | 
						|
		if (ab8500_fg_is_low_curr(di, di->inst_curr)) {
 | 
						|
			/* Detect mode change */
 | 
						|
			if (di->high_curr_mode) {
 | 
						|
				di->high_curr_mode = false;
 | 
						|
				di->high_curr_cnt = 0;
 | 
						|
			}
 | 
						|
 | 
						|
			if (di->recovery_needed) {
 | 
						|
				ab8500_fg_discharge_state_to(di,
 | 
						|
					AB8500_FG_DISCHARGE_INIT_RECOVERY);
 | 
						|
 | 
						|
				queue_delayed_work(di->fg_wq,
 | 
						|
					&di->fg_periodic_work, 0);
 | 
						|
 | 
						|
				break;
 | 
						|
			}
 | 
						|
 | 
						|
			ab8500_fg_calc_cap_discharge_voltage(di, true);
 | 
						|
		} else {
 | 
						|
			mutex_lock(&di->cc_lock);
 | 
						|
			if (!di->flags.conv_done) {
 | 
						|
				/* Wasn't the CC IRQ that got us here */
 | 
						|
				mutex_unlock(&di->cc_lock);
 | 
						|
				dev_dbg(di->dev, "%s CC conv not done\n",
 | 
						|
					__func__);
 | 
						|
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			di->flags.conv_done = false;
 | 
						|
			mutex_unlock(&di->cc_lock);
 | 
						|
 | 
						|
			/* Detect mode change */
 | 
						|
			if (!di->high_curr_mode) {
 | 
						|
				di->high_curr_mode = true;
 | 
						|
				di->high_curr_cnt = 0;
 | 
						|
			}
 | 
						|
 | 
						|
			di->high_curr_cnt +=
 | 
						|
				di->bm->fg_params->accu_high_curr;
 | 
						|
			if (di->high_curr_cnt >
 | 
						|
				di->bm->fg_params->high_curr_time)
 | 
						|
				di->recovery_needed = true;
 | 
						|
 | 
						|
			ab8500_fg_calc_cap_discharge_fg(di);
 | 
						|
		}
 | 
						|
 | 
						|
		ab8500_fg_check_capacity_limits(di, false);
 | 
						|
 | 
						|
		break;
 | 
						|
 | 
						|
	case AB8500_FG_DISCHARGE_WAKEUP:
 | 
						|
		ab8500_fg_calc_cap_discharge_voltage(di, true);
 | 
						|
 | 
						|
		di->fg_samples = SEC_TO_SAMPLE(
 | 
						|
			di->bm->fg_params->accu_high_curr);
 | 
						|
		ab8500_fg_coulomb_counter(di, true);
 | 
						|
		ab8500_fg_discharge_state_to(di,
 | 
						|
				AB8500_FG_DISCHARGE_READOUT);
 | 
						|
 | 
						|
		ab8500_fg_check_capacity_limits(di, false);
 | 
						|
 | 
						|
		break;
 | 
						|
 | 
						|
	default:
 | 
						|
		break;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_algorithm_calibrate() - Internal columb counter offset calibration
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 */
 | 
						|
static void ab8500_fg_algorithm_calibrate(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	switch (di->calib_state) {
 | 
						|
	case AB8500_FG_CALIB_INIT:
 | 
						|
		dev_dbg(di->dev, "Calibration ongoing...\n");
 | 
						|
 | 
						|
		ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG,
 | 
						|
			CC_INT_CAL_N_AVG_MASK, CC_INT_CAL_SAMPLES_8);
 | 
						|
		if (ret < 0)
 | 
						|
			goto err;
 | 
						|
 | 
						|
		ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG,
 | 
						|
			CC_INTAVGOFFSET_ENA, CC_INTAVGOFFSET_ENA);
 | 
						|
		if (ret < 0)
 | 
						|
			goto err;
 | 
						|
		di->calib_state = AB8500_FG_CALIB_WAIT;
 | 
						|
		break;
 | 
						|
	case AB8500_FG_CALIB_END:
 | 
						|
		ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
			AB8500_GAS_GAUGE, AB8500_GASG_CC_CTRL_REG,
 | 
						|
			CC_MUXOFFSET, CC_MUXOFFSET);
 | 
						|
		if (ret < 0)
 | 
						|
			goto err;
 | 
						|
		di->flags.calibrate = false;
 | 
						|
		dev_dbg(di->dev, "Calibration done...\n");
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
		break;
 | 
						|
	case AB8500_FG_CALIB_WAIT:
 | 
						|
		dev_dbg(di->dev, "Calibration WFI\n");
 | 
						|
	default:
 | 
						|
		break;
 | 
						|
	}
 | 
						|
	return;
 | 
						|
err:
 | 
						|
	/* Something went wrong, don't calibrate then */
 | 
						|
	dev_err(di->dev, "failed to calibrate the CC\n");
 | 
						|
	di->flags.calibrate = false;
 | 
						|
	di->calib_state = AB8500_FG_CALIB_INIT;
 | 
						|
	queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_algorithm() - Entry point for the FG algorithm
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Entry point for the battery capacity calculation state machine
 | 
						|
 */
 | 
						|
static void ab8500_fg_algorithm(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	if (di->flags.calibrate)
 | 
						|
		ab8500_fg_algorithm_calibrate(di);
 | 
						|
	else {
 | 
						|
		if (di->flags.charging)
 | 
						|
			ab8500_fg_algorithm_charging(di);
 | 
						|
		else
 | 
						|
			ab8500_fg_algorithm_discharging(di);
 | 
						|
	}
 | 
						|
 | 
						|
	dev_dbg(di->dev, "[FG_DATA] %d %d %d %d %d %d %d %d %d %d "
 | 
						|
		"%d %d %d %d %d %d %d\n",
 | 
						|
		di->bat_cap.max_mah_design,
 | 
						|
		di->bat_cap.max_mah,
 | 
						|
		di->bat_cap.mah,
 | 
						|
		di->bat_cap.permille,
 | 
						|
		di->bat_cap.level,
 | 
						|
		di->bat_cap.prev_mah,
 | 
						|
		di->bat_cap.prev_percent,
 | 
						|
		di->bat_cap.prev_level,
 | 
						|
		di->vbat,
 | 
						|
		di->inst_curr,
 | 
						|
		di->avg_curr,
 | 
						|
		di->accu_charge,
 | 
						|
		di->flags.charging,
 | 
						|
		di->charge_state,
 | 
						|
		di->discharge_state,
 | 
						|
		di->high_curr_mode,
 | 
						|
		di->recovery_needed);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_periodic_work() - Run the FG state machine periodically
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Work queue function for periodic work
 | 
						|
 */
 | 
						|
static void ab8500_fg_periodic_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = container_of(work, struct ab8500_fg,
 | 
						|
		fg_periodic_work.work);
 | 
						|
 | 
						|
	if (di->init_capacity) {
 | 
						|
		/* Get an initial capacity calculation */
 | 
						|
		ab8500_fg_calc_cap_discharge_voltage(di, true);
 | 
						|
		ab8500_fg_check_capacity_limits(di, true);
 | 
						|
		di->init_capacity = false;
 | 
						|
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
	} else if (di->flags.user_cap) {
 | 
						|
		if (check_sysfs_capacity(di)) {
 | 
						|
			ab8500_fg_check_capacity_limits(di, true);
 | 
						|
			if (di->flags.charging)
 | 
						|
				ab8500_fg_charge_state_to(di,
 | 
						|
					AB8500_FG_CHARGE_INIT);
 | 
						|
			else
 | 
						|
				ab8500_fg_discharge_state_to(di,
 | 
						|
					AB8500_FG_DISCHARGE_READOUT_INIT);
 | 
						|
		}
 | 
						|
		di->flags.user_cap = false;
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
	} else
 | 
						|
		ab8500_fg_algorithm(di);
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_check_hw_failure_work() - Check OVV_BAT condition
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Work queue function for checking the OVV_BAT condition
 | 
						|
 */
 | 
						|
static void ab8500_fg_check_hw_failure_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
 | 
						|
	struct ab8500_fg *di = container_of(work, struct ab8500_fg,
 | 
						|
		fg_check_hw_failure_work.work);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If we have had a battery over-voltage situation,
 | 
						|
	 * check ovv-bit to see if it should be reset.
 | 
						|
	 */
 | 
						|
	ret = abx500_get_register_interruptible(di->dev,
 | 
						|
		AB8500_CHARGER, AB8500_CH_STAT_REG,
 | 
						|
		®_value);
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(di->dev, "%s ab8500 read failed\n", __func__);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	if ((reg_value & BATT_OVV) == BATT_OVV) {
 | 
						|
		if (!di->flags.bat_ovv) {
 | 
						|
			dev_dbg(di->dev, "Battery OVV\n");
 | 
						|
			di->flags.bat_ovv = true;
 | 
						|
			power_supply_changed(&di->fg_psy);
 | 
						|
		}
 | 
						|
		/* Not yet recovered from ovv, reschedule this test */
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_check_hw_failure_work,
 | 
						|
				   HZ);
 | 
						|
		} else {
 | 
						|
			dev_dbg(di->dev, "Battery recovered from OVV\n");
 | 
						|
			di->flags.bat_ovv = false;
 | 
						|
			power_supply_changed(&di->fg_psy);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_low_bat_work() - Check LOW_BAT condition
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Work queue function for checking the LOW_BAT condition
 | 
						|
 */
 | 
						|
static void ab8500_fg_low_bat_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	int vbat;
 | 
						|
 | 
						|
	struct ab8500_fg *di = container_of(work, struct ab8500_fg,
 | 
						|
		fg_low_bat_work.work);
 | 
						|
 | 
						|
	vbat = ab8500_fg_bat_voltage(di);
 | 
						|
 | 
						|
	/* Check if LOW_BAT still fulfilled */
 | 
						|
	if (vbat < di->bm->fg_params->lowbat_threshold) {
 | 
						|
		/* Is it time to shut down? */
 | 
						|
		if (di->low_bat_cnt < 1) {
 | 
						|
			di->flags.low_bat = true;
 | 
						|
			dev_warn(di->dev, "Shut down pending...\n");
 | 
						|
		} else {
 | 
						|
			/*
 | 
						|
			* Else we need to re-schedule this check to be able to detect
 | 
						|
			* if the voltage increases again during charging or
 | 
						|
			* due to decreasing load.
 | 
						|
			*/
 | 
						|
			di->low_bat_cnt--;
 | 
						|
			dev_warn(di->dev, "Battery voltage still LOW\n");
 | 
						|
			queue_delayed_work(di->fg_wq, &di->fg_low_bat_work,
 | 
						|
				round_jiffies(LOW_BAT_CHECK_INTERVAL));
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		di->flags.low_bat_delay = false;
 | 
						|
		di->low_bat_cnt = 10;
 | 
						|
		dev_warn(di->dev, "Battery voltage OK again\n");
 | 
						|
	}
 | 
						|
 | 
						|
	/* This is needed to dispatch LOW_BAT */
 | 
						|
	ab8500_fg_check_capacity_limits(di, false);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_battok_calc - calculate the bit pattern corresponding
 | 
						|
 * to the target voltage.
 | 
						|
 * @di:       pointer to the ab8500_fg structure
 | 
						|
 * @target    target voltage
 | 
						|
 *
 | 
						|
 * Returns bit pattern closest to the target voltage
 | 
						|
 * valid return values are 0-14. (0-BATT_OK_MAX_NR_INCREMENTS)
 | 
						|
 */
 | 
						|
 | 
						|
static int ab8500_fg_battok_calc(struct ab8500_fg *di, int target)
 | 
						|
{
 | 
						|
	if (target > BATT_OK_MIN +
 | 
						|
		(BATT_OK_INCREMENT * BATT_OK_MAX_NR_INCREMENTS))
 | 
						|
		return BATT_OK_MAX_NR_INCREMENTS;
 | 
						|
	if (target < BATT_OK_MIN)
 | 
						|
		return 0;
 | 
						|
	return (target - BATT_OK_MIN) / BATT_OK_INCREMENT;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_battok_init_hw_register - init battok levels
 | 
						|
 * @di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
static int ab8500_fg_battok_init_hw_register(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int selected;
 | 
						|
	int sel0;
 | 
						|
	int sel1;
 | 
						|
	int cbp_sel0;
 | 
						|
	int cbp_sel1;
 | 
						|
	int ret;
 | 
						|
	int new_val;
 | 
						|
 | 
						|
	sel0 = di->bm->fg_params->battok_falling_th_sel0;
 | 
						|
	sel1 = di->bm->fg_params->battok_raising_th_sel1;
 | 
						|
 | 
						|
	cbp_sel0 = ab8500_fg_battok_calc(di, sel0);
 | 
						|
	cbp_sel1 = ab8500_fg_battok_calc(di, sel1);
 | 
						|
 | 
						|
	selected = BATT_OK_MIN + cbp_sel0 * BATT_OK_INCREMENT;
 | 
						|
 | 
						|
	if (selected != sel0)
 | 
						|
		dev_warn(di->dev, "Invalid voltage step:%d, using %d %d\n",
 | 
						|
			sel0, selected, cbp_sel0);
 | 
						|
 | 
						|
	selected = BATT_OK_MIN + cbp_sel1 * BATT_OK_INCREMENT;
 | 
						|
 | 
						|
	if (selected != sel1)
 | 
						|
		dev_warn(di->dev, "Invalid voltage step:%d, using %d %d\n",
 | 
						|
			sel1, selected, cbp_sel1);
 | 
						|
 | 
						|
	new_val = cbp_sel0 | (cbp_sel1 << 4);
 | 
						|
 | 
						|
	dev_dbg(di->dev, "using: %x %d %d\n", new_val, cbp_sel0, cbp_sel1);
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_SYS_CTRL2_BLOCK,
 | 
						|
		AB8500_BATT_OK_REG, new_val);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_instant_work() - Run the FG state machine instantly
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Work queue function for instant work
 | 
						|
 */
 | 
						|
static void ab8500_fg_instant_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = container_of(work, struct ab8500_fg, fg_work);
 | 
						|
 | 
						|
	ab8500_fg_algorithm(di);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_cc_data_end_handler() - end of data conversion isr.
 | 
						|
 * @irq:       interrupt number
 | 
						|
 * @_di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns IRQ status(IRQ_HANDLED)
 | 
						|
 */
 | 
						|
static irqreturn_t ab8500_fg_cc_data_end_handler(int irq, void *_di)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = _di;
 | 
						|
	if (!di->nbr_cceoc_irq_cnt) {
 | 
						|
		di->nbr_cceoc_irq_cnt++;
 | 
						|
		complete(&di->ab8500_fg_started);
 | 
						|
	} else {
 | 
						|
		di->nbr_cceoc_irq_cnt = 0;
 | 
						|
		complete(&di->ab8500_fg_complete);
 | 
						|
	}
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_cc_int_calib_handler () - end of calibration isr.
 | 
						|
 * @irq:       interrupt number
 | 
						|
 * @_di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns IRQ status(IRQ_HANDLED)
 | 
						|
 */
 | 
						|
static irqreturn_t ab8500_fg_cc_int_calib_handler(int irq, void *_di)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = _di;
 | 
						|
	di->calib_state = AB8500_FG_CALIB_END;
 | 
						|
	queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_cc_convend_handler() - isr to get battery avg current.
 | 
						|
 * @irq:       interrupt number
 | 
						|
 * @_di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns IRQ status(IRQ_HANDLED)
 | 
						|
 */
 | 
						|
static irqreturn_t ab8500_fg_cc_convend_handler(int irq, void *_di)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = _di;
 | 
						|
 | 
						|
	queue_work(di->fg_wq, &di->fg_acc_cur_work);
 | 
						|
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_batt_ovv_handler() - Battery OVV occured
 | 
						|
 * @irq:       interrupt number
 | 
						|
 * @_di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns IRQ status(IRQ_HANDLED)
 | 
						|
 */
 | 
						|
static irqreturn_t ab8500_fg_batt_ovv_handler(int irq, void *_di)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = _di;
 | 
						|
 | 
						|
	dev_dbg(di->dev, "Battery OVV\n");
 | 
						|
 | 
						|
	/* Schedule a new HW failure check */
 | 
						|
	queue_delayed_work(di->fg_wq, &di->fg_check_hw_failure_work, 0);
 | 
						|
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_lowbatf_handler() - Battery voltage is below LOW threshold
 | 
						|
 * @irq:       interrupt number
 | 
						|
 * @_di:       pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Returns IRQ status(IRQ_HANDLED)
 | 
						|
 */
 | 
						|
static irqreturn_t ab8500_fg_lowbatf_handler(int irq, void *_di)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = _di;
 | 
						|
 | 
						|
	/* Initiate handling in ab8500_fg_low_bat_work() if not already initiated. */
 | 
						|
	if (!di->flags.low_bat_delay) {
 | 
						|
		dev_warn(di->dev, "Battery voltage is below LOW threshold\n");
 | 
						|
		di->flags.low_bat_delay = true;
 | 
						|
		/*
 | 
						|
		 * Start a timer to check LOW_BAT again after some time
 | 
						|
		 * This is done to avoid shutdown on single voltage dips
 | 
						|
		 */
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_low_bat_work,
 | 
						|
			round_jiffies(LOW_BAT_CHECK_INTERVAL));
 | 
						|
	}
 | 
						|
	return IRQ_HANDLED;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_get_property() - get the fg properties
 | 
						|
 * @psy:	pointer to the power_supply structure
 | 
						|
 * @psp:	pointer to the power_supply_property structure
 | 
						|
 * @val:	pointer to the power_supply_propval union
 | 
						|
 *
 | 
						|
 * This function gets called when an application tries to get the
 | 
						|
 * fg properties by reading the sysfs files.
 | 
						|
 * voltage_now:		battery voltage
 | 
						|
 * current_now:		battery instant current
 | 
						|
 * current_avg:		battery average current
 | 
						|
 * charge_full_design:	capacity where battery is considered full
 | 
						|
 * charge_now:		battery capacity in nAh
 | 
						|
 * capacity:		capacity in percent
 | 
						|
 * capacity_level:	capacity level
 | 
						|
 *
 | 
						|
 * Returns error code in case of failure else 0 on success
 | 
						|
 */
 | 
						|
static int ab8500_fg_get_property(struct power_supply *psy,
 | 
						|
	enum power_supply_property psp,
 | 
						|
	union power_supply_propval *val)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If battery is identified as unknown and charging of unknown
 | 
						|
	 * batteries is disabled, we always report 100% capacity and
 | 
						|
	 * capacity level UNKNOWN, since we can't calculate
 | 
						|
	 * remaining capacity
 | 
						|
	 */
 | 
						|
 | 
						|
	switch (psp) {
 | 
						|
	case POWER_SUPPLY_PROP_VOLTAGE_NOW:
 | 
						|
		if (di->flags.bat_ovv)
 | 
						|
			val->intval = BATT_OVV_VALUE * 1000;
 | 
						|
		else
 | 
						|
			val->intval = di->vbat * 1000;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CURRENT_NOW:
 | 
						|
		val->intval = di->inst_curr * 1000;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CURRENT_AVG:
 | 
						|
		val->intval = di->avg_curr * 1000;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_ENERGY_FULL_DESIGN:
 | 
						|
		val->intval = ab8500_fg_convert_mah_to_uwh(di,
 | 
						|
				di->bat_cap.max_mah_design);
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_ENERGY_FULL:
 | 
						|
		val->intval = ab8500_fg_convert_mah_to_uwh(di,
 | 
						|
				di->bat_cap.max_mah);
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_ENERGY_NOW:
 | 
						|
		if (di->flags.batt_unknown && !di->bm->chg_unknown_bat &&
 | 
						|
				di->flags.batt_id_received)
 | 
						|
			val->intval = ab8500_fg_convert_mah_to_uwh(di,
 | 
						|
					di->bat_cap.max_mah);
 | 
						|
		else
 | 
						|
			val->intval = ab8500_fg_convert_mah_to_uwh(di,
 | 
						|
					di->bat_cap.prev_mah);
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CHARGE_FULL_DESIGN:
 | 
						|
		val->intval = di->bat_cap.max_mah_design;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CHARGE_FULL:
 | 
						|
		val->intval = di->bat_cap.max_mah;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CHARGE_NOW:
 | 
						|
		if (di->flags.batt_unknown && !di->bm->chg_unknown_bat &&
 | 
						|
				di->flags.batt_id_received)
 | 
						|
			val->intval = di->bat_cap.max_mah;
 | 
						|
		else
 | 
						|
			val->intval = di->bat_cap.prev_mah;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CAPACITY:
 | 
						|
		if (di->flags.batt_unknown && !di->bm->chg_unknown_bat &&
 | 
						|
				di->flags.batt_id_received)
 | 
						|
			val->intval = 100;
 | 
						|
		else
 | 
						|
			val->intval = di->bat_cap.prev_percent;
 | 
						|
		break;
 | 
						|
	case POWER_SUPPLY_PROP_CAPACITY_LEVEL:
 | 
						|
		if (di->flags.batt_unknown && !di->bm->chg_unknown_bat &&
 | 
						|
				di->flags.batt_id_received)
 | 
						|
			val->intval = POWER_SUPPLY_CAPACITY_LEVEL_UNKNOWN;
 | 
						|
		else
 | 
						|
			val->intval = di->bat_cap.prev_level;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ab8500_fg_get_ext_psy_data(struct device *dev, void *data)
 | 
						|
{
 | 
						|
	struct power_supply *psy;
 | 
						|
	struct power_supply *ext;
 | 
						|
	struct ab8500_fg *di;
 | 
						|
	union power_supply_propval ret;
 | 
						|
	int i, j;
 | 
						|
	bool psy_found = false;
 | 
						|
 | 
						|
	psy = (struct power_supply *)data;
 | 
						|
	ext = dev_get_drvdata(dev);
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * For all psy where the name of your driver
 | 
						|
	 * appears in any supplied_to
 | 
						|
	 */
 | 
						|
	for (i = 0; i < ext->num_supplicants; i++) {
 | 
						|
		if (!strcmp(ext->supplied_to[i], psy->name))
 | 
						|
			psy_found = true;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!psy_found)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	/* Go through all properties for the psy */
 | 
						|
	for (j = 0; j < ext->num_properties; j++) {
 | 
						|
		enum power_supply_property prop;
 | 
						|
		prop = ext->properties[j];
 | 
						|
 | 
						|
		if (ext->get_property(ext, prop, &ret))
 | 
						|
			continue;
 | 
						|
 | 
						|
		switch (prop) {
 | 
						|
		case POWER_SUPPLY_PROP_STATUS:
 | 
						|
			switch (ext->type) {
 | 
						|
			case POWER_SUPPLY_TYPE_BATTERY:
 | 
						|
				switch (ret.intval) {
 | 
						|
				case POWER_SUPPLY_STATUS_UNKNOWN:
 | 
						|
				case POWER_SUPPLY_STATUS_DISCHARGING:
 | 
						|
				case POWER_SUPPLY_STATUS_NOT_CHARGING:
 | 
						|
					if (!di->flags.charging)
 | 
						|
						break;
 | 
						|
					di->flags.charging = false;
 | 
						|
					di->flags.fully_charged = false;
 | 
						|
					if (di->bm->capacity_scaling)
 | 
						|
						ab8500_fg_update_cap_scalers(di);
 | 
						|
					queue_work(di->fg_wq, &di->fg_work);
 | 
						|
					break;
 | 
						|
				case POWER_SUPPLY_STATUS_FULL:
 | 
						|
					if (di->flags.fully_charged)
 | 
						|
						break;
 | 
						|
					di->flags.fully_charged = true;
 | 
						|
					di->flags.force_full = true;
 | 
						|
					/* Save current capacity as maximum */
 | 
						|
					di->bat_cap.max_mah = di->bat_cap.mah;
 | 
						|
					queue_work(di->fg_wq, &di->fg_work);
 | 
						|
					break;
 | 
						|
				case POWER_SUPPLY_STATUS_CHARGING:
 | 
						|
					if (di->flags.charging &&
 | 
						|
						!di->flags.fully_charged)
 | 
						|
						break;
 | 
						|
					di->flags.charging = true;
 | 
						|
					di->flags.fully_charged = false;
 | 
						|
					if (di->bm->capacity_scaling)
 | 
						|
						ab8500_fg_update_cap_scalers(di);
 | 
						|
					queue_work(di->fg_wq, &di->fg_work);
 | 
						|
					break;
 | 
						|
				};
 | 
						|
			default:
 | 
						|
				break;
 | 
						|
			};
 | 
						|
			break;
 | 
						|
		case POWER_SUPPLY_PROP_TECHNOLOGY:
 | 
						|
			switch (ext->type) {
 | 
						|
			case POWER_SUPPLY_TYPE_BATTERY:
 | 
						|
				if (!di->flags.batt_id_received &&
 | 
						|
				    di->bm->batt_id != BATTERY_UNKNOWN) {
 | 
						|
					const struct abx500_battery_type *b;
 | 
						|
 | 
						|
					b = &(di->bm->bat_type[di->bm->batt_id]);
 | 
						|
 | 
						|
					di->flags.batt_id_received = true;
 | 
						|
 | 
						|
					di->bat_cap.max_mah_design =
 | 
						|
						MILLI_TO_MICRO *
 | 
						|
						b->charge_full_design;
 | 
						|
 | 
						|
					di->bat_cap.max_mah =
 | 
						|
						di->bat_cap.max_mah_design;
 | 
						|
 | 
						|
					di->vbat_nom = b->nominal_voltage;
 | 
						|
				}
 | 
						|
 | 
						|
				if (ret.intval)
 | 
						|
					di->flags.batt_unknown = false;
 | 
						|
				else
 | 
						|
					di->flags.batt_unknown = true;
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		case POWER_SUPPLY_PROP_TEMP:
 | 
						|
			switch (ext->type) {
 | 
						|
			case POWER_SUPPLY_TYPE_BATTERY:
 | 
						|
				if (di->flags.batt_id_received)
 | 
						|
					di->bat_temp = ret.intval;
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_init_hw_registers() - Set up FG related registers
 | 
						|
 * @di:		pointer to the ab8500_fg structure
 | 
						|
 *
 | 
						|
 * Set up battery OVV, low battery voltage registers
 | 
						|
 */
 | 
						|
static int ab8500_fg_init_hw_registers(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
 | 
						|
	/* Set VBAT OVV threshold */
 | 
						|
	ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
		AB8500_CHARGER,
 | 
						|
		AB8500_BATT_OVV,
 | 
						|
		BATT_OVV_TH_4P75,
 | 
						|
		BATT_OVV_TH_4P75);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to set BATT_OVV\n");
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Enable VBAT OVV detection */
 | 
						|
	ret = abx500_mask_and_set_register_interruptible(di->dev,
 | 
						|
		AB8500_CHARGER,
 | 
						|
		AB8500_BATT_OVV,
 | 
						|
		BATT_OVV_ENA,
 | 
						|
		BATT_OVV_ENA);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to enable BATT_OVV\n");
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Low Battery Voltage */
 | 
						|
	ret = abx500_set_register_interruptible(di->dev,
 | 
						|
		AB8500_SYS_CTRL2_BLOCK,
 | 
						|
		AB8500_LOW_BAT_REG,
 | 
						|
		ab8500_volt_to_regval(
 | 
						|
			di->bm->fg_params->lowbat_threshold) << 1 |
 | 
						|
		LOW_BAT_ENABLE);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "%s write failed\n", __func__);
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Battery OK threshold */
 | 
						|
	ret = ab8500_fg_battok_init_hw_register(di);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "BattOk init write failed.\n");
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
 | 
						|
	if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
 | 
						|
			abx500_get_chip_id(di->dev) >= AB8500_CUT2P0)
 | 
						|
			|| is_ab8540(di->parent)) {
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8505_RTC_PCUT_MAX_TIME_REG, di->bm->fg_params->pcut_max_time);
 | 
						|
 | 
						|
		if (ret) {
 | 
						|
			dev_err(di->dev, "%s write failed AB8505_RTC_PCUT_MAX_TIME_REG\n", __func__);
 | 
						|
			goto out;
 | 
						|
		};
 | 
						|
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8505_RTC_PCUT_FLAG_TIME_REG, di->bm->fg_params->pcut_flag_time);
 | 
						|
 | 
						|
		if (ret) {
 | 
						|
			dev_err(di->dev, "%s write failed AB8505_RTC_PCUT_FLAG_TIME_REG\n", __func__);
 | 
						|
			goto out;
 | 
						|
		};
 | 
						|
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8505_RTC_PCUT_RESTART_REG, di->bm->fg_params->pcut_max_restart);
 | 
						|
 | 
						|
		if (ret) {
 | 
						|
			dev_err(di->dev, "%s write failed AB8505_RTC_PCUT_RESTART_REG\n", __func__);
 | 
						|
			goto out;
 | 
						|
		};
 | 
						|
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8505_RTC_PCUT_DEBOUNCE_REG, di->bm->fg_params->pcut_debounce_time);
 | 
						|
 | 
						|
		if (ret) {
 | 
						|
			dev_err(di->dev, "%s write failed AB8505_RTC_PCUT_DEBOUNCE_REG\n", __func__);
 | 
						|
			goto out;
 | 
						|
		};
 | 
						|
 | 
						|
		ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
			AB8505_RTC_PCUT_CTL_STATUS_REG, di->bm->fg_params->pcut_enable);
 | 
						|
 | 
						|
		if (ret) {
 | 
						|
			dev_err(di->dev, "%s write failed AB8505_RTC_PCUT_CTL_STATUS_REG\n", __func__);
 | 
						|
			goto out;
 | 
						|
		};
 | 
						|
	}
 | 
						|
out:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_external_power_changed() - callback for power supply changes
 | 
						|
 * @psy:       pointer to the structure power_supply
 | 
						|
 *
 | 
						|
 * This function is the entry point of the pointer external_power_changed
 | 
						|
 * of the structure power_supply.
 | 
						|
 * This function gets executed when there is a change in any external power
 | 
						|
 * supply that this driver needs to be notified of.
 | 
						|
 */
 | 
						|
static void ab8500_fg_external_power_changed(struct power_supply *psy)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	class_for_each_device(power_supply_class, NULL,
 | 
						|
		&di->fg_psy, ab8500_fg_get_ext_psy_data);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * abab8500_fg_reinit_work() - work to reset the FG algorithm
 | 
						|
 * @work:	pointer to the work_struct structure
 | 
						|
 *
 | 
						|
 * Used to reset the current battery capacity to be able to
 | 
						|
 * retrigger a new voltage base capacity calculation. For
 | 
						|
 * test and verification purpose.
 | 
						|
 */
 | 
						|
static void ab8500_fg_reinit_work(struct work_struct *work)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = container_of(work, struct ab8500_fg,
 | 
						|
		fg_reinit_work.work);
 | 
						|
 | 
						|
	if (di->flags.calibrate == false) {
 | 
						|
		dev_dbg(di->dev, "Resetting FG state machine to init.\n");
 | 
						|
		ab8500_fg_clear_cap_samples(di);
 | 
						|
		ab8500_fg_calc_cap_discharge_voltage(di, true);
 | 
						|
		ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_INIT);
 | 
						|
		ab8500_fg_discharge_state_to(di, AB8500_FG_DISCHARGE_INIT);
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
 | 
						|
	} else {
 | 
						|
		dev_err(di->dev, "Residual offset calibration ongoing "
 | 
						|
			"retrying..\n");
 | 
						|
		/* Wait one second until next try*/
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_reinit_work,
 | 
						|
			round_jiffies(1));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_fg_reinit() - forces FG algorithm to reinitialize with current values
 | 
						|
 *
 | 
						|
 * This function can be used to force the FG algorithm to recalculate a new
 | 
						|
 * voltage based battery capacity.
 | 
						|
 */
 | 
						|
void ab8500_fg_reinit(void)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = ab8500_fg_get();
 | 
						|
	/* User won't be notified if a null pointer returned. */
 | 
						|
	if (di != NULL)
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_reinit_work, 0);
 | 
						|
}
 | 
						|
 | 
						|
/* Exposure to the sysfs interface */
 | 
						|
 | 
						|
struct ab8500_fg_sysfs_entry {
 | 
						|
	struct attribute attr;
 | 
						|
	ssize_t (*show)(struct ab8500_fg *, char *);
 | 
						|
	ssize_t (*store)(struct ab8500_fg *, const char *, size_t);
 | 
						|
};
 | 
						|
 | 
						|
static ssize_t charge_full_show(struct ab8500_fg *di, char *buf)
 | 
						|
{
 | 
						|
	return sprintf(buf, "%d\n", di->bat_cap.max_mah);
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t charge_full_store(struct ab8500_fg *di, const char *buf,
 | 
						|
				 size_t count)
 | 
						|
{
 | 
						|
	unsigned long charge_full;
 | 
						|
	ssize_t ret;
 | 
						|
 | 
						|
	ret = kstrtoul(buf, 10, &charge_full);
 | 
						|
 | 
						|
	dev_dbg(di->dev, "Ret %zd charge_full %lu", ret, charge_full);
 | 
						|
 | 
						|
	if (!ret) {
 | 
						|
		di->bat_cap.max_mah = (int) charge_full;
 | 
						|
		ret = count;
 | 
						|
	}
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t charge_now_show(struct ab8500_fg *di, char *buf)
 | 
						|
{
 | 
						|
	return sprintf(buf, "%d\n", di->bat_cap.prev_mah);
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t charge_now_store(struct ab8500_fg *di, const char *buf,
 | 
						|
				 size_t count)
 | 
						|
{
 | 
						|
	unsigned long charge_now;
 | 
						|
	ssize_t ret;
 | 
						|
 | 
						|
	ret = kstrtoul(buf, 10, &charge_now);
 | 
						|
 | 
						|
	dev_dbg(di->dev, "Ret %zd charge_now %lu was %d",
 | 
						|
		ret, charge_now, di->bat_cap.prev_mah);
 | 
						|
 | 
						|
	if (!ret) {
 | 
						|
		di->bat_cap.user_mah = (int) charge_now;
 | 
						|
		di->flags.user_cap = true;
 | 
						|
		ret = count;
 | 
						|
		queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
	}
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static struct ab8500_fg_sysfs_entry charge_full_attr =
 | 
						|
	__ATTR(charge_full, 0644, charge_full_show, charge_full_store);
 | 
						|
 | 
						|
static struct ab8500_fg_sysfs_entry charge_now_attr =
 | 
						|
	__ATTR(charge_now, 0644, charge_now_show, charge_now_store);
 | 
						|
 | 
						|
static ssize_t
 | 
						|
ab8500_fg_show(struct kobject *kobj, struct attribute *attr, char *buf)
 | 
						|
{
 | 
						|
	struct ab8500_fg_sysfs_entry *entry;
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	entry = container_of(attr, struct ab8500_fg_sysfs_entry, attr);
 | 
						|
	di = container_of(kobj, struct ab8500_fg, fg_kobject);
 | 
						|
 | 
						|
	if (!entry->show)
 | 
						|
		return -EIO;
 | 
						|
 | 
						|
	return entry->show(di, buf);
 | 
						|
}
 | 
						|
static ssize_t
 | 
						|
ab8500_fg_store(struct kobject *kobj, struct attribute *attr, const char *buf,
 | 
						|
		size_t count)
 | 
						|
{
 | 
						|
	struct ab8500_fg_sysfs_entry *entry;
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	entry = container_of(attr, struct ab8500_fg_sysfs_entry, attr);
 | 
						|
	di = container_of(kobj, struct ab8500_fg, fg_kobject);
 | 
						|
 | 
						|
	if (!entry->store)
 | 
						|
		return -EIO;
 | 
						|
 | 
						|
	return entry->store(di, buf, count);
 | 
						|
}
 | 
						|
 | 
						|
static const struct sysfs_ops ab8500_fg_sysfs_ops = {
 | 
						|
	.show = ab8500_fg_show,
 | 
						|
	.store = ab8500_fg_store,
 | 
						|
};
 | 
						|
 | 
						|
static struct attribute *ab8500_fg_attrs[] = {
 | 
						|
	&charge_full_attr.attr,
 | 
						|
	&charge_now_attr.attr,
 | 
						|
	NULL,
 | 
						|
};
 | 
						|
 | 
						|
static struct kobj_type ab8500_fg_ktype = {
 | 
						|
	.sysfs_ops = &ab8500_fg_sysfs_ops,
 | 
						|
	.default_attrs = ab8500_fg_attrs,
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_chargalg_sysfs_exit() - de-init of sysfs entry
 | 
						|
 * @di:                pointer to the struct ab8500_chargalg
 | 
						|
 *
 | 
						|
 * This function removes the entry in sysfs.
 | 
						|
 */
 | 
						|
static void ab8500_fg_sysfs_exit(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	kobject_del(&di->fg_kobject);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * ab8500_chargalg_sysfs_init() - init of sysfs entry
 | 
						|
 * @di:                pointer to the struct ab8500_chargalg
 | 
						|
 *
 | 
						|
 * This function adds an entry in sysfs.
 | 
						|
 * Returns error code in case of failure else 0(on success)
 | 
						|
 */
 | 
						|
static int ab8500_fg_sysfs_init(struct ab8500_fg *di)
 | 
						|
{
 | 
						|
	int ret = 0;
 | 
						|
 | 
						|
	ret = kobject_init_and_add(&di->fg_kobject,
 | 
						|
		&ab8500_fg_ktype,
 | 
						|
		NULL, "battery");
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(di->dev, "failed to create sysfs entry\n");
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_flagtime_read(struct device *dev,
 | 
						|
			     struct device_attribute *attr,
 | 
						|
			     char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8505_RTC_PCUT_FLAG_TIME_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_FLAG_TIME_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0x7F));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_flagtime_write(struct device *dev,
 | 
						|
				  struct device_attribute *attr,
 | 
						|
				  const char *buf, size_t count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	long unsigned reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	reg_value = simple_strtoul(buf, NULL, 10);
 | 
						|
 | 
						|
	if (reg_value > 0x7F) {
 | 
						|
		dev_err(dev, "Incorrect parameter, echo 0 (1.98s) - 127 (15.625ms) for flagtime\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8505_RTC_PCUT_FLAG_TIME_REG, (u8)reg_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(dev, "Failed to set AB8505_RTC_PCUT_FLAG_TIME_REG\n");
 | 
						|
 | 
						|
fail:
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_maxtime_read(struct device *dev,
 | 
						|
			     struct device_attribute *attr,
 | 
						|
			     char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8505_RTC_PCUT_MAX_TIME_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_MAX_TIME_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0x7F));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_maxtime_write(struct device *dev,
 | 
						|
				  struct device_attribute *attr,
 | 
						|
				  const char *buf, size_t count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	int reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	reg_value = simple_strtoul(buf, NULL, 10);
 | 
						|
	if (reg_value > 0x7F) {
 | 
						|
		dev_err(dev, "Incorrect parameter, echo 0 (0.0s) - 127 (1.98s) for maxtime\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8505_RTC_PCUT_MAX_TIME_REG, (u8)reg_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(dev, "Failed to set AB8505_RTC_PCUT_MAX_TIME_REG\n");
 | 
						|
 | 
						|
fail:
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_restart_read(struct device *dev,
 | 
						|
			     struct device_attribute *attr,
 | 
						|
			     char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
		AB8505_RTC_PCUT_RESTART_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_RESTART_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0xF));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_restart_write(struct device *dev,
 | 
						|
					     struct device_attribute *attr,
 | 
						|
					     const char *buf, size_t count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	int reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	reg_value = simple_strtoul(buf, NULL, 10);
 | 
						|
	if (reg_value > 0xF) {
 | 
						|
		dev_err(dev, "Incorrect parameter, echo 0 - 15 for number of restart\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_RESTART_REG, (u8)reg_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(dev, "Failed to set AB8505_RTC_PCUT_RESTART_REG\n");
 | 
						|
 | 
						|
fail:
 | 
						|
	return count;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_timer_read(struct device *dev,
 | 
						|
					  struct device_attribute *attr,
 | 
						|
					  char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_TIME_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_TIME_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0x7F));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_restart_counter_read(struct device *dev,
 | 
						|
						    struct device_attribute *attr,
 | 
						|
						    char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_RESTART_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_RESTART_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0xF0) >> 4);
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_read(struct device *dev,
 | 
						|
				    struct device_attribute *attr,
 | 
						|
				    char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		goto fail;
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0x1));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_write(struct device *dev,
 | 
						|
				     struct device_attribute *attr,
 | 
						|
				     const char *buf, size_t count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	int reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	reg_value = simple_strtoul(buf, NULL, 10);
 | 
						|
	if (reg_value > 0x1) {
 | 
						|
		dev_err(dev, "Incorrect parameter, echo 0/1 to disable/enable Pcut feature\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_CTL_STATUS_REG, (u8)reg_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(dev, "Failed to set AB8505_RTC_PCUT_CTL_STATUS_REG\n");
 | 
						|
 | 
						|
fail:
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_flag_read(struct device *dev,
 | 
						|
					 struct device_attribute *attr,
 | 
						|
					 char *buf)
 | 
						|
{
 | 
						|
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_CTL_STATUS_REG,  ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_CTL_STATUS_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", ((reg_value & 0x10) >> 4));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_debounce_read(struct device *dev,
 | 
						|
					     struct device_attribute *attr,
 | 
						|
					     char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_DEBOUNCE_REG,  ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_DEBOUNCE_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", (reg_value & 0x7));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_debounce_write(struct device *dev,
 | 
						|
					      struct device_attribute *attr,
 | 
						|
					      const char *buf, size_t count)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	int reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	reg_value = simple_strtoul(buf, NULL, 10);
 | 
						|
	if (reg_value > 0x7) {
 | 
						|
		dev_err(dev, "Incorrect parameter, echo 0 to 7 for debounce setting\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = abx500_set_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_DEBOUNCE_REG, (u8)reg_value);
 | 
						|
 | 
						|
	if (ret < 0)
 | 
						|
		dev_err(dev, "Failed to set AB8505_RTC_PCUT_DEBOUNCE_REG\n");
 | 
						|
 | 
						|
fail:
 | 
						|
	return count;
 | 
						|
}
 | 
						|
 | 
						|
static ssize_t ab8505_powercut_enable_status_read(struct device *dev,
 | 
						|
						  struct device_attribute *attr,
 | 
						|
						  char *buf)
 | 
						|
{
 | 
						|
	int ret;
 | 
						|
	u8 reg_value;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	ret = abx500_get_register_interruptible(di->dev, AB8500_RTC,
 | 
						|
						AB8505_RTC_PCUT_CTL_STATUS_REG, ®_value);
 | 
						|
 | 
						|
	if (ret < 0) {
 | 
						|
		dev_err(dev, "Failed to read AB8505_RTC_PCUT_CTL_STATUS_REG\n");
 | 
						|
		goto fail;
 | 
						|
	}
 | 
						|
 | 
						|
	return scnprintf(buf, PAGE_SIZE, "%d\n", ((reg_value & 0x20) >> 5));
 | 
						|
 | 
						|
fail:
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static struct device_attribute ab8505_fg_sysfs_psy_attrs[] = {
 | 
						|
	__ATTR(powercut_flagtime, (S_IRUGO | S_IWUSR | S_IWGRP),
 | 
						|
		ab8505_powercut_flagtime_read, ab8505_powercut_flagtime_write),
 | 
						|
	__ATTR(powercut_maxtime, (S_IRUGO | S_IWUSR | S_IWGRP),
 | 
						|
		ab8505_powercut_maxtime_read, ab8505_powercut_maxtime_write),
 | 
						|
	__ATTR(powercut_restart_max, (S_IRUGO | S_IWUSR | S_IWGRP),
 | 
						|
		ab8505_powercut_restart_read, ab8505_powercut_restart_write),
 | 
						|
	__ATTR(powercut_timer, S_IRUGO, ab8505_powercut_timer_read, NULL),
 | 
						|
	__ATTR(powercut_restart_counter, S_IRUGO,
 | 
						|
		ab8505_powercut_restart_counter_read, NULL),
 | 
						|
	__ATTR(powercut_enable, (S_IRUGO | S_IWUSR | S_IWGRP),
 | 
						|
		ab8505_powercut_read, ab8505_powercut_write),
 | 
						|
	__ATTR(powercut_flag, S_IRUGO, ab8505_powercut_flag_read, NULL),
 | 
						|
	__ATTR(powercut_debounce_time, (S_IRUGO | S_IWUSR | S_IWGRP),
 | 
						|
		ab8505_powercut_debounce_read, ab8505_powercut_debounce_write),
 | 
						|
	__ATTR(powercut_enable_status, S_IRUGO,
 | 
						|
		ab8505_powercut_enable_status_read, NULL),
 | 
						|
};
 | 
						|
 | 
						|
static int ab8500_fg_sysfs_psy_create_attrs(struct device *dev)
 | 
						|
{
 | 
						|
	unsigned int i, j;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
 | 
						|
	     abx500_get_chip_id(dev->parent) >= AB8500_CUT2P0)
 | 
						|
	    || is_ab8540(di->parent)) {
 | 
						|
		for (j = 0; j < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); j++)
 | 
						|
			if (device_create_file(dev, &ab8505_fg_sysfs_psy_attrs[j]))
 | 
						|
				goto sysfs_psy_create_attrs_failed_ab8505;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
sysfs_psy_create_attrs_failed_ab8505:
 | 
						|
	dev_err(dev, "Failed creating sysfs psy attrs for ab8505.\n");
 | 
						|
	while (j--)
 | 
						|
		device_remove_file(dev, &ab8505_fg_sysfs_psy_attrs[i]);
 | 
						|
 | 
						|
	return -EIO;
 | 
						|
}
 | 
						|
 | 
						|
static void ab8500_fg_sysfs_psy_remove_attrs(struct device *dev)
 | 
						|
{
 | 
						|
	unsigned int i;
 | 
						|
	struct power_supply *psy = dev_get_drvdata(dev);
 | 
						|
	struct ab8500_fg *di;
 | 
						|
 | 
						|
	di = to_ab8500_fg_device_info(psy);
 | 
						|
 | 
						|
	if (((is_ab8505(di->parent) || is_ab9540(di->parent)) &&
 | 
						|
	     abx500_get_chip_id(dev->parent) >= AB8500_CUT2P0)
 | 
						|
	    || is_ab8540(di->parent)) {
 | 
						|
		for (i = 0; i < ARRAY_SIZE(ab8505_fg_sysfs_psy_attrs); i++)
 | 
						|
			(void)device_remove_file(dev, &ab8505_fg_sysfs_psy_attrs[i]);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Exposure to the sysfs interface <<END>> */
 | 
						|
 | 
						|
#if defined(CONFIG_PM)
 | 
						|
static int ab8500_fg_resume(struct platform_device *pdev)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = platform_get_drvdata(pdev);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Change state if we're not charging. If we're charging we will wake
 | 
						|
	 * up on the FG IRQ
 | 
						|
	 */
 | 
						|
	if (!di->flags.charging) {
 | 
						|
		ab8500_fg_discharge_state_to(di, AB8500_FG_DISCHARGE_WAKEUP);
 | 
						|
		queue_work(di->fg_wq, &di->fg_work);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int ab8500_fg_suspend(struct platform_device *pdev,
 | 
						|
	pm_message_t state)
 | 
						|
{
 | 
						|
	struct ab8500_fg *di = platform_get_drvdata(pdev);
 | 
						|
 | 
						|
	flush_delayed_work(&di->fg_periodic_work);
 | 
						|
	flush_work(&di->fg_work);
 | 
						|
	flush_work(&di->fg_acc_cur_work);
 | 
						|
	flush_delayed_work(&di->fg_reinit_work);
 | 
						|
	flush_delayed_work(&di->fg_low_bat_work);
 | 
						|
	flush_delayed_work(&di->fg_check_hw_failure_work);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If the FG is enabled we will disable it before going to suspend
 | 
						|
	 * only if we're not charging
 | 
						|
	 */
 | 
						|
	if (di->flags.fg_enabled && !di->flags.charging)
 | 
						|
		ab8500_fg_coulomb_counter(di, false);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
#else
 | 
						|
#define ab8500_fg_suspend      NULL
 | 
						|
#define ab8500_fg_resume       NULL
 | 
						|
#endif
 | 
						|
 | 
						|
static int ab8500_fg_remove(struct platform_device *pdev)
 | 
						|
{
 | 
						|
	int ret = 0;
 | 
						|
	struct ab8500_fg *di = platform_get_drvdata(pdev);
 | 
						|
 | 
						|
	list_del(&di->node);
 | 
						|
 | 
						|
	/* Disable coulomb counter */
 | 
						|
	ret = ab8500_fg_coulomb_counter(di, false);
 | 
						|
	if (ret)
 | 
						|
		dev_err(di->dev, "failed to disable coulomb counter\n");
 | 
						|
 | 
						|
	destroy_workqueue(di->fg_wq);
 | 
						|
	ab8500_fg_sysfs_exit(di);
 | 
						|
 | 
						|
	flush_scheduled_work();
 | 
						|
	ab8500_fg_sysfs_psy_remove_attrs(di->fg_psy.dev);
 | 
						|
	power_supply_unregister(&di->fg_psy);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
/* ab8500 fg driver interrupts and their respective isr */
 | 
						|
static struct ab8500_fg_interrupts ab8500_fg_irq[] = {
 | 
						|
	{"NCONV_ACCU", ab8500_fg_cc_convend_handler},
 | 
						|
	{"BATT_OVV", ab8500_fg_batt_ovv_handler},
 | 
						|
	{"LOW_BAT_F", ab8500_fg_lowbatf_handler},
 | 
						|
	{"CC_INT_CALIB", ab8500_fg_cc_int_calib_handler},
 | 
						|
	{"CCEOC", ab8500_fg_cc_data_end_handler},
 | 
						|
};
 | 
						|
 | 
						|
static char *supply_interface[] = {
 | 
						|
	"ab8500_chargalg",
 | 
						|
	"ab8500_usb",
 | 
						|
};
 | 
						|
 | 
						|
static int ab8500_fg_probe(struct platform_device *pdev)
 | 
						|
{
 | 
						|
	struct device_node *np = pdev->dev.of_node;
 | 
						|
	struct abx500_bm_data *plat = pdev->dev.platform_data;
 | 
						|
	struct ab8500_fg *di;
 | 
						|
	int i, irq;
 | 
						|
	int ret = 0;
 | 
						|
 | 
						|
	di = devm_kzalloc(&pdev->dev, sizeof(*di), GFP_KERNEL);
 | 
						|
	if (!di) {
 | 
						|
		dev_err(&pdev->dev, "%s no mem for ab8500_fg\n", __func__);
 | 
						|
		return -ENOMEM;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!plat) {
 | 
						|
		dev_err(&pdev->dev, "no battery management data supplied\n");
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
	di->bm = plat;
 | 
						|
 | 
						|
	if (np) {
 | 
						|
		ret = ab8500_bm_of_probe(&pdev->dev, np, di->bm);
 | 
						|
		if (ret) {
 | 
						|
			dev_err(&pdev->dev, "failed to get battery information\n");
 | 
						|
			return ret;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	mutex_init(&di->cc_lock);
 | 
						|
 | 
						|
	/* get parent data */
 | 
						|
	di->dev = &pdev->dev;
 | 
						|
	di->parent = dev_get_drvdata(pdev->dev.parent);
 | 
						|
	di->gpadc = ab8500_gpadc_get("ab8500-gpadc.0");
 | 
						|
 | 
						|
	di->fg_psy.name = "ab8500_fg";
 | 
						|
	di->fg_psy.type = POWER_SUPPLY_TYPE_BATTERY;
 | 
						|
	di->fg_psy.properties = ab8500_fg_props;
 | 
						|
	di->fg_psy.num_properties = ARRAY_SIZE(ab8500_fg_props);
 | 
						|
	di->fg_psy.get_property = ab8500_fg_get_property;
 | 
						|
	di->fg_psy.supplied_to = supply_interface;
 | 
						|
	di->fg_psy.num_supplicants = ARRAY_SIZE(supply_interface),
 | 
						|
	di->fg_psy.external_power_changed = ab8500_fg_external_power_changed;
 | 
						|
 | 
						|
	di->bat_cap.max_mah_design = MILLI_TO_MICRO *
 | 
						|
		di->bm->bat_type[di->bm->batt_id].charge_full_design;
 | 
						|
 | 
						|
	di->bat_cap.max_mah = di->bat_cap.max_mah_design;
 | 
						|
 | 
						|
	di->vbat_nom = di->bm->bat_type[di->bm->batt_id].nominal_voltage;
 | 
						|
 | 
						|
	di->init_capacity = true;
 | 
						|
 | 
						|
	ab8500_fg_charge_state_to(di, AB8500_FG_CHARGE_INIT);
 | 
						|
	ab8500_fg_discharge_state_to(di, AB8500_FG_DISCHARGE_INIT);
 | 
						|
 | 
						|
	/* Create a work queue for running the FG algorithm */
 | 
						|
	di->fg_wq = create_singlethread_workqueue("ab8500_fg_wq");
 | 
						|
	if (di->fg_wq == NULL) {
 | 
						|
		dev_err(di->dev, "failed to create work queue\n");
 | 
						|
		return -ENOMEM;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Init work for running the fg algorithm instantly */
 | 
						|
	INIT_WORK(&di->fg_work, ab8500_fg_instant_work);
 | 
						|
 | 
						|
	/* Init work for getting the battery accumulated current */
 | 
						|
	INIT_WORK(&di->fg_acc_cur_work, ab8500_fg_acc_cur_work);
 | 
						|
 | 
						|
	/* Init work for reinitialising the fg algorithm */
 | 
						|
	INIT_DEFERRABLE_WORK(&di->fg_reinit_work,
 | 
						|
		ab8500_fg_reinit_work);
 | 
						|
 | 
						|
	/* Work delayed Queue to run the state machine */
 | 
						|
	INIT_DEFERRABLE_WORK(&di->fg_periodic_work,
 | 
						|
		ab8500_fg_periodic_work);
 | 
						|
 | 
						|
	/* Work to check low battery condition */
 | 
						|
	INIT_DEFERRABLE_WORK(&di->fg_low_bat_work,
 | 
						|
		ab8500_fg_low_bat_work);
 | 
						|
 | 
						|
	/* Init work for HW failure check */
 | 
						|
	INIT_DEFERRABLE_WORK(&di->fg_check_hw_failure_work,
 | 
						|
		ab8500_fg_check_hw_failure_work);
 | 
						|
 | 
						|
	/* Reset battery low voltage flag */
 | 
						|
	di->flags.low_bat = false;
 | 
						|
 | 
						|
	/* Initialize low battery counter */
 | 
						|
	di->low_bat_cnt = 10;
 | 
						|
 | 
						|
	/* Initialize OVV, and other registers */
 | 
						|
	ret = ab8500_fg_init_hw_registers(di);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to initialize registers\n");
 | 
						|
		goto free_inst_curr_wq;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Consider battery unknown until we're informed otherwise */
 | 
						|
	di->flags.batt_unknown = true;
 | 
						|
	di->flags.batt_id_received = false;
 | 
						|
 | 
						|
	/* Register FG power supply class */
 | 
						|
	ret = power_supply_register(di->dev, &di->fg_psy);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to register FG psy\n");
 | 
						|
		goto free_inst_curr_wq;
 | 
						|
	}
 | 
						|
 | 
						|
	di->fg_samples = SEC_TO_SAMPLE(di->bm->fg_params->init_timer);
 | 
						|
	ab8500_fg_coulomb_counter(di, true);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Initialize completion used to notify completion and start
 | 
						|
	 * of inst current
 | 
						|
	 */
 | 
						|
	init_completion(&di->ab8500_fg_started);
 | 
						|
	init_completion(&di->ab8500_fg_complete);
 | 
						|
 | 
						|
	/* Register interrupts */
 | 
						|
	for (i = 0; i < ARRAY_SIZE(ab8500_fg_irq); i++) {
 | 
						|
		irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
 | 
						|
		ret = request_threaded_irq(irq, NULL, ab8500_fg_irq[i].isr,
 | 
						|
			IRQF_SHARED | IRQF_NO_SUSPEND,
 | 
						|
			ab8500_fg_irq[i].name, di);
 | 
						|
 | 
						|
		if (ret != 0) {
 | 
						|
			dev_err(di->dev, "failed to request %s IRQ %d: %d\n"
 | 
						|
				, ab8500_fg_irq[i].name, irq, ret);
 | 
						|
			goto free_irq;
 | 
						|
		}
 | 
						|
		dev_dbg(di->dev, "Requested %s IRQ %d: %d\n",
 | 
						|
			ab8500_fg_irq[i].name, irq, ret);
 | 
						|
	}
 | 
						|
	di->irq = platform_get_irq_byname(pdev, "CCEOC");
 | 
						|
	disable_irq(di->irq);
 | 
						|
	di->nbr_cceoc_irq_cnt = 0;
 | 
						|
 | 
						|
	platform_set_drvdata(pdev, di);
 | 
						|
 | 
						|
	ret = ab8500_fg_sysfs_init(di);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to create sysfs entry\n");
 | 
						|
		goto free_irq;
 | 
						|
	}
 | 
						|
 | 
						|
	ret = ab8500_fg_sysfs_psy_create_attrs(di->fg_psy.dev);
 | 
						|
	if (ret) {
 | 
						|
		dev_err(di->dev, "failed to create FG psy\n");
 | 
						|
		ab8500_fg_sysfs_exit(di);
 | 
						|
		goto free_irq;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Calibrate the fg first time */
 | 
						|
	di->flags.calibrate = true;
 | 
						|
	di->calib_state = AB8500_FG_CALIB_INIT;
 | 
						|
 | 
						|
	/* Use room temp as default value until we get an update from driver. */
 | 
						|
	di->bat_temp = 210;
 | 
						|
 | 
						|
	/* Run the FG algorithm */
 | 
						|
	queue_delayed_work(di->fg_wq, &di->fg_periodic_work, 0);
 | 
						|
 | 
						|
	list_add_tail(&di->node, &ab8500_fg_list);
 | 
						|
 | 
						|
	return ret;
 | 
						|
 | 
						|
free_irq:
 | 
						|
	power_supply_unregister(&di->fg_psy);
 | 
						|
 | 
						|
	/* We also have to free all successfully registered irqs */
 | 
						|
	for (i = i - 1; i >= 0; i--) {
 | 
						|
		irq = platform_get_irq_byname(pdev, ab8500_fg_irq[i].name);
 | 
						|
		free_irq(irq, di);
 | 
						|
	}
 | 
						|
free_inst_curr_wq:
 | 
						|
	destroy_workqueue(di->fg_wq);
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
static const struct of_device_id ab8500_fg_match[] = {
 | 
						|
	{ .compatible = "stericsson,ab8500-fg", },
 | 
						|
	{ },
 | 
						|
};
 | 
						|
 | 
						|
static struct platform_driver ab8500_fg_driver = {
 | 
						|
	.probe = ab8500_fg_probe,
 | 
						|
	.remove = ab8500_fg_remove,
 | 
						|
	.suspend = ab8500_fg_suspend,
 | 
						|
	.resume = ab8500_fg_resume,
 | 
						|
	.driver = {
 | 
						|
		.name = "ab8500-fg",
 | 
						|
		.owner = THIS_MODULE,
 | 
						|
		.of_match_table = ab8500_fg_match,
 | 
						|
	},
 | 
						|
};
 | 
						|
 | 
						|
static int __init ab8500_fg_init(void)
 | 
						|
{
 | 
						|
	return platform_driver_register(&ab8500_fg_driver);
 | 
						|
}
 | 
						|
 | 
						|
static void __exit ab8500_fg_exit(void)
 | 
						|
{
 | 
						|
	platform_driver_unregister(&ab8500_fg_driver);
 | 
						|
}
 | 
						|
 | 
						|
subsys_initcall_sync(ab8500_fg_init);
 | 
						|
module_exit(ab8500_fg_exit);
 | 
						|
 | 
						|
MODULE_LICENSE("GPL v2");
 | 
						|
MODULE_AUTHOR("Johan Palsson, Karl Komierowski");
 | 
						|
MODULE_ALIAS("platform:ab8500-fg");
 | 
						|
MODULE_DESCRIPTION("AB8500 Fuel Gauge driver");
 |