aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/max77693-muic.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/misc/max77693-muic.c')
-rw-r--r--drivers/misc/max77693-muic.c2520
1 files changed, 2520 insertions, 0 deletions
diff --git a/drivers/misc/max77693-muic.c b/drivers/misc/max77693-muic.c
new file mode 100644
index 00000000000..a902597fde2
--- /dev/null
+++ b/drivers/misc/max77693-muic.c
@@ -0,0 +1,2520 @@
+/*
+ * max77693-muic.c - MUIC driver for the Maxim 77693
+ *
+ * Copyright (C) 2012 Samsung Electronics
+ * <sukdong.kim@samsung.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ */
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/i2c.h>
+#include <linux/gpio.h>
+#include <plat/gpio-cfg.h>
+#include <linux/slab.h>
+#include <linux/interrupt.h>
+#include <linux/err.h>
+#include <linux/platform_device.h>
+#include <linux/workqueue.h>
+#include <linux/input.h>
+#include <linux/mfd/max77693.h>
+#include <linux/mfd/max77693-private.h>
+#include <linux/host_notify.h>
+#include <plat/udc-hs.h>
+#ifdef CONFIG_USBHUB_USB3803
+#include <linux/usb3803.h>
+#endif
+#include <linux/delay.h>
+#include <linux/extcon.h>
+
+#define DEV_NAME "max77693-muic"
+
+/* for providing API */
+static struct max77693_muic_info *gInfo;
+
+/* For restore charger interrupt states */
+static u8 chg_int_state;
+
+#ifdef CONFIG_LTE_VIA_SWITCH
+/* For saving uart path during CP booting */
+static int cpbooting;
+#endif
+
+/* MAX77693 MUIC CHG_TYP setting values */
+enum {
+ /* No Valid voltage at VB (Vvb < Vvbdet) */
+ CHGTYP_NO_VOLTAGE = 0x00,
+ /* Unknown (D+/D- does not present a valid USB charger signature) */
+ CHGTYP_USB = 0x01,
+ /* Charging Downstream Port */
+ CHGTYP_DOWNSTREAM_PORT = 0x02,
+ /* Dedicated Charger (D+/D- shorted) */
+ CHGTYP_DEDICATED_CHGR = 0x03,
+ /* Special 500mA charger, max current 500mA */
+ CHGTYP_500MA = 0x04,
+ /* Special 1A charger, max current 1A */
+ CHGTYP_1A = 0x05,
+ /* Reserved for Future Use */
+ CHGTYP_RFU = 0x06,
+ /* Dead Battery Charging, max current 100mA */
+ CHGTYP_DB_100MA = 0x07,
+ CHGTYP_MAX,
+
+ CHGTYP_INIT,
+ CHGTYP_MIN = CHGTYP_NO_VOLTAGE
+};
+
+enum {
+ ADC_GND = 0x00,
+ ADC_MHL = 0x01,
+ ADC_DOCK_PREV_KEY = 0x04,
+ ADC_DOCK_NEXT_KEY = 0x07,
+ ADC_DOCK_VOL_DN = 0x0a, /* 0x01010 14.46K ohm */
+ ADC_DOCK_VOL_UP = 0x0b, /* 0x01011 17.26K ohm */
+ ADC_DOCK_PLAY_PAUSE_KEY = 0x0d,
+ ADC_SMARTDOCK = 0x10, /* 0x10000 40.2K ohm */
+ ADC_CEA936ATYPE1_CHG = 0x17, /* 0x10111 200K ohm */
+ ADC_JIG_USB_OFF = 0x18, /* 0x11000 255K ohm */
+ ADC_JIG_USB_ON = 0x19, /* 0x11001 301K ohm */
+ ADC_DESKDOCK = 0x1a, /* 0x11010 365K ohm */
+ ADC_CEA936ATYPE2_CHG = 0x1b, /* 0x11011 442K ohm */
+ ADC_JIG_UART_OFF = 0x1c, /* 0x11100 523K ohm */
+ ADC_JIG_UART_ON = 0x1d, /* 0x11101 619K ohm */
+ ADC_CARDOCK = 0x1d, /* 0x11101 619K ohm */
+ ADC_OPEN = 0x1f
+};
+
+enum {
+ DOCK_KEY_NONE = 0,
+ DOCK_KEY_VOL_UP_PRESSED,
+ DOCK_KEY_VOL_UP_RELEASED,
+ DOCK_KEY_VOL_DOWN_PRESSED,
+ DOCK_KEY_VOL_DOWN_RELEASED,
+ DOCK_KEY_PREV_PRESSED,
+ DOCK_KEY_PREV_RELEASED,
+ DOCK_KEY_PLAY_PAUSE_PRESSED,
+ DOCK_KEY_PLAY_PAUSE_RELEASED,
+ DOCK_KEY_NEXT_PRESSED,
+ DOCK_KEY_NEXT_RELEASED,
+};
+
+struct max77693_muic_info {
+ struct device *dev;
+ struct max77693_dev *max77693;
+ struct i2c_client *muic;
+ struct max77693_muic_data *muic_data;
+ int irq_adc;
+ int irq_chgtype;
+ int irq_vbvolt;
+ int irq_adc1k;
+ int mansw;
+ bool is_default_uart_path_cp;
+
+ enum cable_type_muic cable_type;
+ struct delayed_work init_work;
+ struct delayed_work usb_work;
+ struct delayed_work mhl_work;
+ struct mutex mutex;
+
+ bool is_usb_ready;
+ bool is_mhl_ready;
+
+ struct input_dev *input;
+ int previous_key;
+ bool is_adc_open_prev;
+#ifdef CONFIG_EXTCON
+ struct extcon_dev *edev;
+#endif
+};
+
+static int if_muic_info;
+static int switch_sel;
+static int if_pmic_rev;
+
+/* func : get_if_pmic_inifo
+ * switch_sel value get from bootloader comand line
+ * switch_sel data consist 8 bits (xxxxzzzz)
+ * first 4bits(zzzz) mean path infomation.
+ * next 4bits(xxxx) mean if pmic version info
+ */
+static int get_if_pmic_inifo(char *str)
+{
+ get_option(&str, &if_muic_info);
+ switch_sel = if_muic_info & 0x0f;
+ if_pmic_rev = (if_muic_info & 0xf0) >> 4;
+ pr_info("%s %s: switch_sel: %x if_pmic_rev:%x\n",
+ __FILE__, __func__, switch_sel, if_pmic_rev);
+ return if_muic_info;
+}
+__setup("pmic_info=", get_if_pmic_inifo);
+
+int get_switch_sel(void)
+{
+ return switch_sel;
+}
+
+static int max77693_muic_get_comp2_comn1_pass2
+ (struct max77693_muic_info *info)
+{
+ int ret;
+ u8 val;
+
+ ret = max77693_read_reg(info->muic, MAX77693_MUIC_REG_CTRL1, &val);
+ val = val & CLEAR_IDBEN_MICEN_MASK;
+ dev_info(info->dev, "func:%s ret:%d val:%x\n", __func__, ret, val);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg\n", __func__);
+ return -EINVAL;
+ }
+ return val;
+}
+
+static int max77693_muic_set_comp2_comn1_pass2
+ (struct max77693_muic_info *info, int type, int path)
+{
+ /* type 1 == usb, type 2 == uart */
+ u8 cntl1_val, cntl1_msk;
+ int ret = 0;
+ int val;
+
+ dev_info(info->dev, "func: %s type: %d path: %d\n",
+ __func__, type, path);
+ if (type == 0) {
+ if (path == AP_USB_MODE) {
+ info->muic_data->sw_path = AP_USB_MODE;
+ val = MAX77693_MUIC_CTRL1_BIN_1_001;
+ } else if (path == CP_USB_MODE) {
+ info->muic_data->sw_path = CP_USB_MODE;
+ val = MAX77693_MUIC_CTRL1_BIN_4_100;
+ } else {
+ dev_err(info->dev, "func: %s invalid usb path\n"
+ , __func__);
+ return -EINVAL;
+ }
+ } else if (type == 1) {
+ if (path == UART_PATH_AP) {
+ info->muic_data->sw_path = UART_PATH_AP;
+ if (info->is_default_uart_path_cp)
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ } else if (path == UART_PATH_CP) {
+ info->muic_data->sw_path = UART_PATH_CP;
+ if (info->is_default_uart_path_cp)
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+#ifdef CONFIG_LTE_VIA_SWITCH
+ dev_info(info->dev, "%s: cpbooting is %s\n",
+ __func__,
+ cpbooting ?
+ "started. skip path set" : "done. set path");
+ if (!cpbooting) {
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL,
+ GPIO_LEVEL_HIGH);
+ dev_info(info->dev,
+ "%s: LTE_GPIO_LEVEL_HIGH"
+ , __func__);
+ } else {
+ dev_err(info->dev,
+ "%s: ERR_LTE_GPIO_SET_HIGH\n"
+ , __func__);
+ return -EINVAL;
+ }
+ }
+#endif
+ }
+#ifdef CONFIG_LTE_VIA_SWITCH
+ else if (path == UART_PATH_LTE) {
+ info->muic_data->sw_path = UART_PATH_LTE;
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL,
+ GPIO_LEVEL_LOW);
+ dev_info(info->dev, "%s: LTE_GPIO_LEVEL_LOW\n"
+ , __func__);
+ } else {
+ dev_err(info->dev, "%s: ERR_LTE_GPIO_SET_LOW\n"
+ , __func__);
+ return -EINVAL;
+ }
+ }
+#endif
+ else {
+ dev_err(info->dev, "func: %s invalid uart path\n"
+ , __func__);
+ return -EINVAL;
+ }
+ } else {
+ dev_err(info->dev, "func: %s invalid path type(%d)\n"
+ , __func__, type);
+ return -EINVAL;
+ }
+
+ cntl1_val = (val << COMN1SW_SHIFT) | (val << COMP2SW_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK;
+
+ max77693_update_reg(info->muic, MAX77693_MUIC_REG_CTRL1, cntl1_val,
+ cntl1_msk);
+
+ return ret;
+}
+
+static int max77693_muic_set_usb_path_pass2
+ (struct max77693_muic_info *info, int path)
+{
+ int ret = 0;
+ ret = max77693_muic_set_comp2_comn1_pass2
+ (info, 0/*usb*/, path);
+ sysfs_notify(&switch_dev->kobj, NULL, "usb_sel");
+ return ret;
+}
+
+static int max77693_muic_get_usb_path_pass2
+ (struct max77693_muic_info *info)
+{
+ u8 val;
+
+ val = max77693_muic_get_comp2_comn1_pass2(info);
+ if (val == CTRL1_AP_USB)
+ return AP_USB_MODE;
+ else if (val == CTRL1_CP_USB)
+ return CP_USB_MODE;
+ else if (val == CTRL1_AUDIO)
+ return AUDIO_MODE;
+ else
+ return -EINVAL;
+}
+
+static int max77693_muic_set_uart_path_pass2
+ (struct max77693_muic_info *info, int path)
+{
+ int ret = 0;
+ ret = max77693_muic_set_comp2_comn1_pass2
+ (info, 1/*uart*/, path);
+ return ret;
+
+}
+
+static int max77693_muic_get_uart_path_pass2
+ (struct max77693_muic_info *info)
+{
+ u8 val;
+
+ val = max77693_muic_get_comp2_comn1_pass2(info);
+
+ if (val == CTRL1_AP_UART) {
+ if (info->is_default_uart_path_cp)
+ return UART_PATH_CP;
+ else
+ return UART_PATH_AP;
+ } else if (val == CTRL1_CP_UART) {
+#ifdef CONFIG_LTE_VIA_SWITCH
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ if (gpio_get_value(GPIO_LTE_VIA_UART_SEL))
+ return UART_PATH_CP;
+ else
+ return UART_PATH_LTE;
+ } else {
+ dev_info(info->dev, "%s: ERR_UART_PATH_LTE\n"
+ , __func__);
+ return -EINVAL;
+ }
+#endif
+#ifndef CONFIG_LTE_VIA_SWITCH
+ if (info->is_default_uart_path_cp)
+ return UART_PATH_AP;
+ else
+ return UART_PATH_CP;
+#endif
+ } else {
+ return -EINVAL;
+ }
+}
+
+static ssize_t max77693_muic_show_usb_state(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ dev_info(info->dev, "func:%s info->cable_type:%d\n",
+ __func__, info->cable_type);
+ switch (info->cable_type) {
+ case CABLE_TYPE_USB_MUIC:
+ case CABLE_TYPE_JIG_USB_OFF_MUIC:
+ case CABLE_TYPE_JIG_USB_ON_MUIC:
+ return sprintf(buf, "USB_STATE_CONFIGURED\n");
+ default:
+ break;
+ }
+
+ return sprintf(buf, "USB_STATE_NOTCONFIGURED\n");
+}
+
+static ssize_t max77693_muic_show_device(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ dev_info(info->dev, "func:%s info->cable_type:%d\n",
+ __func__, info->cable_type);
+
+ switch (info->cable_type) {
+ case CABLE_TYPE_NONE_MUIC:
+ return sprintf(buf, "No cable\n");
+ case CABLE_TYPE_USB_MUIC:
+ return sprintf(buf, "USB\n");
+ case CABLE_TYPE_OTG_MUIC:
+ return sprintf(buf, "OTG\n");
+ case CABLE_TYPE_TA_MUIC:
+ return sprintf(buf, "TA\n");
+ case CABLE_TYPE_DESKDOCK_MUIC:
+ return sprintf(buf, "Desk Dock\n");
+ case CABLE_TYPE_CARDOCK_MUIC:
+ return sprintf(buf, "Car Dock\n");
+ case CABLE_TYPE_JIG_UART_OFF_MUIC:
+ return sprintf(buf, "JIG UART OFF\n");
+ case CABLE_TYPE_JIG_UART_OFF_VB_MUIC:
+ return sprintf(buf, "JIG UART OFF/VB\n");
+ case CABLE_TYPE_JIG_UART_ON_MUIC:
+ return sprintf(buf, "JIG UART ON\n");
+ case CABLE_TYPE_JIG_USB_OFF_MUIC:
+ return sprintf(buf, "JIG USB OFF\n");
+ case CABLE_TYPE_JIG_USB_ON_MUIC:
+ return sprintf(buf, "JIG USB ON\n");
+ case CABLE_TYPE_MHL_MUIC:
+ return sprintf(buf, "mHL\n");
+ case CABLE_TYPE_MHL_VB_MUIC:
+ return sprintf(buf, "mHL charging\n");
+ default:
+ break;
+ }
+
+ return sprintf(buf, "UNKNOWN\n");
+}
+
+static ssize_t max77693_muic_show_manualsw(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+
+ dev_info(info->dev, "func:%s ap(0),cp(1),vps(2)sw_path:%d(%d)\n",
+ __func__, info->muic_data->sw_path,
+ gpio_get_value(GPIO_USB_SEL));/*For debuging*/
+
+ switch (info->muic_data->sw_path) {
+ case AP_USB_MODE:
+ return sprintf(buf, "PDA\n");
+ case CP_USB_MODE:
+ return sprintf(buf, "MODEM\n");
+ case AUDIO_MODE:
+ return sprintf(buf, "Audio\n");
+ default:
+ break;
+ }
+
+ return sprintf(buf, "UNKNOWN\n");
+}
+
+static ssize_t max77693_muic_set_manualsw(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+
+ dev_info(info->dev, "func:%s buf:%s,count:%d\n", __func__, buf, count);
+
+ if (!strncasecmp(buf, "PDA", 3)) {
+ info->muic_data->sw_path = AP_USB_MODE;
+ dev_info(info->dev, "%s: AP_USB_MODE\n", __func__);
+ } else if (!strncasecmp(buf, "MODEM", 5)) {
+ info->muic_data->sw_path = CP_USB_MODE;
+ dev_info(info->dev, "%s: CP_USB_MODE\n", __func__);
+ } else
+ dev_warn(info->dev, "%s: Wrong command\n", __func__);
+
+ return count;
+}
+
+static ssize_t max77693_muic_show_adc(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int ret;
+ u8 val;
+
+ ret = max77693_read_reg(info->muic, MAX77693_MUIC_REG_STATUS1, &val);
+ dev_info(info->dev, "func:%s ret:%d val:%x\n", __func__, ret, val);
+
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg\n", __func__);
+ return sprintf(buf, "UNKNOWN\n");
+ }
+
+ return sprintf(buf, "%x\n", (val & STATUS1_ADC_MASK));
+}
+
+static ssize_t max77693_muic_show_audio_path(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int ret;
+ u8 val;
+
+ ret = max77693_read_reg(info->muic, MAX77693_MUIC_REG_CTRL1, &val);
+ dev_info(info->dev, "func:%s ret:%d val:%x\n", __func__, ret, val);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg\n", __func__);
+ return sprintf(buf, "UNKNOWN\n");
+ }
+
+ return sprintf(buf, "%x\n", val);
+}
+
+static ssize_t max77693_muic_set_audio_path(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ struct i2c_client *client = info->muic;
+ u8 cntl1_val, cntl1_msk;
+ u8 val;
+ dev_info(info->dev, "func:%s buf:%s\n", __func__, buf);
+ if (!strncmp(buf, "0", 1))
+ val = MAX77693_MUIC_CTRL1_BIN_0_000;
+ else if (!strncmp(buf, "1", 1))
+ val = MAX77693_MUIC_CTRL1_BIN_2_010;
+ else {
+ dev_warn(info->dev, "%s: Wrong command\n", __func__);
+ return count;
+ }
+
+ cntl1_val = (val << COMN1SW_SHIFT) | (val << COMP2SW_SHIFT) |
+ (0 << MICEN_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK | MICEN_MASK;
+
+ max77693_update_reg(client, MAX77693_MUIC_REG_CTRL1, cntl1_val,
+ cntl1_msk);
+ dev_info(info->dev, "MUIC cntl1_val:%x, cntl1_msk:%x\n", cntl1_val,
+ cntl1_msk);
+
+ cntl1_val = MAX77693_MUIC_CTRL1_BIN_0_000;
+ max77693_read_reg(client, MAX77693_MUIC_REG_CTRL1, &cntl1_val);
+ dev_info(info->dev, "%s: CNTL1(0x%02x)\n", __func__, cntl1_val);
+
+ return count;
+}
+
+static ssize_t max77693_muic_show_otg_test(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int ret;
+ u8 val;
+
+ ret = max77693_read_reg(info->muic, MAX77693_MUIC_REG_CDETCTRL1, &val);
+ dev_info(info->dev, "func:%s ret:%d val:%x buf%s\n",
+ __func__, ret, val, buf);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg\n", __func__);
+ return sprintf(buf, "UNKNOWN\n");
+ }
+ val &= CHGDETEN_MASK;
+
+ return sprintf(buf, "%x\n", val);
+}
+
+static ssize_t max77693_muic_set_otg_test(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ struct i2c_client *client = info->muic;
+ u8 val;
+
+ dev_info(info->dev, "func:%s buf:%s\n", __func__, buf);
+ if (!strncmp(buf, "0", 1))
+ val = 0;
+ else if (!strncmp(buf, "1", 1))
+ val = 1;
+ else {
+ dev_warn(info->dev, "%s: Wrong command\n", __func__);
+ return count;
+ }
+
+ max77693_update_reg(client, MAX77693_MUIC_REG_CDETCTRL1,
+ val << CHGDETEN_SHIFT, CHGDETEN_MASK);
+
+ val = 0;
+ max77693_read_reg(client, MAX77693_MUIC_REG_CDETCTRL1, &val);
+ dev_info(info->dev, "%s: CDETCTRL(0x%02x)\n", __func__, val);
+
+ return count;
+}
+
+static void max77693_muic_set_adcdbset(struct max77693_muic_info *info,
+ int value)
+{
+ int ret;
+ u8 val;
+ dev_info(info->dev, "func:%s value:%x\n", __func__, value);
+ if (value > 3) {
+ dev_err(info->dev, "%s: invalid value(%x)\n", __func__, value);
+ return;
+ }
+
+ if (!info->muic) {
+ dev_err(info->dev, "%s: no muic i2c client\n", __func__);
+ return;
+ }
+
+ val = value << CTRL3_ADCDBSET_SHIFT;
+ dev_info(info->dev, "%s: ADCDBSET(0x%02x)\n", __func__, val);
+ ret = max77693_update_reg(info->muic, MAX77693_MUIC_REG_CTRL3, val,
+ CTRL3_ADCDBSET_MASK);
+ if (ret < 0)
+ dev_err(info->dev, "%s: fail to update reg\n", __func__);
+}
+
+static ssize_t max77693_muic_show_adc_debounce_time(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int ret;
+ u8 val;
+ dev_info(info->dev, "func:%s buf:%s\n", __func__, buf);
+
+ if (!info->muic)
+ return sprintf(buf, "No I2C client\n");
+
+ ret = max77693_read_reg(info->muic, MAX77693_MUIC_REG_CTRL3, &val);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg\n", __func__);
+ return sprintf(buf, "UNKNOWN\n");
+ }
+ val &= CTRL3_ADCDBSET_MASK;
+ val = val >> CTRL3_ADCDBSET_SHIFT;
+ dev_info(info->dev, "func:%s val:%x\n", __func__, val);
+ return sprintf(buf, "%x\n", val);
+}
+
+static ssize_t max77693_muic_set_adc_debounce_time(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int value;
+
+ sscanf(buf, "%d", &value);
+ value = (value & 0x3);
+
+ dev_info(info->dev, "%s: Do nothing\n", __func__);
+
+ return count;
+}
+
+static ssize_t max77693_muic_set_uart_sel(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+
+ if (info->max77693->pmic_rev < MAX77693_REV_PASS2) {
+ if (!strncasecmp(buf, "AP", 2)) {
+ info->muic_data->uart_path = UART_PATH_AP;
+ if (gpio_is_valid(GPIO_UART_SEL)) {
+ gpio_set_value(GPIO_UART_SEL, GPIO_LEVEL_HIGH);
+ dev_info(info->dev, "%s: UART_PATH_AP\n"
+ , __func__);
+ } else {
+ dev_err(info->dev, "%s: Change(AP) fail!!"
+ , __func__);
+ }
+ } else if (!strncasecmp(buf, "CP", 2)) {
+ info->muic_data->uart_path = UART_PATH_CP;
+ if (gpio_is_valid(GPIO_UART_SEL)
+#ifdef CONFIG_LTE_VIA_SWITCH
+ && gpio_is_valid(GPIO_LTE_VIA_UART_SEL)
+#endif
+ ) {
+ gpio_set_value(GPIO_UART_SEL, GPIO_LEVEL_LOW);
+#ifdef CONFIG_LTE_VIA_SWITCH
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL
+ , GPIO_LEVEL_HIGH);
+#endif
+ dev_info(info->dev, "%s: UART_PATH_CP\n"
+ , __func__);
+ } else {
+ dev_err(info->dev, "%s: Change(CP) fail!!"
+ , __func__);
+ }
+ }
+#ifdef CONFIG_LTE_VIA_SWITCH
+ else if (!strncasecmp(buf, "LTE", 3)) {
+ info->muic_data->uart_path = UART_PATH_LTE;
+ if (gpio_is_valid(GPIO_UART_SEL) &&
+ gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ gpio_set_value(GPIO_UART_SEL, GPIO_LEVEL_LOW);
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL
+ , GPIO_LEVEL_LOW);
+ dev_info(info->dev, "%s: UART_PATH_LTE\n"
+ , __func__);
+ } else {
+ dev_err(info->dev, "%s: Change(LTE) fail!!"
+ , __func__);
+ }
+ }
+#endif
+ else
+ dev_warn(info->dev, "%s: Wrong command\n", __func__);
+ } else if (info->max77693->pmic_rev >= MAX77693_REV_PASS2) {
+ if (!strncasecmp(buf, "AP", 2)) {
+ int ret = max77693_muic_set_uart_path_pass2
+ (info, UART_PATH_AP);
+ if (ret >= 0)
+ info->muic_data->uart_path = UART_PATH_AP;
+ else
+ dev_err(info->dev, "%s: Change(AP) fail!!"
+ , __func__);
+ } else if (!strncasecmp(buf, "CP", 2)) {
+ int ret = max77693_muic_set_uart_path_pass2
+ (info, UART_PATH_CP);
+ if (ret >= 0)
+ info->muic_data->uart_path = UART_PATH_CP;
+ else
+ dev_err(info->dev, "%s: Change(CP) fail!!"
+ , __func__);
+ }
+#ifdef CONFIG_LTE_VIA_SWITCH
+ else if (!strncasecmp(buf, "LTE", 3)) {
+ int ret = max77693_muic_set_uart_path_pass2
+ (info, UART_PATH_LTE);
+ if (ret >= 0)
+ info->muic_data->uart_path = UART_PATH_LTE;
+ else
+ dev_err(info->dev, "%s: Change(LTE) fail!!"
+ , __func__);
+ }
+#endif
+ else {
+ dev_warn(info->dev, "%s: Wrong command\n"
+ , __func__);
+ }
+ }
+ return count;
+}
+
+static ssize_t max77693_muic_show_uart_sel(struct device *dev,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ if (info->max77693->pmic_rev < MAX77693_REV_PASS2) {
+ switch (info->muic_data->uart_path) {
+ case UART_PATH_AP:
+ if (gpio_get_value(GPIO_UART_SEL) == GPIO_LEVEL_HIGH)
+ return sprintf(buf, "AP\n");
+ else
+ return sprintf(buf, "ERR_AP\n");
+ break;
+ case UART_PATH_CP:
+ if (gpio_get_value(GPIO_UART_SEL) == GPIO_LEVEL_LOW
+#ifdef CONFIG_LTE_VIA_SWITCH
+ && gpio_get_value(GPIO_LTE_VIA_UART_SEL)
+ == GPIO_LEVEL_HIGH
+#endif
+ ) {
+ return sprintf(buf, "CP\n");
+ } else {
+ return sprintf(buf, "ERR_CP\n");
+ }
+ break;
+#ifdef CONFIG_LTE_VIA_SWITCH
+ case UART_PATH_LTE:
+ if (gpio_get_value(GPIO_UART_SEL) == GPIO_LEVEL_LOW &&
+ gpio_get_value(GPIO_LTE_VIA_UART_SEL)
+ == GPIO_LEVEL_LOW) {
+ return sprintf(buf, "LTE\n");
+ } else {
+ return sprintf(buf, "ERR_LTE\n");
+ }
+ break;
+#endif
+ default:
+ break;
+ }
+ } else if (info->max77693->pmic_rev >= MAX77693_REV_PASS2) {
+ int val = max77693_muic_get_uart_path_pass2(info);
+ switch (info->muic_data->uart_path) {
+ case UART_PATH_AP:
+ if (val == UART_PATH_AP)
+ return sprintf(buf, "AP\n");
+ else
+ return sprintf(buf, "ERR_AP\n");
+ break;
+ case UART_PATH_CP:
+ if (val == UART_PATH_CP)
+ return sprintf(buf, "CP\n");
+ else
+ return sprintf(buf, "ERR_CP\n");
+ break;
+#ifdef CONFIG_LTE_VIA_SWITCH
+ case UART_PATH_LTE:
+ if (val == UART_PATH_LTE)
+ return sprintf(buf, "LTE\n");
+ else
+ return sprintf(buf, "ERR_LTE\n");
+ break;
+#endif
+ default:
+ break;
+ }
+ }
+ return sprintf(buf, "UNKNOWN\n");
+}
+
+#ifdef CONFIG_LTE_VIA_SWITCH
+static ssize_t max77693_muic_show_check_cpboot(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ if (cpbooting)
+ return sprintf(buf, "start\n");
+ else
+ return sprintf(buf, "done\n");
+}
+
+static ssize_t max77693_muic_set_check_cpboot(struct device *dev,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+
+ if (!strncasecmp(buf, "start", 5)) {
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ dev_info(info->dev, "%s: Fix CP-UART path to LTE during CP Booting",
+ __func__);
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL, GPIO_LEVEL_LOW);
+ } else {
+ dev_err(info->dev, "%s: ERR_LTE_GPIO_SET_LOW\n",
+ __func__);
+ return -EINVAL;
+ }
+ cpbooting = 1;
+ } else if (!strncasecmp(buf, "done", 4)) {
+ if (info->muic_data->uart_path == UART_PATH_CP) {
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ dev_info(info->dev, "%s: Reroute CP-UART path to VIA after CP Booting",
+ __func__);
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL,
+ GPIO_LEVEL_HIGH);
+ } else {
+ dev_err(info->dev, "%s: ERR_LTE_GPIO_SET_HIGH\n",
+ __func__);
+ return -EINVAL;
+ }
+ }
+ cpbooting = 0;
+ } else {
+ dev_warn(info->dev, "%s: Wrong command : %s\n", __func__, buf);
+ }
+
+ return count;
+}
+#endif
+
+static DEVICE_ATTR(uart_sel, 0664, max77693_muic_show_uart_sel,
+ max77693_muic_set_uart_sel);
+static DEVICE_ATTR(usb_state, S_IRUGO, max77693_muic_show_usb_state, NULL);
+static DEVICE_ATTR(device, S_IRUGO, max77693_muic_show_device, NULL);
+static DEVICE_ATTR(usb_sel, 0664,
+ max77693_muic_show_manualsw, max77693_muic_set_manualsw);
+static DEVICE_ATTR(adc, S_IRUGO, max77693_muic_show_adc, NULL);
+static DEVICE_ATTR(audio_path, 0664,
+ max77693_muic_show_audio_path, max77693_muic_set_audio_path);
+static DEVICE_ATTR(otg_test, 0664,
+ max77693_muic_show_otg_test, max77693_muic_set_otg_test);
+static DEVICE_ATTR(adc_debounce_time, 0664,
+ max77693_muic_show_adc_debounce_time,
+ max77693_muic_set_adc_debounce_time);
+#ifdef CONFIG_LTE_VIA_SWITCH
+static DEVICE_ATTR(check_cpboot, 0664,
+ max77693_muic_show_check_cpboot,
+ max77693_muic_set_check_cpboot);
+#endif
+
+static struct attribute *max77693_muic_attributes[] = {
+ &dev_attr_uart_sel.attr,
+ &dev_attr_usb_state.attr,
+ &dev_attr_device.attr,
+ &dev_attr_usb_sel.attr,
+ &dev_attr_adc.attr,
+ &dev_attr_audio_path.attr,
+ &dev_attr_otg_test.attr,
+ &dev_attr_adc_debounce_time.attr,
+#ifdef CONFIG_LTE_VIA_SWITCH
+ &dev_attr_check_cpboot.attr,
+#endif
+ NULL
+};
+
+static const struct attribute_group max77693_muic_group = {
+ .attrs = max77693_muic_attributes,
+};
+
+#if defined(CONFIG_MACH_M0_CTC)
+/*0 : usb is not conneted to CP 1 : usb is connected to CP */
+int cp_usb_state;
+
+int max7693_muic_cp_usb_state(void)
+{
+ return cp_usb_state;
+}
+EXPORT_SYMBOL(max7693_muic_cp_usb_state);
+#endif
+
+static int max77693_muic_set_usb_path(struct max77693_muic_info *info, int path)
+{
+ struct i2c_client *client = info->muic;
+ struct max77693_muic_data *mdata = info->muic_data;
+ int ret;
+ int gpio_val;
+ u8 cntl1_val, cntl1_msk;
+ int val;
+ dev_info(info->dev, "func:%s path:%d\n", __func__, path);
+ if (mdata->set_safeout) {
+ ret = mdata->set_safeout(path);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to set safout!\n",
+ __func__);
+ return ret;
+ }
+ }
+ switch (path) {
+ case AP_USB_MODE:
+ dev_info(info->dev, "%s: AP_USB_MODE\n", __func__);
+ gpio_val = 0;
+ val = MAX77693_MUIC_CTRL1_BIN_1_001;
+ cntl1_val = (val << COMN1SW_SHIFT) | (val << COMP2SW_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK;
+ break;
+ case CP_USB_MODE:
+ dev_info(info->dev, "%s: CP_USB_MODE\n", __func__);
+ gpio_val = 1;
+ if (info->max77693->pmic_rev >= MAX77693_REV_PASS2)
+ val = MAX77693_MUIC_CTRL1_BIN_4_100;
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ cntl1_val = (val << COMN1SW_SHIFT) | (val << COMP2SW_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK;
+ break;
+ case AUDIO_MODE:
+ dev_info(info->dev, "%s: AUDIO_MODE\n", __func__);
+ gpio_val = 0;
+ /* SL1, SR2 */
+ cntl1_val = (MAX77693_MUIC_CTRL1_BIN_2_010 << COMN1SW_SHIFT)
+ | (MAX77693_MUIC_CTRL1_BIN_2_010 << COMP2SW_SHIFT) |
+ (0 << MICEN_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK | MICEN_MASK;
+ break;
+ default:
+ dev_warn(info->dev, "%s: invalid path(%d)\n", __func__, path);
+ return -EINVAL;
+ }
+ if (info->max77693->pmic_rev < MAX77693_REV_PASS2) {
+ if (gpio_is_valid(info->muic_data->gpio_usb_sel))
+ gpio_direction_output(mdata->gpio_usb_sel, gpio_val);
+ }
+
+ dev_info(info->dev, "%s: Set manual path\n", __func__);
+#if defined(CONFIG_SND_USE_MUIC_SWITCH)
+ if (info->cable_type != CABLE_TYPE_CARDOCK_MUIC
+ && info->cable_type != CABLE_TYPE_DESKDOCK_MUIC)
+#endif
+ max77693_update_reg(client, MAX77693_MUIC_REG_CTRL1, cntl1_val,
+ cntl1_msk);
+ max77693_update_reg(client, MAX77693_MUIC_REG_CTRL2,
+ CTRL2_CPEn1_LOWPWD0,
+ CTRL2_CPEn_MASK | CTRL2_LOWPWD_MASK);
+
+ cntl1_val = MAX77693_MUIC_CTRL1_BIN_0_000;
+ max77693_read_reg(client, MAX77693_MUIC_REG_CTRL1, &cntl1_val);
+ dev_info(info->dev, "%s: CNTL1(0x%02x)\n", __func__, cntl1_val);
+
+ cntl1_val = MAX77693_MUIC_CTRL1_BIN_0_000;
+ max77693_read_reg(client, MAX77693_MUIC_REG_CTRL2, &cntl1_val);
+ dev_info(info->dev, "%s: CNTL2(0x%02x)\n", __func__, cntl1_val);
+
+ sysfs_notify(&switch_dev->kobj, NULL, "usb_sel");
+ return 0;
+}
+
+int max77693_muic_get_charging_type(void)
+{
+ return gInfo->cable_type;
+}
+
+static int max77693_muic_set_charging_type(struct max77693_muic_info *info,
+ bool force_disable)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+ int ret = 0;
+ dev_info(info->dev, "func:%s force_disable:%d\n",
+ __func__, force_disable);
+ if (mdata->charger_cb) {
+ if (force_disable)
+ ret = mdata->charger_cb(CABLE_TYPE_NONE_MUIC);
+ else
+ ret = mdata->charger_cb(info->cable_type);
+ }
+
+ if (ret) {
+ dev_err(info->dev, "%s: error from charger_cb(%d)\n", __func__,
+ ret);
+ return ret;
+ }
+ return 0;
+}
+
+static int max77693_muic_handle_dock_vol_key(struct max77693_muic_info *info,
+ u8 status1)
+{
+ struct input_dev *input = info->input;
+ int pre_key = info->previous_key;
+ unsigned int code;
+ int state;
+ u8 adc;
+
+ adc = status1 & STATUS1_ADC_MASK;
+ dev_info(info->dev,
+ "func:%s status1:%x adc:%x cable_type:%d\n",
+ __func__, status1, adc, info->cable_type);
+ if (info->cable_type != CABLE_TYPE_DESKDOCK_MUIC)
+ return 0;
+
+ if (adc == ADC_OPEN) {
+ switch (pre_key) {
+ case DOCK_KEY_VOL_UP_PRESSED:
+ code = KEY_VOLUMEUP;
+ state = 0;
+ info->previous_key = DOCK_KEY_VOL_UP_RELEASED;
+ break;
+ case DOCK_KEY_VOL_DOWN_PRESSED:
+ code = KEY_VOLUMEDOWN;
+ state = 0;
+ info->previous_key = DOCK_KEY_VOL_DOWN_RELEASED;
+ break;
+ case DOCK_KEY_PREV_PRESSED:
+ code = KEY_PREVIOUSSONG;
+ state = 0;
+ info->previous_key = DOCK_KEY_PREV_RELEASED;
+ break;
+ case DOCK_KEY_PLAY_PAUSE_PRESSED:
+ code = KEY_PLAYPAUSE;
+ state = 0;
+ info->previous_key = DOCK_KEY_PLAY_PAUSE_RELEASED;
+ break;
+ case DOCK_KEY_NEXT_PRESSED:
+ code = KEY_NEXTSONG;
+ state = 0;
+ info->previous_key = DOCK_KEY_NEXT_RELEASED;
+ break;
+ default:
+ return 0;
+ }
+ input_event(input, EV_KEY, code, state);
+ input_sync(input);
+ return 0;
+ }
+
+ if (pre_key == DOCK_KEY_NONE) {
+ /*
+ if (adc != ADC_DOCK_VOL_UP && adc != ADC_DOCK_VOL_DN && \
+ adc != ADC_DOCK_PREV_KEY && adc != ADC_DOCK_PLAY_PAUSE_KEY \
+ && adc != ADC_DOCK_NEXT_KEY)
+ */
+ if ((adc < 0x03) || (adc > 0x0d))
+ return 0;
+ }
+
+ dev_info(info->dev, "%s: dock vol key(%d)\n", __func__, pre_key);
+
+ switch (adc) {
+ case ADC_DOCK_VOL_UP:
+ code = KEY_VOLUMEUP;
+ state = 1;
+ info->previous_key = DOCK_KEY_VOL_UP_PRESSED;
+ break;
+ case ADC_DOCK_VOL_DN:
+ code = KEY_VOLUMEDOWN;
+ state = 1;
+ info->previous_key = DOCK_KEY_VOL_DOWN_PRESSED;
+ break;
+ case ADC_DOCK_PREV_KEY-1 ... ADC_DOCK_PREV_KEY+1:
+ code = KEY_PREVIOUSSONG;
+ state = 1;
+ info->previous_key = DOCK_KEY_PREV_PRESSED;
+ break;
+ case ADC_DOCK_PLAY_PAUSE_KEY-1 ... ADC_DOCK_PLAY_PAUSE_KEY+1:
+ code = KEY_PLAYPAUSE;
+ state = 1;
+ info->previous_key = DOCK_KEY_PLAY_PAUSE_PRESSED;
+ break;
+ case ADC_DOCK_NEXT_KEY-1 ... ADC_DOCK_NEXT_KEY+1:
+ code = KEY_NEXTSONG;
+ state = 1;
+ info->previous_key = DOCK_KEY_NEXT_PRESSED;
+ break;
+ case ADC_DESKDOCK: /* key release routine */
+ if (pre_key == DOCK_KEY_VOL_UP_PRESSED) {
+ code = KEY_VOLUMEUP;
+ state = 0;
+ info->previous_key = DOCK_KEY_VOL_UP_RELEASED;
+ } else if (pre_key == DOCK_KEY_VOL_DOWN_PRESSED) {
+ code = KEY_VOLUMEDOWN;
+ state = 0;
+ info->previous_key = DOCK_KEY_VOL_DOWN_RELEASED;
+ } else if (pre_key == DOCK_KEY_PREV_PRESSED) {
+ code = KEY_PREVIOUSSONG;
+ state = 0;
+ info->previous_key = DOCK_KEY_PREV_RELEASED;
+ } else if (pre_key == DOCK_KEY_PLAY_PAUSE_PRESSED) {
+ code = KEY_PLAYPAUSE;
+ state = 0;
+ info->previous_key = DOCK_KEY_PLAY_PAUSE_RELEASED;
+ } else if (pre_key == DOCK_KEY_NEXT_PRESSED) {
+ code = KEY_NEXTSONG;
+ state = 0;
+ info->previous_key = DOCK_KEY_NEXT_RELEASED;
+ } else {
+ dev_warn(info->dev, "%s:%d should not reach here\n",
+ __func__, __LINE__);
+ return 0;
+ }
+ break;
+ default:
+ dev_warn(info->dev, "%s: unsupported ADC(0x%02x)\n", __func__,
+ adc);
+ return 0;
+ }
+
+ input_event(input, EV_KEY, code, state);
+ input_sync(input);
+
+ return 1;
+}
+
+static int max77693_muic_attach_usb_type(struct max77693_muic_info *info,
+ int adc)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+ int ret, path;
+ dev_info(info->dev, "func:%s adc:%x cable_type:%d\n",
+ __func__, adc, info->cable_type);
+ if (info->cable_type == CABLE_TYPE_MHL_MUIC
+ || info->cable_type == CABLE_TYPE_MHL_VB_MUIC) {
+ dev_warn(info->dev, "%s: mHL was attached!\n", __func__);
+ return 0;
+ }
+
+ switch (adc) {
+ case ADC_JIG_USB_OFF:
+ if (info->cable_type == CABLE_TYPE_JIG_USB_OFF_MUIC) {
+ dev_info(info->dev, "%s: duplicated(JIG USB OFF)\n",
+ __func__);
+ return 0;
+ }
+
+ dev_info(info->dev, "%s:JIG USB BOOTOFF\n", __func__);
+ info->cable_type = CABLE_TYPE_JIG_USB_OFF_MUIC;
+ path = AP_USB_MODE;
+ break;
+ case ADC_JIG_USB_ON:
+ if (info->cable_type == CABLE_TYPE_JIG_USB_ON_MUIC) {
+ dev_info(info->dev, "%s: duplicated(JIG USB ON)\n",
+ __func__);
+ return 0;
+ }
+
+ dev_info(info->dev, "%s:JIG USB BOOTON\n", __func__);
+ info->cable_type = CABLE_TYPE_JIG_USB_ON_MUIC;
+ path = AP_USB_MODE;
+ break;
+ case ADC_OPEN:
+ if (info->cable_type == CABLE_TYPE_USB_MUIC) {
+ dev_info(info->dev, "%s: duplicated(USB)\n", __func__);
+ return 0;
+ }
+
+ dev_info(info->dev, "%s:USB\n", __func__);
+ info->cable_type = CABLE_TYPE_USB_MUIC;
+ path = AP_USB_MODE;
+ break;
+ default:
+ dev_info(info->dev, "%s: Unkown cable(0x%x)\n", __func__, adc);
+ return 0;
+ }
+
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret) {
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ return ret;
+ }
+
+ if (mdata->sw_path == CP_USB_MODE) {
+ info->cable_type = CABLE_TYPE_USB_MUIC;
+#if defined(CONFIG_MACH_M0_CTC)
+ if (system_rev < 11) {
+ gpio_direction_output(GPIO_USB_BOOT_EN, 1);
+ } else if (system_rev == 11) {
+ gpio_direction_output(GPIO_USB_BOOT_EN, 1);
+ gpio_direction_output(GPIO_USB_BOOT_EN_REV06, 1);
+ } else {
+ gpio_direction_output(GPIO_USB_BOOT_EN_REV06, 1);
+ }
+ cp_usb_state = 1;
+#endif
+ max77693_muic_set_usb_path(info, CP_USB_MODE);
+ return 0;
+ }
+
+ max77693_muic_set_usb_path(info, path);
+
+ if (path == AP_USB_MODE) {
+ if (mdata->usb_cb && info->is_usb_ready)
+#ifdef CONFIG_USBHUB_USB3803
+ /* setting usb hub in Diagnostic(hub) mode */
+ usb3803_set_mode(USB_3803_MODE_HUB);
+#endif /* CONFIG_USBHUB_USB3803 */
+ mdata->usb_cb(USB_CABLE_ATTACHED);
+ }
+
+ return 0;
+}
+
+static int max77693_muic_attach_dock_type(struct max77693_muic_info *info,
+ int adc)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+ int path;
+ dev_info(info->dev, "func:%s adc:%x, open(%d)\n",
+ __func__, adc, info->is_adc_open_prev);
+ /*Workaround for unstable adc*/
+ if (info->is_adc_open_prev == false) {
+ return 0;
+ }
+ switch (adc) {
+ case ADC_DESKDOCK:
+ /* Desk Dock */
+ if (info->cable_type == CABLE_TYPE_DESKDOCK_MUIC) {
+ dev_info(info->dev, "%s: duplicated(DeskDock)\n",
+ __func__);
+ return 0;
+ }
+ dev_info(info->dev, "%s:DeskDock\n", __func__);
+ info->cable_type = CABLE_TYPE_DESKDOCK_MUIC;
+ path = AUDIO_MODE;
+
+ if (mdata->deskdock_cb)
+ mdata->deskdock_cb(MAX77693_MUIC_ATTACHED);
+ break;
+ case ADC_CARDOCK:
+ /* Car Dock */
+ if (info->cable_type == CABLE_TYPE_CARDOCK_MUIC) {
+ dev_info(info->dev, "%s: duplicated(CarDock)\n",
+ __func__);
+ return 0;
+ }
+ dev_info(info->dev, "%s:CarDock\n", __func__);
+ info->cable_type = CABLE_TYPE_CARDOCK_MUIC;
+ path = AUDIO_MODE;
+
+ if (mdata->cardock_cb)
+ mdata->cardock_cb(MAX77693_MUIC_ATTACHED);
+ break;
+ default:
+ dev_info(info->dev, "%s: should not reach here(0x%x)\n",
+ __func__, adc);
+ return 0;
+ }
+
+ max77693_muic_set_usb_path(info, path);
+
+ return 0;
+}
+
+static void max77693_muic_attach_mhl(struct max77693_muic_info *info, u8 chgtyp)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+
+ dev_info(info->dev, "func:%s chgtyp:%x\n", __func__, chgtyp);
+
+ if (info->cable_type == CABLE_TYPE_USB_MUIC) {
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_CABLE_DETACHED);
+
+ max77693_muic_set_charging_type(info, true);
+ }
+#if 0
+ if (info->cable_type == CABLE_TYPE_MHL) {
+ dev_info(info->dev, "%s: duplicated(MHL)\n", __func__);
+ return;
+ }
+#endif
+ info->cable_type = CABLE_TYPE_MHL_MUIC;
+
+#ifdef CONFIG_EXTCON
+ if (info->edev && info->is_mhl_ready)
+ extcon_set_cable_state(info->edev, "MHL", true);
+#else
+ if (mdata->mhl_cb && info->is_mhl_ready)
+ mdata->mhl_cb(MAX77693_MUIC_ATTACHED);
+#endif
+
+ if (chgtyp == CHGTYP_USB) {
+ info->cable_type = CABLE_TYPE_MHL_VB_MUIC;
+ max77693_muic_set_charging_type(info, false);
+ }
+}
+
+static void max77693_muic_handle_jig_uart(struct max77693_muic_info *info,
+ u8 vbvolt)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+ enum cable_type_muic prev_ct = info->cable_type;
+ bool is_otgtest = false;
+ u8 cntl1_val, cntl1_msk;
+ u8 val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ dev_info(info->dev, "func:%s vbvolt:%x cable_type:%d\n",
+ __func__, vbvolt, info->cable_type);
+ dev_info(info->dev, "%s: JIG UART/BOOTOFF(0x%x)\n", __func__, vbvolt);
+
+ if (info->max77693->pmic_rev >= MAX77693_REV_PASS2) {
+ if (info->muic_data->uart_path == UART_PATH_AP) {
+ if (info->is_default_uart_path_cp)
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ } else if (info->muic_data->uart_path == UART_PATH_CP) {
+ if (info->is_default_uart_path_cp)
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+#ifdef CONFIG_LTE_VIA_SWITCH
+ dev_info(info->dev, "%s: cpbooting is %s\n",
+ __func__,
+ cpbooting ?
+ "started. skip path set" : "done. set path");
+ if (!cpbooting) {
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL,
+ GPIO_LEVEL_HIGH);
+ dev_info(info->dev,
+ "%s: LTE_GPIO_LEVEL_HIGH"
+ , __func__);
+ } else {
+ dev_err(info->dev,
+ "%s: ERR_LTE_GPIO_SET_HIGH\n"
+ , __func__);
+ }
+ }
+#endif
+ }
+#ifdef CONFIG_LTE_VIA_SWITCH
+ else if (info->muic_data->uart_path == UART_PATH_LTE) {
+ val = MAX77693_MUIC_CTRL1_BIN_5_101;
+ /*TODO must modify H/W rev.5*/
+ if (gpio_is_valid(GPIO_LTE_VIA_UART_SEL)) {
+ gpio_set_value(GPIO_LTE_VIA_UART_SEL,
+ GPIO_LEVEL_LOW);
+ dev_info(info->dev, "%s: LTE_GPIO_LEVEL_LOW\n"
+ , __func__);
+ } else
+ dev_err(info->dev, "%s: ERR_LTE_GPIO_SET_LOW\n"
+ , __func__);
+ }
+#endif
+ else
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+ } else
+ val = MAX77693_MUIC_CTRL1_BIN_3_011;
+
+ cntl1_val = (val << COMN1SW_SHIFT) | (val << COMP2SW_SHIFT);
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK;
+ max77693_update_reg(info->muic, MAX77693_MUIC_REG_CTRL1, cntl1_val,
+ cntl1_msk);
+
+ max77693_update_reg(info->muic, MAX77693_MUIC_REG_CTRL2,
+ CTRL2_CPEn1_LOWPWD0,
+ CTRL2_CPEn_MASK | CTRL2_LOWPWD_MASK);
+
+ if (vbvolt & STATUS2_VBVOLT_MASK) {
+ if (mdata->host_notify_cb) {
+ if (mdata->host_notify_cb(1) == NOTIFY_TEST_MODE) {
+ is_otgtest = true;
+ dev_info(info->dev, "%s: OTG TEST\n", __func__);
+ }
+ }
+
+ info->cable_type = CABLE_TYPE_JIG_UART_OFF_VB_MUIC;
+ max77693_muic_set_charging_type(info, is_otgtest);
+
+ } else {
+ info->cable_type = CABLE_TYPE_JIG_UART_OFF_MUIC;
+#if 0
+ if (mdata->uart_path == UART_PATH_CP && mdata->jig_uart_cb)
+ mdata->jig_uart_cb(UART_PATH_CP);
+#endif
+ if (prev_ct == CABLE_TYPE_JIG_UART_OFF_VB_MUIC) {
+ max77693_muic_set_charging_type(info, false);
+
+ if (mdata->host_notify_cb)
+ mdata->host_notify_cb(0);
+ }
+ }
+}
+
+void max77693_otg_control(struct max77693_muic_info *info, int enable)
+{
+ u8 int_mask, cdetctrl1, chg_cnfg_00;
+ pr_info("%s: enable(%d)\n", __func__, enable);
+
+ if (enable) {
+ /* disable charger interrupt */
+ max77693_read_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_INT_MASK, &int_mask);
+ chg_int_state = int_mask;
+ int_mask |= (1 << 4); /* disable chgin intr */
+ int_mask |= (1 << 6); /* disable chg */
+ int_mask &= ~(1 << 0); /* enable byp intr */
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_INT_MASK, int_mask);
+
+ /* disable charger detection */
+ max77693_read_reg(info->max77693->muic,
+ MAX77693_MUIC_REG_CDETCTRL1, &cdetctrl1);
+ cdetctrl1 &= ~(1 << 0);
+ max77693_write_reg(info->max77693->muic,
+ MAX77693_MUIC_REG_CDETCTRL1, cdetctrl1);
+
+ /* OTG on, boost on, DIS_MUIC_CTRL=1 */
+ max77693_read_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, &chg_cnfg_00);
+ chg_cnfg_00 &= ~(CHG_CNFG_00_CHG_MASK
+ | CHG_CNFG_00_OTG_MASK
+ | CHG_CNFG_00_BUCK_MASK
+ | CHG_CNFG_00_BOOST_MASK
+ | CHG_CNFG_00_DIS_MUIC_CTRL_MASK);
+ chg_cnfg_00 |= (CHG_CNFG_00_OTG_MASK
+ | CHG_CNFG_00_BOOST_MASK
+ | CHG_CNFG_00_DIS_MUIC_CTRL_MASK);
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, chg_cnfg_00);
+ } else {
+ /* OTG off, boost off, (buck on),
+ DIS_MUIC_CTRL = 0 unless CHG_ENA = 1 */
+ max77693_read_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, &chg_cnfg_00);
+ chg_cnfg_00 &= ~(CHG_CNFG_00_OTG_MASK
+ | CHG_CNFG_00_BOOST_MASK
+ | CHG_CNFG_00_DIS_MUIC_CTRL_MASK);
+ chg_cnfg_00 |= CHG_CNFG_00_BUCK_MASK;
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, chg_cnfg_00);
+
+ mdelay(50);
+
+ /* enable charger detection */
+ max77693_read_reg(info->max77693->muic,
+ MAX77693_MUIC_REG_CDETCTRL1, &cdetctrl1);
+ cdetctrl1 |= (1 << 0);
+ max77693_write_reg(info->max77693->muic,
+ MAX77693_MUIC_REG_CDETCTRL1, cdetctrl1);
+
+ /* enable charger interrupt */
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_INT_MASK, chg_int_state);
+ max77693_read_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_INT_MASK, &int_mask);
+ }
+
+ pr_info("%s: INT_MASK(0x%x), CDETCTRL1(0x%x), CHG_CNFG_00(0x%x)\n",
+ __func__, int_mask, cdetctrl1, chg_cnfg_00);
+}
+
+void max77693_powered_otg_control(struct max77693_muic_info *info, int enable)
+{
+ pr_info("%s: enable(%d)\n", __func__, enable);
+
+ if (enable) {
+ /* OTG on, boost on */
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, 0x05);
+
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_02, 0x0E);
+ } else {
+ /* OTG off, boost off, (buck on) */
+ max77693_write_reg(info->max77693->i2c,
+ MAX77693_CHG_REG_CHG_CNFG_00, 0x04);
+ }
+}
+/* use in mach for otg */
+void otg_control(int enable)
+{
+ pr_debug("%s: enable(%d)\n", __func__, enable);
+
+ max77693_otg_control(gInfo, enable);
+}
+
+/* use in mach for powered-otg */
+void powered_otg_control(int enable)
+{
+ pr_debug("%s: enable(%d)\n", __func__, enable);
+
+ max77693_powered_otg_control(gInfo, enable);
+}
+
+static int max77693_muic_handle_attach(struct max77693_muic_info *info,
+ u8 status1, u8 status2, int irq)
+{
+ struct max77693_muic_data *mdata = info->muic_data;
+ u8 adc, vbvolt, chgtyp, chgdetrun, adc1k;
+ int ret = 0;
+
+ adc = status1 & STATUS1_ADC_MASK;
+ adc1k = status1 & STATUS1_ADC1K_MASK;
+ chgtyp = status2 & STATUS2_CHGTYP_MASK;
+ vbvolt = status2 & STATUS2_VBVOLT_MASK;
+ chgdetrun = status2 & STATUS2_CHGDETRUN_MASK;
+
+ dev_info(info->dev, "func:%s st1:%x st2:%x cable_type:%d\n",
+ __func__, status1, status2, info->cable_type);
+ /* Workaround for Factory mode.
+ * Abandon adc interrupt of approximately +-100K range
+ * if previous cable status was JIG UART BOOT OFF.
+ */
+ if (info->cable_type == CABLE_TYPE_JIG_UART_OFF_MUIC ||
+ info->cable_type == CABLE_TYPE_JIG_UART_OFF_VB_MUIC) {
+ if (adc == (ADC_JIG_UART_OFF + 1) ||
+ adc == (ADC_JIG_UART_OFF - 1)) {
+ /* Workaround for factory mode in MUIC PASS2
+ * In uart path cp, adc is unstable state
+ * MUIC PASS2 turn to AP_UART mode automatically
+ * So, in this state set correct path manually.
+ * !! NEEDED ONLY IF PMIC PASS2 !!
+ */
+ if (info->muic_data->uart_path == UART_PATH_CP
+ && info->max77693->pmic_rev >= MAX77693_REV_PASS2)
+ max77693_muic_handle_jig_uart(info, vbvolt);
+ dev_warn(info->dev, "%s: abandon ADC\n", __func__);
+ return 0;
+ }
+ }
+
+ if (info->cable_type == CABLE_TYPE_DESKDOCK_MUIC
+ && adc != ADC_DESKDOCK) {
+ dev_warn(info->dev, "%s: assume deskdock detach\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ max77693_muic_set_charging_type(info, false);
+ info->is_adc_open_prev = false;
+ if (mdata->deskdock_cb)
+ mdata->deskdock_cb(MAX77693_MUIC_DETACHED);
+ } else if (info->cable_type == CABLE_TYPE_CARDOCK_MUIC
+ && adc != ADC_CARDOCK) {
+ dev_warn(info->dev, "%s: assume cardock detach\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ max77693_muic_set_charging_type(info, false);
+ info->is_adc_open_prev = false;
+ if (mdata->cardock_cb)
+ mdata->cardock_cb(MAX77693_MUIC_DETACHED);
+ }
+
+ /* 1Kohm ID regiter detection (mHL)
+ * Old MUIC : ADC value:0x00 or 0x01, ADCLow:1
+ * New MUIC : ADC value is not set(Open), ADCLow:1, ADCError:1
+ */
+ if (adc1k) {
+ if (irq == info->irq_adc
+ || irq == info->irq_chgtype
+ || irq == info->irq_vbvolt) {
+ dev_warn(info->dev,
+ "%s: Ignore irq:%d at MHL detection\n",
+ __func__, irq);
+ if (vbvolt) {
+ dev_info(info->dev, "%s: call charger_cb(%d)"
+ , __func__, vbvolt);
+ max77693_muic_set_charging_type(info, false);
+ } else {
+ dev_info(info->dev, "%s: call charger_cb(%d)"
+ , __func__, vbvolt);
+ max77693_muic_set_charging_type(info, true);
+ }
+ return 0;
+ }
+ max77693_muic_attach_mhl(info, chgtyp);
+ return 0;
+ }
+
+ switch (adc) {
+ case ADC_GND:
+ if (chgtyp == CHGTYP_NO_VOLTAGE) {
+ if (info->cable_type == CABLE_TYPE_OTG_MUIC) {
+ dev_info(info->dev,
+ "%s: duplicated(OTG)\n", __func__);
+ break;
+ }
+
+ info->cable_type = CABLE_TYPE_OTG_MUIC;
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_OTGHOST_ATTACHED);
+
+ msleep(40);
+
+ max77693_muic_set_usb_path(info, AP_USB_MODE);
+ } else if (chgtyp == CHGTYP_USB ||
+ chgtyp == CHGTYP_DOWNSTREAM_PORT ||
+ chgtyp == CHGTYP_DEDICATED_CHGR ||
+ chgtyp == CHGTYP_500MA || chgtyp == CHGTYP_1A) {
+ dev_info(info->dev, "%s: OTG charging pump\n",
+ __func__);
+ ret = max77693_muic_set_charging_type(info, false);
+ }
+ break;
+ case ADC_SMARTDOCK:
+ if (info->cable_type == CABLE_TYPE_SMARTDOCK_MUIC) {
+ dev_info(info->dev,
+ "%s: duplicated(SMARTDOCK)\n", __func__);
+ break;
+ }
+ dev_info(info->dev, "func:%s Attach SmartDock\n", __func__);
+ info->cable_type = CABLE_TYPE_SMARTDOCK_MUIC;
+ max77693_muic_set_usb_path(info, AP_USB_MODE);
+ msleep(40);
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_POWERED_HOST_ATTACHED);
+#ifdef CONFIG_EXTCON
+ if (info->edev && info->is_mhl_ready)
+ extcon_set_cable_state(info->edev, "MHL", true);
+#else
+ if (mdata->mhl_cb && info->is_mhl_ready)
+ mdata->mhl_cb(MAX77693_MUIC_ATTACHED);
+#endif
+ max77693_muic_set_charging_type(info, false);
+ break;
+ case ADC_JIG_UART_OFF:
+ max77693_muic_handle_jig_uart(info, vbvolt);
+#if defined(CONFIG_SEC_MODEM_M0_TD)
+ gpio_set_value(GPIO_AP_CP_INT1, GPIO_LEVEL_HIGH);
+#endif
+ mdata->jig_state(true);
+ break;
+ case ADC_JIG_USB_OFF:
+ case ADC_JIG_USB_ON:
+ if (vbvolt & STATUS2_VBVOLT_MASK) {
+ dev_info(info->dev, "%s: SKIP_JIG_USB\n", __func__);
+ ret = max77693_muic_attach_usb_type(info, adc);
+ }
+#if defined(CONFIG_SEC_MODEM_M0_TD)
+ gpio_set_value(GPIO_AP_CP_INT1, GPIO_LEVEL_HIGH);
+#endif
+ mdata->jig_state(true);
+ break;
+ case ADC_DESKDOCK:
+ case ADC_CARDOCK:
+ max77693_muic_attach_dock_type(info, adc);
+ if (chgtyp == CHGTYP_USB ||
+ chgtyp == CHGTYP_DOWNSTREAM_PORT ||
+ chgtyp == CHGTYP_DEDICATED_CHGR ||
+ chgtyp == CHGTYP_500MA || chgtyp == CHGTYP_1A)
+ ret = max77693_muic_set_charging_type(info, false);
+ else if (chgtyp == CHGTYP_NO_VOLTAGE && !chgdetrun)
+ ret = max77693_muic_set_charging_type(info, !vbvolt);
+ /* For MAX77693 IC doesn`t occur chgtyp IRQ
+ * because of audio noise prevention.
+ * So, If below condition is set,
+ * we do charging at CARDOCK.
+ */
+ break;
+ case ADC_CEA936ATYPE1_CHG:
+ case ADC_CEA936ATYPE2_CHG:
+ case ADC_OPEN:
+ switch (chgtyp) {
+ case CHGTYP_USB:
+ case CHGTYP_DOWNSTREAM_PORT:
+ if (adc == ADC_CEA936ATYPE1_CHG
+ || adc == ADC_CEA936ATYPE2_CHG)
+ break;
+ if (info->cable_type == CABLE_TYPE_MHL_MUIC) {
+ dev_info(info->dev, "%s: MHL(charging)\n",
+ __func__);
+ info->cable_type = CABLE_TYPE_MHL_VB_MUIC;
+ ret = max77693_muic_set_charging_type(info,
+ false);
+ return ret;
+ }
+#ifdef CONFIG_EXTCON
+ if (info->edev)
+ extcon_set_cable_state(info->edev,
+ "USB", true);
+#endif
+ ret = max77693_muic_attach_usb_type(info, adc);
+ break;
+ case CHGTYP_DEDICATED_CHGR:
+ case CHGTYP_500MA:
+ case CHGTYP_1A:
+ dev_info(info->dev, "%s:TA\n", __func__);
+ info->cable_type = CABLE_TYPE_TA_MUIC;
+#ifdef CONFIG_EXTCON
+ if (info->edev)
+ extcon_set_cable_state(info->edev,
+ "TA", true);
+#endif
+#ifdef CONFIG_USBHUB_USB3803
+ /* setting usb hub in default mode (standby) */
+ usb3803_set_mode(USB_3803_MODE_STANDBY);
+#endif /* CONFIG_USBHUB_USB3803 */
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret)
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ break;
+ default:
+ break;
+ }
+ break;
+ default:
+ dev_warn(info->dev, "%s: unsupported adc=0x%x\n", __func__,
+ adc);
+ break;
+ }
+ return ret;
+}
+
+static int max77693_muic_handle_detach(struct max77693_muic_info *info, int irq)
+{
+ struct i2c_client *client = info->muic;
+ struct max77693_muic_data *mdata = info->muic_data;
+ enum cable_type_muic prev_ct = CABLE_TYPE_NONE_MUIC;
+ u8 cntl2_val;
+ int ret = 0;
+ dev_info(info->dev, "func:%s\n", __func__);
+
+ info->is_adc_open_prev = true;
+ /* Workaround: irq doesn't occur after detaching mHL cable */
+ max77693_write_reg(client, MAX77693_MUIC_REG_CTRL1,
+ MAX77693_MUIC_CTRL1_BIN_0_000);
+
+ /* Enable Factory Accessory Detection State Machine */
+ max77693_update_reg(client, MAX77693_MUIC_REG_CTRL2,
+ (1 << CTRL2_ACCDET_SHIFT), CTRL2_ACCDET_MASK);
+
+ max77693_update_reg(client, MAX77693_MUIC_REG_CTRL2,
+ CTRL2_CPEn0_LOWPWD1,
+ CTRL2_CPEn_MASK | CTRL2_LOWPWD_MASK);
+
+ max77693_read_reg(client, MAX77693_MUIC_REG_CTRL2, &cntl2_val);
+ dev_info(info->dev, "%s: CNTL2(0x%02x)\n", __func__, cntl2_val);
+
+#if defined(CONFIG_MACH_M0_CTC)
+ if (cp_usb_state != 0) {
+ if (system_rev < 11) {
+ gpio_direction_output(GPIO_USB_BOOT_EN, 0);
+ } else if (system_rev == 11) {
+ gpio_direction_output(GPIO_USB_BOOT_EN, 0);
+ gpio_direction_output(GPIO_USB_BOOT_EN_REV06, 0);
+ } else {
+ gpio_direction_output(GPIO_USB_BOOT_EN_REV06, 0);
+ }
+ cp_usb_state = 0;
+ }
+#endif
+
+#ifdef CONFIG_USBHUB_USB3803
+ /* setting usb hub in default mode (standby) */
+ usb3803_set_mode(USB_3803_MODE_STANDBY);
+#endif /* CONFIG_USBHUB_USB3803 */
+ info->previous_key = DOCK_KEY_NONE;
+
+ if (info->cable_type == CABLE_TYPE_NONE_MUIC) {
+ dev_info(info->dev, "%s: duplicated(NONE)\n", __func__);
+ return 0;
+ }
+#if 0
+ if (mdata->jig_uart_cb)
+ mdata->jig_uart_cb(UART_PATH_AP);
+#endif
+
+ switch (info->cable_type) {
+ case CABLE_TYPE_OTG_MUIC:
+ dev_info(info->dev, "%s: OTG\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_OTGHOST_DETACHED);
+ break;
+ case CABLE_TYPE_USB_MUIC:
+ case CABLE_TYPE_JIG_USB_OFF_MUIC:
+ case CABLE_TYPE_JIG_USB_ON_MUIC:
+#ifdef CONFIG_EXTCON
+ if (info->edev)
+ extcon_set_cable_state(info->edev, "USB", false);
+#endif
+ dev_info(info->dev, "%s: USB(0x%x)\n", __func__,
+ info->cable_type);
+ prev_ct = info->cable_type;
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret) {
+ info->cable_type = prev_ct;
+ break;
+ }
+
+ if (mdata->sw_path == CP_USB_MODE)
+ return 0;
+
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_CABLE_DETACHED);
+ break;
+ case CABLE_TYPE_DESKDOCK_MUIC:
+ dev_info(info->dev, "%s: DESKDOCK\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret) {
+ info->cable_type = CABLE_TYPE_DESKDOCK_MUIC;
+ break;
+ }
+ if (mdata->deskdock_cb)
+ mdata->deskdock_cb(MAX77693_MUIC_DETACHED);
+ break;
+ case CABLE_TYPE_CARDOCK_MUIC:
+ dev_info(info->dev, "%s: CARDOCK\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret) {
+ info->cable_type = CABLE_TYPE_CARDOCK_MUIC;
+ break;
+ }
+ if (mdata->cardock_cb)
+ mdata->cardock_cb(MAX77693_MUIC_DETACHED);
+ break;
+ case CABLE_TYPE_TA_MUIC:
+#ifdef CONFIG_EXTCON
+ if (info->edev)
+ extcon_set_cable_state(info->edev, "TA", false);
+#endif
+ dev_info(info->dev, "%s: TA\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret)
+ info->cable_type = CABLE_TYPE_TA_MUIC;
+ break;
+ case CABLE_TYPE_JIG_UART_ON_MUIC:
+ dev_info(info->dev, "%s: JIG UART/BOOTON\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ break;
+ case CABLE_TYPE_JIG_UART_OFF_MUIC:
+ dev_info(info->dev, "%s: JIG UART/BOOTOFF\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ break;
+ case CABLE_TYPE_SMARTDOCK_MUIC:
+ dev_info(info->dev, "%s: SMARTDOCK\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ if (mdata->usb_cb && info->is_usb_ready)
+ mdata->usb_cb(USB_POWERED_HOST_DETACHED);
+ ret = max77693_muic_set_charging_type(info, false);
+#ifdef CONFIG_EXTCON
+ if (info->edev && info->is_mhl_ready)
+ extcon_set_cable_state(info->edev, "MHL", false);
+#else
+ if (mdata->mhl_cb && info->is_mhl_ready)
+ mdata->mhl_cb(MAX77693_MUIC_DETACHED);
+#endif
+ max77693_muic_set_charging_type(info, false);
+ break;
+ case CABLE_TYPE_JIG_UART_OFF_VB_MUIC:
+ dev_info(info->dev, "%s: JIG UART/OFF/VB\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret)
+ info->cable_type = CABLE_TYPE_JIG_UART_OFF_VB_MUIC;
+ break;
+ case CABLE_TYPE_MHL_MUIC:
+ if (irq == info->irq_adc || irq == info->irq_chgtype) {
+ dev_warn(info->dev, "Detech mhl: Ignore irq:%d\n", irq);
+ break;
+ }
+ dev_info(info->dev, "%s: MHL\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ max77693_muic_set_charging_type(info, false);
+#ifdef CONFIG_EXTCON
+ if (info->edev && info->is_mhl_ready)
+ extcon_set_cable_state(info->edev, "MHL", false);
+#else
+ if (mdata->mhl_cb && info->is_mhl_ready)
+ mdata->mhl_cb(MAX77693_MUIC_DETACHED);
+#endif
+
+ break;
+ case CABLE_TYPE_MHL_VB_MUIC:
+ if (irq == info->irq_adc || irq == info->irq_chgtype) {
+ dev_warn(info->dev,
+ "Detech vbMhl: Ignore irq:%d\n", irq);
+ break;
+ }
+ dev_info(info->dev, "%s: MHL VBUS\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+ max77693_muic_set_charging_type(info, false);
+
+#ifdef CONFIG_EXTCON
+ if (info->edev && info->is_mhl_ready)
+ extcon_set_cable_state(info->edev, "MHL", false);
+#else
+ if (mdata->mhl_cb && info->is_mhl_ready)
+ mdata->mhl_cb(MAX77693_MUIC_DETACHED);
+#endif
+ break;
+ case CABLE_TYPE_UNKNOWN_MUIC:
+ dev_info(info->dev, "%s: UNKNOWN\n", __func__);
+ info->cable_type = CABLE_TYPE_NONE_MUIC;
+
+ ret = max77693_muic_set_charging_type(info, false);
+ if (ret)
+ info->cable_type = CABLE_TYPE_UNKNOWN_MUIC;
+ break;
+ default:
+ dev_info(info->dev, "%s:invalid cable type %d\n",
+ __func__, info->cable_type);
+ break;
+ }
+
+ /* jig state clear */
+ mdata->jig_state(false);
+#if defined(CONFIG_SEC_MODEM_M0_TD)
+ gpio_set_value(GPIO_AP_CP_INT1, GPIO_LEVEL_LOW);
+#endif
+ return ret;
+}
+
+static void max77693_muic_detect_dev(struct max77693_muic_info *info, int irq)
+{
+ struct i2c_client *client = info->muic;
+ u8 status[2];
+ u8 adc, chgtyp, adcerr;
+ int intr = INT_ATTACH;
+ int ret;
+ u8 cntl1_val;
+
+ ret = max77693_read_reg(client, MAX77693_MUIC_REG_CTRL1, &cntl1_val);
+ dev_info(info->dev, "func:%s CONTROL1:%x\n", __func__, cntl1_val);
+
+ ret = max77693_bulk_read(client, MAX77693_MUIC_REG_STATUS1, 2, status);
+ dev_info(info->dev, "func:%s irq:%d ret:%d\n", __func__, irq, ret);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg(%d)\n", __func__,
+ ret);
+ return;
+ }
+
+ dev_info(info->dev, "%s: STATUS1:0x%x, 2:0x%x\n", __func__,
+ status[0], status[1]);
+
+ if ((irq == info->irq_adc) &&
+ max77693_muic_handle_dock_vol_key(info, status[0])) {
+ dev_info(info->dev,
+ "max77693_muic_handle_dock_vol_key(irq_adc:%x)", irq);
+ return;
+ }
+
+ adc = status[0] & STATUS1_ADC_MASK;
+ adcerr = status[0] & STATUS1_ADCERR_MASK;
+ chgtyp = status[1] & STATUS2_CHGTYP_MASK;
+
+ dev_info(info->dev, "adc:%x adcerr:%x chgtyp:%xcable_type:%x\n", adc,
+ adcerr, chgtyp, info->cable_type);
+
+ if (!adcerr && adc == ADC_OPEN) {
+ if (chgtyp == CHGTYP_NO_VOLTAGE)
+ intr = INT_DETACH;
+ else if (chgtyp == CHGTYP_USB ||
+ chgtyp == CHGTYP_DOWNSTREAM_PORT ||
+ chgtyp == CHGTYP_DEDICATED_CHGR ||
+ chgtyp == CHGTYP_500MA || chgtyp == CHGTYP_1A) {
+ if (info->cable_type == CABLE_TYPE_OTG_MUIC ||
+ info->cable_type == CABLE_TYPE_DESKDOCK_MUIC ||
+ info->cable_type == CABLE_TYPE_CARDOCK_MUIC)
+ intr = INT_DETACH;
+ }
+ }
+
+ if (intr == INT_ATTACH) {
+ dev_info(info->dev, "%s: ATTACHED\n", __func__);
+ max77693_muic_handle_attach(info, status[0], status[1], irq);
+ } else {
+ dev_info(info->dev, "%s: DETACHED\n", __func__);
+ max77693_muic_handle_detach(info, irq);
+ }
+ return;
+}
+static irqreturn_t max77693_muic_irq(int irq, void *data)
+{
+ struct max77693_muic_info *info = data;
+ dev_info(info->dev, "%s: irq:%d\n", __func__, irq);
+
+ mutex_lock(&info->mutex);
+ max77693_muic_detect_dev(info, irq);
+ mutex_unlock(&info->mutex);
+
+ return IRQ_HANDLED;
+}
+
+#define REQUEST_IRQ(_irq, _name) \
+do { \
+ ret = request_threaded_irq(_irq, NULL, max77693_muic_irq, \
+ 0, _name, info); \
+ if (ret < 0) \
+ dev_err(info->dev, "Failed to request IRQ #%d: %d\n", \
+ _irq, ret); \
+} while (0)
+
+static int max77693_muic_irq_init(struct max77693_muic_info *info)
+{
+ int ret;
+ u8 val;
+
+ dev_info(info->dev, "func:%s\n", __func__);
+ dev_info(info->dev, "%s: system_rev=%x\n", __func__, system_rev);
+
+ /* INTMASK1 3:ADC1K 2:ADCErr 1:ADCLow 0:ADC */
+ /* INTMASK2 0:Chgtype */
+ max77693_write_reg(info->muic, MAX77693_MUIC_REG_INTMASK1, 0x09);
+ max77693_write_reg(info->muic, MAX77693_MUIC_REG_INTMASK2, 0x11);
+ max77693_write_reg(info->muic, MAX77693_MUIC_REG_INTMASK3, 0x00);
+
+ REQUEST_IRQ(info->irq_adc, "muic-adc");
+ REQUEST_IRQ(info->irq_chgtype, "muic-chgtype");
+ REQUEST_IRQ(info->irq_vbvolt, "muic-vbvolt");
+ REQUEST_IRQ(info->irq_adc1k, "muic-adc1k");
+
+ dev_info(info->dev, "adc:%d chgtype:%d adc1k:%d vbvolt:%d",
+ info->irq_adc, info->irq_chgtype,
+ info->irq_adc1k, info->irq_vbvolt);
+
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INTMASK1, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INTMASK1, val);
+
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INTMASK2, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INTMASK2, val);
+
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INTMASK3, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INTMASK3, val);
+
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INT1, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INT1, val);
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INT2, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INT2, val);
+ max77693_read_reg(info->muic, MAX77693_MUIC_REG_INT3, &val);
+ dev_info(info->dev, "%s: reg=%x, val=%x\n", __func__,
+ MAX77693_MUIC_REG_INT3, val);
+ return 0;
+}
+
+#define CHECK_GPIO(_gpio, _name) \
+do { \
+ if (!_gpio) { \
+ dev_err(&pdev->dev, _name " GPIO defined as 0 !\n"); \
+ WARN_ON(!_gpio); \
+ ret = -EIO; \
+ goto err_kfree; \
+ } \
+} while (0)
+
+static void max77693_muic_init_detect(struct work_struct *work)
+{
+ struct max77693_muic_info *info =
+ container_of(work, struct max77693_muic_info, init_work.work);
+ dev_info(info->dev, "func:%s\n", __func__);
+ mutex_lock(&info->mutex);
+ max77693_muic_detect_dev(info, -1);
+ mutex_unlock(&info->mutex);
+}
+
+static void max77693_muic_usb_detect(struct work_struct *work)
+{
+ struct max77693_muic_info *info =
+ container_of(work, struct max77693_muic_info, usb_work.work);
+ struct max77693_muic_data *mdata = info->muic_data;
+
+ dev_info(info->dev, "func:%s info->muic_data->sw_path:%d\n",
+ __func__, info->muic_data->sw_path);
+
+ mutex_lock(&info->mutex);
+ info->is_usb_ready = true;
+
+ if (info->muic_data->sw_path != CP_USB_MODE) {
+ if (mdata->usb_cb) {
+ switch (info->cable_type) {
+ case CABLE_TYPE_USB_MUIC:
+ case CABLE_TYPE_JIG_USB_OFF_MUIC:
+ case CABLE_TYPE_JIG_USB_ON_MUIC:
+#ifdef CONFIG_USBHUB_USB3803
+ /* setting usb hub in Diagnostic(hub) mode */
+ usb3803_set_mode(USB_3803_MODE_HUB);
+#endif /* CONFIG_USBHUB_USB3803 */
+ mdata->usb_cb(USB_CABLE_ATTACHED);
+ break;
+ case CABLE_TYPE_OTG_MUIC:
+ mdata->usb_cb(USB_OTGHOST_ATTACHED);
+ break;
+ case CABLE_TYPE_SMARTDOCK_MUIC:
+ mdata->usb_cb(USB_POWERED_HOST_ATTACHED);
+ break;
+ default:
+ break;
+ }
+ }
+ }
+ mutex_unlock(&info->mutex);
+}
+
+static void max77693_muic_mhl_detect(struct work_struct *work)
+{
+ struct max77693_muic_info *info =
+ container_of(work, struct max77693_muic_info, mhl_work.work);
+ struct max77693_muic_data *mdata = info->muic_data;
+
+ dev_info(info->dev, "func:%s cable_type:%d\n", __func__,
+ info->cable_type);
+ mutex_lock(&info->mutex);
+ info->is_mhl_ready = true;
+
+ if (info->cable_type == CABLE_TYPE_MHL_MUIC ||
+ info->cable_type == CABLE_TYPE_MHL_VB_MUIC) {
+#ifdef CONFIG_EXTCON
+ if (info->edev)
+ extcon_set_cable_state(info->edev, "MHL", true);
+#else
+ if (mdata->mhl_cb)
+ mdata->mhl_cb(MAX77693_MUIC_ATTACHED);
+#endif
+ }
+ mutex_unlock(&info->mutex);
+}
+
+static int uart_switch_init(struct max77693_muic_info *info)
+{
+ int ret, val;
+
+ if (info->max77693->pmic_rev < MAX77693_REV_PASS2) {
+ ret = gpio_request(GPIO_UART_SEL, "UART_SEL");
+ if (ret < 0) {
+ pr_err("Failed to request GPIO_UART_SEL!\n");
+ return -ENODEV;
+ }
+ s3c_gpio_setpull(GPIO_UART_SEL, S3C_GPIO_PULL_NONE);
+ val = gpio_get_value(GPIO_UART_SEL);
+ gpio_direction_output(GPIO_UART_SEL, val);
+ pr_info("func: %s uart_gpio val: %d\n", __func__, val);
+ }
+#ifdef CONFIG_LTE_VIA_SWITCH
+ ret = gpio_request(GPIO_LTE_VIA_UART_SEL, "LTE_VIA_SEL");
+ if (ret < 0) {
+ pr_err("Failed to request GPIO_LTE_VIA_UART_SEL!\n");
+ return -ENODEV;
+ }
+ s3c_gpio_setpull(GPIO_LTE_VIA_UART_SEL, S3C_GPIO_PULL_NONE);
+ val = gpio_get_value(GPIO_LTE_VIA_UART_SEL);
+ gpio_direction_output(GPIO_LTE_VIA_UART_SEL, val);
+ pr_info("func: %s lte_gpio val: %d\n", __func__, val);
+#endif
+/*
+#ifndef CONFIG_LTE_VIA_SWITCH
+ gpio_export(GPIO_UART_SEL, 1);
+ gpio_export_link(switch_dev, "uart_sel", GPIO_UART_SEL);
+#endif
+*/
+ return 0;
+}
+
+int max77693_muic_get_status1_adc1k_value(void)
+{
+ u8 adc1k;
+ int ret;
+
+ ret = max77693_read_reg(gInfo->muic,
+ MAX77693_MUIC_REG_STATUS1, &adc1k);
+ if (ret) {
+ dev_err(gInfo->dev, "%s: fail to read muic reg(%d)\n",
+ __func__, ret);
+ return -EINVAL;
+ }
+ adc1k = adc1k & STATUS1_ADC1K_MASK ? 1 : 0;
+
+ pr_info("func:%s, adc1k: %d\n", __func__, adc1k);
+ /* -1:err, 0:adc1k not detected, 1:adc1k detected */
+ return adc1k;
+}
+
+int max77693_muic_get_status1_adc_value(void)
+{
+ u8 adc;
+ int ret;
+
+ ret = max77693_read_reg(gInfo->muic,
+ MAX77693_MUIC_REG_STATUS1, &adc);
+ if (ret) {
+ dev_err(gInfo->dev, "%s: fail to read muic reg(%d)\n",
+ __func__, ret);
+ return -EINVAL;
+ }
+
+ return adc & STATUS1_ADC_MASK;
+}
+
+/*
+* func: max77693_muic_set_audio_switch
+* arg: bool enable(true:set vps path, false:set path open)
+* return: only 0 success
+*/
+int max77693_muic_set_audio_switch(bool enable)
+{
+ struct i2c_client *client = gInfo->muic;
+ u8 cntl1_val, cntl1_msk;
+ int ret;
+ pr_info("func:%s enable(%d)", __func__, enable);
+
+ if (enable) {
+ cntl1_val = (MAX77693_MUIC_CTRL1_BIN_2_010 << COMN1SW_SHIFT)
+ | (MAX77693_MUIC_CTRL1_BIN_2_010 << COMP2SW_SHIFT) |
+ (0 << MICEN_SHIFT);
+ } else {
+ cntl1_val = 0x3f;
+ }
+ cntl1_msk = COMN1SW_MASK | COMP2SW_MASK | MICEN_MASK;
+
+ ret = max77693_update_reg(client, MAX77693_MUIC_REG_CTRL1, cntl1_val,
+ cntl1_msk);
+ cntl1_val = MAX77693_MUIC_CTRL1_BIN_0_000;
+ max77693_read_reg(client, MAX77693_MUIC_REG_CTRL1, &cntl1_val);
+ dev_info(gInfo->dev, "%s: CNTL1(0x%02x)\n", __func__, cntl1_val);
+ return ret;
+}
+
+void max77693_update_jig_state(struct max77693_muic_info *info)
+{
+ struct i2c_client *client = info->muic;
+ struct max77693_muic_data *mdata = info->muic_data;
+ u8 reg_data, adc;
+ int ret, jig_state;
+
+ ret = max77693_read_reg(client, MAX77693_MUIC_REG_STATUS1, &reg_data);
+ if (ret) {
+ dev_err(info->dev, "%s: fail to read muic reg(%d)\n",
+ __func__, ret);
+ return;
+ }
+ adc = reg_data & STATUS1_ADC_MASK;
+
+ switch (adc) {
+ case ADC_JIG_UART_OFF:
+ case ADC_JIG_USB_OFF:
+ case ADC_JIG_USB_ON:
+ jig_state = true;
+ break;
+ default:
+ jig_state = false;
+ break;
+ }
+
+ mdata->jig_state(jig_state);
+}
+
+static int __devinit max77693_muic_probe(struct platform_device *pdev)
+{
+ struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent);
+ struct max77693_platform_data *pdata = dev_get_platdata(max77693->dev);
+ struct max77693_muic_info *info;
+ struct input_dev *input;
+ int ret;
+ pr_info("func:%s\n", __func__);
+ info = kzalloc(sizeof(struct max77693_muic_info), GFP_KERNEL);
+ if (!info) {
+ dev_err(&pdev->dev, "%s: failed to allocate info\n", __func__);
+ ret = -ENOMEM;
+ goto err_return;
+ }
+ input = input_allocate_device();
+ if (!input) {
+ dev_err(&pdev->dev, "%s: failed to allocate input\n", __func__);
+ ret = -ENOMEM;
+ goto err_kfree;
+ }
+ info->dev = &pdev->dev;
+ info->max77693 = max77693;
+ info->muic = max77693->muic;
+ info->input = input;
+ info->irq_adc = max77693->irq_base + MAX77693_MUIC_IRQ_INT1_ADC;
+ info->irq_chgtype = max77693->irq_base + MAX77693_MUIC_IRQ_INT2_CHGTYP;
+ info->irq_vbvolt = max77693->irq_base + MAX77693_MUIC_IRQ_INT2_VBVOLT;
+ info->irq_adc1k = max77693->irq_base + MAX77693_MUIC_IRQ_INT1_ADC1K;
+ info->muic_data = pdata->muic;
+ info->is_adc_open_prev = true;
+
+ if (pdata->is_default_uart_path_cp)
+ info->is_default_uart_path_cp =
+ pdata->is_default_uart_path_cp();
+ else
+ info->is_default_uart_path_cp = false;
+ info->cable_type = CABLE_TYPE_UNKNOWN_MUIC;
+ info->muic_data->sw_path = AP_USB_MODE;
+ gInfo = info;
+
+ platform_set_drvdata(pdev, info);
+ dev_info(info->dev, "adc:%d chgtype:%d, adc1k%d\n",
+ info->irq_adc, info->irq_chgtype, info->irq_adc1k);
+
+ input->name = pdev->name;
+ input->phys = "deskdock-key/input0";
+ input->dev.parent = &pdev->dev;
+
+ input->id.bustype = BUS_HOST;
+ input->id.vendor = 0x0001;
+ input->id.product = 0x0001;
+ input->id.version = 0x0001;
+
+ /* Enable auto repeat feature of Linux input subsystem */
+ __set_bit(EV_REP, input->evbit);
+
+ input_set_capability(input, EV_KEY, KEY_VOLUMEUP);
+ input_set_capability(input, EV_KEY, KEY_VOLUMEDOWN);
+ input_set_capability(input, EV_KEY, KEY_PLAYPAUSE);
+ input_set_capability(input, EV_KEY, KEY_PREVIOUSSONG);
+ input_set_capability(input, EV_KEY, KEY_NEXTSONG);
+
+ ret = input_register_device(input);
+ if (ret) {
+ dev_err(info->dev, "%s: Unable to register input device, "
+ "error: %d\n", __func__, ret);
+ goto err_input;
+ }
+
+ ret = uart_switch_init(info);
+ if (ret) {
+ pr_err("Failed to initialize uart\n");
+ goto err_input;
+ }
+
+ if (info->is_default_uart_path_cp)
+ dev_info(info->dev, "H/W rev connected UT1 UR2 pin to CP UART");
+ else
+ dev_info(info->dev, "H/W rev connected UT1 UR2 pin to AP UART");
+ /*PASS1 */
+ if (max77693->pmic_rev < MAX77693_REV_PASS2 &&
+ gpio_is_valid(info->muic_data->gpio_usb_sel)) {
+ CHECK_GPIO(info->muic_data->gpio_usb_sel, "USB_SEL");
+
+ if (info->muic_data->cfg_uart_gpio)
+ info->muic_data->uart_path =
+ info->muic_data->cfg_uart_gpio();
+
+ ret = gpio_request(info->muic_data->gpio_usb_sel, "USB_SEL");
+ if (ret) {
+ dev_info(info->dev, "%s: fail to request gpio(%d)\n",
+ __func__, ret);
+ goto err_input;
+ }
+ if (gpio_get_value(info->muic_data->gpio_usb_sel)) {
+ dev_info(info->dev, "%s: CP USB\n", __func__);
+ info->muic_data->sw_path = CP_USB_MODE;
+ }
+ } else if (max77693->pmic_rev >= MAX77693_REV_PASS2) {
+ /*PASS2 */
+ int switch_sel = get_switch_sel();
+ if (switch_sel & MAX77693_SWITCH_SEL_1st_BIT_USB)
+ info->muic_data->sw_path = AP_USB_MODE;
+ else
+ info->muic_data->sw_path = CP_USB_MODE;
+ if (switch_sel & MAX77693_SWITCH_SEL_2nd_BIT_UART)
+ info->muic_data->uart_path = UART_PATH_AP;
+ else {
+ info->muic_data->uart_path = UART_PATH_CP;
+#ifdef CONFIG_LTE_VIA_SWITCH
+ if (switch_sel & MAX77693_SWITCH_SEL_3rd_BIT_LTE_UART)
+ info->muic_data->uart_path = UART_PATH_LTE;
+#endif
+ }
+ pr_info("%s: switch_sel: %x\n", __func__, switch_sel);
+ }
+ /* create sysfs group */
+ ret = sysfs_create_group(&switch_dev->kobj, &max77693_muic_group);
+ dev_set_drvdata(switch_dev, info);
+ if (ret) {
+ dev_err(&pdev->dev,
+ "failed to create max77693 muic attribute group\n");
+ goto fail;
+ }
+
+#ifdef CONFIG_EXTCON
+ /* External connector */
+ info->edev = kzalloc(sizeof(struct extcon_dev), GFP_KERNEL);
+ if (!info->edev) {
+ pr_err("Failed to allocate memory for extcon device\n");
+ ret = -ENOMEM;
+ goto fail;
+ }
+ info->edev->name = DEV_NAME;
+ info->edev->supported_cable = extcon_cable_name;
+ ret = extcon_dev_register(info->edev, NULL);
+ if (ret) {
+ pr_err("Failed to register extcon device\n");
+ kfree(info->edev);
+ goto fail;
+ }
+#endif
+
+ if (info->muic_data->init_cb)
+ info->muic_data->init_cb();
+
+ mutex_init(&info->mutex);
+
+ /* Set ADC debounce time: 25ms */
+ max77693_muic_set_adcdbset(info, 2);
+
+ ret = max77693_muic_irq_init(info);
+ if (ret < 0) {
+ dev_err(&pdev->dev, "Failed to initialize MUIC irq:%d\n", ret);
+ goto fail;
+ }
+
+ /* init jig state */
+ max77693_update_jig_state(info);
+
+ /* initial cable detection */
+ INIT_DELAYED_WORK(&info->init_work, max77693_muic_init_detect);
+ schedule_delayed_work(&info->init_work, msecs_to_jiffies(3000));
+
+ INIT_DELAYED_WORK(&info->usb_work, max77693_muic_usb_detect);
+ schedule_delayed_work(&info->usb_work, msecs_to_jiffies(17000));
+
+ INIT_DELAYED_WORK(&info->mhl_work, max77693_muic_mhl_detect);
+ schedule_delayed_work(&info->mhl_work, msecs_to_jiffies(25000));
+
+ return 0;
+
+ fail:
+ if (info->irq_adc)
+ free_irq(info->irq_adc, NULL);
+ if (info->irq_chgtype)
+ free_irq(info->irq_chgtype, NULL);
+ if (info->irq_vbvolt)
+ free_irq(info->irq_vbvolt, NULL);
+ if (info->irq_adc1k)
+ free_irq(info->irq_adc1k, NULL);
+ mutex_destroy(&info->mutex);
+ err_input:
+ platform_set_drvdata(pdev, NULL);
+ input_free_device(input);
+ err_kfree:
+ kfree(info);
+ err_return:
+ return ret;
+}
+
+static int __devexit max77693_muic_remove(struct platform_device *pdev)
+{
+ struct max77693_muic_info *info = platform_get_drvdata(pdev);
+ sysfs_remove_group(&switch_dev->kobj, &max77693_muic_group);
+
+ if (info) {
+ dev_info(info->dev, "func:%s\n", __func__);
+ input_unregister_device(info->input);
+ cancel_delayed_work(&info->init_work);
+ cancel_delayed_work(&info->usb_work);
+ cancel_delayed_work(&info->mhl_work);
+ free_irq(info->irq_adc, info);
+ free_irq(info->irq_chgtype, info);
+ free_irq(info->irq_vbvolt, info);
+ free_irq(info->irq_adc1k, info);
+#ifndef CONFIG_TARGET_LOCALE_NA
+ gpio_free(info->muic_data->gpio_usb_sel);
+#endif /* CONFIG_TARGET_LOCALE_NA */
+ mutex_destroy(&info->mutex);
+ kfree(info);
+ }
+ return 0;
+}
+
+void max77693_muic_shutdown(struct device *dev)
+{
+ struct max77693_muic_info *info = dev_get_drvdata(dev);
+ int ret;
+ u8 val;
+ dev_info(info->dev, "func:%s\n", __func__);
+ if (!info->muic) {
+ dev_err(info->dev, "%s: no muic i2c client\n", __func__);
+ return;
+ }
+
+ dev_info(info->dev, "%s: JIGSet: auto detection\n", __func__);
+ val = (0 << CTRL3_JIGSET_SHIFT) | (0 << CTRL3_BOOTSET_SHIFT);
+
+ ret = max77693_update_reg(info->muic, MAX77693_MUIC_REG_CTRL3, val,
+ CTRL3_JIGSET_MASK | CTRL3_BOOTSET_MASK);
+ if (ret < 0) {
+ dev_err(info->dev, "%s: fail to update reg\n", __func__);
+ return;
+ }
+}
+
+static struct platform_driver max77693_muic_driver = {
+ .driver = {
+ .name = DEV_NAME,
+ .owner = THIS_MODULE,
+ .shutdown = max77693_muic_shutdown,
+ },
+ .probe = max77693_muic_probe,
+ .remove = __devexit_p(max77693_muic_remove),
+};
+
+static int __init max77693_muic_init(void)
+{
+ pr_info("func:%s\n", __func__);
+ return platform_driver_register(&max77693_muic_driver);
+}
+module_init(max77693_muic_init);
+
+static void __exit max77693_muic_exit(void)
+{
+ pr_info("func:%s\n", __func__);
+ platform_driver_unregister(&max77693_muic_driver);
+}
+module_exit(max77693_muic_exit);
+
+
+MODULE_DESCRIPTION("Maxim MAX77693 MUIC driver");
+MODULE_AUTHOR("<sukdong.kim@samsung.com>");
+MODULE_LICENSE("GPL");