xref: /OK3568_Linux_fs/u-boot/drivers/power/fuel_gauge/fg_max17042.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  *  Copyright (C) 2012 Samsung Electronics
3*4882a593Smuzhiyun  *  Lukasz Majewski <l.majewski@samsung.com>
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * SPDX-License-Identifier:	GPL-2.0+
6*4882a593Smuzhiyun  */
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <common.h>
9*4882a593Smuzhiyun #include <power/pmic.h>
10*4882a593Smuzhiyun #include <power/max17042_fg.h>
11*4882a593Smuzhiyun #include <i2c.h>
12*4882a593Smuzhiyun #include <power/max8997_pmic.h>
13*4882a593Smuzhiyun #include <power/power_chrg.h>
14*4882a593Smuzhiyun #include <power/battery.h>
15*4882a593Smuzhiyun #include <power/fg_battery_cell_params.h>
16*4882a593Smuzhiyun #include <errno.h>
17*4882a593Smuzhiyun 
fg_write_regs(struct pmic * p,u8 addr,u16 * data,int num)18*4882a593Smuzhiyun static int fg_write_regs(struct pmic *p, u8 addr, u16 *data, int num)
19*4882a593Smuzhiyun {
20*4882a593Smuzhiyun 	int ret = 0;
21*4882a593Smuzhiyun 	int i;
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun 	for (i = 0; i < num; i++, addr++) {
24*4882a593Smuzhiyun 		ret = pmic_reg_write(p, addr, *(data + i));
25*4882a593Smuzhiyun 		if (ret)
26*4882a593Smuzhiyun 			return ret;
27*4882a593Smuzhiyun 	}
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun 	return 0;
30*4882a593Smuzhiyun }
31*4882a593Smuzhiyun 
fg_read_regs(struct pmic * p,u8 addr,u16 * data,int num)32*4882a593Smuzhiyun static int fg_read_regs(struct pmic *p, u8 addr, u16 *data, int num)
33*4882a593Smuzhiyun {
34*4882a593Smuzhiyun 	unsigned int dat;
35*4882a593Smuzhiyun 	int ret = 0;
36*4882a593Smuzhiyun 	int i;
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun 	for (i = 0; i < num; i++, addr++) {
39*4882a593Smuzhiyun 		ret = pmic_reg_read(p, addr, &dat);
40*4882a593Smuzhiyun 		if (ret)
41*4882a593Smuzhiyun 			return ret;
42*4882a593Smuzhiyun 
43*4882a593Smuzhiyun 		*(data + i) = (u16)dat;
44*4882a593Smuzhiyun 	}
45*4882a593Smuzhiyun 
46*4882a593Smuzhiyun 	return 0;
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun 
fg_write_and_verify(struct pmic * p,u8 addr,u16 data)49*4882a593Smuzhiyun static int fg_write_and_verify(struct pmic *p, u8 addr, u16 data)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun 	unsigned int val = data;
52*4882a593Smuzhiyun 	int ret = 0;
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun 	ret |= pmic_reg_write(p, addr, val);
55*4882a593Smuzhiyun 	ret |= pmic_reg_read(p, addr, &val);
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	if (ret)
58*4882a593Smuzhiyun 		return ret;
59*4882a593Smuzhiyun 
60*4882a593Smuzhiyun 	if (((u16) val) == data)
61*4882a593Smuzhiyun 		return 0;
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	return -1;
64*4882a593Smuzhiyun }
65*4882a593Smuzhiyun 
por_fuelgauge_init(struct pmic * p)66*4882a593Smuzhiyun static void por_fuelgauge_init(struct pmic *p)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun 	u16 r_data0[16], r_data1[16], r_data2[16];
69*4882a593Smuzhiyun 	u32 rewrite_count = 5;
70*4882a593Smuzhiyun 	u32 check_count;
71*4882a593Smuzhiyun 	u32 lock_count;
72*4882a593Smuzhiyun 	u32 i = 0;
73*4882a593Smuzhiyun 	u32 val;
74*4882a593Smuzhiyun 	s32 ret = 0;
75*4882a593Smuzhiyun 	char *status_msg;
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	/* Delay 500 ms */
78*4882a593Smuzhiyun 	mdelay(500);
79*4882a593Smuzhiyun 	/* Initilize Configuration */
80*4882a593Smuzhiyun 	pmic_reg_write(p, MAX17042_CONFIG, 0x2310);
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun rewrite_model:
83*4882a593Smuzhiyun 	check_count = 5;
84*4882a593Smuzhiyun 	lock_count = 5;
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun 	if (!rewrite_count--) {
87*4882a593Smuzhiyun 		status_msg = "init failed!";
88*4882a593Smuzhiyun 		goto error;
89*4882a593Smuzhiyun 	}
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun 	/* Unlock Model Access */
92*4882a593Smuzhiyun 	pmic_reg_write(p, MAX17042_MLOCKReg1, MODEL_UNLOCK1);
93*4882a593Smuzhiyun 	pmic_reg_write(p, MAX17042_MLOCKReg2, MODEL_UNLOCK2);
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	/* Write/Read/Verify the Custom Model */
96*4882a593Smuzhiyun 	ret = fg_write_regs(p, MAX17042_MODEL1, cell_character0,
97*4882a593Smuzhiyun 			     ARRAY_SIZE(cell_character0));
98*4882a593Smuzhiyun 	if (ret)
99*4882a593Smuzhiyun 		goto rewrite_model;
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	ret = fg_write_regs(p, MAX17042_MODEL2, cell_character1,
102*4882a593Smuzhiyun 			     ARRAY_SIZE(cell_character1));
103*4882a593Smuzhiyun 	if (ret)
104*4882a593Smuzhiyun 		goto rewrite_model;
105*4882a593Smuzhiyun 
106*4882a593Smuzhiyun 	ret = fg_write_regs(p, MAX17042_MODEL3, cell_character2,
107*4882a593Smuzhiyun 			     ARRAY_SIZE(cell_character2));
108*4882a593Smuzhiyun 	if (ret)
109*4882a593Smuzhiyun 		goto rewrite_model;
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun check_model:
112*4882a593Smuzhiyun 	if (!check_count--) {
113*4882a593Smuzhiyun 		if (rewrite_count)
114*4882a593Smuzhiyun 			goto rewrite_model;
115*4882a593Smuzhiyun 		else
116*4882a593Smuzhiyun 			status_msg = "check failed!";
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 		goto error;
119*4882a593Smuzhiyun 	}
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL1, r_data0, ARRAY_SIZE(r_data0));
122*4882a593Smuzhiyun 	if (ret)
123*4882a593Smuzhiyun 		goto check_model;
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL2, r_data1, ARRAY_SIZE(r_data1));
126*4882a593Smuzhiyun 	if (ret)
127*4882a593Smuzhiyun 		goto check_model;
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL3, r_data2, ARRAY_SIZE(r_data2));
130*4882a593Smuzhiyun 	if (ret)
131*4882a593Smuzhiyun 		goto check_model;
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun 	for (i = 0; i < 16; i++) {
134*4882a593Smuzhiyun 		if ((cell_character0[i] != r_data0[i])
135*4882a593Smuzhiyun 		    || (cell_character1[i] != r_data1[i])
136*4882a593Smuzhiyun 		    || (cell_character2[i] != r_data2[i]))
137*4882a593Smuzhiyun 			goto rewrite_model;
138*4882a593Smuzhiyun 		}
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun lock_model:
141*4882a593Smuzhiyun 	if (!lock_count--) {
142*4882a593Smuzhiyun 		if (rewrite_count)
143*4882a593Smuzhiyun 			goto rewrite_model;
144*4882a593Smuzhiyun 		else
145*4882a593Smuzhiyun 			status_msg = "lock failed!";
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 		goto error;
148*4882a593Smuzhiyun 	}
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun 	/* Lock model access */
151*4882a593Smuzhiyun 	pmic_reg_write(p, MAX17042_MLOCKReg1, MODEL_LOCK1);
152*4882a593Smuzhiyun 	pmic_reg_write(p, MAX17042_MLOCKReg2, MODEL_LOCK2);
153*4882a593Smuzhiyun 
154*4882a593Smuzhiyun 	/* Verify the model access is locked */
155*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL1, r_data0, ARRAY_SIZE(r_data0));
156*4882a593Smuzhiyun 	if (ret)
157*4882a593Smuzhiyun 		goto lock_model;
158*4882a593Smuzhiyun 
159*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL2, r_data1, ARRAY_SIZE(r_data1));
160*4882a593Smuzhiyun 	if (ret)
161*4882a593Smuzhiyun 		goto lock_model;
162*4882a593Smuzhiyun 
163*4882a593Smuzhiyun 	ret = fg_read_regs(p, MAX17042_MODEL3, r_data2, ARRAY_SIZE(r_data2));
164*4882a593Smuzhiyun 	if (ret)
165*4882a593Smuzhiyun 		goto lock_model;
166*4882a593Smuzhiyun 
167*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(r_data0); i++) {
168*4882a593Smuzhiyun 		/* Check if model locked */
169*4882a593Smuzhiyun 		if (r_data0[i] || r_data1[i] || r_data2[i])
170*4882a593Smuzhiyun 			goto lock_model;
171*4882a593Smuzhiyun 	}
172*4882a593Smuzhiyun 
173*4882a593Smuzhiyun 	/* Write Custom Parameters */
174*4882a593Smuzhiyun 	fg_write_and_verify(p, MAX17042_RCOMP0, RCOMP0);
175*4882a593Smuzhiyun 	fg_write_and_verify(p, MAX17042_TEMPCO, TempCo);
176*4882a593Smuzhiyun 
177*4882a593Smuzhiyun 	/* Delay at least 350mS */
178*4882a593Smuzhiyun 	mdelay(350);
179*4882a593Smuzhiyun 
180*4882a593Smuzhiyun 	/* Initialization Complete */
181*4882a593Smuzhiyun 	pmic_reg_read(p, MAX17042_STATUS, &val);
182*4882a593Smuzhiyun 	/* Write and Verify Status with POR bit Cleared */
183*4882a593Smuzhiyun 	fg_write_and_verify(p, MAX17042_STATUS, val & ~MAX17042_POR);
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun 	/* Delay at least 350 ms */
186*4882a593Smuzhiyun 	mdelay(350);
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun 	status_msg = "OK!";
189*4882a593Smuzhiyun error:
190*4882a593Smuzhiyun 	debug("%s: model init status: %s\n", p->name, status_msg);
191*4882a593Smuzhiyun 	return;
192*4882a593Smuzhiyun }
193*4882a593Smuzhiyun 
power_update_battery(struct pmic * p,struct pmic * bat)194*4882a593Smuzhiyun static int power_update_battery(struct pmic *p, struct pmic *bat)
195*4882a593Smuzhiyun {
196*4882a593Smuzhiyun 	struct power_battery *pb = bat->pbat;
197*4882a593Smuzhiyun 	unsigned int val;
198*4882a593Smuzhiyun 	int ret = 0;
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun 	if (pmic_probe(p)) {
201*4882a593Smuzhiyun 		puts("Can't find max17042 fuel gauge\n");
202*4882a593Smuzhiyun 		return -ENODEV;
203*4882a593Smuzhiyun 	}
204*4882a593Smuzhiyun 
205*4882a593Smuzhiyun 	ret |= pmic_reg_read(p, MAX17042_VFSOC, &val);
206*4882a593Smuzhiyun 	pb->bat->state_of_chrg = (val >> 8);
207*4882a593Smuzhiyun 
208*4882a593Smuzhiyun 	pmic_reg_read(p, MAX17042_VCELL, &val);
209*4882a593Smuzhiyun 	debug("vfsoc: 0x%x\n", val);
210*4882a593Smuzhiyun 	pb->bat->voltage_uV = ((val & 0xFFUL) >> 3) + ((val & 0xFF00) >> 3);
211*4882a593Smuzhiyun 	pb->bat->voltage_uV = (pb->bat->voltage_uV * 625);
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	pmic_reg_read(p, 0x05, &val);
214*4882a593Smuzhiyun 	pb->bat->capacity = val >> 2;
215*4882a593Smuzhiyun 
216*4882a593Smuzhiyun 	return ret;
217*4882a593Smuzhiyun }
218*4882a593Smuzhiyun 
power_check_battery(struct pmic * p,struct pmic * bat)219*4882a593Smuzhiyun static int power_check_battery(struct pmic *p, struct pmic *bat)
220*4882a593Smuzhiyun {
221*4882a593Smuzhiyun 	struct power_battery *pb = bat->pbat;
222*4882a593Smuzhiyun 	unsigned int val;
223*4882a593Smuzhiyun 	int ret = 0;
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun 	if (pmic_probe(p)) {
226*4882a593Smuzhiyun 		puts("Can't find max17042 fuel gauge\n");
227*4882a593Smuzhiyun 		return -ENODEV;
228*4882a593Smuzhiyun 	}
229*4882a593Smuzhiyun 
230*4882a593Smuzhiyun 	ret |= pmic_reg_read(p, MAX17042_STATUS, &val);
231*4882a593Smuzhiyun 	debug("fg status: 0x%x\n", val);
232*4882a593Smuzhiyun 
233*4882a593Smuzhiyun 	if (val & MAX17042_POR)
234*4882a593Smuzhiyun 		por_fuelgauge_init(p);
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun 	ret |= pmic_reg_read(p, MAX17042_VERSION, &val);
237*4882a593Smuzhiyun 	pb->bat->version = val;
238*4882a593Smuzhiyun 
239*4882a593Smuzhiyun 	power_update_battery(p, bat);
240*4882a593Smuzhiyun 	debug("fg ver: 0x%x\n", pb->bat->version);
241*4882a593Smuzhiyun 	printf("BAT: state_of_charge(SOC):%d%%\n",
242*4882a593Smuzhiyun 	       pb->bat->state_of_chrg);
243*4882a593Smuzhiyun 
244*4882a593Smuzhiyun 	printf("     voltage: %d.%6.6d [V] (expected to be %d [mAh])\n",
245*4882a593Smuzhiyun 	       pb->bat->voltage_uV / 1000000,
246*4882a593Smuzhiyun 	       pb->bat->voltage_uV % 1000000,
247*4882a593Smuzhiyun 	       pb->bat->capacity);
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun 	if (pb->bat->voltage_uV > 3850000)
250*4882a593Smuzhiyun 		pb->bat->state = EXT_SOURCE;
251*4882a593Smuzhiyun 	else if (pb->bat->voltage_uV < 3600000 || pb->bat->state_of_chrg < 5)
252*4882a593Smuzhiyun 		pb->bat->state = CHARGE;
253*4882a593Smuzhiyun 	else
254*4882a593Smuzhiyun 		pb->bat->state = NORMAL;
255*4882a593Smuzhiyun 
256*4882a593Smuzhiyun 	return ret;
257*4882a593Smuzhiyun }
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun static struct power_fg power_fg_ops = {
260*4882a593Smuzhiyun 	.fg_battery_check = power_check_battery,
261*4882a593Smuzhiyun 	.fg_battery_update = power_update_battery,
262*4882a593Smuzhiyun };
263*4882a593Smuzhiyun 
power_fg_init(unsigned char bus)264*4882a593Smuzhiyun int power_fg_init(unsigned char bus)
265*4882a593Smuzhiyun {
266*4882a593Smuzhiyun 	static const char name[] = "MAX17042_FG";
267*4882a593Smuzhiyun 	struct pmic *p = pmic_alloc();
268*4882a593Smuzhiyun 
269*4882a593Smuzhiyun 	if (!p) {
270*4882a593Smuzhiyun 		printf("%s: POWER allocation error!\n", __func__);
271*4882a593Smuzhiyun 		return -ENOMEM;
272*4882a593Smuzhiyun 	}
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun 	debug("Board Fuel Gauge init\n");
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun 	p->name = name;
277*4882a593Smuzhiyun 	p->interface = PMIC_I2C;
278*4882a593Smuzhiyun 	p->number_of_regs = FG_NUM_OF_REGS;
279*4882a593Smuzhiyun 	p->hw.i2c.addr = MAX17042_I2C_ADDR;
280*4882a593Smuzhiyun 	p->hw.i2c.tx_num = 2;
281*4882a593Smuzhiyun 	p->sensor_byte_order = PMIC_SENSOR_BYTE_ORDER_BIG;
282*4882a593Smuzhiyun 	p->bus = bus;
283*4882a593Smuzhiyun 
284*4882a593Smuzhiyun 	p->fg = &power_fg_ops;
285*4882a593Smuzhiyun 	return 0;
286*4882a593Smuzhiyun }
287