// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (C) 2022 Richtek Technology Corp. * * Author: ChiYuan Huang */ #include #include #include #include #include #include #include #include #include "mt6370.h" #define MT6370_REG_DEV_INFO 0x100 #define MT6370_REG_CHG_IRQ1 0x1C0 #define MT6370_REG_CHG_MASK1 0x1E0 #define MT6370_REG_MAXADDR 0x1FF #define MT6370_VENID_MASK GENMASK(7, 4) #define MT6370_NUM_IRQREGS 16 #define MT6370_USBC_I2CADDR 0x4E #define MT6370_MAX_ADDRLEN 2 #define MT6370_VENID_RT5081 0x8 #define MT6370_VENID_RT5081A 0xA #define MT6370_VENID_MT6370 0xE #define MT6370_VENID_MT6371 0xF #define MT6370_VENID_MT6372P 0x9 #define MT6370_VENID_MT6372CP 0xB static const struct regmap_irq mt6370_irqs[] = { REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHGON, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_TREG, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_AICR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_MIVR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_PWR_RDY, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FL_CHG_VINOVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_VSYSUV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_VSYSOV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_VBATOV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_VINOVPCHG, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TS_BAT_COLD, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TS_BAT_COOL, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TS_BAT_WARM, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TS_BAT_HOT, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TS_STATC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_FAULT, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_STATC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_TMR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_BATABS, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_ADPBAD, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_RVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_TSHUTDOWN, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_IINMEAS, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_ICCMEAS, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHGDET_DONE, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_WDTMR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_SSFINISH, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_RECHG, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_TERM, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHG_IEOC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_ADC_DONE, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_PUMPX_DONE, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_BST_BATUV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_BST_MIDOV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_BST_OLP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_ATTACH, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DETACH, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_HVDCP_STPDONE, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_HVDCP_VBUSDET_DONE, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_HVDCP_DET, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_CHGDET, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DCDT, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHG_VGOK, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHG_WDTMR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHG_UC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHG_OC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DIRCHG_OV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OVPCTRL_SWON, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OVPCTRL_UVP_D, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OVPCTRL_UVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OVPCTRL_OVP_D, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OVPCTRL_OVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED_STRBPIN, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED_TORPIN, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED_TX, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED_LVF, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED2_SHORT, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED1_SHORT, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED2_STRB, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED1_STRB, 8), REGMAP_IRQ_REG_LINE(mT6370_IRQ_FLED2_STRB_TO, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED1_STRB_TO, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED2_TOR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_FLED1_TOR, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_OTP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_VDDA_OVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_VDDA_UV, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_LDO_OC, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_BLED_OCP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_BLED_OVP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DSV_VNEG_OCP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DSV_VPOS_OCP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DSV_BST_OCP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DSV_VNEG_SCP, 8), REGMAP_IRQ_REG_LINE(MT6370_IRQ_DSV_VPOS_SCP, 8), }; static const struct regmap_irq_chip mt6370_irq_chip = { .name = "mt6370-irqs", .status_base = MT6370_REG_CHG_IRQ1, .mask_base = MT6370_REG_CHG_MASK1, .num_regs = MT6370_NUM_IRQREGS, .irqs = mt6370_irqs, .num_irqs = ARRAY_SIZE(mt6370_irqs), }; static const struct resource mt6370_regulator_irqs[] = { DEFINE_RES_IRQ_NAMED(MT6370_IRQ_DSV_VPOS_SCP, "db_vpos_scp"), DEFINE_RES_IRQ_NAMED(MT6370_IRQ_DSV_VNEG_SCP, "db_vneg_scp"), DEFINE_RES_IRQ_NAMED(MT6370_IRQ_DSV_BST_OCP, "db_vbst_ocp"), DEFINE_RES_IRQ_NAMED(MT6370_IRQ_DSV_VPOS_OCP, "db_vpos_ocp"), DEFINE_RES_IRQ_NAMED(MT6370_IRQ_DSV_VNEG_OCP, "db_vneg_ocp"), DEFINE_RES_IRQ_NAMED(MT6370_IRQ_LDO_OC, "ldo_oc"), }; static const struct mfd_cell mt6370_devices[] = { MFD_CELL_OF("mt6370-adc", NULL, NULL, 0, 0, "mediatek,mt6370-adc"), MFD_CELL_OF("mt6370-charger", NULL, NULL, 0, 0, "mediatek,mt6370-charger"), MFD_CELL_OF("mt6370-flashlight", NULL, NULL, 0, 0, "mediatek,mt6370-flashlight"), MFD_CELL_OF("mt6370-indicator", NULL, NULL, 0, 0, "mediatek,mt6370-indicator"), MFD_CELL_OF("mt6370-tcpc", NULL, NULL, 0, 0, "mediatek,mt6370-tcpc"), MFD_CELL_RES("mt6370-regulator", mt6370_regulator_irqs), }; static const struct mfd_cell mt6370_exclusive_devices[] = { MFD_CELL_OF("mt6370-backlight", NULL, NULL, 0, 0, "mediatek,mt6370-backlight"), }; static const struct mfd_cell mt6372_exclusive_devices[] = { MFD_CELL_OF("mt6370-backlight", NULL, NULL, 0, 0, "mediatek,mt6372-backlight"), }; static int mt6370_check_vendor_info(struct device *dev, struct regmap *rmap, int *vid) { unsigned int devinfo; int ret; ret = regmap_read(rmap, MT6370_REG_DEV_INFO, &devinfo); if (ret) return ret; *vid = FIELD_GET(MT6370_VENID_MASK, devinfo); switch (*vid) { case MT6370_VENID_RT5081: case MT6370_VENID_RT5081A: case MT6370_VENID_MT6370: case MT6370_VENID_MT6371: case MT6370_VENID_MT6372P: case MT6370_VENID_MT6372CP: return 0; default: dev_err(dev, "Unknown Vendor ID 0x%02x\n", devinfo); return -ENODEV; } } static int mt6370_regmap_read(void *context, const void *reg_buf, size_t reg_size, void *val_buf, size_t val_size) { struct mt6370_info *info = context; const u8 *u8_buf = reg_buf; u8 bank_idx, bank_addr; int ret; bank_idx = u8_buf[0]; bank_addr = u8_buf[1]; ret = i2c_smbus_read_i2c_block_data(info->i2c[bank_idx], bank_addr, val_size, val_buf); if (ret < 0) return ret; if (ret != val_size) return -EIO; return 0; } static int mt6370_regmap_write(void *context, const void *data, size_t count) { struct mt6370_info *info = context; const u8 *u8_buf = data; u8 bank_idx, bank_addr; int len = count - MT6370_MAX_ADDRLEN; bank_idx = u8_buf[0]; bank_addr = u8_buf[1]; return i2c_smbus_write_i2c_block_data(info->i2c[bank_idx], bank_addr, len, data + MT6370_MAX_ADDRLEN); } static const struct regmap_bus mt6370_regmap_bus = { .read = mt6370_regmap_read, .write = mt6370_regmap_write, }; static const struct regmap_config mt6370_regmap_config = { .reg_bits = 16, .val_bits = 8, .reg_format_endian = REGMAP_ENDIAN_BIG, .max_register = MT6370_REG_MAXADDR, }; static int mt6370_probe(struct i2c_client *i2c) { struct mt6370_info *info; struct i2c_client *usbc_i2c; struct regmap *regmap; struct device *dev = &i2c->dev; int ret, vid; info = devm_kzalloc(dev, sizeof(*info), GFP_KERNEL); if (!info) return -ENOMEM; usbc_i2c = devm_i2c_new_dummy_device(dev, i2c->adapter, MT6370_USBC_I2CADDR); if (IS_ERR(usbc_i2c)) return dev_err_probe(dev, PTR_ERR(usbc_i2c), "Failed to register USBC I2C client\n"); /* Assign I2C client for PMU and TypeC */ info->i2c[MT6370_PMU_I2C] = i2c; info->i2c[MT6370_USBC_I2C] = usbc_i2c; regmap = devm_regmap_init(dev, &mt6370_regmap_bus, info, &mt6370_regmap_config); if (IS_ERR(regmap)) return dev_err_probe(dev, PTR_ERR(regmap), "Failed to init regmap\n"); ret = mt6370_check_vendor_info(dev, regmap, &vid); if (ret) return dev_err_probe(dev, ret, "Failed to check vendor info\n"); ret = devm_regmap_add_irq_chip(dev, regmap, i2c->irq, IRQF_ONESHOT, -1, &mt6370_irq_chip, &info->irq_data); if (ret) return dev_err_probe(dev, ret, "Failed to add irq chip\n"); switch (vid) { case MT6370_VENID_MT6372P: case MT6370_VENID_MT6372CP: ret = devm_mfd_add_devices(dev, PLATFORM_DEVID_AUTO, mt6372_exclusive_devices, ARRAY_SIZE(mt6372_exclusive_devices), NULL, 0, regmap_irq_get_domain(info->irq_data)); break; default: ret = devm_mfd_add_devices(dev, PLATFORM_DEVID_AUTO, mt6370_exclusive_devices, ARRAY_SIZE(mt6370_exclusive_devices), NULL, 0, regmap_irq_get_domain(info->irq_data)); break; } if (ret) return dev_err_probe(dev, ret, "Failed to add the exclusive devices\n"); return devm_mfd_add_devices(dev, PLATFORM_DEVID_AUTO, mt6370_devices, ARRAY_SIZE(mt6370_devices), NULL, 0, regmap_irq_get_domain(info->irq_data)); } static const struct of_device_id mt6370_match_table[] = { { .compatible = "mediatek,mt6370" }, {} }; MODULE_DEVICE_TABLE(of, mt6370_match_table); static struct i2c_driver mt6370_driver = { .driver = { .name = "mt6370", .of_match_table = mt6370_match_table, }, .probe_new = mt6370_probe, }; module_i2c_driver(mt6370_driver); MODULE_AUTHOR("ChiYuan Huang "); MODULE_DESCRIPTION("MediaTek MT6370 SubPMIC Driver"); MODULE_LICENSE("GPL v2");