| /* SPDX-License-Identifier: GPL-2.0 */ |
| /* |
| * Copyright 2020 Google, LLC |
| * |
| * 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. |
| */ |
| |
| #define pr_fmt(fmt) KBUILD_MODNAME ": %s " fmt, __func__ |
| |
| #include <linux/ctype.h> |
| #include <linux/i2c.h> |
| #include <linux/of.h> |
| #include <linux/of_gpio.h> |
| #include <linux/of_irq.h> |
| #include <linux/gpio.h> |
| #include <linux/gpio/driver.h> |
| #include <linux/module.h> |
| #include <linux/regmap.h> |
| #include <linux/interrupt.h> |
| #include <linux/debugfs.h> |
| #include <linux/seq_file.h> |
| #include <linux/irqdomain.h> |
| #include <linux/irq.h> |
| #include <linux/irqchip.h> |
| #include "max_m5.h" |
| #include "max77759.h" |
| #include "max77759_maxq.h" |
| #include "gbms_storage.h" |
| #include "google_bms.h" |
| |
| |
| enum max77729_pmic_register { |
| MAX77729_PMIC_ID = 0x00, |
| MAX77729_PMIC_REVISION = 0x01, |
| MAX77729_PMIC_MAINCTRL = 0x02, |
| MAX77729_PMIC_INTSRC = 0x22, |
| MAX77729_PMIC_INTSRCMASK = 0x23, |
| MAX77729_PMIC_TOPSYS_INT = 0x24, |
| MAX77729_PMIC_TOPSYS_INT_MASK = 0x26, |
| }; |
| |
| #define MUSBC 0x08 |
| #define MFUEL 0x04 |
| #define MTOPS 0x02 |
| #define MCHGR 0x01 |
| |
| #define MAX77759_GPIO_DIR_IN 0 |
| #define MAX77759_GPIO_DIR_OUT 1 |
| |
| #define MAX77759_GPIO5_DIR_MASK (1 << 2) |
| #define MAX77759_GPIO5_DIR(x) ((x) << 2) |
| #define MAX77759_GPIO5_VAL_MASK (1 << 3) |
| #define MAX77759_GPIO5_VAL(x) ((x) << 3) |
| #define MAX77759_GPIO6_DIR_MASK (1 << 5) |
| #define MAX77759_GPIO6_DIR(x) ((x) << 5) |
| #define MAX77759_GPIO6_VAL_MASK (1 << 6) |
| #define MAX77759_GPIO6_VAL(x) ((x) << 6) |
| |
| #define MAX77759_GPIO5_OFF 4 |
| #define MAX77759_GPIO6_OFF 5 |
| #define MAX77759_MIN_GPIO_OFF 4 |
| #define MAX77759_MAX_GPIO_OFF 5 |
| #define MAX77759_NUM_GPIOS 6 |
| |
| #define MAX77759_GPIO_CONTROL_READ 0x23 |
| #define MAX77759_GPIO_CONTROL_WRITE 0x24 |
| |
| #define MDEFAULT (0xF0 | ~(MUSBC | MFUEL | MCHGR)) |
| |
| |
| #define MAX77729_PMIC_INTSRCMASK_DEFAULT MDEFAULT |
| #define MAX77729_PMIC_TOPSYS_INT_MASK_DEFAULT 0xff |
| |
| #define MAX77759_PMIC_INTSRCMASK_DEFAULT \ |
| ~(MAX77759_PMIC_INTSRCMASK_MAXQ_INT_M | \ |
| MAX77759_PMIC_INTSRCMASK_CHGR_INT_M) |
| |
| /* b/156527175: *_PMIC_TOPSYS_INT_MASK_SPR_7 is reserved */ |
| #define MAX77759_PMIC_TOPSYS_INT_MASK_MASK \ |
| (MAX77759_PMIC_TOPSYS_INT_MASK_TSHDN_INT_M | \ |
| MAX77759_PMIC_TOPSYS_INT_MASK_SYSOVLO_INT_M | \ |
| MAX77759_PMIC_TOPSYS_INT_MASK_SYSUVLO_INT_M) |
| |
| #define MAX77759_PMIC_TOPSYS_INT_MASK_DEFAULT \ |
| (MAX77759_PMIC_TOPSYS_INT_MASK_TSHDN_INT_M) |
| |
| #define MAX77759_STORAGE_SIZE 16 |
| #define MAX77759_STORAGE_BASE (MAX77759_PMIC_AP_DATAOUT0 + MAX77759_STORAGE_SIZE) |
| |
| struct max77729_pmic_data { |
| struct device *dev; |
| struct regmap *regmap; |
| uint8_t pmic_id; |
| uint8_t rev_id; |
| |
| #if IS_ENABLED(CONFIG_GPIOLIB) |
| struct mutex irq_lock; |
| struct gpio_chip gpio; |
| |
| /* threaded irq */ |
| int irq_trig_falling[2]; |
| u8 irq_trig_u; |
| u8 irq_mask; |
| u8 irq_mask_u; |
| #endif |
| struct max77759_maxq *maxq; |
| |
| struct i2c_client *fg_i2c_client; |
| struct i2c_client *pmic_i2c_client; |
| void *ovp_client_data; |
| struct mutex io_lock; |
| int batt_id; |
| |
| atomic_t sysuvlo_cnt; |
| atomic_t sysovlo_cnt; |
| struct dentry *de; |
| |
| struct delayed_work storage_init_work; |
| |
| /* debug interface, register to read or write */ |
| u32 debug_reg_address; |
| |
| }; |
| |
| static bool max77729_pmic_is_reg(struct device *dev, unsigned int reg) |
| { |
| int ret; |
| |
| switch (reg) { |
| case MAX77729_PMIC_ID: |
| case MAX77729_PMIC_REVISION: |
| case MAX77729_PMIC_MAINCTRL: |
| case MAX77729_PMIC_INTSRC: |
| case MAX77729_PMIC_INTSRCMASK: |
| case MAX77729_PMIC_TOPSYS_INT: |
| case MAX77729_PMIC_TOPSYS_INT_MASK: |
| ret = true; |
| break; |
| case MAX77759_PMIC_I2C_CNFG: |
| case MAX77759_PMIC_SWRESET: |
| case MAX77759_PMIC_CONTROL_FG: |
| ret = true; |
| break; |
| case MAX77759_PMIC_DEVICE_ID: |
| case MAX77759_PMIC_DEVICE_REV: |
| case MAX77759_PMIC_FW_REV: |
| case MAX77759_PMIC_FW_SUB_REV: |
| case MAX77759_PMIC_UIC_INT1...MAX77759_PMIC_UIC_INT4_M: |
| case MAX77759_PMIC_AP_DATAOUT0...MAX77759_PMIC_AP_DATAIN32: |
| case MAX77759_PMIC_UIC_SWRST: |
| ret = true; |
| break; |
| default: |
| ret = false; |
| break; |
| } |
| |
| return ret; |
| } |
| |
| static struct regmap_config max777x9_pmic_regmap_cfg = { |
| .name = "max777x9_pmic", |
| .reg_bits = 8, |
| .val_bits = 8, |
| .val_format_endian = REGMAP_ENDIAN_NATIVE, |
| .max_register = MAX77729_PMIC_INTSRCMASK, |
| .readable_reg = max77729_pmic_is_reg, |
| .volatile_reg = max77729_pmic_is_reg, |
| }; |
| |
| static inline int max77729_pmic_readn(struct max77729_pmic_data *data, |
| int addr, u8 *val, int len) |
| { |
| int rc; |
| |
| rc = regmap_bulk_read(data->regmap, addr, val, len); |
| if (rc < 0) |
| pr_err("regmap_read failed for address %04x rc=%d\n", |
| addr, rc); |
| |
| return rc; |
| } |
| |
| static inline int max77729_pmic_writen(struct max77729_pmic_data *data, |
| int addr, const u8 *val, int len) |
| { |
| int rc; |
| |
| rc = regmap_bulk_write(data->regmap, addr, val, len); |
| if (rc < 0) |
| pr_err("regmap_write failed for address %04x rc=%d\n", |
| addr, rc); |
| |
| return 0; |
| } |
| |
| |
| #define max77729_pmic_rd8(data, addr, val) \ |
| max77729_pmic_readn(data, addr, val, 1) |
| #define max77729_pmic_wr8(data, addr, val) \ |
| max77729_pmic_writen(data, addr, (const u8[]){ val }, 1) |
| |
| /* no need for caching */ |
| static inline int max77729_pmic_rmw8(struct max77729_pmic_data *data, |
| int reg, u8 mask, u8 value) |
| { |
| return regmap_write_bits(data->regmap, reg, mask, value); |
| } |
| |
| int max777x9_pmic_reg_read(struct i2c_client *client, |
| u8 addr, u8 *val, int len) |
| { |
| struct max77729_pmic_data *data; |
| |
| if (!client) |
| return -EINVAL; |
| |
| data = i2c_get_clientdata(client); |
| if (!data || !data->regmap) |
| return -ENXIO; |
| |
| return max77729_pmic_readn(data, addr, val, len); |
| } |
| EXPORT_SYMBOL_GPL(max777x9_pmic_reg_read); |
| |
| int max777x9_pmic_reg_write(struct i2c_client *client, |
| u8 addr, const u8 *val, int len) |
| { |
| struct max77729_pmic_data *data; |
| |
| if (!client) |
| return -EINVAL; |
| |
| data = i2c_get_clientdata(client); |
| if (!data || !data->regmap) |
| return -ENXIO; |
| |
| return max77729_pmic_writen(data, addr, val, len); |
| } |
| EXPORT_SYMBOL_GPL(max777x9_pmic_reg_write); |
| |
| int max777x9_pmic_reg_update(struct i2c_client *client, |
| u8 reg, u8 mask, u8 value) |
| { |
| struct max77729_pmic_data *data; |
| |
| if (!client) |
| return -EINVAL; |
| |
| data = i2c_get_clientdata(client); |
| if (!data || !data->regmap) |
| return -ENXIO; |
| |
| return max77729_pmic_rmw8(data, reg, mask, value); |
| } |
| EXPORT_SYMBOL_GPL(max777x9_pmic_reg_update); |
| |
| |
| int max777x9_pmic_get_id(struct i2c_client *client, u8 *id, u8 *rev) |
| { |
| struct max77729_pmic_data *data; |
| |
| if (!client) |
| return -EINVAL; |
| |
| data = i2c_get_clientdata(client); |
| if (!data) |
| return -ENXIO; |
| |
| *rev = data->rev_id; |
| *id = data->pmic_id; |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(max777x9_pmic_get_id); |
| |
| static int max77729_gpio_to_irq_mask(int offset, unsigned int *mask, unsigned int *val) |
| { |
| if (offset != MAX77759_GPIO5_OFF && offset != MAX77759_GPIO6_OFF) |
| return -EINVAL; |
| |
| /* gpio5 is bit 0 of MAX77759_PMIC_UIC_INT1, gpio6 is bit 1 */ |
| *mask = 1 << (offset - MAX77759_GPIO5_OFF); |
| *val = 1 << (offset - MAX77759_GPIO5_OFF) ; |
| return 0; |
| } |
| |
| static int max77729_gpio_clear_int(struct max77729_pmic_data *data, unsigned int offset) |
| { |
| unsigned int mask, val; |
| int ret; |
| |
| ret = max77729_gpio_to_irq_mask(offset, &mask, &val); |
| if (ret == 0) |
| ret = max77729_pmic_rmw8(data, MAX77759_PMIC_UIC_INT1, mask, val); |
| |
| pr_debug("offset=%d clear int1 mask=%x val=%x (%d)\n", |
| offset, mask, val, ret); |
| |
| return ret; |
| } |
| |
| static irqreturn_t max777x9_pmic_route_irq(struct max77729_pmic_data *data, |
| int offset) |
| { |
| int ret = 0, sub_irq; |
| |
| /* NOTE: clearing before handle_nested_irq() assumes EDGE-type IRQ */ |
| ret = max77729_gpio_clear_int(data, offset); |
| if (ret < 0) |
| pr_err("gpio%d cannot clear int1 (%d)", offset + 1, ret); |
| |
| sub_irq = irq_find_mapping(data->gpio.irq.domain, offset); |
| pr_debug("offset=%d sub_irq=%d\n", offset, sub_irq); |
| if (sub_irq) |
| handle_nested_irq(sub_irq); |
| |
| return ret; |
| } |
| |
| /* this interrupt is read to clear, in max77759 it should be write to clear */ |
| static irqreturn_t max777x9_pmic_irq(int irq, void *ptr) |
| { |
| struct max77729_pmic_data *data = ptr; |
| uint8_t intsrc = 0, uic_int[4]; |
| int ret; |
| |
| /* INTSRC is read to clear on MW and max77729f */ |
| ret = max77729_pmic_rd8(data, MAX77729_PMIC_INTSRC, &intsrc); |
| if (ret < 0) { |
| dev_err_ratelimited(data->dev, "INTSRC: read error %d\n", ret); |
| return IRQ_NONE; |
| } |
| |
| if (intsrc == 0) |
| return IRQ_NONE; |
| |
| /* just clear for max77729f */ |
| pr_debug("irq=%d INTSRC:%x\n", irq, intsrc); |
| if (data->pmic_id != MAX77759_PMIC_PMIC_ID_MW) |
| return IRQ_HANDLED; |
| |
| /* UIC_INT are write to clear */ |
| if (intsrc & MAX77759_PMIC_INTSRC_MAXQ_INT) { |
| ret = max77729_pmic_readn(data, MAX77759_PMIC_UIC_INT1, |
| uic_int, sizeof(uic_int)); |
| if (ret < 0) { |
| dev_err_ratelimited(data->dev, |
| "UIC_INT1: read error %d\n", ret); |
| return IRQ_NONE; |
| } |
| |
| /* TODO: implement / handle comms with maxq */ |
| if (uic_int[0] & MAX77759_PMIC_UIC_INT1_APCMDRESI) { |
| maxq_irq(data->maxq); |
| max77729_pmic_wr8(data, MAX77759_PMIC_UIC_INT1, |
| MAX77759_PMIC_UIC_INT1_APCMDRESI); |
| } |
| |
| if (uic_int[0] & MAX77759_PMIC_UIC_INT1_GPIO5I) |
| max777x9_pmic_route_irq(data, MAX77759_GPIO5_OFF); |
| if (uic_int[0] & MAX77759_PMIC_UIC_INT1_GPIO6I) |
| max777x9_pmic_route_irq(data, MAX77759_GPIO6_OFF); |
| } |
| |
| if (intsrc & MAX77759_PMIC_TOPSYS_INT_SYSUVLO_INT) |
| atomic_inc(&data->sysuvlo_cnt); |
| |
| if (intsrc & MAX77759_PMIC_TOPSYS_INT_SYSOVLO_INT) |
| atomic_inc(&data->sysovlo_cnt); |
| |
| if (intsrc & MAX77759_PMIC_INTSRC_TOPSYS_INT) { |
| uint8_t tsi; |
| |
| ret = max77729_pmic_rd8(data, MAX77729_PMIC_TOPSYS_INT, &tsi); |
| if (ret < 0) { |
| dev_err_ratelimited(data->dev, |
| "TOPSYS_INT: read error %d\n", ret); |
| return IRQ_NONE; |
| } |
| |
| ret = max77729_pmic_wr8(data, MAX77729_PMIC_TOPSYS_INT, tsi); |
| if (ret < 0) { |
| dev_err_ratelimited(data->dev, |
| "TOPSYS_INT:%x clr error %d\n", tsi, ret); |
| return IRQ_NONE; |
| } |
| |
| pr_info("TOPSYS_INT:%x\n", tsi); |
| |
| /* TODO: handle TSHDN_INT, SYSOVLO_INT, SYSUVLO_INT */ |
| } |
| |
| /* just clear CHG_INT, no FG intr for MW */ |
| |
| return IRQ_HANDLED; |
| } |
| |
| /* |
| * Bootloader has everything masked clear this on boot |
| * GPIO irqs are enabled later |
| */ |
| static int max777x9_pmic_set_irqmask(struct max77729_pmic_data *data) |
| { |
| int ret; |
| |
| if (data->pmic_id == MAX77759_PMIC_PMIC_ID_MW) { |
| const u8 uic_mask[] = {0x7f, 0xff, 0xff, 0xff}; |
| u8 reg; |
| |
| ret = max77729_pmic_rd8(data, MAX77759_PMIC_INTSRC, ®); |
| if (ret < 0 || reg) |
| dev_info(data->dev, "INTSRC :%x (%d)\n", reg, ret); |
| |
| ret = max77729_pmic_rd8(data, MAX77759_PMIC_TOPSYS_INT, ®); |
| if (ret < 0 || reg) |
| dev_info(data->dev, "TOPSYS_INT :%x (%d)\n", reg, ret); |
| |
| ret = max77729_pmic_wr8(data, MAX77759_PMIC_INTSRCMASK, |
| MAX77759_PMIC_INTSRCMASK_DEFAULT); |
| |
| /* b/156527175, *_PMIC_TOPSYS_INT_MASK_SPR_7 is reserved */ |
| ret |= max77729_pmic_rmw8(data, MAX77759_PMIC_TOPSYS_INT_MASK, |
| MAX77759_PMIC_TOPSYS_INT_MASK_MASK, |
| MAX77759_PMIC_TOPSYS_INT_MASK_DEFAULT); |
| |
| /* clear all, unmask MAX77759_PMIC_UIC_INT1_APCMDRESI */ |
| ret |= max77729_pmic_wr8(data, MAX77759_PMIC_UIC_INT1, |
| MAX77759_PMIC_UIC_INT1_GPIO5I | |
| MAX77759_PMIC_UIC_INT1_GPIO6I | |
| MAX77759_PMIC_UIC_INT1_APCMDRESI); |
| ret |= max77729_pmic_writen(data, MAX77759_PMIC_UIC_INT1_M, |
| uic_mask, sizeof(uic_mask)); |
| } else { |
| ret = max77729_pmic_wr8(data, MAX77729_PMIC_INTSRCMASK, |
| MAX77729_PMIC_INTSRCMASK_DEFAULT); |
| ret |= max77729_pmic_wr8(data, MAX77729_PMIC_TOPSYS_INT_MASK, |
| MAX77729_PMIC_TOPSYS_INT_MASK_DEFAULT); |
| } |
| |
| return ret ? -EIO : 0; |
| } |
| |
| static int max77759_find_fg(struct max77729_pmic_data *data) |
| { |
| struct device_node *dn; |
| |
| if (data->fg_i2c_client) |
| return 0; |
| |
| dn = of_parse_phandle(data->dev->of_node, "max77759,max_m5", 0); |
| if (!dn) |
| return -ENXIO; |
| |
| data->fg_i2c_client = of_find_i2c_device_by_node(dn); |
| if (!data->fg_i2c_client) |
| return -EAGAIN; |
| |
| return 0; |
| } |
| |
| #define NTC_CURVE_THRESHOLD 185 |
| #define NTC_CURVE_1_BASE 960 |
| #define NTC_CURVE_1_SHIFT 2 |
| #define NTC_CURVE_2_BASE 730 |
| #define NTC_CURVE_2_SHIFT 3 |
| |
| /* |
| * WARNING: The FG will behave erratically when the recovery path fails. |
| */ |
| static int max77759_read_thm(struct max77729_pmic_data *data, int mux, |
| unsigned int *value) |
| { |
| unsigned int ain0, config, check_config = 0; |
| u8 pmic_ctrl, check_pmic = 0; |
| int tmp, ret; |
| |
| if (!data->fg_i2c_client) |
| return -EINVAL; |
| |
| /* TODO: prevent the FG from loading the FG model */ |
| |
| /* set TEX=1 in Config 0x1D, make sure that TEN is enabled */ |
| ret = max_m5_reg_read(data->fg_i2c_client, MAX77759_FG_CONFIG, &config); |
| if (ret == 0) { |
| const u16 val = config | MAX77759_FG_CONFIG_TEN | MAX77759_FG_CONFIG_TEX; |
| |
| /* TEN should be enabled for this to work */ |
| WARN_ON(!(config & MAX77759_FG_CONFIG_TEN)); |
| |
| ret = max_m5_reg_write(data->fg_i2c_client, MAX77759_FG_CONFIG, |
| val); |
| |
| pr_info("%s: config:%x->%x (%d)\n", __func__, config, val, ret); |
| } |
| if (ret == -ENODEV) { |
| pr_err("%s: no support for max_m5 FG (%d)\n", __func__, ret); |
| *value = 25; |
| return 0; |
| } else if (ret < 0) { |
| pr_err("%s: cannot change FG Config (%d)\n", __func__, ret); |
| return -EIO; |
| } |
| |
| /* set THMIO_MUX */ |
| ret = max77729_pmic_rd8(data, MAX77759_PMIC_CONTROL_FG, &pmic_ctrl); |
| if (ret == 0) { |
| const u8 val = _pmic_control_fg_thmio_mux_set(pmic_ctrl, mux); |
| |
| ret = max77729_pmic_wr8(data, MAX77759_PMIC_CONTROL_FG, val); |
| |
| pr_info("%s: pmic_ctrl:%x->%x (%d)\n", __func__, pmic_ctrl, val, ret); |
| } |
| |
| if (ret < 0) { |
| pr_err("%s: cannot change MUX config (%d)\n", __func__, ret); |
| goto restore_fg; |
| } |
| |
| /* msleep is uninterruptible */ |
| msleep(1500); |
| |
| ret = max_m5_reg_read(data->fg_i2c_client, MAX77759_FG_AIN0, &ain0); |
| pr_debug("%s: AIN0=%d (%d)\n", __func__, ain0, ret); |
| if (ret < 0) { |
| pr_err("%s: cannot read AIN0 (%d)\n", __func__, ret); |
| } else if (mux == THMIO_MUX_USB_TEMP || mux == THMIO_MUX_BATT_PACK) { |
| /* convert form 1.8V to 2.4V and get higher 10 bits */ |
| const unsigned int conv_adc = ((ain0 * 1800) / 2400) >> 6; |
| |
| /* Temp = (rawadc < 185)? (960-rawadc/4) : (730-rawadc/8) */ |
| /* unit: 0.1 degree C */ |
| if (conv_adc < NTC_CURVE_THRESHOLD) |
| *value = NTC_CURVE_1_BASE - ((conv_adc * 10) >> NTC_CURVE_1_SHIFT); |
| else |
| *value = NTC_CURVE_2_BASE - ((conv_adc * 10) >> NTC_CURVE_2_SHIFT); |
| } else { |
| /* AIN0 is ratiometric on THM, 0xffff = 100%, lsb is 2^-16 */ |
| *value = (100000 * (unsigned long)ain0) / (0x10000 - ain0); |
| } |
| |
| /* restore THMIO_MUX */ |
| tmp = max77729_pmic_wr8(data, MAX77759_PMIC_CONTROL_FG, pmic_ctrl); |
| WARN_ON(tmp != 0); |
| |
| /* And reset the pmic if cannot restore (b/191319560) */ |
| tmp = max77729_pmic_rd8(data, MAX77759_PMIC_CONTROL_FG, &check_pmic); |
| if (tmp != 0 || pmic_ctrl != check_pmic) { |
| dev_err(data->dev, "Cannot restore TMUX ret=%d\n", tmp); |
| BUG_ON(tmp != 0 || pmic_ctrl != check_pmic); |
| } |
| |
| restore_fg: |
| |
| /* Clear TEX=0 in Config, restore 0x1D (b/191319560) */ |
| config &= ~MAX77759_FG_CONFIG_TEX; |
| tmp = max_m5_reg_write(data->fg_i2c_client, MAX77759_FG_CONFIG, config); |
| WARN_ON(tmp != 0); |
| |
| /* And reset the FG if this fails (b/191319560) */ |
| tmp = max_m5_reg_read(data->fg_i2c_client, MAX77759_FG_CONFIG, &check_config); |
| if (tmp != 0 || config != check_config) { |
| tmp = max17x0x_sw_reset(data->fg_i2c_client); |
| dev_err(data->dev, "Cannot restore FG Config, FG reset ret=%d\n", tmp); |
| BUG_ON(tmp != 0); |
| } else if (!(check_config & MAX77759_FG_CONFIG_TEN)) { |
| dev_warn(data->dev, "TEN bit is not set in Config=%x\n", check_config); |
| } |
| |
| /* TODO: allow the FG to load the FG model */ |
| |
| pr_info("%s: check_pmic=%x check_config=%x (%d)\n", __func__, |
| check_pmic, check_config, ret); |
| |
| return ret; |
| } |
| |
| /* THMIO_MUX=0 in CONTROL_FG (0x51) */ |
| int max77759_read_batt_conn(struct i2c_client *client, int *temp) |
| { |
| struct max77729_pmic_data *data = i2c_get_clientdata(client); |
| unsigned int val; |
| int ret; |
| |
| mutex_lock(&data->io_lock); |
| ret = max77759_find_fg(data); |
| if (ret == 0) |
| ret = max77759_read_thm(data, THMIO_MUX_BATT_PACK, &val); |
| mutex_unlock(&data->io_lock); |
| |
| if (ret == 0) { |
| /* TODO: b/160737498 convert voltage to temperature */ |
| *temp = val; |
| } |
| |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(max77759_read_batt_conn); |
| |
| /* THMIO_MUX=1 in CONTROL_FG (0x51) */ |
| int max77759_read_usb_temp(struct i2c_client *client, int *temp) |
| { |
| struct max77729_pmic_data *data = i2c_get_clientdata(client); |
| unsigned int val; |
| int ret; |
| |
| mutex_lock(&data->io_lock); |
| ret = max77759_find_fg(data); |
| if (ret == 0) |
| ret = max77759_read_thm(data, THMIO_MUX_USB_TEMP, &val); |
| if (ret == 0) |
| *temp = val; |
| mutex_unlock(&data->io_lock); |
| |
| return ret; |
| } |
| EXPORT_SYMBOL_GPL(max77759_read_usb_temp); |
| |
| /* THMIO_MUX=2 in CONTROL_FG (0x51) */ |
| int max77759_read_batt_id(struct i2c_client *client, unsigned int *id) |
| { |
| struct max77729_pmic_data *data = i2c_get_clientdata(client); |
| unsigned int val; |
| int ret; |
| |
| mutex_lock(&data->io_lock); |
| ret = max77759_find_fg(data); |
| if (ret == 0) |
| ret = max77759_read_thm(data, THMIO_MUX_BATT_ID, &val); |
| if (ret == 0) |
| *id = val; |
| mutex_unlock(&data->io_lock); |
| |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(max77759_read_batt_id); |
| |
| /* must use repeated starts b/152373060 */ |
| static int max77729_pmic_read_id(struct i2c_client *i2c) |
| { |
| struct i2c_msg xfer[2]; |
| u8 reg = MAX77729_PMIC_ID; |
| u8 pmic_id; |
| int ret; |
| |
| xfer[0].addr = i2c->addr; |
| xfer[0].flags = 0; |
| xfer[0].len = 1; |
| xfer[0].buf = ® |
| |
| xfer[1].addr = i2c->addr; |
| xfer[1].flags = I2C_M_RD; |
| xfer[1].len = 1; |
| xfer[1].buf = &pmic_id; |
| |
| ret = i2c_transfer(i2c->adapter, xfer, 2); |
| if (ret == 2) |
| return pmic_id; |
| return -EIO; |
| } |
| |
| /* cahe the value */ |
| static int debug_batt_thm_id_get(void *d, u64 *val) |
| { |
| struct max77729_pmic_data *data = d; |
| int ret, value; |
| |
| ret = max77759_read_batt_id(data->pmic_i2c_client, &value); |
| if (ret == 0) |
| *val = value; |
| return ret; |
| } |
| DEFINE_SIMPLE_ATTRIBUTE(debug_batt_thm_id_fops, debug_batt_thm_id_get, NULL, "%llu\n"); |
| |
| static int debug_batt_thm_conn_get(void *d, u64 *val) |
| { |
| struct max77729_pmic_data *data = d; |
| int ret, value; |
| |
| ret = max77759_read_batt_conn(data->pmic_i2c_client, &value); |
| if (ret == 0) |
| *val = value; |
| |
| return ret; |
| } |
| DEFINE_SIMPLE_ATTRIBUTE(debug_batt_thm_conn_fops, debug_batt_thm_conn_get, NULL, "%llu\n"); |
| |
| static int max777x9_pmic_debug_reg_read(void *d, u64 *val) |
| { |
| struct max77729_pmic_data *data = d; |
| u8 reg = 0; |
| int ret; |
| |
| ret = max77729_pmic_rd8(data, data->debug_reg_address, ®); |
| if (ret) |
| return ret; |
| *val = reg; |
| return 0; |
| } |
| |
| static int max777x9_pmic_debug_reg_write(void *d, u64 val) |
| { |
| struct max77729_pmic_data *data = d; |
| u8 reg = (u8) val; |
| |
| pr_warn("debug write reg 0x%x, 0x%x", data->debug_reg_address, reg); |
| return max77729_pmic_wr8(data, data->debug_reg_address, reg); |
| } |
| DEFINE_SIMPLE_ATTRIBUTE(debug_reg_rw_fops, max777x9_pmic_debug_reg_read, |
| max777x9_pmic_debug_reg_write, "%02llx\n"); |
| |
| static ssize_t max777x9_pmic_debug_show_reg_all(struct file *filp, char __user *buf, |
| size_t count, loff_t *ppos) |
| { |
| struct max77729_pmic_data *data = (struct max77729_pmic_data *)filp->private_data; |
| u32 reg_address; |
| u8 reg = 0; |
| char *tmp; |
| int ret = 0, len = 0; |
| |
| if (!data->regmap) { |
| pr_err("Failed to read, no regmap\n"); |
| return -EIO; |
| } |
| |
| tmp = kmalloc(PAGE_SIZE, GFP_KERNEL); |
| if (!tmp) |
| return -ENOMEM; |
| |
| for (reg_address = 0; reg_address <= 0xFF; reg_address++) { |
| /* reasonable registers */ |
| if (!max77729_pmic_is_reg(data->dev, reg_address)) |
| continue; |
| |
| ret = max77729_pmic_rd8(data, reg_address, ®); |
| if (ret < 0) |
| continue; |
| |
| len += scnprintf(tmp + len, PAGE_SIZE - len, "%02x: %02x\n", reg_address, reg); |
| } |
| |
| if (len > 0) |
| len = simple_read_from_buffer(buf, count, ppos, tmp, strlen(tmp)); |
| |
| kfree(tmp); |
| |
| return len; |
| } |
| |
| BATTERY_DEBUG_ATTRIBUTE(debug_all_reg_fops, max777x9_pmic_debug_show_reg_all, NULL); |
| |
| static int max77759_pmic_storage_iter(int index, gbms_tag_t *tag, void *ptr) |
| { |
| if (index < 0 || index > (GBMS_TAG_RRS7 - GBMS_TAG_RRS0)) |
| return -ENOENT; |
| |
| *tag = GBMS_TAG_RRS0 + index; |
| return 0; |
| } |
| |
| static int max77759_pmic_storage_read(gbms_tag_t tag, void *buff, size_t size, void *ptr) |
| { |
| const int base = MAX77759_STORAGE_BASE + tag - GBMS_TAG_RRS0; |
| struct max77729_pmic_data *data = ptr; |
| int ret; |
| |
| if (tag < GBMS_TAG_RRS0 || tag > GBMS_TAG_RRS7) |
| return -ENOENT; |
| if ((tag + size - 1) > GBMS_TAG_RRS7) |
| return -ERANGE; |
| |
| ret = max77729_pmic_readn(data, base, buff, size); |
| if (ret < 0) |
| ret = -EIO; |
| return ret; |
| } |
| |
| static int max77759_pmic_storage_write(gbms_tag_t tag, const void *buff, size_t size, void *ptr) |
| { |
| const int base = MAX77759_STORAGE_BASE + tag - GBMS_TAG_RRS0; |
| struct max77729_pmic_data *data = ptr; |
| int ret; |
| |
| if (tag < GBMS_TAG_RRS0 || tag > GBMS_TAG_RRS7) |
| return -ENOENT; |
| if ((tag + size - 1) > GBMS_TAG_RRS7) |
| return -ERANGE; |
| |
| ret = max77729_pmic_writen(data, base, buff, size); |
| if (ret < 0) |
| ret = -EIO; |
| return ret; |
| } |
| |
| static struct gbms_storage_desc max77759_pmic_storage_dsc = { |
| .iter = max77759_pmic_storage_iter, |
| .read = max77759_pmic_storage_read, |
| .write = max77759_pmic_storage_write, |
| }; |
| |
| #define STORAGE_INIT_DELAY_MS 100 |
| #define STORAGE_INIT_MAX_RETRY 3 |
| static void max777x9_pmic_storage_init_work(struct work_struct *work) |
| { |
| struct max77729_pmic_data *data = container_of(work, struct max77729_pmic_data, |
| storage_init_work.work); |
| static int retry_cnt; |
| int ret = 0; |
| |
| ret = gbms_storage_register(&max77759_pmic_storage_dsc, |
| "max777x9_pmic_storage", data); |
| |
| if (ret == 0) { |
| pr_info("register storage done\n"); |
| } else if (retry_cnt >= STORAGE_INIT_MAX_RETRY) { |
| pr_info("register storage:%d retry_cnt=%d, stop retry.\n", ret, retry_cnt); |
| } else { |
| schedule_delayed_work(&data->storage_init_work, |
| msecs_to_jiffies(STORAGE_INIT_DELAY_MS)); |
| retry_cnt++; |
| } |
| |
| return; |
| } |
| |
| static int dbg_init_fs(struct max77729_pmic_data *data) |
| { |
| data->de = debugfs_create_dir("max77729_pmic", 0); |
| if (IS_ERR_OR_NULL(data->de)) |
| return -EINVAL; |
| |
| debugfs_create_atomic_t("sysuvlo_cnt", 0644, data->de, &data->sysuvlo_cnt); |
| debugfs_create_atomic_t("sysovlo_cnt", 0644, data->de, &data->sysovlo_cnt); |
| |
| debugfs_create_file("batt_id", 0400, data->de, data, |
| &debug_batt_thm_id_fops); |
| debugfs_create_file("batt_thm_conn", 0400, data->de, data, |
| &debug_batt_thm_conn_fops); |
| |
| debugfs_create_u32("address", 0600, data->de, &data->debug_reg_address); |
| debugfs_create_file("data", 0600, data->de, data, &debug_reg_rw_fops); |
| debugfs_create_file("registers", 0444, data->de, data, &debug_all_reg_fops); |
| |
| return 0; |
| } |
| |
| #if IS_ENABLED(CONFIG_GPIOLIB) |
| |
| /* offset is gpionum - 1 */ |
| static int max77759_gpio_get_direction(struct gpio_chip *chip, |
| unsigned int offset) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(chip); |
| int rc; |
| uint8_t val; |
| |
| if ((offset < MAX77759_MIN_GPIO_OFF) || (offset > MAX77759_MAX_GPIO_OFF)) |
| return -EINVAL; |
| |
| rc = maxq_gpio_control_read(data->maxq, &val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode read 0x23 failed\n"); |
| return rc; |
| } |
| |
| if (offset == MAX77759_GPIO5_OFF) |
| return !(val & MAX77759_GPIO5_DIR_MASK); |
| |
| return !(val & MAX77759_GPIO6_DIR_MASK); |
| } |
| |
| /* offset is gpionum - 1 */ |
| static int max77759_gpio_get(struct gpio_chip *chip, unsigned int offset) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(chip); |
| int rc; |
| uint8_t val; |
| |
| if ((offset < MAX77759_MIN_GPIO_OFF) || (offset > MAX77759_MAX_GPIO_OFF)) |
| return -EINVAL; |
| |
| rc = maxq_gpio_control_read(data->maxq, &val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode read 0x23 failed\n"); |
| return rc; |
| } |
| |
| if (offset == MAX77759_GPIO5_OFF) |
| return !!(val & MAX77759_GPIO5_VAL_MASK); |
| |
| return !!(val & MAX77759_GPIO6_VAL_MASK); |
| } |
| |
| /* offset is gpionum - 1 */ |
| static void max77759_gpio_set(struct gpio_chip *chip, |
| unsigned int offset, int value) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(chip); |
| int rc; |
| uint8_t val; |
| uint8_t new_val; |
| uint8_t dir; |
| |
| if ((offset < MAX77759_MIN_GPIO_OFF) || (offset > MAX77759_MAX_GPIO_OFF)) |
| return; |
| |
| rc = maxq_gpio_control_read(data->maxq, &val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode read 0x23 failed\n"); |
| return; |
| } |
| |
| if (offset == MAX77759_GPIO5_OFF) { |
| dir = !(val & MAX77759_GPIO5_DIR_MASK); |
| if (dir != GPIOF_DIR_OUT) { |
| dev_err(data->dev, "not output\n"); |
| return; |
| } |
| new_val = val & ~MAX77759_GPIO5_VAL_MASK; |
| new_val |= MAX77759_GPIO5_VAL(value); |
| } else { /* MAX77759_GPIO6_OFF */ |
| dir = !(val & MAX77759_GPIO6_DIR_MASK); |
| if (dir != GPIOF_DIR_OUT) { |
| dev_err(data->dev, "not output\n"); |
| return; |
| } |
| new_val = val & ~MAX77759_GPIO6_VAL_MASK; |
| new_val |= MAX77759_GPIO6_VAL(value); |
| } |
| |
| if (new_val != val) { |
| rc = maxq_gpio_control_write(data->maxq, new_val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode write 0x24 failed\n"); |
| return; |
| } |
| } |
| } |
| |
| /* offset is gpionum - 1 */ |
| static int max77759_gpio_direction_input(struct gpio_chip *chip, |
| unsigned int offset) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(chip); |
| int rc; |
| uint8_t val; |
| uint8_t new_val; |
| |
| if ((offset < MAX77759_MIN_GPIO_OFF) || (offset > MAX77759_MAX_GPIO_OFF)) |
| return -EINVAL; |
| |
| rc = maxq_gpio_control_read(data->maxq, &val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode read 0x23 failed\n"); |
| return rc; |
| } |
| |
| if (offset == MAX77759_GPIO5_OFF) { |
| new_val = val & ~MAX77759_GPIO5_DIR_MASK; |
| new_val |= MAX77759_GPIO5_DIR(MAX77759_GPIO_DIR_IN); |
| } else { /* MAX77759_GPIO6_OFF */ |
| new_val = val & ~MAX77759_GPIO6_DIR_MASK; |
| new_val |= MAX77759_GPIO6_DIR(MAX77759_GPIO_DIR_IN); |
| } |
| |
| if (new_val != val) { |
| rc = maxq_gpio_control_write(data->maxq, new_val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode write 0x24 failed\n"); |
| return rc; |
| } |
| } |
| |
| return 0; |
| } |
| |
| /* offset is gpionum - 1 */ |
| static int max77759_gpio_direction_output(struct gpio_chip *chip, |
| unsigned int offset, int value) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(chip); |
| int rc; |
| uint8_t val; |
| uint8_t new_val; |
| |
| if ((offset < MAX77759_MIN_GPIO_OFF) || (offset > MAX77759_MAX_GPIO_OFF)) |
| return -EINVAL; |
| |
| rc = maxq_gpio_control_read(data->maxq, &val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode read 0x23 failed\n"); |
| return rc; |
| } |
| |
| if (offset == MAX77759_GPIO5_OFF) { |
| new_val = val & ~MAX77759_GPIO5_DIR_MASK; |
| new_val |= MAX77759_GPIO5_DIR(MAX77759_GPIO_DIR_OUT); |
| } else { /* MAX77759_GPIO6_OFF */ |
| new_val = val & ~MAX77759_GPIO6_DIR_MASK; |
| new_val |= MAX77759_GPIO6_DIR(MAX77759_GPIO_DIR_OUT); |
| } |
| |
| if (new_val != val) { |
| rc = maxq_gpio_control_write(data->maxq, new_val); |
| if (rc < 0) { |
| dev_err(data->dev, "opcode write 0x24 failed\n"); |
| return rc; |
| } |
| } |
| |
| return 0; |
| } |
| |
| /* d->hwirq is same as offset */ |
| static void max77729_gpio_irq_mask(struct irq_data *d) |
| { |
| struct gpio_chip *gc = irq_data_get_irq_chip_data(d); |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| |
| data->irq_mask |= 1 << d->hwirq; |
| data->irq_mask_u |= 1 << d->hwirq; |
| } |
| |
| static void max77729_gpio_irq_unmask(struct irq_data *d) |
| { |
| struct gpio_chip *gc = irq_data_get_irq_chip_data(d); |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| |
| data->irq_mask &= ~(1 << d->hwirq); |
| data->irq_mask_u |= 1 << d->hwirq; |
| } |
| |
| static void max77729_gpio_irq_enable(struct irq_data *d) |
| { |
| max77729_gpio_irq_unmask(d); |
| } |
| |
| static void max77729_gpio_irq_disable(struct irq_data *d) |
| { |
| max77729_gpio_irq_mask(d); |
| } |
| |
| /* called in atomic context */ |
| static int max77729_gpio_set_irq_type(struct irq_data *d, unsigned int type) |
| { |
| struct gpio_chip *gc = irq_data_get_irq_chip_data(d); |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| const int index = d->hwirq - MAX77759_GPIO5_OFF; |
| |
| switch (type) { |
| case IRQF_TRIGGER_FALLING: |
| data->irq_trig_falling[index] = 1; |
| break; |
| case IRQF_TRIGGER_RISING: |
| data->irq_trig_falling[index] = 0; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| data->irq_trig_u |= 1 << d->hwirq; |
| return 0; |
| } |
| |
| static void max77729_gpio_bus_lock(struct irq_data *d) |
| { |
| struct gpio_chip *gc = irq_data_get_irq_chip_data(d); |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| |
| mutex_lock(&data->irq_lock); |
| } |
| |
| static int max77729_gpio_bus_irq_update_trig(struct max77729_pmic_data *data, |
| int offset, int trig) |
| { |
| int ret; |
| |
| /* direction works with offset, trigger works with gpio number */ |
| ret = max77759_gpio_direction_input(&data->gpio, offset); |
| if (ret == 0) |
| ret = maxq_gpio_trigger_write(data->maxq, offset + 1, trig); |
| |
| pr_debug("gpio%d: trig=%d (%d)\n", offset + 1, trig, ret); |
| |
| return ret; |
| } |
| |
| static int max77729_gpio_bus_irq_update_mask(struct max77729_pmic_data *data, |
| int offset, int value) |
| { |
| unsigned int mask = 0, val = 0; |
| int ret; |
| |
| ret = max77729_gpio_to_irq_mask(offset, &mask, &val); |
| if (ret == 0) |
| ret = max77729_pmic_rmw8(data, MAX77759_PMIC_UIC_INT1_M, mask, |
| value ? val : 0); |
| if (ret < 0) { |
| dev_err(data->dev, "gpio%d: cannot change mask=%x to %x (%d)\n", |
| offset + 1, mask, value ? val : 0, ret); |
| return ret; |
| } |
| |
| pr_debug("gpio%d: value=%d mask=%x, val=%x (%d)\n", |
| offset +1, value, mask, val, ret); |
| |
| return 0; |
| } |
| |
| /* cannot call any maxq function in atomic */ |
| static void max77729_gpio_bus_sync_unlock(struct irq_data *d) |
| { |
| struct gpio_chip *gc = irq_data_get_irq_chip_data(d); |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| unsigned int offset, value; |
| |
| while (data->irq_trig_u) { |
| offset = __ffs(data->irq_trig_u); |
| value = data->irq_trig_falling[offset - MAX77759_GPIO5_OFF]; |
| |
| max77729_gpio_bus_irq_update_trig(data, offset, value); |
| data->irq_trig_u &= ~(1 << offset); |
| } |
| |
| while (data->irq_mask_u) { |
| offset = __ffs(data->irq_mask_u); |
| value = data->irq_mask & (1 << offset); |
| |
| max77729_gpio_bus_irq_update_mask(data, offset, value); |
| data->irq_mask_u &= ~(1 << offset); |
| } |
| |
| mutex_unlock(&data->irq_lock); |
| } |
| |
| /* bits 4 and 5 */ |
| static void max77729_gpio_set_irq_valid_mask(struct gpio_chip *chip, |
| unsigned long *valid_mask, |
| unsigned int ngpios) |
| { |
| bitmap_clear(valid_mask, 0, ngpios); |
| *valid_mask = (1 << MAX77759_GPIO5_OFF) | (1 << MAX77759_GPIO6_OFF); |
| } |
| |
| /* only support 5 and 6, 5 is output */ |
| static int max77729_gpio_irq_init_hw(struct gpio_chip *gc) |
| { |
| struct max77729_pmic_data *data = gpiochip_get_data(gc); |
| const u8 mask_gpio = MAX77759_PMIC_UIC_INT1_GPIO5I | |
| MAX77759_PMIC_UIC_INT1_GPIO6I; |
| int ret; |
| |
| /* mask both */ |
| ret = max77729_pmic_rmw8(data, MAX77759_PMIC_UIC_INT1_M, |
| mask_gpio, mask_gpio); |
| if (ret < 0) |
| dev_err(data->dev, "cannot mask IRQs\n"); |
| |
| /* ...and clear both */ |
| ret = max77729_pmic_wr8(data, MAX77759_PMIC_UIC_INT1, |
| MAX77759_PMIC_UIC_INT1_GPIO5I | |
| MAX77759_PMIC_UIC_INT1_GPIO6I); |
| if (ret < 0) |
| dev_err(data->dev, "cannot clear IRQs\n"); |
| |
| return 0; |
| } |
| static struct irq_chip max77729_gpio_irq_chip = { |
| .name = "max777x9_irq", |
| .irq_enable = max77729_gpio_irq_enable, |
| .irq_disable = max77729_gpio_irq_disable, |
| .irq_mask = max77729_gpio_irq_mask, |
| .irq_unmask = max77729_gpio_irq_unmask, |
| .irq_set_type =max77729_gpio_set_irq_type, |
| .irq_bus_lock = max77729_gpio_bus_lock, |
| .irq_bus_sync_unlock = max77729_gpio_bus_sync_unlock, |
| }; |
| |
| #endif |
| |
| /* ----------------------------------------------------------------------- */ |
| |
| |
| static int max77729_pmic_probe(struct i2c_client *client, |
| const struct i2c_device_id *id) |
| { |
| struct device *dev = &client->dev; |
| struct max77729_pmic_data *data; |
| int irq_gpio, pmic_id, ret =0; |
| |
| pmic_id = max77729_pmic_read_id(client); |
| if (pmic_id < 0) |
| return -ENODEV; |
| if (pmic_id == MAX77759_PMIC_PMIC_ID_MW) |
| max777x9_pmic_regmap_cfg.max_register = 0xe0; |
| |
| data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL); |
| if (!data) |
| return -ENOMEM; |
| |
| data->dev = dev; |
| data->pmic_id = pmic_id; |
| data->batt_id = -1; |
| mutex_init(&data->io_lock); |
| atomic_set(&data->sysuvlo_cnt, 0); |
| atomic_set(&data->sysovlo_cnt, 0); |
| i2c_set_clientdata(client, data); |
| data->pmic_i2c_client = client; |
| |
| INIT_DELAYED_WORK(&data->storage_init_work, max777x9_pmic_storage_init_work); |
| |
| data->regmap = devm_regmap_init_i2c(client, &max777x9_pmic_regmap_cfg); |
| if (IS_ERR(data->regmap)) { |
| dev_err(dev, "Failed to initialize regmap\n"); |
| return -EINVAL; |
| } |
| |
| if (pmic_id == MAX77759_PMIC_PMIC_ID_MW) { |
| const int poll_en = of_property_read_bool(dev->of_node, |
| "goog,maxq-poll"); |
| u8 pmic_ctrl; |
| |
| |
| data->maxq = maxq_init(dev, data->regmap, poll_en); |
| if (IS_ERR_OR_NULL(data->maxq)) { |
| dev_err(dev, "Maxq init failed!\n"); |
| ret = PTR_ERR(data->maxq); |
| } |
| |
| ret = max77729_pmic_rd8(data, MAX77759_PMIC_CONTROL_FG, &pmic_ctrl); |
| WARN_ON(ret != 0 ||((pmic_ctrl & MAX77759_PMIC_CONTROL_FG_THMIO_MUX_MASK) |
| != THMIO_MUX_BATT_PACK)); |
| if (ret == 0) { |
| const u8 val = _pmic_control_fg_thmio_mux_set(pmic_ctrl, |
| THMIO_MUX_BATT_PACK); |
| |
| ret = max77729_pmic_wr8(data, MAX77759_PMIC_CONTROL_FG, val); |
| WARN_ON(ret != 0); |
| } |
| } |
| |
| irq_gpio = of_get_named_gpio(dev->of_node, "max777x9,irq-gpio", 0); |
| if (irq_gpio < 0) { |
| dev_err(dev, "irq is not defined\n"); |
| } else { |
| client->irq = gpio_to_irq(irq_gpio); |
| |
| /* NOTE: all interrupts are masked here */ |
| ret = devm_request_threaded_irq(data->dev, client->irq, NULL, |
| max777x9_pmic_irq, |
| IRQF_TRIGGER_LOW | |
| IRQF_SHARED | |
| IRQF_ONESHOT, |
| "max777x9_pmic", |
| data); |
| if (ret < 0) { |
| dev_err(dev, "failed get irq thread\n"); |
| } else { |
| /* force clear pending before unmasking */ |
| max777x9_pmic_irq(0, data); |
| |
| /* NOTE: only enable the maxq interrupt */ |
| ret = max777x9_pmic_set_irqmask(data); |
| if (ret < 0) |
| dev_err(dev, "failed to apply irq mask\n"); |
| } |
| } |
| |
| ret = dbg_init_fs(data); |
| if (ret < 0) |
| dev_err(dev, "Failed to initialize debug fs\n"); |
| |
| if (pmic_id == MAX77759_PMIC_PMIC_ID_MW) { |
| u8 rev_reg; |
| int rc = 0; |
| |
| rc = max77729_pmic_rd8(data, MAX77759_PMIC_PMIC_REVISION, &rev_reg); |
| if (rc < 0) { |
| dev_err(dev, "Failed to read revision\n"); |
| data->rev_id = 0; |
| } else { |
| data->rev_id = _pmic_pmic_revision_rev_get(rev_reg); |
| } |
| |
| if (data->rev_id == MAX77759_PMIC_REV_A0) |
| schedule_delayed_work(&data->storage_init_work, 0); |
| } |
| |
| #if IS_ENABLED(CONFIG_GPIOLIB) |
| mutex_init(&data->irq_lock); |
| |
| if (pmic_id == MAX77759_PMIC_PMIC_ID_MW) { |
| struct gpio_irq_chip *girq = &data->gpio.irq; |
| |
| /* Setup GPIO controller */ |
| data->gpio.owner = THIS_MODULE; |
| data->gpio.parent = dev; |
| data->gpio.label = "max777x9_gpio"; |
| data->gpio.get_direction = max77759_gpio_get_direction; |
| data->gpio.direction_input = max77759_gpio_direction_input; |
| data->gpio.direction_output = max77759_gpio_direction_output; |
| data->gpio.get = max77759_gpio_get; |
| data->gpio.set = max77759_gpio_set; |
| data->gpio.ngpio = MAX77759_NUM_GPIOS; |
| data->gpio.can_sleep = true; |
| data->gpio.base = -1; |
| data->gpio.of_node = of_find_node_by_name(dev->of_node, |
| data->gpio.label); |
| if (!data->gpio.of_node) |
| dev_err(dev, "Failed to find %s DT node\n", data->gpio.label); |
| |
| /* check regmap-irq */ |
| girq->chip = &max77729_gpio_irq_chip; |
| girq->default_type = IRQ_TYPE_NONE; |
| girq->handler = handle_simple_irq; |
| girq->parent_handler = NULL; |
| girq->num_parents = 0; |
| girq->parents = NULL; |
| girq->threaded = true; |
| girq->init_hw = max77729_gpio_irq_init_hw; |
| girq->init_valid_mask = max77729_gpio_set_irq_valid_mask; |
| girq->first = 0; |
| |
| ret = devm_gpiochip_add_data(dev, &data->gpio, data); |
| if (ret) |
| dev_err(dev, "Failed to initialize gpio chip\n"); |
| |
| } |
| #endif |
| |
| dev_info(dev, "probe_done pmic_id = %x, rev_id= %x\n", pmic_id, data->rev_id); |
| return ret; |
| } |
| |
| static int max77729_pmic_remove(struct i2c_client *client) |
| { |
| struct max77729_pmic_data *data = i2c_get_clientdata(client); |
| |
| maxq_remove(data->maxq); |
| return 0; |
| } |
| |
| static const struct of_device_id max77729_pmic_of_match_table[] = { |
| { .compatible = "maxim,max77729pmic" }, |
| { .compatible = "maxim,max77759pmic" }, |
| {}, |
| }; |
| MODULE_DEVICE_TABLE(of, max77729_pmic_of_match_table); |
| |
| static const struct i2c_device_id max77729_pmic_id[] = { |
| {"max77729_pmic", 0}, |
| {} |
| }; |
| MODULE_DEVICE_TABLE(i2c, max77729_pmic_id); |
| |
| static struct i2c_driver max77729_pmic_i2c_driver = { |
| .driver = { |
| .name = "max777x9-pmic", |
| .owner = THIS_MODULE, |
| .of_match_table = max77729_pmic_of_match_table, |
| }, |
| .id_table = max77729_pmic_id, |
| .probe = max77729_pmic_probe, |
| .remove = max77729_pmic_remove, |
| }; |
| |
| module_i2c_driver(max77729_pmic_i2c_driver); |
| MODULE_DESCRIPTION("Maxim 77729 PMIC driver"); |
| MODULE_LICENSE("GPL"); |