| /* | 
 |  * Driver for LP8727 Micro/Mini USB IC with integrated charger | 
 |  * | 
 |  *			Copyright (C) 2011 Texas Instruments | 
 |  *			Copyright (C) 2011 National Semiconductor | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License version 2 as | 
 |  * published by the Free Software Foundation. | 
 |  * | 
 |  */ | 
 |  | 
 | #include <linux/module.h> | 
 | #include <linux/slab.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/i2c.h> | 
 | #include <linux/power_supply.h> | 
 | #include <linux/platform_data/lp8727.h> | 
 | #include <linux/of.h> | 
 |  | 
 | #define LP8788_NUM_INTREGS	2 | 
 | #define DEFAULT_DEBOUNCE_MSEC	270 | 
 |  | 
 | /* Registers */ | 
 | #define LP8727_CTRL1		0x1 | 
 | #define LP8727_CTRL2		0x2 | 
 | #define LP8727_SWCTRL		0x3 | 
 | #define LP8727_INT1		0x4 | 
 | #define LP8727_INT2		0x5 | 
 | #define LP8727_STATUS1		0x6 | 
 | #define LP8727_STATUS2		0x7 | 
 | #define LP8727_CHGCTRL2		0x9 | 
 |  | 
 | /* CTRL1 register */ | 
 | #define LP8727_CP_EN		BIT(0) | 
 | #define LP8727_ADC_EN		BIT(1) | 
 | #define LP8727_ID200_EN		BIT(4) | 
 |  | 
 | /* CTRL2 register */ | 
 | #define LP8727_CHGDET_EN	BIT(1) | 
 | #define LP8727_INT_EN		BIT(6) | 
 |  | 
 | /* SWCTRL register */ | 
 | #define LP8727_SW_DM1_DM	(0x0 << 0) | 
 | #define LP8727_SW_DM1_HiZ	(0x7 << 0) | 
 | #define LP8727_SW_DP2_DP	(0x0 << 3) | 
 | #define LP8727_SW_DP2_HiZ	(0x7 << 3) | 
 |  | 
 | /* INT1 register */ | 
 | #define LP8727_IDNO		(0xF << 0) | 
 | #define LP8727_VBUS		BIT(4) | 
 |  | 
 | /* STATUS1 register */ | 
 | #define LP8727_CHGSTAT		(3 << 4) | 
 | #define LP8727_CHPORT		BIT(6) | 
 | #define LP8727_DCPORT		BIT(7) | 
 | #define LP8727_STAT_EOC		0x30 | 
 |  | 
 | /* STATUS2 register */ | 
 | #define LP8727_TEMP_STAT	(3 << 5) | 
 | #define LP8727_TEMP_SHIFT	5 | 
 |  | 
 | /* CHGCTRL2 register */ | 
 | #define LP8727_ICHG_SHIFT	4 | 
 |  | 
 | enum lp8727_dev_id { | 
 | 	LP8727_ID_NONE, | 
 | 	LP8727_ID_TA, | 
 | 	LP8727_ID_DEDICATED_CHG, | 
 | 	LP8727_ID_USB_CHG, | 
 | 	LP8727_ID_USB_DS, | 
 | 	LP8727_ID_MAX, | 
 | }; | 
 |  | 
 | enum lp8727_die_temp { | 
 | 	LP8788_TEMP_75C, | 
 | 	LP8788_TEMP_95C, | 
 | 	LP8788_TEMP_115C, | 
 | 	LP8788_TEMP_135C, | 
 | }; | 
 |  | 
 | struct lp8727_psy { | 
 | 	struct power_supply ac; | 
 | 	struct power_supply usb; | 
 | 	struct power_supply batt; | 
 | }; | 
 |  | 
 | struct lp8727_chg { | 
 | 	struct device *dev; | 
 | 	struct i2c_client *client; | 
 | 	struct mutex xfer_lock; | 
 | 	struct lp8727_psy *psy; | 
 | 	struct lp8727_platform_data *pdata; | 
 |  | 
 | 	/* Charger Data */ | 
 | 	enum lp8727_dev_id devid; | 
 | 	struct lp8727_chg_param *chg_param; | 
 |  | 
 | 	/* Interrupt Handling */ | 
 | 	int irq; | 
 | 	struct delayed_work work; | 
 | 	unsigned long debounce_jiffies; | 
 | }; | 
 |  | 
 | static int lp8727_read_bytes(struct lp8727_chg *pchg, u8 reg, u8 *data, u8 len) | 
 | { | 
 | 	s32 ret; | 
 |  | 
 | 	mutex_lock(&pchg->xfer_lock); | 
 | 	ret = i2c_smbus_read_i2c_block_data(pchg->client, reg, len, data); | 
 | 	mutex_unlock(&pchg->xfer_lock); | 
 |  | 
 | 	return (ret != len) ? -EIO : 0; | 
 | } | 
 |  | 
 | static inline int lp8727_read_byte(struct lp8727_chg *pchg, u8 reg, u8 *data) | 
 | { | 
 | 	return lp8727_read_bytes(pchg, reg, data, 1); | 
 | } | 
 |  | 
 | static int lp8727_write_byte(struct lp8727_chg *pchg, u8 reg, u8 data) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	mutex_lock(&pchg->xfer_lock); | 
 | 	ret = i2c_smbus_write_byte_data(pchg->client, reg, data); | 
 | 	mutex_unlock(&pchg->xfer_lock); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | static bool lp8727_is_charger_attached(const char *name, int id) | 
 | { | 
 | 	if (!strcmp(name, "ac")) | 
 | 		return id == LP8727_ID_TA || id == LP8727_ID_DEDICATED_CHG; | 
 | 	else if (!strcmp(name, "usb")) | 
 | 		return id == LP8727_ID_USB_CHG; | 
 |  | 
 | 	return id >= LP8727_ID_TA && id <= LP8727_ID_USB_CHG; | 
 | } | 
 |  | 
 | static int lp8727_init_device(struct lp8727_chg *pchg) | 
 | { | 
 | 	u8 val; | 
 | 	int ret; | 
 | 	u8 intstat[LP8788_NUM_INTREGS]; | 
 |  | 
 | 	/* clear interrupts */ | 
 | 	ret = lp8727_read_bytes(pchg, LP8727_INT1, intstat, LP8788_NUM_INTREGS); | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	val = LP8727_ID200_EN | LP8727_ADC_EN | LP8727_CP_EN; | 
 | 	ret = lp8727_write_byte(pchg, LP8727_CTRL1, val); | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	val = LP8727_INT_EN | LP8727_CHGDET_EN; | 
 | 	return lp8727_write_byte(pchg, LP8727_CTRL2, val); | 
 | } | 
 |  | 
 | static int lp8727_is_dedicated_charger(struct lp8727_chg *pchg) | 
 | { | 
 | 	u8 val; | 
 |  | 
 | 	lp8727_read_byte(pchg, LP8727_STATUS1, &val); | 
 | 	return val & LP8727_DCPORT; | 
 | } | 
 |  | 
 | static int lp8727_is_usb_charger(struct lp8727_chg *pchg) | 
 | { | 
 | 	u8 val; | 
 |  | 
 | 	lp8727_read_byte(pchg, LP8727_STATUS1, &val); | 
 | 	return val & LP8727_CHPORT; | 
 | } | 
 |  | 
 | static inline void lp8727_ctrl_switch(struct lp8727_chg *pchg, u8 sw) | 
 | { | 
 | 	lp8727_write_byte(pchg, LP8727_SWCTRL, sw); | 
 | } | 
 |  | 
 | static void lp8727_id_detection(struct lp8727_chg *pchg, u8 id, int vbusin) | 
 | { | 
 | 	struct lp8727_platform_data *pdata = pchg->pdata; | 
 | 	u8 devid = LP8727_ID_NONE; | 
 | 	u8 swctrl = LP8727_SW_DM1_HiZ | LP8727_SW_DP2_HiZ; | 
 |  | 
 | 	switch (id) { | 
 | 	case 0x5: | 
 | 		devid = LP8727_ID_TA; | 
 | 		pchg->chg_param = pdata ? pdata->ac : NULL; | 
 | 		break; | 
 | 	case 0xB: | 
 | 		if (lp8727_is_dedicated_charger(pchg)) { | 
 | 			pchg->chg_param = pdata ? pdata->ac : NULL; | 
 | 			devid = LP8727_ID_DEDICATED_CHG; | 
 | 		} else if (lp8727_is_usb_charger(pchg)) { | 
 | 			pchg->chg_param = pdata ? pdata->usb : NULL; | 
 | 			devid = LP8727_ID_USB_CHG; | 
 | 			swctrl = LP8727_SW_DM1_DM | LP8727_SW_DP2_DP; | 
 | 		} else if (vbusin) { | 
 | 			devid = LP8727_ID_USB_DS; | 
 | 			swctrl = LP8727_SW_DM1_DM | LP8727_SW_DP2_DP; | 
 | 		} | 
 | 		break; | 
 | 	default: | 
 | 		devid = LP8727_ID_NONE; | 
 | 		pchg->chg_param = NULL; | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	pchg->devid = devid; | 
 | 	lp8727_ctrl_switch(pchg, swctrl); | 
 | } | 
 |  | 
 | static void lp8727_enable_chgdet(struct lp8727_chg *pchg) | 
 | { | 
 | 	u8 val; | 
 |  | 
 | 	lp8727_read_byte(pchg, LP8727_CTRL2, &val); | 
 | 	val |= LP8727_CHGDET_EN; | 
 | 	lp8727_write_byte(pchg, LP8727_CTRL2, val); | 
 | } | 
 |  | 
 | static void lp8727_delayed_func(struct work_struct *_work) | 
 | { | 
 | 	struct lp8727_chg *pchg = container_of(_work, struct lp8727_chg, | 
 | 						work.work); | 
 | 	u8 intstat[LP8788_NUM_INTREGS]; | 
 | 	u8 idno; | 
 | 	u8 vbus; | 
 |  | 
 | 	if (lp8727_read_bytes(pchg, LP8727_INT1, intstat, LP8788_NUM_INTREGS)) { | 
 | 		dev_err(pchg->dev, "can not read INT registers\n"); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	idno = intstat[0] & LP8727_IDNO; | 
 | 	vbus = intstat[0] & LP8727_VBUS; | 
 |  | 
 | 	lp8727_id_detection(pchg, idno, vbus); | 
 | 	lp8727_enable_chgdet(pchg); | 
 |  | 
 | 	power_supply_changed(&pchg->psy->ac); | 
 | 	power_supply_changed(&pchg->psy->usb); | 
 | 	power_supply_changed(&pchg->psy->batt); | 
 | } | 
 |  | 
 | static irqreturn_t lp8727_isr_func(int irq, void *ptr) | 
 | { | 
 | 	struct lp8727_chg *pchg = ptr; | 
 |  | 
 | 	schedule_delayed_work(&pchg->work, pchg->debounce_jiffies); | 
 | 	return IRQ_HANDLED; | 
 | } | 
 |  | 
 | static int lp8727_setup_irq(struct lp8727_chg *pchg) | 
 | { | 
 | 	int ret; | 
 | 	int irq = pchg->client->irq; | 
 | 	unsigned delay_msec = pchg->pdata ? pchg->pdata->debounce_msec : | 
 | 						DEFAULT_DEBOUNCE_MSEC; | 
 |  | 
 | 	INIT_DELAYED_WORK(&pchg->work, lp8727_delayed_func); | 
 |  | 
 | 	if (irq <= 0) { | 
 | 		dev_warn(pchg->dev, "invalid irq number: %d\n", irq); | 
 | 		return 0; | 
 | 	} | 
 |  | 
 | 	ret = request_threaded_irq(irq,	NULL, lp8727_isr_func, | 
 | 				IRQF_TRIGGER_FALLING | IRQF_ONESHOT, | 
 | 				"lp8727_irq", pchg); | 
 |  | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	pchg->irq = irq; | 
 | 	pchg->debounce_jiffies = msecs_to_jiffies(delay_msec); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void lp8727_release_irq(struct lp8727_chg *pchg) | 
 | { | 
 | 	cancel_delayed_work_sync(&pchg->work); | 
 |  | 
 | 	if (pchg->irq) | 
 | 		free_irq(pchg->irq, pchg); | 
 | } | 
 |  | 
 | static enum power_supply_property lp8727_charger_prop[] = { | 
 | 	POWER_SUPPLY_PROP_ONLINE, | 
 | }; | 
 |  | 
 | static enum power_supply_property lp8727_battery_prop[] = { | 
 | 	POWER_SUPPLY_PROP_STATUS, | 
 | 	POWER_SUPPLY_PROP_HEALTH, | 
 | 	POWER_SUPPLY_PROP_PRESENT, | 
 | 	POWER_SUPPLY_PROP_VOLTAGE_NOW, | 
 | 	POWER_SUPPLY_PROP_CAPACITY, | 
 | 	POWER_SUPPLY_PROP_TEMP, | 
 | }; | 
 |  | 
 | static char *battery_supplied_to[] = { | 
 | 	"main_batt", | 
 | }; | 
 |  | 
 | static int lp8727_charger_get_property(struct power_supply *psy, | 
 | 				       enum power_supply_property psp, | 
 | 				       union power_supply_propval *val) | 
 | { | 
 | 	struct lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent); | 
 |  | 
 | 	if (psp != POWER_SUPPLY_PROP_ONLINE) | 
 | 		return -EINVAL; | 
 |  | 
 | 	val->intval = lp8727_is_charger_attached(psy->name, pchg->devid); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static bool lp8727_is_high_temperature(enum lp8727_die_temp temp) | 
 | { | 
 | 	switch (temp) { | 
 | 	case LP8788_TEMP_95C: | 
 | 	case LP8788_TEMP_115C: | 
 | 	case LP8788_TEMP_135C: | 
 | 		return true; | 
 | 	default: | 
 | 		return false; | 
 | 	} | 
 | } | 
 |  | 
 | static int lp8727_battery_get_property(struct power_supply *psy, | 
 | 				       enum power_supply_property psp, | 
 | 				       union power_supply_propval *val) | 
 | { | 
 | 	struct lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent); | 
 | 	struct lp8727_platform_data *pdata = pchg->pdata; | 
 | 	enum lp8727_die_temp temp; | 
 | 	u8 read; | 
 |  | 
 | 	switch (psp) { | 
 | 	case POWER_SUPPLY_PROP_STATUS: | 
 | 		if (!lp8727_is_charger_attached(psy->name, pchg->devid)) { | 
 | 			val->intval = POWER_SUPPLY_STATUS_DISCHARGING; | 
 | 			return 0; | 
 | 		} | 
 |  | 
 | 		lp8727_read_byte(pchg, LP8727_STATUS1, &read); | 
 |  | 
 | 		val->intval = (read & LP8727_CHGSTAT) == LP8727_STAT_EOC ? | 
 | 				POWER_SUPPLY_STATUS_FULL : | 
 | 				POWER_SUPPLY_STATUS_CHARGING; | 
 | 		break; | 
 | 	case POWER_SUPPLY_PROP_HEALTH: | 
 | 		lp8727_read_byte(pchg, LP8727_STATUS2, &read); | 
 | 		temp = (read & LP8727_TEMP_STAT) >> LP8727_TEMP_SHIFT; | 
 |  | 
 | 		val->intval = lp8727_is_high_temperature(temp) ? | 
 | 			POWER_SUPPLY_HEALTH_OVERHEAT : | 
 | 			POWER_SUPPLY_HEALTH_GOOD; | 
 | 		break; | 
 | 	case POWER_SUPPLY_PROP_PRESENT: | 
 | 		if (!pdata) | 
 | 			return -EINVAL; | 
 |  | 
 | 		if (pdata->get_batt_present) | 
 | 			val->intval = pdata->get_batt_present(); | 
 | 		break; | 
 | 	case POWER_SUPPLY_PROP_VOLTAGE_NOW: | 
 | 		if (!pdata) | 
 | 			return -EINVAL; | 
 |  | 
 | 		if (pdata->get_batt_level) | 
 | 			val->intval = pdata->get_batt_level(); | 
 | 		break; | 
 | 	case POWER_SUPPLY_PROP_CAPACITY: | 
 | 		if (!pdata) | 
 | 			return -EINVAL; | 
 |  | 
 | 		if (pdata->get_batt_capacity) | 
 | 			val->intval = pdata->get_batt_capacity(); | 
 | 		break; | 
 | 	case POWER_SUPPLY_PROP_TEMP: | 
 | 		if (!pdata) | 
 | 			return -EINVAL; | 
 |  | 
 | 		if (pdata->get_batt_temp) | 
 | 			val->intval = pdata->get_batt_temp(); | 
 | 		break; | 
 | 	default: | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void lp8727_charger_changed(struct power_supply *psy) | 
 | { | 
 | 	struct lp8727_chg *pchg = dev_get_drvdata(psy->dev->parent); | 
 | 	u8 eoc_level; | 
 | 	u8 ichg; | 
 | 	u8 val; | 
 |  | 
 | 	/* skip if no charger exists */ | 
 | 	if (!lp8727_is_charger_attached(psy->name, pchg->devid)) | 
 | 		return; | 
 |  | 
 | 	/* update charging parameters */ | 
 | 	if (pchg->chg_param) { | 
 | 		eoc_level = pchg->chg_param->eoc_level; | 
 | 		ichg = pchg->chg_param->ichg; | 
 | 		val = (ichg << LP8727_ICHG_SHIFT) | eoc_level; | 
 | 		lp8727_write_byte(pchg, LP8727_CHGCTRL2, val); | 
 | 	} | 
 | } | 
 |  | 
 | static int lp8727_register_psy(struct lp8727_chg *pchg) | 
 | { | 
 | 	struct lp8727_psy *psy; | 
 |  | 
 | 	psy = devm_kzalloc(pchg->dev, sizeof(*psy), GFP_KERNEL); | 
 | 	if (!psy) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	pchg->psy = psy; | 
 |  | 
 | 	psy->ac.name = "ac"; | 
 | 	psy->ac.type = POWER_SUPPLY_TYPE_MAINS; | 
 | 	psy->ac.properties = lp8727_charger_prop; | 
 | 	psy->ac.num_properties = ARRAY_SIZE(lp8727_charger_prop); | 
 | 	psy->ac.get_property = lp8727_charger_get_property; | 
 | 	psy->ac.supplied_to = battery_supplied_to; | 
 | 	psy->ac.num_supplicants = ARRAY_SIZE(battery_supplied_to); | 
 |  | 
 | 	if (power_supply_register(pchg->dev, &psy->ac)) | 
 | 		goto err_psy_ac; | 
 |  | 
 | 	psy->usb.name = "usb"; | 
 | 	psy->usb.type = POWER_SUPPLY_TYPE_USB; | 
 | 	psy->usb.properties = lp8727_charger_prop; | 
 | 	psy->usb.num_properties = ARRAY_SIZE(lp8727_charger_prop); | 
 | 	psy->usb.get_property = lp8727_charger_get_property; | 
 | 	psy->usb.supplied_to = battery_supplied_to; | 
 | 	psy->usb.num_supplicants = ARRAY_SIZE(battery_supplied_to); | 
 |  | 
 | 	if (power_supply_register(pchg->dev, &psy->usb)) | 
 | 		goto err_psy_usb; | 
 |  | 
 | 	psy->batt.name = "main_batt"; | 
 | 	psy->batt.type = POWER_SUPPLY_TYPE_BATTERY; | 
 | 	psy->batt.properties = lp8727_battery_prop; | 
 | 	psy->batt.num_properties = ARRAY_SIZE(lp8727_battery_prop); | 
 | 	psy->batt.get_property = lp8727_battery_get_property; | 
 | 	psy->batt.external_power_changed = lp8727_charger_changed; | 
 |  | 
 | 	if (power_supply_register(pchg->dev, &psy->batt)) | 
 | 		goto err_psy_batt; | 
 |  | 
 | 	return 0; | 
 |  | 
 | err_psy_batt: | 
 | 	power_supply_unregister(&psy->usb); | 
 | err_psy_usb: | 
 | 	power_supply_unregister(&psy->ac); | 
 | err_psy_ac: | 
 | 	return -EPERM; | 
 | } | 
 |  | 
 | static void lp8727_unregister_psy(struct lp8727_chg *pchg) | 
 | { | 
 | 	struct lp8727_psy *psy = pchg->psy; | 
 |  | 
 | 	if (!psy) | 
 | 		return; | 
 |  | 
 | 	power_supply_unregister(&psy->ac); | 
 | 	power_supply_unregister(&psy->usb); | 
 | 	power_supply_unregister(&psy->batt); | 
 | } | 
 |  | 
 | #ifdef CONFIG_OF | 
 | static struct lp8727_chg_param | 
 | *lp8727_parse_charge_pdata(struct device *dev, struct device_node *np) | 
 | { | 
 | 	struct lp8727_chg_param *param; | 
 |  | 
 | 	param = devm_kzalloc(dev, sizeof(*param), GFP_KERNEL); | 
 | 	if (!param) | 
 | 		goto out; | 
 |  | 
 | 	of_property_read_u8(np, "eoc-level", (u8 *)¶m->eoc_level); | 
 | 	of_property_read_u8(np, "charging-current", (u8 *)¶m->ichg); | 
 | out: | 
 | 	return param; | 
 | } | 
 |  | 
 | static int lp8727_parse_dt(struct device *dev) | 
 | { | 
 | 	struct device_node *np = dev->of_node; | 
 | 	struct device_node *child; | 
 | 	struct lp8727_platform_data *pdata; | 
 | 	const char *type; | 
 |  | 
 | 	/* If charging parameter is not defined, just skip parsing the dt */ | 
 | 	if (of_get_child_count(np) == 0) | 
 | 		goto out; | 
 |  | 
 | 	pdata = devm_kzalloc(dev, sizeof(*pdata), GFP_KERNEL); | 
 | 	if (!pdata) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	of_property_read_u32(np, "debounce-ms", &pdata->debounce_msec); | 
 |  | 
 | 	for_each_child_of_node(np, child) { | 
 | 		of_property_read_string(child, "charger-type", &type); | 
 |  | 
 | 		if (!strcmp(type, "ac")) | 
 | 			pdata->ac = lp8727_parse_charge_pdata(dev, child); | 
 |  | 
 | 		if (!strcmp(type, "usb")) | 
 | 			pdata->usb = lp8727_parse_charge_pdata(dev, child); | 
 | 	} | 
 |  | 
 | 	dev->platform_data = pdata; | 
 | out: | 
 | 	return 0; | 
 | } | 
 | #else | 
 | static int lp8727_parse_dt(struct device *dev) | 
 | { | 
 | 	return 0; | 
 | } | 
 | #endif | 
 |  | 
 | static int lp8727_probe(struct i2c_client *cl, const struct i2c_device_id *id) | 
 | { | 
 | 	struct lp8727_chg *pchg; | 
 | 	int ret; | 
 |  | 
 | 	if (!i2c_check_functionality(cl->adapter, I2C_FUNC_SMBUS_I2C_BLOCK)) | 
 | 		return -EIO; | 
 |  | 
 | 	if (cl->dev.of_node) { | 
 | 		ret = lp8727_parse_dt(&cl->dev); | 
 | 		if (ret) | 
 | 			return ret; | 
 | 	} | 
 |  | 
 | 	pchg = devm_kzalloc(&cl->dev, sizeof(*pchg), GFP_KERNEL); | 
 | 	if (!pchg) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	pchg->client = cl; | 
 | 	pchg->dev = &cl->dev; | 
 | 	pchg->pdata = cl->dev.platform_data; | 
 | 	i2c_set_clientdata(cl, pchg); | 
 |  | 
 | 	mutex_init(&pchg->xfer_lock); | 
 |  | 
 | 	ret = lp8727_init_device(pchg); | 
 | 	if (ret) { | 
 | 		dev_err(pchg->dev, "i2c communication err: %d", ret); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	ret = lp8727_register_psy(pchg); | 
 | 	if (ret) { | 
 | 		dev_err(pchg->dev, "power supplies register err: %d", ret); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	ret = lp8727_setup_irq(pchg); | 
 | 	if (ret) { | 
 | 		dev_err(pchg->dev, "irq handler err: %d", ret); | 
 | 		lp8727_unregister_psy(pchg); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int lp8727_remove(struct i2c_client *cl) | 
 | { | 
 | 	struct lp8727_chg *pchg = i2c_get_clientdata(cl); | 
 |  | 
 | 	lp8727_release_irq(pchg); | 
 | 	lp8727_unregister_psy(pchg); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct of_device_id lp8727_dt_ids[] = { | 
 | 	{ .compatible = "ti,lp8727", }, | 
 | 	{ } | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, lp8727_dt_ids); | 
 |  | 
 | static const struct i2c_device_id lp8727_ids[] = { | 
 | 	{"lp8727", 0}, | 
 | 	{ } | 
 | }; | 
 | MODULE_DEVICE_TABLE(i2c, lp8727_ids); | 
 |  | 
 | static struct i2c_driver lp8727_driver = { | 
 | 	.driver = { | 
 | 		   .name = "lp8727", | 
 | 		   .of_match_table = of_match_ptr(lp8727_dt_ids), | 
 | 		   }, | 
 | 	.probe = lp8727_probe, | 
 | 	.remove = lp8727_remove, | 
 | 	.id_table = lp8727_ids, | 
 | }; | 
 | module_i2c_driver(lp8727_driver); | 
 |  | 
 | MODULE_DESCRIPTION("TI/National Semiconductor LP8727 charger driver"); | 
 | MODULE_AUTHOR("Milo Kim <milo.kim@ti.com>, Daniel Jeong <daniel.jeong@ti.com>"); | 
 | MODULE_LICENSE("GPL"); |