xref: /linux/drivers/power/supply/twl6030_charger.c (revision 448ecd5771e255629bef0fb16c9b78c4bbd7bd56)
1*b45cdcebSAndreas Kemnade // SPDX-License-Identifier: GPL-2.0
2*b45cdcebSAndreas Kemnade /*
3*b45cdcebSAndreas Kemnade  * TWL6030 charger
4*b45cdcebSAndreas Kemnade  *
5*b45cdcebSAndreas Kemnade  * Copyright (C) 2024 Andreas Kemnade <andreas@kemnade.info>
6*b45cdcebSAndreas Kemnade  *
7*b45cdcebSAndreas Kemnade  * based on older 6030 driver found in a v3.0 vendor kernel
8*b45cdcebSAndreas Kemnade  *
9*b45cdcebSAndreas Kemnade  * based on twl4030_bci_battery.c by TI
10*b45cdcebSAndreas Kemnade  * Copyright (C) 2008 Texas Instruments, Inc.
11*b45cdcebSAndreas Kemnade  */
12*b45cdcebSAndreas Kemnade 
13*b45cdcebSAndreas Kemnade #include <linux/init.h>
14*b45cdcebSAndreas Kemnade #include <linux/module.h>
15*b45cdcebSAndreas Kemnade #include <linux/slab.h>
16*b45cdcebSAndreas Kemnade #include <linux/err.h>
17*b45cdcebSAndreas Kemnade #include <linux/of.h>
18*b45cdcebSAndreas Kemnade #include <linux/bits.h>
19*b45cdcebSAndreas Kemnade #include <linux/platform_device.h>
20*b45cdcebSAndreas Kemnade #include <linux/interrupt.h>
21*b45cdcebSAndreas Kemnade #include <linux/mfd/twl.h>
22*b45cdcebSAndreas Kemnade #include <linux/power_supply.h>
23*b45cdcebSAndreas Kemnade #include <linux/notifier.h>
24*b45cdcebSAndreas Kemnade #include <linux/usb/otg.h>
25*b45cdcebSAndreas Kemnade #include <linux/iio/consumer.h>
26*b45cdcebSAndreas Kemnade #include <linux/devm-helpers.h>
27*b45cdcebSAndreas Kemnade 
28*b45cdcebSAndreas Kemnade #define CONTROLLER_INT_MASK	0x00
29*b45cdcebSAndreas Kemnade #define CONTROLLER_CTRL1	0x01
30*b45cdcebSAndreas Kemnade #define CONTROLLER_WDG		0x02
31*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1	0x03
32*b45cdcebSAndreas Kemnade #define CHARGERUSB_INT_STATUS	0x04
33*b45cdcebSAndreas Kemnade #define CHARGERUSB_INT_MASK	0x05
34*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1	0x06
35*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT2	0x07
36*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRL1	0x08
37*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRL2	0x09
38*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRL3	0x0A
39*b45cdcebSAndreas Kemnade #define CHARGERUSB_STAT1	0x0B
40*b45cdcebSAndreas Kemnade #define CHARGERUSB_VOREG	0x0C
41*b45cdcebSAndreas Kemnade #define CHARGERUSB_VICHRG	0x0D
42*b45cdcebSAndreas Kemnade #define CHARGERUSB_CINLIMIT	0x0E
43*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRLLIMIT1	0x0F
44*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRLLIMIT2	0x10
45*b45cdcebSAndreas Kemnade #define ANTICOLLAPSE_CTRL1	0x11
46*b45cdcebSAndreas Kemnade #define ANTICOLLAPSE_CTRL2	0x12
47*b45cdcebSAndreas Kemnade 
48*b45cdcebSAndreas Kemnade /* TWL6032 registers 0xDA to 0xDE - TWL6032_MODULE_CHARGER */
49*b45cdcebSAndreas Kemnade #define CONTROLLER_CTRL2	0x00
50*b45cdcebSAndreas Kemnade #define CONTROLLER_VSEL_COMP	0x01
51*b45cdcebSAndreas Kemnade #define CHARGERUSB_VSYSREG	0x02
52*b45cdcebSAndreas Kemnade #define CHARGERUSB_VICHRG_PC	0x03
53*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS		0x04
54*b45cdcebSAndreas Kemnade 
55*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_CRYSTL_OSC_OK	0x40
56*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_END_OF_CHARGE	0x20
57*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_VBATOV		0x10
58*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_VSYSOV		0x08
59*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_DPPM_STS	0x04
60*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_CV_STS		0x02
61*b45cdcebSAndreas Kemnade #define LINEAR_CHRG_STS_CC_STS		0x01
62*b45cdcebSAndreas Kemnade 
63*b45cdcebSAndreas Kemnade #define FG_REG_00	0x00
64*b45cdcebSAndreas Kemnade #define FG_REG_01	0x01
65*b45cdcebSAndreas Kemnade #define FG_REG_02	0x02
66*b45cdcebSAndreas Kemnade #define FG_REG_03	0x03
67*b45cdcebSAndreas Kemnade #define FG_REG_04	0x04
68*b45cdcebSAndreas Kemnade #define FG_REG_05	0x05
69*b45cdcebSAndreas Kemnade #define FG_REG_06	0x06
70*b45cdcebSAndreas Kemnade #define FG_REG_07	0x07
71*b45cdcebSAndreas Kemnade #define FG_REG_08	0x08
72*b45cdcebSAndreas Kemnade #define FG_REG_09	0x09
73*b45cdcebSAndreas Kemnade #define FG_REG_10	0x0A
74*b45cdcebSAndreas Kemnade #define FG_REG_11	0x0B
75*b45cdcebSAndreas Kemnade 
76*b45cdcebSAndreas Kemnade /* CONTROLLER_INT_MASK */
77*b45cdcebSAndreas Kemnade #define MVAC_FAULT		BIT(7)
78*b45cdcebSAndreas Kemnade #define MAC_EOC			BIT(6)
79*b45cdcebSAndreas Kemnade #define LINCH_GATED		BIT(5)
80*b45cdcebSAndreas Kemnade #define MBAT_REMOVED		BIT(4)
81*b45cdcebSAndreas Kemnade #define MFAULT_WDG		BIT(3)
82*b45cdcebSAndreas Kemnade #define MBAT_TEMP		BIT(2)
83*b45cdcebSAndreas Kemnade #define MVBUS_DET		BIT(1)
84*b45cdcebSAndreas Kemnade #define MVAC_DET		BIT(0)
85*b45cdcebSAndreas Kemnade 
86*b45cdcebSAndreas Kemnade /* CONTROLLER_CTRL1 */
87*b45cdcebSAndreas Kemnade #define CONTROLLER_CTRL1_EN_LINCH	BIT(5)
88*b45cdcebSAndreas Kemnade #define CONTROLLER_CTRL1_EN_CHARGER	BIT(4)
89*b45cdcebSAndreas Kemnade #define CONTROLLER_CTRL1_SEL_CHARGER	BIT(3)
90*b45cdcebSAndreas Kemnade 
91*b45cdcebSAndreas Kemnade /* CONTROLLER_STAT1 */
92*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_EXTCHRG_STATZ	BIT(7)
93*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_LINCH_GATED	BIT(6)
94*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_CHRG_DET_N	BIT(5)
95*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_FAULT_WDG	BIT(4)
96*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_VAC_DET	BIT(3)
97*b45cdcebSAndreas Kemnade #define VAC_DET	BIT(3)
98*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_VBUS_DET	BIT(2)
99*b45cdcebSAndreas Kemnade #define VBUS_DET	BIT(2)
100*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_BAT_REMOVED	BIT(1)
101*b45cdcebSAndreas Kemnade #define CONTROLLER_STAT1_BAT_TEMP_OVRANGE BIT(0)
102*b45cdcebSAndreas Kemnade 
103*b45cdcebSAndreas Kemnade /* CHARGERUSB_INT_STATUS */
104*b45cdcebSAndreas Kemnade #define EN_LINCH		BIT(4)
105*b45cdcebSAndreas Kemnade #define CURRENT_TERM_INT	BIT(3)
106*b45cdcebSAndreas Kemnade #define CHARGERUSB_STAT		BIT(2)
107*b45cdcebSAndreas Kemnade #define CHARGERUSB_THMREG	BIT(1)
108*b45cdcebSAndreas Kemnade #define CHARGERUSB_FAULT	BIT(0)
109*b45cdcebSAndreas Kemnade 
110*b45cdcebSAndreas Kemnade /* CHARGERUSB_INT_MASK */
111*b45cdcebSAndreas Kemnade #define MASK_MCURRENT_TERM		BIT(3)
112*b45cdcebSAndreas Kemnade #define MASK_MCHARGERUSB_STAT		BIT(2)
113*b45cdcebSAndreas Kemnade #define MASK_MCHARGERUSB_THMREG		BIT(1)
114*b45cdcebSAndreas Kemnade #define MASK_MCHARGERUSB_FAULT		BIT(0)
115*b45cdcebSAndreas Kemnade 
116*b45cdcebSAndreas Kemnade /* CHARGERUSB_STATUS_INT1 */
117*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_TMREG	BIT(7)
118*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_NO_BAT	BIT(6)
119*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_BST_OCP	BIT(5)
120*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_TH_SHUTD	BIT(4)
121*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_BAT_OVP	BIT(3)
122*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_POOR_SRC	BIT(2)
123*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_SLP_MODE	BIT(1)
124*b45cdcebSAndreas Kemnade #define CHARGERUSB_STATUS_INT1_VBUS_OVP	BIT(0)
125*b45cdcebSAndreas Kemnade 
126*b45cdcebSAndreas Kemnade /* CHARGERUSB_STATUS_INT2 */
127*b45cdcebSAndreas Kemnade #define ICCLOOP		BIT(3)
128*b45cdcebSAndreas Kemnade #define CURRENT_TERM	BIT(2)
129*b45cdcebSAndreas Kemnade #define CHARGE_DONE	BIT(1)
130*b45cdcebSAndreas Kemnade #define ANTICOLLAPSE	BIT(0)
131*b45cdcebSAndreas Kemnade 
132*b45cdcebSAndreas Kemnade /* CHARGERUSB_CTRL1 */
133*b45cdcebSAndreas Kemnade #define SUSPEND_BOOT	BIT(7)
134*b45cdcebSAndreas Kemnade #define OPA_MODE	BIT(6)
135*b45cdcebSAndreas Kemnade #define HZ_MODE		BIT(5)
136*b45cdcebSAndreas Kemnade #define TERM		BIT(4)
137*b45cdcebSAndreas Kemnade 
138*b45cdcebSAndreas Kemnade /* CHARGERUSB_CTRL2 */
139*b45cdcebSAndreas Kemnade #define UA_TO_VITERM(x) (((x) / 50000 - 1) << 5)
140*b45cdcebSAndreas Kemnade 
141*b45cdcebSAndreas Kemnade /* CHARGERUSB_CTRL3 */
142*b45cdcebSAndreas Kemnade #define VBUSCHRG_LDO_OVRD	BIT(7)
143*b45cdcebSAndreas Kemnade #define CHARGE_ONCE		BIT(6)
144*b45cdcebSAndreas Kemnade #define BST_HW_PR_DIS		BIT(5)
145*b45cdcebSAndreas Kemnade #define AUTOSUPPLY		BIT(3)
146*b45cdcebSAndreas Kemnade #define BUCK_HSILIM		BIT(0)
147*b45cdcebSAndreas Kemnade 
148*b45cdcebSAndreas Kemnade /* CHARGERUSB_VOREG */
149*b45cdcebSAndreas Kemnade #define UV_TO_VOREG(x) (((x) - 3500000) / 20000)
150*b45cdcebSAndreas Kemnade #define VOREG_TO_UV(x) (((x) & 0x3F) * 20000 + 3500000)
151*b45cdcebSAndreas Kemnade #define CHARGERUSB_VOREG_3P52		0x01
152*b45cdcebSAndreas Kemnade #define CHARGERUSB_VOREG_4P0		0x19
153*b45cdcebSAndreas Kemnade #define CHARGERUSB_VOREG_4P2		0x23
154*b45cdcebSAndreas Kemnade #define CHARGERUSB_VOREG_4P76		0x3F
155*b45cdcebSAndreas Kemnade 
156*b45cdcebSAndreas Kemnade /* CHARGERUSB_VICHRG */
157*b45cdcebSAndreas Kemnade /*
158*b45cdcebSAndreas Kemnade  * might be inaccurate for < 500 mA, diffent scale might apply,
159*b45cdcebSAndreas Kemnade  * either starting from 100 mA or 300 mA
160*b45cdcebSAndreas Kemnade  */
161*b45cdcebSAndreas Kemnade #define UA_TO_VICHRG(x) (((x) / 100000) - 1)
162*b45cdcebSAndreas Kemnade #define VICHRG_TO_UA(x) (((x) & 0xf) * 100000 + 100000)
163*b45cdcebSAndreas Kemnade 
164*b45cdcebSAndreas Kemnade /* CHARGERUSB_CINLIMIT */
165*b45cdcebSAndreas Kemnade #define CHARGERUSB_CIN_LIMIT_100	0x1
166*b45cdcebSAndreas Kemnade #define CHARGERUSB_CIN_LIMIT_300	0x5
167*b45cdcebSAndreas Kemnade #define CHARGERUSB_CIN_LIMIT_500	0x9
168*b45cdcebSAndreas Kemnade #define CHARGERUSB_CIN_LIMIT_NONE	0xF
169*b45cdcebSAndreas Kemnade 
170*b45cdcebSAndreas Kemnade /* CHARGERUSB_CTRLLIMIT2 */
171*b45cdcebSAndreas Kemnade #define CHARGERUSB_CTRLLIMIT2_1500	0x0E
172*b45cdcebSAndreas Kemnade #define		LOCK_LIMIT		BIT(4)
173*b45cdcebSAndreas Kemnade 
174*b45cdcebSAndreas Kemnade /* ANTICOLLAPSE_CTRL2 */
175*b45cdcebSAndreas Kemnade #define BUCK_VTH_SHIFT			5
176*b45cdcebSAndreas Kemnade 
177*b45cdcebSAndreas Kemnade /* FG_REG_00 */
178*b45cdcebSAndreas Kemnade #define CC_ACTIVE_MODE_SHIFT	6
179*b45cdcebSAndreas Kemnade #define CC_AUTOCLEAR		BIT(2)
180*b45cdcebSAndreas Kemnade #define CC_CAL_EN		BIT(1)
181*b45cdcebSAndreas Kemnade #define CC_PAUSE		BIT(0)
182*b45cdcebSAndreas Kemnade 
183*b45cdcebSAndreas Kemnade #define REG_TOGGLE1		0x90
184*b45cdcebSAndreas Kemnade #define REG_PWDNSTATUS1		0x93
185*b45cdcebSAndreas Kemnade #define FGDITHS			BIT(7)
186*b45cdcebSAndreas Kemnade #define FGDITHR			BIT(6)
187*b45cdcebSAndreas Kemnade #define FGS			BIT(5)
188*b45cdcebSAndreas Kemnade #define FGR			BIT(4)
189*b45cdcebSAndreas Kemnade #define BBSPOR_CFG		0xE6
190*b45cdcebSAndreas Kemnade #define	BB_CHG_EN		BIT(3)
191*b45cdcebSAndreas Kemnade 
192*b45cdcebSAndreas Kemnade struct twl6030_charger_info {
193*b45cdcebSAndreas Kemnade 	struct device		*dev;
194*b45cdcebSAndreas Kemnade 	struct power_supply	*usb;
195*b45cdcebSAndreas Kemnade 	struct power_supply_battery_info *binfo;
196*b45cdcebSAndreas Kemnade 	struct work_struct	work;
197*b45cdcebSAndreas Kemnade 	int			irq_chg;
198*b45cdcebSAndreas Kemnade 	int			input_current_limit;
199*b45cdcebSAndreas Kemnade 	struct iio_channel	*channel_vusb;
200*b45cdcebSAndreas Kemnade 	struct delayed_work	charger_monitor;
201*b45cdcebSAndreas Kemnade 	bool			extended_current_range;
202*b45cdcebSAndreas Kemnade };
203*b45cdcebSAndreas Kemnade 
204*b45cdcebSAndreas Kemnade struct twl6030_charger_chip_data {
205*b45cdcebSAndreas Kemnade 	bool extended_current_range;
206*b45cdcebSAndreas Kemnade };
207*b45cdcebSAndreas Kemnade 
twl6030_charger_read(u8 reg,u8 * val)208*b45cdcebSAndreas Kemnade static int twl6030_charger_read(u8 reg, u8 *val)
209*b45cdcebSAndreas Kemnade {
210*b45cdcebSAndreas Kemnade 	return twl_i2c_read_u8(TWL_MODULE_MAIN_CHARGE, val, reg);
211*b45cdcebSAndreas Kemnade }
212*b45cdcebSAndreas Kemnade 
twl6030_charger_write(u8 reg,u8 val)213*b45cdcebSAndreas Kemnade static int twl6030_charger_write(u8 reg, u8 val)
214*b45cdcebSAndreas Kemnade {
215*b45cdcebSAndreas Kemnade 	return twl_i2c_write_u8(TWL_MODULE_MAIN_CHARGE, val, reg);
216*b45cdcebSAndreas Kemnade }
217*b45cdcebSAndreas Kemnade 
twl6030_config_cinlimit_reg(struct twl6030_charger_info * charger,unsigned int ua)218*b45cdcebSAndreas Kemnade static int twl6030_config_cinlimit_reg(struct twl6030_charger_info *charger,
219*b45cdcebSAndreas Kemnade 				       unsigned int ua)
220*b45cdcebSAndreas Kemnade {
221*b45cdcebSAndreas Kemnade 	if (ua >= 50000 && ua <= 750000) {
222*b45cdcebSAndreas Kemnade 		ua = (ua - 50000) / 50000;
223*b45cdcebSAndreas Kemnade 	} else if ((ua > 750000) && (ua <= 1500000) && charger->extended_current_range) {
224*b45cdcebSAndreas Kemnade 		ua = ((ua % 100000) ? 0x30 : 0x20) + ((ua - 100000) / 100000);
225*b45cdcebSAndreas Kemnade 	} else {
226*b45cdcebSAndreas Kemnade 		if (ua < 50000) {
227*b45cdcebSAndreas Kemnade 			dev_err(charger->dev, "invalid input current limit\n");
228*b45cdcebSAndreas Kemnade 			return -EINVAL;
229*b45cdcebSAndreas Kemnade 		}
230*b45cdcebSAndreas Kemnade 		/* This is no current limit */
231*b45cdcebSAndreas Kemnade 		ua = 0x0F;
232*b45cdcebSAndreas Kemnade 	}
233*b45cdcebSAndreas Kemnade 
234*b45cdcebSAndreas Kemnade 	return twl6030_charger_write(CHARGERUSB_CINLIMIT, ua);
235*b45cdcebSAndreas Kemnade }
236*b45cdcebSAndreas Kemnade 
237*b45cdcebSAndreas Kemnade /*
238*b45cdcebSAndreas Kemnade  * rewriting all stuff here, resets to extremely conservative defaults were
239*b45cdcebSAndreas Kemnade  * seen under some circumstances, like charge voltage to 3.5V
240*b45cdcebSAndreas Kemnade  */
twl6030_enable_usb(struct twl6030_charger_info * charger)241*b45cdcebSAndreas Kemnade static int twl6030_enable_usb(struct twl6030_charger_info *charger)
242*b45cdcebSAndreas Kemnade {
243*b45cdcebSAndreas Kemnade 	int ret;
244*b45cdcebSAndreas Kemnade 
245*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_write(CHARGERUSB_VICHRG,
246*b45cdcebSAndreas Kemnade 				    UA_TO_VICHRG(charger->binfo->constant_charge_current_max_ua));
247*b45cdcebSAndreas Kemnade 	if (ret < 0)
248*b45cdcebSAndreas Kemnade 		return ret;
249*b45cdcebSAndreas Kemnade 
250*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_write(CONTROLLER_WDG, 0xff);
251*b45cdcebSAndreas Kemnade 	if (ret < 0)
252*b45cdcebSAndreas Kemnade 		return ret;
253*b45cdcebSAndreas Kemnade 
254*b45cdcebSAndreas Kemnade 	charger->input_current_limit = 500000;
255*b45cdcebSAndreas Kemnade 	ret = twl6030_config_cinlimit_reg(charger, charger->input_current_limit);
256*b45cdcebSAndreas Kemnade 	if (ret < 0)
257*b45cdcebSAndreas Kemnade 		return ret;
258*b45cdcebSAndreas Kemnade 
259*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_write(CHARGERUSB_CINLIMIT, CHARGERUSB_CIN_LIMIT_500);
260*b45cdcebSAndreas Kemnade 	if (ret < 0)
261*b45cdcebSAndreas Kemnade 		return ret;
262*b45cdcebSAndreas Kemnade 
263*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_write(CHARGERUSB_VOREG,
264*b45cdcebSAndreas Kemnade 				    UV_TO_VOREG(charger->binfo->constant_charge_voltage_max_uv));
265*b45cdcebSAndreas Kemnade 	if (ret < 0)
266*b45cdcebSAndreas Kemnade 		return ret;
267*b45cdcebSAndreas Kemnade 
268*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_write(CHARGERUSB_CTRL1, TERM);
269*b45cdcebSAndreas Kemnade 	if (ret < 0)
270*b45cdcebSAndreas Kemnade 		return ret;
271*b45cdcebSAndreas Kemnade 
272*b45cdcebSAndreas Kemnade 	if (charger->binfo->charge_term_current_ua != -EINVAL) {
273*b45cdcebSAndreas Kemnade 		ret = twl6030_charger_write(CHARGERUSB_CTRL2,
274*b45cdcebSAndreas Kemnade 					    UA_TO_VITERM(charger->binfo->charge_term_current_ua));
275*b45cdcebSAndreas Kemnade 		if (ret < 0)
276*b45cdcebSAndreas Kemnade 			return ret;
277*b45cdcebSAndreas Kemnade 	}
278*b45cdcebSAndreas Kemnade 
279*b45cdcebSAndreas Kemnade 	return twl6030_charger_write(CONTROLLER_CTRL1, CONTROLLER_CTRL1_EN_CHARGER);
280*b45cdcebSAndreas Kemnade }
281*b45cdcebSAndreas Kemnade 
twl6030_charger_wdg(struct work_struct * data)282*b45cdcebSAndreas Kemnade static void twl6030_charger_wdg(struct work_struct *data)
283*b45cdcebSAndreas Kemnade {
284*b45cdcebSAndreas Kemnade 	struct twl6030_charger_info *charger =
285*b45cdcebSAndreas Kemnade 		container_of(data, struct twl6030_charger_info,
286*b45cdcebSAndreas Kemnade 			     charger_monitor.work);
287*b45cdcebSAndreas Kemnade 
288*b45cdcebSAndreas Kemnade 	u8 val;
289*b45cdcebSAndreas Kemnade 	u8 int_stat;
290*b45cdcebSAndreas Kemnade 	u8 stat_int1;
291*b45cdcebSAndreas Kemnade 	u8 stat_int2;
292*b45cdcebSAndreas Kemnade 
293*b45cdcebSAndreas Kemnade 	twl6030_charger_read(CONTROLLER_STAT1, &val);
294*b45cdcebSAndreas Kemnade 	twl6030_charger_read(CHARGERUSB_INT_STATUS, &int_stat);
295*b45cdcebSAndreas Kemnade 	twl6030_charger_read(CHARGERUSB_STATUS_INT1, &stat_int1);
296*b45cdcebSAndreas Kemnade 	twl6030_charger_read(CHARGERUSB_STATUS_INT2, &stat_int2);
297*b45cdcebSAndreas Kemnade 	dev_dbg(charger->dev,
298*b45cdcebSAndreas Kemnade 		"wdg: stat1: %02x %s INT_STATUS %02x STATUS_INT1 %02x STATUS_INT2 %02x\n",
299*b45cdcebSAndreas Kemnade 		val, (val & VBUS_DET) ? "usb online" :  "usb offline",
300*b45cdcebSAndreas Kemnade 		int_stat, stat_int1, stat_int2);
301*b45cdcebSAndreas Kemnade 
302*b45cdcebSAndreas Kemnade 	twl6030_charger_write(CONTROLLER_WDG, 0xff);
303*b45cdcebSAndreas Kemnade 	schedule_delayed_work(&charger->charger_monitor,
304*b45cdcebSAndreas Kemnade 			      msecs_to_jiffies(10000));
305*b45cdcebSAndreas Kemnade }
306*b45cdcebSAndreas Kemnade 
twl6030_charger_interrupt(int irq,void * arg)307*b45cdcebSAndreas Kemnade static irqreturn_t twl6030_charger_interrupt(int irq, void *arg)
308*b45cdcebSAndreas Kemnade {
309*b45cdcebSAndreas Kemnade 	struct twl6030_charger_info *charger = arg;
310*b45cdcebSAndreas Kemnade 	u8 val;
311*b45cdcebSAndreas Kemnade 	u8 int_stat;
312*b45cdcebSAndreas Kemnade 	u8 stat_int1;
313*b45cdcebSAndreas Kemnade 	u8 stat_int2;
314*b45cdcebSAndreas Kemnade 
315*b45cdcebSAndreas Kemnade 	if (twl6030_charger_read(CONTROLLER_STAT1, &val) < 0)
316*b45cdcebSAndreas Kemnade 		return IRQ_HANDLED;
317*b45cdcebSAndreas Kemnade 
318*b45cdcebSAndreas Kemnade 	if (twl6030_charger_read(CHARGERUSB_INT_STATUS, &int_stat) < 0)
319*b45cdcebSAndreas Kemnade 		return IRQ_HANDLED;
320*b45cdcebSAndreas Kemnade 
321*b45cdcebSAndreas Kemnade 	if (twl6030_charger_read(CHARGERUSB_STATUS_INT1, &stat_int1) < 0)
322*b45cdcebSAndreas Kemnade 		return IRQ_HANDLED;
323*b45cdcebSAndreas Kemnade 
324*b45cdcebSAndreas Kemnade 	if (twl6030_charger_read(CHARGERUSB_STATUS_INT2, &stat_int2) < 0)
325*b45cdcebSAndreas Kemnade 		return IRQ_HANDLED;
326*b45cdcebSAndreas Kemnade 
327*b45cdcebSAndreas Kemnade 	dev_dbg(charger->dev,
328*b45cdcebSAndreas Kemnade 		"charger irq: stat1: %02x %s INT_STATUS %02x STATUS_INT1 %02x STATUS_INT2 %02x\n",
329*b45cdcebSAndreas Kemnade 		val, (val & VBUS_DET) ? "usb online" :  "usb offline",
330*b45cdcebSAndreas Kemnade 		int_stat, stat_int1, stat_int2);
331*b45cdcebSAndreas Kemnade 	power_supply_changed(charger->usb);
332*b45cdcebSAndreas Kemnade 
333*b45cdcebSAndreas Kemnade 	if (val & VBUS_DET) {
334*b45cdcebSAndreas Kemnade 		if (twl6030_charger_read(CONTROLLER_CTRL1, &val) < 0)
335*b45cdcebSAndreas Kemnade 			return IRQ_HANDLED;
336*b45cdcebSAndreas Kemnade 
337*b45cdcebSAndreas Kemnade 		if (!(val & CONTROLLER_CTRL1_EN_CHARGER)) {
338*b45cdcebSAndreas Kemnade 			if (twl6030_enable_usb(charger) < 0)
339*b45cdcebSAndreas Kemnade 				return IRQ_HANDLED;
340*b45cdcebSAndreas Kemnade 
341*b45cdcebSAndreas Kemnade 			schedule_delayed_work(&charger->charger_monitor,
342*b45cdcebSAndreas Kemnade 					      msecs_to_jiffies(10000));
343*b45cdcebSAndreas Kemnade 		}
344*b45cdcebSAndreas Kemnade 	} else {
345*b45cdcebSAndreas Kemnade 		cancel_delayed_work(&charger->charger_monitor);
346*b45cdcebSAndreas Kemnade 	}
347*b45cdcebSAndreas Kemnade 	return IRQ_HANDLED;
348*b45cdcebSAndreas Kemnade }
349*b45cdcebSAndreas Kemnade 
twl6030_charger_usb_get_property(struct power_supply * psy,enum power_supply_property psp,union power_supply_propval * val)350*b45cdcebSAndreas Kemnade static int twl6030_charger_usb_get_property(struct power_supply *psy,
351*b45cdcebSAndreas Kemnade 					    enum power_supply_property psp,
352*b45cdcebSAndreas Kemnade 					    union power_supply_propval *val)
353*b45cdcebSAndreas Kemnade {
354*b45cdcebSAndreas Kemnade 	struct twl6030_charger_info *charger = power_supply_get_drvdata(psy);
355*b45cdcebSAndreas Kemnade 	int ret;
356*b45cdcebSAndreas Kemnade 	u8 stat1;
357*b45cdcebSAndreas Kemnade 	u8 intstat;
358*b45cdcebSAndreas Kemnade 
359*b45cdcebSAndreas Kemnade 	ret = twl6030_charger_read(CONTROLLER_STAT1, &stat1);
360*b45cdcebSAndreas Kemnade 	if (ret)
361*b45cdcebSAndreas Kemnade 		return ret;
362*b45cdcebSAndreas Kemnade 
363*b45cdcebSAndreas Kemnade 	switch (psp) {
364*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_STATUS:
365*b45cdcebSAndreas Kemnade 		if (!(stat1 & VBUS_DET)) {
366*b45cdcebSAndreas Kemnade 			val->intval = POWER_SUPPLY_STATUS_NOT_CHARGING;
367*b45cdcebSAndreas Kemnade 			break;
368*b45cdcebSAndreas Kemnade 		}
369*b45cdcebSAndreas Kemnade 		ret = twl6030_charger_read(CHARGERUSB_STATUS_INT2, &intstat);
370*b45cdcebSAndreas Kemnade 		if (ret)
371*b45cdcebSAndreas Kemnade 			return ret;
372*b45cdcebSAndreas Kemnade 
373*b45cdcebSAndreas Kemnade 		if (intstat & CHARGE_DONE)
374*b45cdcebSAndreas Kemnade 			val->intval = POWER_SUPPLY_STATUS_FULL;
375*b45cdcebSAndreas Kemnade 		else if (intstat & CURRENT_TERM)
376*b45cdcebSAndreas Kemnade 			val->intval = POWER_SUPPLY_STATUS_NOT_CHARGING;
377*b45cdcebSAndreas Kemnade 		else
378*b45cdcebSAndreas Kemnade 			val->intval = POWER_SUPPLY_STATUS_CHARGING;
379*b45cdcebSAndreas Kemnade 		break;
380*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_VOLTAGE_NOW:
381*b45cdcebSAndreas Kemnade 		if (!charger->channel_vusb)
382*b45cdcebSAndreas Kemnade 			return -ENODATA;
383*b45cdcebSAndreas Kemnade 
384*b45cdcebSAndreas Kemnade 		ret = iio_read_channel_processed_scale(charger->channel_vusb, &val->intval, 1000);
385*b45cdcebSAndreas Kemnade 		if (ret < 0)
386*b45cdcebSAndreas Kemnade 			return ret;
387*b45cdcebSAndreas Kemnade 
388*b45cdcebSAndreas Kemnade 		break;
389*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_ONLINE:
390*b45cdcebSAndreas Kemnade 		val->intval = !!(stat1 & VBUS_DET);
391*b45cdcebSAndreas Kemnade 		break;
392*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
393*b45cdcebSAndreas Kemnade 		val->intval = charger->input_current_limit;
394*b45cdcebSAndreas Kemnade 		break;
395*b45cdcebSAndreas Kemnade 	default:
396*b45cdcebSAndreas Kemnade 		return -EINVAL;
397*b45cdcebSAndreas Kemnade 	}
398*b45cdcebSAndreas Kemnade 
399*b45cdcebSAndreas Kemnade 	return 0;
400*b45cdcebSAndreas Kemnade }
401*b45cdcebSAndreas Kemnade 
twl6030_charger_usb_set_property(struct power_supply * psy,enum power_supply_property psp,const union power_supply_propval * val)402*b45cdcebSAndreas Kemnade static int twl6030_charger_usb_set_property(struct power_supply *psy,
403*b45cdcebSAndreas Kemnade 					    enum power_supply_property psp,
404*b45cdcebSAndreas Kemnade 					    const union power_supply_propval *val)
405*b45cdcebSAndreas Kemnade {
406*b45cdcebSAndreas Kemnade 	struct twl6030_charger_info *charger = power_supply_get_drvdata(psy);
407*b45cdcebSAndreas Kemnade 
408*b45cdcebSAndreas Kemnade 	switch (psp) {
409*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
410*b45cdcebSAndreas Kemnade 		charger->input_current_limit = val->intval;
411*b45cdcebSAndreas Kemnade 		return twl6030_config_cinlimit_reg(charger, charger->input_current_limit);
412*b45cdcebSAndreas Kemnade 	default:
413*b45cdcebSAndreas Kemnade 		return -EINVAL;
414*b45cdcebSAndreas Kemnade 	}
415*b45cdcebSAndreas Kemnade 
416*b45cdcebSAndreas Kemnade 	return 0;
417*b45cdcebSAndreas Kemnade }
418*b45cdcebSAndreas Kemnade 
twl6030_charger_usb_property_is_writeable(struct power_supply * psy,enum power_supply_property psp)419*b45cdcebSAndreas Kemnade static int twl6030_charger_usb_property_is_writeable(struct power_supply *psy,
420*b45cdcebSAndreas Kemnade 						     enum power_supply_property psp)
421*b45cdcebSAndreas Kemnade {
422*b45cdcebSAndreas Kemnade 	dev_info(&psy->dev, "is %d writeable?\n", (int)psp);
423*b45cdcebSAndreas Kemnade 	switch (psp) {
424*b45cdcebSAndreas Kemnade 	case POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT:
425*b45cdcebSAndreas Kemnade 		return true;
426*b45cdcebSAndreas Kemnade 	default:
427*b45cdcebSAndreas Kemnade 		return false;
428*b45cdcebSAndreas Kemnade 	}
429*b45cdcebSAndreas Kemnade }
430*b45cdcebSAndreas Kemnade 
431*b45cdcebSAndreas Kemnade static enum power_supply_property twl6030_charger_props[] = {
432*b45cdcebSAndreas Kemnade 	POWER_SUPPLY_PROP_STATUS,
433*b45cdcebSAndreas Kemnade 	POWER_SUPPLY_PROP_ONLINE,
434*b45cdcebSAndreas Kemnade 	POWER_SUPPLY_PROP_VOLTAGE_NOW,
435*b45cdcebSAndreas Kemnade 	POWER_SUPPLY_PROP_INPUT_CURRENT_LIMIT,
436*b45cdcebSAndreas Kemnade };
437*b45cdcebSAndreas Kemnade 
438*b45cdcebSAndreas Kemnade static const struct power_supply_desc twl6030_charger_usb_desc = {
439*b45cdcebSAndreas Kemnade 	.name		= "twl6030_usb",
440*b45cdcebSAndreas Kemnade 	.type		= POWER_SUPPLY_TYPE_USB,
441*b45cdcebSAndreas Kemnade 	.properties	= twl6030_charger_props,
442*b45cdcebSAndreas Kemnade 	.num_properties	= ARRAY_SIZE(twl6030_charger_props),
443*b45cdcebSAndreas Kemnade 	.get_property	= twl6030_charger_usb_get_property,
444*b45cdcebSAndreas Kemnade 	.set_property	= twl6030_charger_usb_set_property,
445*b45cdcebSAndreas Kemnade 	.property_is_writeable	= twl6030_charger_usb_property_is_writeable,
446*b45cdcebSAndreas Kemnade };
447*b45cdcebSAndreas Kemnade 
twl6030_charger_probe(struct platform_device * pdev)448*b45cdcebSAndreas Kemnade static int twl6030_charger_probe(struct platform_device *pdev)
449*b45cdcebSAndreas Kemnade {
450*b45cdcebSAndreas Kemnade 	struct twl6030_charger_info *charger;
451*b45cdcebSAndreas Kemnade 	const struct twl6030_charger_chip_data *chip_data;
452*b45cdcebSAndreas Kemnade 	struct power_supply_config psy_cfg = {};
453*b45cdcebSAndreas Kemnade 	int ret;
454*b45cdcebSAndreas Kemnade 	u8 val;
455*b45cdcebSAndreas Kemnade 
456*b45cdcebSAndreas Kemnade 	charger = devm_kzalloc(&pdev->dev, sizeof(*charger), GFP_KERNEL);
457*b45cdcebSAndreas Kemnade 	if (!charger)
458*b45cdcebSAndreas Kemnade 		return -ENOMEM;
459*b45cdcebSAndreas Kemnade 
460*b45cdcebSAndreas Kemnade 	charger->dev = &pdev->dev;
461*b45cdcebSAndreas Kemnade 	charger->irq_chg = platform_get_irq(pdev, 0);
462*b45cdcebSAndreas Kemnade 
463*b45cdcebSAndreas Kemnade 	chip_data = device_get_match_data(&pdev->dev);
464*b45cdcebSAndreas Kemnade 	if (!chip_data)
465*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, -EINVAL, "missing chip data\n");
466*b45cdcebSAndreas Kemnade 
467*b45cdcebSAndreas Kemnade 	charger->extended_current_range = chip_data->extended_current_range;
468*b45cdcebSAndreas Kemnade 	platform_set_drvdata(pdev, charger);
469*b45cdcebSAndreas Kemnade 	psy_cfg.drv_data = charger;
470*b45cdcebSAndreas Kemnade 	psy_cfg.fwnode = dev_fwnode(&pdev->dev);
471*b45cdcebSAndreas Kemnade 
472*b45cdcebSAndreas Kemnade 	charger->channel_vusb = devm_iio_channel_get(&pdev->dev, "vusb");
473*b45cdcebSAndreas Kemnade 	if (IS_ERR(charger->channel_vusb)) {
474*b45cdcebSAndreas Kemnade 		ret = PTR_ERR(charger->channel_vusb);
475*b45cdcebSAndreas Kemnade 		if (ret == -EPROBE_DEFER)
476*b45cdcebSAndreas Kemnade 			return ret;	/* iio not ready */
477*b45cdcebSAndreas Kemnade 		dev_warn(&pdev->dev, "could not request vusb iio channel (%d)",
478*b45cdcebSAndreas Kemnade 			 ret);
479*b45cdcebSAndreas Kemnade 		charger->channel_vusb = NULL;
480*b45cdcebSAndreas Kemnade 	}
481*b45cdcebSAndreas Kemnade 
482*b45cdcebSAndreas Kemnade 	charger->usb = devm_power_supply_register(&pdev->dev,
483*b45cdcebSAndreas Kemnade 						  &twl6030_charger_usb_desc,
484*b45cdcebSAndreas Kemnade 						  &psy_cfg);
485*b45cdcebSAndreas Kemnade 	if (IS_ERR(charger->usb))
486*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, PTR_ERR(charger->usb),
487*b45cdcebSAndreas Kemnade 				     "Failed to register usb\n");
488*b45cdcebSAndreas Kemnade 
489*b45cdcebSAndreas Kemnade 	ret = power_supply_get_battery_info(charger->usb, &charger->binfo);
490*b45cdcebSAndreas Kemnade 	if (ret < 0)
491*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, ret,
492*b45cdcebSAndreas Kemnade 				     "Failed to get battery info\n");
493*b45cdcebSAndreas Kemnade 
494*b45cdcebSAndreas Kemnade 	dev_info(&pdev->dev, "battery with vmax %d imax: %d\n",
495*b45cdcebSAndreas Kemnade 		 charger->binfo->constant_charge_voltage_max_uv,
496*b45cdcebSAndreas Kemnade 		 charger->binfo->constant_charge_current_max_ua);
497*b45cdcebSAndreas Kemnade 
498*b45cdcebSAndreas Kemnade 	if (charger->binfo->constant_charge_voltage_max_uv == -EINVAL) {
499*b45cdcebSAndreas Kemnade 		ret = twl6030_charger_read(CHARGERUSB_CTRLLIMIT1, &val);
500*b45cdcebSAndreas Kemnade 		if (ret < 0)
501*b45cdcebSAndreas Kemnade 			return ret;
502*b45cdcebSAndreas Kemnade 
503*b45cdcebSAndreas Kemnade 		charger->binfo->constant_charge_voltage_max_uv =
504*b45cdcebSAndreas Kemnade 			VOREG_TO_UV(val);
505*b45cdcebSAndreas Kemnade 	}
506*b45cdcebSAndreas Kemnade 
507*b45cdcebSAndreas Kemnade 	if (charger->binfo->constant_charge_voltage_max_uv > 4760000 ||
508*b45cdcebSAndreas Kemnade 	    charger->binfo->constant_charge_voltage_max_uv < 350000)
509*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, -EINVAL,
510*b45cdcebSAndreas Kemnade 				     "Invalid charge voltage\n");
511*b45cdcebSAndreas Kemnade 
512*b45cdcebSAndreas Kemnade 	if (charger->binfo->constant_charge_current_max_ua == -EINVAL) {
513*b45cdcebSAndreas Kemnade 		ret = twl6030_charger_read(CHARGERUSB_CTRLLIMIT2, &val);
514*b45cdcebSAndreas Kemnade 		if (ret < 0)
515*b45cdcebSAndreas Kemnade 			return ret;
516*b45cdcebSAndreas Kemnade 
517*b45cdcebSAndreas Kemnade 		charger->binfo->constant_charge_current_max_ua = VICHRG_TO_UA(val);
518*b45cdcebSAndreas Kemnade 	}
519*b45cdcebSAndreas Kemnade 
520*b45cdcebSAndreas Kemnade 	if (charger->binfo->constant_charge_current_max_ua < 100000 ||
521*b45cdcebSAndreas Kemnade 	    charger->binfo->constant_charge_current_max_ua > 1500000) {
522*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, -EINVAL,
523*b45cdcebSAndreas Kemnade 			 "Invalid charge current\n");
524*b45cdcebSAndreas Kemnade 	}
525*b45cdcebSAndreas Kemnade 
526*b45cdcebSAndreas Kemnade 	if ((charger->binfo->charge_term_current_ua != -EINVAL) &&
527*b45cdcebSAndreas Kemnade 	    (charger->binfo->charge_term_current_ua > 400000 ||
528*b45cdcebSAndreas Kemnade 	     charger->binfo->charge_term_current_ua < 50000)) {
529*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, -EINVAL,
530*b45cdcebSAndreas Kemnade 			"Invalid charge termination current\n");
531*b45cdcebSAndreas Kemnade 	}
532*b45cdcebSAndreas Kemnade 
533*b45cdcebSAndreas Kemnade 	ret = devm_delayed_work_autocancel(&pdev->dev,
534*b45cdcebSAndreas Kemnade 					   &charger->charger_monitor,
535*b45cdcebSAndreas Kemnade 					   twl6030_charger_wdg);
536*b45cdcebSAndreas Kemnade 	if (ret < 0)
537*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, ret,
538*b45cdcebSAndreas Kemnade 				     "Failed to register delayed work\n");
539*b45cdcebSAndreas Kemnade 
540*b45cdcebSAndreas Kemnade 	ret = devm_request_threaded_irq(&pdev->dev, charger->irq_chg, NULL,
541*b45cdcebSAndreas Kemnade 					twl6030_charger_interrupt,
542*b45cdcebSAndreas Kemnade 					IRQF_ONESHOT, pdev->name,
543*b45cdcebSAndreas Kemnade 					charger);
544*b45cdcebSAndreas Kemnade 	if (ret < 0)
545*b45cdcebSAndreas Kemnade 		return dev_err_probe(&pdev->dev, ret,
546*b45cdcebSAndreas Kemnade 				     "could not request irq %d\n",
547*b45cdcebSAndreas Kemnade 				     charger->irq_chg);
548*b45cdcebSAndreas Kemnade 
549*b45cdcebSAndreas Kemnade 	/* turing to charging to configure things */
550*b45cdcebSAndreas Kemnade 	twl6030_charger_write(CONTROLLER_CTRL1, 0);
551*b45cdcebSAndreas Kemnade 	twl6030_charger_interrupt(0, charger);
552*b45cdcebSAndreas Kemnade 
553*b45cdcebSAndreas Kemnade 	return 0;
554*b45cdcebSAndreas Kemnade }
555*b45cdcebSAndreas Kemnade 
556*b45cdcebSAndreas Kemnade static const struct twl6030_charger_chip_data twl6030_data = {
557*b45cdcebSAndreas Kemnade 	.extended_current_range = false,
558*b45cdcebSAndreas Kemnade };
559*b45cdcebSAndreas Kemnade 
560*b45cdcebSAndreas Kemnade static const struct twl6030_charger_chip_data twl6032_data = {
561*b45cdcebSAndreas Kemnade 	.extended_current_range = true,
562*b45cdcebSAndreas Kemnade };
563*b45cdcebSAndreas Kemnade 
564*b45cdcebSAndreas Kemnade static const struct of_device_id twl_charger_of_match[] = {
565*b45cdcebSAndreas Kemnade 	{.compatible = "ti,twl6030-charger", .data = &twl6030_data},
566*b45cdcebSAndreas Kemnade 	{.compatible = "ti,twl6032-charger", .data = &twl6032_data},
567*b45cdcebSAndreas Kemnade 	{ }
568*b45cdcebSAndreas Kemnade };
569*b45cdcebSAndreas Kemnade MODULE_DEVICE_TABLE(of, twl_charger_of_match);
570*b45cdcebSAndreas Kemnade 
571*b45cdcebSAndreas Kemnade static struct platform_driver twl6030_charger_driver = {
572*b45cdcebSAndreas Kemnade 	.probe = twl6030_charger_probe,
573*b45cdcebSAndreas Kemnade 	.driver	= {
574*b45cdcebSAndreas Kemnade 		.name	= "twl6030_charger",
575*b45cdcebSAndreas Kemnade 		.of_match_table = twl_charger_of_match,
576*b45cdcebSAndreas Kemnade 	},
577*b45cdcebSAndreas Kemnade };
578*b45cdcebSAndreas Kemnade module_platform_driver(twl6030_charger_driver);
579*b45cdcebSAndreas Kemnade 
580*b45cdcebSAndreas Kemnade MODULE_DESCRIPTION("TWL6030 Battery Charger Interface driver");
581*b45cdcebSAndreas Kemnade MODULE_LICENSE("GPL");
582