Initial Contribution

msm-2.6.38: tag AU_LINUX_ANDROID_GINGERBREAD.02.03.04.00.142

Signed-off-by: Bryan Huntsman <bryanh@codeaurora.org>
diff --git a/drivers/power/smb137b.c b/drivers/power/smb137b.c
new file mode 100644
index 0000000..7ff8e28
--- /dev/null
+++ b/drivers/power/smb137b.c
@@ -0,0 +1,857 @@
+/* Copyright (c) 2010-2011 Code Aurora Forum. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ */
+
+#include <linux/i2c.h>
+#include <linux/gpio.h>
+#include <linux/errno.h>
+#include <linux/delay.h>
+#include <linux/module.h>
+#include <linux/debugfs.h>
+#include <linux/workqueue.h>
+#include <linux/interrupt.h>
+#include <linux/slab.h>
+#include <linux/i2c/smb137b.h>
+#include <linux/power_supply.h>
+#include <linux/msm-charger.h>
+
+#define SMB137B_MASK(BITS, POS)  ((unsigned char)(((1 << BITS) - 1) << POS))
+
+#define CHG_CURRENT_REG		0x00
+#define FAST_CHG_CURRENT_MASK		SMB137B_MASK(3, 5)
+#define PRE_CHG_CURRENT_MASK		SMB137B_MASK(2, 3)
+#define TERM_CHG_CURRENT_MASK		SMB137B_MASK(2, 1)
+
+#define INPUT_CURRENT_LIMIT_REG	0x01
+#define IN_CURRENT_MASK			SMB137B_MASK(3, 5)
+#define IN_CURRENT_LIMIT_EN_BIT		BIT(2)
+#define IN_CURRENT_DET_THRESH_MASK	SMB137B_MASK(2, 0)
+
+#define FLOAT_VOLTAGE_REG	0x02
+#define STAT_OUT_POLARITY_BIT		BIT(7)
+#define FLOAT_VOLTAGE_MASK		SMB137B_MASK(7, 0)
+
+#define CONTROL_A_REG		0x03
+#define AUTO_RECHARGE_DIS_BIT		BIT(7)
+#define CURR_CYCLE_TERM_BIT		BIT(6)
+#define PRE_TO_FAST_V_MASK		SMB137B_MASK(3, 3)
+#define TEMP_BEHAV_BIT			BIT(2)
+#define THERM_NTC_CURR_MODE_BIT		BIT(1)
+#define THERM_NTC_47KOHM_BIT		BIT(0)
+
+#define CONTROL_B_REG		0x04
+#define STAT_OUTPUT_MODE_MASK		SMB137B_MASK(2, 6)
+#define BATT_OV_ENDS_CYCLE_BIT		BIT(5)
+#define AUTO_PRE_TO_FAST_DIS_BIT	BIT(4)
+#define SAFETY_TIMER_EN_BIT		BIT(3)
+#define OTG_LBR_WD_EN_BIT		BIT(2)
+#define CHG_WD_TIMER_EN_BIT		BIT(1)
+#define IRQ_OP_MASK			BIT(0)
+
+#define PIN_CTRL_REG		0x05
+#define AUTO_CHG_EN_BIT			BIT(7)
+#define AUTO_LBR_EN_BIT			BIT(6)
+#define OTG_LBR_BIT			BIT(5)
+#define I2C_PIN_BIT			BIT(4)
+#define PIN_EN_CTRL_MASK		SMB137B_MASK(2, 2)
+#define OTG_LBR_PIN_CTRL_MASK		SMB137B_MASK(2, 0)
+
+#define OTG_LBR_CTRL_REG	0x06
+#define BATT_MISSING_DET_EN_BIT		BIT(7)
+#define AUTO_RECHARGE_THRESH_MASK	BIT(6)
+#define USB_DP_DN_DET_EN_MASK		BIT(5)
+#define OTG_LBR_BATT_CURRENT_LIMIT_MASK	SMB137B_MASK(2, 3)
+#define OTG_LBR_UVLO_THRESH_MASK	SMB137B_MASK(3, 0)
+
+#define FAULT_INTR_REG		0x07
+#define SAFETY_TIMER_EXP_MASK		SMB137B_MASK(1, 7)
+#define BATT_TEMP_UNSAFE_MASK		SMB137B_MASK(1, 6)
+#define INPUT_OVLO_IVLO_MASK		SMB137B_MASK(1, 5)
+#define BATT_OVLO_MASK			SMB137B_MASK(1, 4)
+#define INTERNAL_OVER_TEMP_MASK		SMB137B_MASK(1, 2)
+#define ENTER_TAPER_CHG_MASK		SMB137B_MASK(1, 1)
+#define CHG_MASK			SMB137B_MASK(1, 0)
+
+#define CELL_TEMP_MON_REG	0x08
+#define THERMISTOR_CURR_MASK		SMB137B_MASK(2, 6)
+#define LOW_TEMP_CHG_INHIBIT_MASK	SMB137B_MASK(3, 3)
+#define HIGH_TEMP_CHG_INHIBIT_MASK	SMB137B_MASK(3, 0)
+
+#define	SAFETY_TIMER_THERMAL_SHUTDOWN_REG	0x09
+#define DCIN_OVLO_SEL_MASK		SMB137B_MASK(2, 7)
+#define RELOAD_EN_INPUT_VOLTAGE_MASK	SMB137B_MASK(1, 6)
+#define THERM_SHUTDN_EN_MASK		SMB137B_MASK(1, 5)
+#define STANDBY_WD_TIMER_EN_MASK		SMB137B_MASK(1, 4)
+#define COMPLETE_CHG_TMOUT_MASK		SMB137B_MASK(2, 2)
+#define PRE_CHG_TMOUT_MASK		SMB137B_MASK(2, 0)
+
+#define	VSYS_REG	0x0A
+#define	VSYS_MASK			SMB137B_MASK(3, 4)
+
+#define IRQ_RESET_REG	0x30
+
+#define COMMAND_A_REG	0x31
+#define	VOLATILE_REGS_WRITE_PERM_BIT	BIT(7)
+#define	POR_BIT				BIT(6)
+#define	FAST_CHG_SETTINGS_BIT		BIT(5)
+#define	BATT_CHG_EN_BIT			BIT(4)
+#define	USBIN_MODE_500_BIT		BIT(3)
+#define	USBIN_MODE_HCMODE_BIT		BIT(2)
+#define	OTG_LBR_EN_BIT			BIT(1)
+#define	STAT_OE_BIT			BIT(0)
+
+#define STATUS_A_REG	0x32
+#define INTERNAL_TEMP_IRQ_STAT		BIT(4)
+#define DCIN_OV_IRQ_STAT		BIT(3)
+#define DCIN_UV_IRQ_STAT		BIT(2)
+#define USBIN_OV_IRQ_STAT		BIT(1)
+#define USBIN_UV_IRQ_STAT		BIT(0)
+
+#define STATUS_B_REG	0x33
+#define USB_PIN_STAT			BIT(7)
+#define USB51_MODE_STAT			BIT(6)
+#define USB51_HC_MODE_STAT		BIT(5)
+#define INTERNAL_TEMP_LIMIT_B_STAT	BIT(4)
+#define DC_IN_OV_STAT			BIT(3)
+#define DC_IN_UV_STAT			BIT(2)
+#define USB_IN_OV_STAT			BIT(1)
+#define USB_IN_UV_STAT			BIT(0)
+
+#define	STATUS_C_REG	0x34
+#define AUTO_IN_CURR_LIMIT_MASK		SMB137B_MASK(4, 4)
+#define AUTO_IN_CURR_LIMIT_STAT		BIT(3)
+#define AUTO_SOURCE_DET_COMP_STAT_MASK	SMB137B_MASK(2, 1)
+#define AUTO_SOURCE_DET_RESULT_STAT	BIT(0)
+
+#define	STATUS_D_REG	0x35
+#define	VBATT_LESS_THAN_VSYS_STAT	BIT(7)
+#define	USB_FAIL_STAT			BIT(6)
+#define	BATT_TEMP_STAT_MASK		SMB137B_MASK(2, 4)
+#define	INTERNAL_TEMP_LIMIT_STAT	BIT(2)
+#define	OTG_LBR_MODE_EN_STAT		BIT(1)
+#define	OTG_LBR_VBATT_UVLO_STAT		BIT(0)
+
+#define	STATUS_E_REG	0x36
+#define	CHARGE_CYCLE_COUNT_STAT		BIT(7)
+#define	CHARGER_TERM_STAT		BIT(6)
+#define	SAFETY_TIMER_STAT_MASK		SMB137B_MASK(2, 4)
+#define	CHARGER_ERROR_STAT		BIT(3)
+#define	CHARGING_STAT_E			SMB137B_MASK(2, 1)
+#define	CHARGING_EN			BIT(0)
+
+#define	STATUS_F_REG	0x37
+#define	WD_IRQ_ACTIVE_STAT		BIT(7)
+#define	OTG_OVERCURRENT_STAT		BIT(6)
+#define	BATT_PRESENT_STAT		BIT(4)
+#define	BATT_OV_LATCHED_STAT		BIT(3)
+#define	CHARGER_OVLO_STAT		BIT(2)
+#define	CHARGER_UVLO_STAT		BIT(1)
+#define	BATT_LOW_STAT			BIT(0)
+
+#define	STATUS_G_REG	0x38
+#define	CHARGE_TIMEOUT_IRQ_STAT		BIT(7)
+#define	PRECHARGE_TIMEOUT_IRQ_STAT	BIT(6)
+#define	BATT_HOT_IRQ_STAT		BIT(5)
+#define	BATT_COLD_IRQ_STAT		BIT(4)
+#define	BATT_OV_IRQ_STAT		BIT(3)
+#define	TAPER_CHG_IRQ_STAT		BIT(2)
+#define	FAST_CHG_IRQ_STAT		BIT(1)
+#define	CHARGING_IRQ_STAT		BIT(0)
+
+#define	STATUS_H_REG	0x39
+#define	CHARGE_TIMEOUT_STAT		BIT(7)
+#define	PRECHARGE_TIMEOUT_STAT		BIT(6)
+#define	BATT_HOT_STAT			BIT(5)
+#define	BATT_COLD_STAT			BIT(4)
+#define	BATT_OV_STAT			BIT(3)
+#define	TAPER_CHG_STAT			BIT(2)
+#define	FAST_CHG_STAT			BIT(1)
+#define	CHARGING_STAT_H			BIT(0)
+
+#define DEV_ID_REG	0x3B
+
+#define COMMAND_B_REG	0x3C
+#define	THERM_NTC_CURR_VERRIDE		BIT(7)
+
+#define SMB137B_CHG_PERIOD	((HZ) * 150)
+
+#define INPUT_CURRENT_REG_DEFAULT	0xE1
+#define INPUT_CURRENT_REG_MIN		0x01
+#define	COMMAND_A_REG_DEFAULT		0xA0
+#define	COMMAND_A_REG_OTG_MODE		0xA2
+
+#define	PIN_CTRL_REG_DEFAULT		0x00
+#define	PIN_CTRL_REG_CHG_OFF		0x04
+
+#define	FAST_CHG_E_STATUS 0x2
+
+#define SMB137B_DEFAULT_BATT_RATING   950
+struct smb137b_data {
+	struct i2c_client *client;
+	struct delayed_work charge_work;
+
+	bool charging;
+	int chgcurrent;
+	int cur_charging_mode;
+	int max_system_voltage;
+	int min_system_voltage;
+
+	int valid_n_gpio;
+
+	int batt_status;
+	int batt_chg_type;
+	int batt_present;
+	int min_design;
+	int max_design;
+	int batt_mah_rating;
+
+	int usb_status;
+
+	u8 dev_id_reg;
+	struct msm_hardware_charger adapter_hw_chg;
+};
+
+static unsigned int disabled;
+static DEFINE_MUTEX(init_lock);
+static unsigned int init_otg_power;
+
+enum charger_stat {
+	SMB137B_ABSENT,
+	SMB137B_PRESENT,
+	SMB137B_ENUMERATED,
+};
+
+static struct smb137b_data *usb_smb137b_chg;
+
+static int smb137b_read_reg(struct i2c_client *client, int reg,
+	u8 *val)
+{
+	s32 ret;
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = i2c_get_clientdata(client);
+	ret = i2c_smbus_read_byte_data(smb137b_chg->client, reg);
+	if (ret < 0) {
+		dev_err(&smb137b_chg->client->dev,
+			"i2c read fail: can't read from %02x: %d\n", reg, ret);
+		return ret;
+	} else
+		*val = ret;
+
+	return 0;
+}
+
+static int smb137b_write_reg(struct i2c_client *client, int reg,
+	u8 val)
+{
+	s32 ret;
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = i2c_get_clientdata(client);
+	ret = i2c_smbus_write_byte_data(smb137b_chg->client, reg, val);
+	if (ret < 0) {
+		dev_err(&smb137b_chg->client->dev,
+			"i2c write fail: can't write %02x to %02x: %d\n",
+			val, reg, ret);
+		return ret;
+	}
+	return 0;
+}
+
+static ssize_t id_reg_show(struct device *dev, struct device_attribute *attr,
+		char *buf)
+{
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = i2c_get_clientdata(to_i2c_client(dev));
+
+	return sprintf(buf, "%02x\n", smb137b_chg->dev_id_reg);
+}
+static DEVICE_ATTR(id_reg, S_IRUGO | S_IWUSR, id_reg_show, NULL);
+
+#ifdef DEBUG
+static void smb137b_dbg_print_status_regs(struct smb137b_data *smb137b_chg)
+{
+	int ret;
+	u8 temp;
+
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_A_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s A=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_B_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s B=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_C_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s C=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_D_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s D=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_E_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s E=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_F_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s F=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_G_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s G=0x%x\n", __func__, temp);
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_H_REG, &temp);
+	dev_dbg(&smb137b_chg->client->dev, "%s H=0x%x\n", __func__, temp);
+}
+#else
+static void smb137b_dbg_print_status_regs(struct smb137b_data *smb137b_chg)
+{
+}
+#endif
+
+static int smb137b_start_charging(struct msm_hardware_charger *hw_chg,
+		int chg_voltage, int chg_current)
+{
+	int cmd_val = COMMAND_A_REG_DEFAULT;
+	u8 temp = 0;
+	int ret = 0;
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = container_of(hw_chg, struct smb137b_data, adapter_hw_chg);
+	if (disabled) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s called when disabled\n", __func__);
+		goto out;
+	}
+
+	if (smb137b_chg->charging == true
+		&& smb137b_chg->chgcurrent == chg_current)
+		/* we are already charging with the same current*/
+		 dev_err(&smb137b_chg->client->dev,
+			 "%s charge with same current %d called again\n",
+			  __func__, chg_current);
+
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+	if (chg_current < 500)
+		cmd_val &= ~USBIN_MODE_500_BIT;
+	else if (chg_current == 500)
+		cmd_val |= USBIN_MODE_500_BIT;
+	else
+		cmd_val |= USBIN_MODE_HCMODE_BIT;
+
+	smb137b_chg->chgcurrent = chg_current;
+	smb137b_chg->cur_charging_mode = cmd_val;
+
+	/* Due to non-volatile reload feature,always enable volatile
+	 * mirror writes before modifying any 00h~09h control register.
+	 * Current mode needs to be programmed according to what's detected
+	 * Otherwise default 100mA mode might cause VOUTL drop and fail
+	 * the system in case of dead battery.
+	 */
+	ret = smb137b_write_reg(smb137b_chg->client,
+					COMMAND_A_REG, cmd_val);
+	if (ret) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't write to command_reg\n", __func__);
+		goto out;
+	}
+	ret = smb137b_write_reg(smb137b_chg->client,
+					PIN_CTRL_REG, PIN_CTRL_REG_DEFAULT);
+	if (ret) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't write to pin ctrl reg\n", __func__);
+		goto out;
+	}
+	smb137b_chg->charging = true;
+	smb137b_chg->batt_status = POWER_SUPPLY_STATUS_CHARGING;
+	smb137b_chg->batt_chg_type = POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
+
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_E_REG, &temp);
+	if (ret) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't read status e reg %d\n", __func__, ret);
+	} else {
+		if (temp & CHARGER_ERROR_STAT) {
+			dev_err(&smb137b_chg->client->dev,
+				"%s chg error E=0x%x\n", __func__, temp);
+			smb137b_dbg_print_status_regs(smb137b_chg);
+		}
+		if (((temp & CHARGING_STAT_E) >> 1) >= FAST_CHG_E_STATUS)
+			smb137b_chg->batt_chg_type
+						= POWER_SUPPLY_CHARGE_TYPE_FAST;
+	}
+	/*schedule charge_work to keep track of battery charging state*/
+	schedule_delayed_work(&smb137b_chg->charge_work, SMB137B_CHG_PERIOD);
+
+out:
+	return ret;
+}
+
+static int smb137b_stop_charging(struct msm_hardware_charger *hw_chg)
+{
+	int ret = 0;
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = container_of(hw_chg, struct smb137b_data, adapter_hw_chg);
+
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+	if (smb137b_chg->charging == false)
+		return 0;
+
+	smb137b_chg->charging = false;
+	smb137b_chg->batt_status = POWER_SUPPLY_STATUS_DISCHARGING;
+	smb137b_chg->batt_chg_type = POWER_SUPPLY_CHARGE_TYPE_NONE;
+
+	ret = smb137b_write_reg(smb137b_chg->client, COMMAND_A_REG,
+				smb137b_chg->cur_charging_mode);
+	if (ret) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't write to command_reg\n", __func__);
+		goto out;
+	}
+
+	ret = smb137b_write_reg(smb137b_chg->client,
+					PIN_CTRL_REG, PIN_CTRL_REG_CHG_OFF);
+	if (ret)
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't write to pin ctrl reg\n", __func__);
+
+out:
+	return ret;
+}
+
+static int smb137b_charger_switch(struct msm_hardware_charger *hw_chg)
+{
+	struct smb137b_data *smb137b_chg;
+
+	smb137b_chg = container_of(hw_chg, struct smb137b_data, adapter_hw_chg);
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+	return 0;
+}
+
+static irqreturn_t smb137b_valid_handler(int irq, void *dev_id)
+{
+	int val;
+	struct smb137b_data *smb137b_chg;
+	struct i2c_client *client = dev_id;
+
+	smb137b_chg = i2c_get_clientdata(client);
+
+	pr_debug("%s Cable Detected USB inserted\n", __func__);
+	/*extra delay needed to allow CABLE_DET_N settling down and debounce
+	 before	trying to sample its correct value*/
+	usleep_range(1000, 1200);
+	val = gpio_get_value_cansleep(smb137b_chg->valid_n_gpio);
+	if (val < 0) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s gpio_get_value failed for %d ret=%d\n", __func__,
+			smb137b_chg->valid_n_gpio, val);
+		goto err;
+	}
+	dev_dbg(&smb137b_chg->client->dev, "%s val=%d\n", __func__, val);
+
+	if (val) {
+		if (smb137b_chg->usb_status != SMB137B_ABSENT) {
+			smb137b_chg->usb_status = SMB137B_ABSENT;
+			msm_charger_notify_event(&smb137b_chg->adapter_hw_chg,
+					CHG_REMOVED_EVENT);
+		}
+	} else {
+		if (smb137b_chg->usb_status == SMB137B_ABSENT) {
+			smb137b_chg->usb_status = SMB137B_PRESENT;
+			msm_charger_notify_event(&smb137b_chg->adapter_hw_chg,
+					CHG_INSERTED_EVENT);
+		}
+	}
+err:
+	return IRQ_HANDLED;
+}
+
+#ifdef CONFIG_DEBUG_FS
+static struct dentry *dent;
+static int debug_fs_otg;
+static int otg_get(void *data, u64 *value)
+{
+	*value = debug_fs_otg;
+	return 0;
+}
+static int otg_set(void *data, u64 value)
+{
+	smb137b_otg_power(debug_fs_otg);
+	return 0;
+}
+DEFINE_SIMPLE_ATTRIBUTE(smb137b_otg_fops, otg_get, otg_set, "%llu\n");
+
+static void smb137b_create_debugfs_entries(struct smb137b_data *smb137b_chg)
+{
+	dent = debugfs_create_dir("smb137b", NULL);
+	if (dent) {
+		debugfs_create_file("otg", 0644, dent, NULL, &smb137b_otg_fops);
+	}
+}
+static void smb137b_destroy_debugfs_entries(void)
+{
+	if (dent)
+		debugfs_remove_recursive(dent);
+}
+#else
+static void smb137b_create_debugfs_entries(struct smb137b_data *smb137b_chg)
+{
+}
+static void smb137b_destroy_debugfs_entries(void)
+{
+}
+#endif
+
+static int set_disable_status_param(const char *val, struct kernel_param *kp)
+{
+	int ret;
+
+	ret = param_set_int(val, kp);
+	if (ret)
+		return ret;
+
+	if (usb_smb137b_chg && disabled)
+		msm_charger_notify_event(&usb_smb137b_chg->adapter_hw_chg,
+				CHG_DONE_EVENT);
+
+	pr_debug("%s disabled =%d\n", __func__, disabled);
+	return 0;
+}
+module_param_call(disabled, set_disable_status_param, param_get_uint,
+					&disabled, 0644);
+static void smb137b_charge_sm(struct work_struct *smb137b_work)
+{
+	int ret;
+	struct smb137b_data *smb137b_chg;
+	u8 temp = 0;
+	int notify_batt_changed = 0;
+
+	smb137b_chg = container_of(smb137b_work, struct smb137b_data,
+			charge_work.work);
+
+	/*if not charging, exit smb137b charging state transition*/
+	if (!smb137b_chg->charging)
+		return;
+
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+
+	ret = smb137b_read_reg(smb137b_chg->client, STATUS_F_REG, &temp);
+	if (ret) {
+		dev_err(&smb137b_chg->client->dev,
+			"%s couldn't read status f reg %d\n", __func__, ret);
+		goto out;
+	}
+	if (smb137b_chg->batt_present != !(temp & BATT_PRESENT_STAT)) {
+		smb137b_chg->batt_present = !(temp & BATT_PRESENT_STAT);
+		notify_batt_changed = 1;
+	}
+
+	if (!smb137b_chg->batt_present)
+		smb137b_chg->batt_chg_type = POWER_SUPPLY_CHARGE_TYPE_NONE;
+
+	if (!smb137b_chg->batt_present && smb137b_chg->charging)
+		msm_charger_notify_event(&smb137b_chg->adapter_hw_chg,
+				CHG_DONE_EVENT);
+
+	if (smb137b_chg->batt_present
+		&& smb137b_chg->charging
+		&& smb137b_chg->batt_chg_type
+			!= POWER_SUPPLY_CHARGE_TYPE_FAST) {
+		ret = smb137b_read_reg(smb137b_chg->client,
+						STATUS_E_REG, &temp);
+		if (ret) {
+			dev_err(&smb137b_chg->client->dev,
+				"%s couldn't read cntrl reg\n", __func__);
+			goto out;
+
+		} else {
+			if (temp & CHARGER_ERROR_STAT) {
+				dev_err(&smb137b_chg->client->dev,
+					"%s error E=0x%x\n", __func__, temp);
+				smb137b_dbg_print_status_regs(smb137b_chg);
+			}
+			if (((temp & CHARGING_STAT_E) >> 1)
+					>= FAST_CHG_E_STATUS) {
+				smb137b_chg->batt_chg_type
+					= POWER_SUPPLY_CHARGE_TYPE_FAST;
+				notify_batt_changed = 1;
+				msm_charger_notify_event(
+					&smb137b_chg->adapter_hw_chg,
+					CHG_BATT_BEGIN_FAST_CHARGING);
+			} else {
+				smb137b_chg->batt_chg_type
+					= POWER_SUPPLY_CHARGE_TYPE_TRICKLE;
+			}
+		}
+	}
+
+out:
+	schedule_delayed_work(&smb137b_chg->charge_work,
+					SMB137B_CHG_PERIOD);
+}
+
+static void __smb137b_otg_power(int on)
+{
+	int ret;
+
+	if (on) {
+		ret = smb137b_write_reg(usb_smb137b_chg->client,
+					PIN_CTRL_REG, PIN_CTRL_REG_CHG_OFF);
+		if (ret) {
+			pr_err("%s turning off charging in pin_ctrl err=%d\n",
+								__func__, ret);
+			/*
+			 * don't change the command register if charging in
+			 * pin control cannot be turned off
+			 */
+			return;
+		}
+
+		ret = smb137b_write_reg(usb_smb137b_chg->client,
+			COMMAND_A_REG, COMMAND_A_REG_OTG_MODE);
+		if (ret)
+			pr_err("%s failed turning on OTG mode ret=%d\n",
+								__func__, ret);
+	} else {
+		ret = smb137b_write_reg(usb_smb137b_chg->client,
+			COMMAND_A_REG, COMMAND_A_REG_DEFAULT);
+		if (ret)
+			pr_err("%s failed turning off OTG mode ret=%d\n",
+								__func__, ret);
+		ret = smb137b_write_reg(usb_smb137b_chg->client,
+				PIN_CTRL_REG, PIN_CTRL_REG_DEFAULT);
+		if (ret)
+			pr_err("%s failed writing to pn_ctrl ret=%d\n",
+								__func__, ret);
+	}
+}
+static int __devinit smb137b_probe(struct i2c_client *client,
+				    const struct i2c_device_id *id)
+{
+	const struct smb137b_platform_data *pdata;
+	struct smb137b_data *smb137b_chg;
+	int ret = 0;
+
+	pdata = client->dev.platform_data;
+
+	if (pdata == NULL) {
+		dev_err(&client->dev, "%s no platform data\n", __func__);
+		ret = -EINVAL;
+		goto out;
+	}
+
+	if (!i2c_check_functionality(client->adapter,
+				I2C_FUNC_SMBUS_BYTE_DATA)) {
+		ret = -EIO;
+		goto out;
+	}
+
+	smb137b_chg = kzalloc(sizeof(*smb137b_chg), GFP_KERNEL);
+	if (!smb137b_chg) {
+		ret = -ENOMEM;
+		goto out;
+	}
+
+	INIT_DELAYED_WORK(&smb137b_chg->charge_work, smb137b_charge_sm);
+	smb137b_chg->client = client;
+	smb137b_chg->valid_n_gpio = pdata->valid_n_gpio;
+	smb137b_chg->batt_mah_rating = pdata->batt_mah_rating;
+	if (smb137b_chg->batt_mah_rating == 0)
+		smb137b_chg->batt_mah_rating = SMB137B_DEFAULT_BATT_RATING;
+
+	/*It supports USB-WALL charger and PC USB charger */
+	smb137b_chg->adapter_hw_chg.type = CHG_TYPE_USB;
+	smb137b_chg->adapter_hw_chg.rating = pdata->batt_mah_rating;
+	smb137b_chg->adapter_hw_chg.name = "smb137b-charger";
+	smb137b_chg->adapter_hw_chg.start_charging = smb137b_start_charging;
+	smb137b_chg->adapter_hw_chg.stop_charging = smb137b_stop_charging;
+	smb137b_chg->adapter_hw_chg.charging_switched = smb137b_charger_switch;
+
+	if (pdata->chg_detection_config)
+		ret = pdata->chg_detection_config();
+	if (ret) {
+		dev_err(&client->dev, "%s valid config failed ret=%d\n",
+			__func__, ret);
+		goto free_smb137b_chg;
+	}
+
+	ret = gpio_request(pdata->valid_n_gpio, "smb137b_charger_valid");
+	if (ret) {
+		dev_err(&client->dev, "%s gpio_request failed for %d ret=%d\n",
+			__func__, pdata->valid_n_gpio, ret);
+		goto free_smb137b_chg;
+	}
+
+	i2c_set_clientdata(client, smb137b_chg);
+
+	ret = msm_charger_register(&smb137b_chg->adapter_hw_chg);
+	if (ret) {
+		dev_err(&client->dev, "%s msm_charger_register\
+			failed for ret=%d\n", __func__, ret);
+		goto free_valid_gpio;
+	}
+
+	ret = irq_set_irq_wake(client->irq, 1);
+	if (ret) {
+		dev_err(&client->dev, "%s failed for irq_set_irq_wake %d ret =%d\n",
+			 __func__, client->irq, ret);
+		goto unregister_charger;
+	}
+
+	ret = request_threaded_irq(client->irq, NULL,
+				   smb137b_valid_handler,
+				   IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING,
+				   "smb137b_charger_valid", client);
+	if (ret) {
+		dev_err(&client->dev,
+			"%s request_threaded_irq failed for %d ret =%d\n",
+			__func__, client->irq, ret);
+		goto disable_irq_wake;
+	}
+
+	ret = gpio_get_value_cansleep(smb137b_chg->valid_n_gpio);
+	if (ret < 0) {
+		dev_err(&client->dev,
+			"%s gpio_get_value failed for %d ret=%d\n", __func__,
+			pdata->valid_n_gpio, ret);
+		/* assume absent */
+		ret = 1;
+	}
+	if (!ret) {
+		msm_charger_notify_event(&smb137b_chg->adapter_hw_chg,
+			CHG_INSERTED_EVENT);
+		smb137b_chg->usb_status = SMB137B_PRESENT;
+	}
+
+	ret = smb137b_read_reg(smb137b_chg->client, DEV_ID_REG,
+			&smb137b_chg->dev_id_reg);
+
+	ret = device_create_file(&smb137b_chg->client->dev, &dev_attr_id_reg);
+
+	/* TODO read min_design and max_design from chip registers */
+	smb137b_chg->min_design = 3200;
+	smb137b_chg->max_design = 4200;
+
+	smb137b_chg->batt_status = POWER_SUPPLY_STATUS_DISCHARGING;
+	smb137b_chg->batt_chg_type = POWER_SUPPLY_CHARGE_TYPE_NONE;
+
+	device_init_wakeup(&client->dev, 1);
+
+	mutex_lock(&init_lock);
+	usb_smb137b_chg = smb137b_chg;
+	if (init_otg_power)
+		__smb137b_otg_power(init_otg_power);
+	mutex_unlock(&init_lock);
+
+	smb137b_create_debugfs_entries(smb137b_chg);
+	dev_dbg(&client->dev,
+		"%s OK device_id = %x chg_state=%d\n", __func__,
+		smb137b_chg->dev_id_reg, smb137b_chg->usb_status);
+	return 0;
+
+disable_irq_wake:
+	irq_set_irq_wake(client->irq, 0);
+unregister_charger:
+	msm_charger_unregister(&smb137b_chg->adapter_hw_chg);
+free_valid_gpio:
+	gpio_free(pdata->valid_n_gpio);
+free_smb137b_chg:
+	kfree(smb137b_chg);
+out:
+	return ret;
+}
+
+void smb137b_otg_power(int on)
+{
+	pr_debug("%s Enter on=%d\n", __func__, on);
+
+	mutex_lock(&init_lock);
+	if (!usb_smb137b_chg) {
+		init_otg_power = !!on;
+		pr_warning("%s called when not initialized\n", __func__);
+		mutex_unlock(&init_lock);
+		return;
+	}
+	__smb137b_otg_power(on);
+	mutex_unlock(&init_lock);
+}
+
+static int __devexit smb137b_remove(struct i2c_client *client)
+{
+	const struct smb137b_platform_data *pdata;
+	struct smb137b_data *smb137b_chg = i2c_get_clientdata(client);
+
+	pdata = client->dev.platform_data;
+	device_init_wakeup(&client->dev, 0);
+	irq_set_irq_wake(client->irq, 0);
+	free_irq(client->irq, client);
+	gpio_free(pdata->valid_n_gpio);
+	cancel_delayed_work_sync(&smb137b_chg->charge_work);
+
+	msm_charger_notify_event(&smb137b_chg->adapter_hw_chg,
+			 CHG_REMOVED_EVENT);
+	msm_charger_unregister(&smb137b_chg->adapter_hw_chg);
+	smb137b_destroy_debugfs_entries();
+	kfree(smb137b_chg);
+	return 0;
+}
+
+#ifdef CONFIG_PM
+static int smb137b_suspend(struct device *dev)
+{
+	struct smb137b_data *smb137b_chg = dev_get_drvdata(dev);
+
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+	if (smb137b_chg->charging)
+		return -EBUSY;
+	return 0;
+}
+
+static int smb137b_resume(struct device *dev)
+{
+	struct smb137b_data *smb137b_chg = dev_get_drvdata(dev);
+
+	dev_dbg(&smb137b_chg->client->dev, "%s\n", __func__);
+	return 0;
+}
+
+static const struct dev_pm_ops smb137b_pm_ops = {
+	.suspend = smb137b_suspend,
+	.resume = smb137b_resume,
+};
+#endif
+
+static const struct i2c_device_id smb137b_id[] = {
+	{"smb137b", 0},
+	{},
+};
+MODULE_DEVICE_TABLE(i2c, smb137b_id);
+
+static struct i2c_driver smb137b_driver = {
+	.driver = {
+		   .name = "smb137b",
+		   .owner = THIS_MODULE,
+#ifdef CONFIG_PM
+		   .pm = &smb137b_pm_ops,
+#endif
+	},
+	.probe = smb137b_probe,
+	.remove = __devexit_p(smb137b_remove),
+	.id_table = smb137b_id,
+};
+
+static int __init smb137b_init(void)
+{
+	return i2c_add_driver(&smb137b_driver);
+}
+module_init(smb137b_init);
+
+static void __exit smb137b_exit(void)
+{
+	return i2c_del_driver(&smb137b_driver);
+}
+module_exit(smb137b_exit);
+
+MODULE_AUTHOR("Abhijeet Dharmapurikar <adharmap@codeaurora.org>");
+MODULE_DESCRIPTION("Driver for SMB137B Charger chip");
+MODULE_LICENSE("GPL v2");
+MODULE_ALIAS("i2c:smb137b");