diff options
Diffstat (limited to 'drivers/misc/max77693-muic.c')
-rw-r--r-- | drivers/misc/max77693-muic.c | 2520 |
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, ®_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"); |