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