// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (c) 2018, 2020 The Linux Foundation. All rights reserved. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved. */ #define pr_fmt(fmt) "SCHG-FLASH: %s: " fmt, __func__ #include #include #include #include #include #include #include #include #include #include #include "smblite-lib.h" #include "schgm-flashlite.h" #define IS_BETWEEN(left, right, value) \ (((left) >= (right) && (left) >= (value) \ && (value) >= (right)) \ || ((left) <= (right) && (left) <= (value) \ && (value) <= (right))) irqreturn_t schgm_flashlite_default_irq_handler(int irq, void *data) { struct smb_irq_data *irq_data = data; pr_debug("IRQ: %s\n", irq_data->name); return IRQ_HANDLED; } irqreturn_t schgm_flashlite_ilim2_irq_handler(int irq, void *data) { struct smb_irq_data *irq_data = data; struct smb_charger *chg = irq_data->parent_data; int rc; rc = smblite_lib_write(chg, SCHGM_FLASH_S2_LATCH_RESET_CMD_REG, FLASH_S2_LATCH_RESET_BIT); if (rc < 0) pr_err("Couldn't reset S2_LATCH reset rc=%d\n", rc); return IRQ_HANDLED; } irqreturn_t schgm_flashlite_state_change_irq_handler(int irq, void *data) { struct smb_irq_data *irq_data = data; struct smb_charger *chg = irq_data->parent_data; int rc; u8 reg; rc = smblite_lib_read(chg, SCHGM_FLASH_STATUS_3_REG, ®); if (rc < 0) pr_err("Couldn't read flash status_3 rc=%d\n", rc); else pr_debug("Flash status changed state=[%x]\n", (reg && FLASH_STATE_MASK)); return IRQ_HANDLED; } #define FIXED_MODE 0 #define ADAPTIVE_MODE 1 static void schgm_flashlite_parse_dt(struct smb_charger *chg) { struct device_node *node = chg->dev->of_node; u32 val; int rc; chg->flash_derating_soc = -EINVAL; rc = of_property_read_u32(node, "qcom,flash-derating-soc", &val); if (!rc) { if (IS_BETWEEN(0, 100, val)) chg->flash_derating_soc = (val * 255) / 100; } chg->flash_disable_soc = -EINVAL; rc = of_property_read_u32(node, "qcom,flash-disable-soc", &val); if (!rc) { if (IS_BETWEEN(0, 100, val)) chg->flash_disable_soc = (val * 255) / 100; } } bool is_flashlite_active(struct smb_charger *chg) { return chg->flash_active ? true : false; } int schgm_flashlite_get_vreg_ok(struct smb_charger *chg, int *val) { int rc, vreg_state; u8 stat = 0; if (!chg->flash_init_done) return -EPERM; rc = smblite_lib_read(chg, SCHGM_FLASH_STATUS_2_REG, &stat); if (rc < 0) { pr_err("Couldn't read FLASH STATUS_2 rc=%d\n", rc); return rc; } vreg_state = !!(stat & VREG_OK_BIT); /* If VREG_OK is not set check for flash error */ if (!vreg_state) { rc = smblite_lib_read(chg, SCHGM_FLASH_STATUS_3_REG, &stat); if (rc < 0) { pr_err("Couldn't read FLASH_STATUS_3 rc=%d\n", rc); return rc; } if ((stat & FLASH_STATE_MASK) == FLASH_ERROR_VAL) { vreg_state = -EFAULT; rc = smblite_lib_read(chg, SCHGM_FLASH_STATUS_5_REG, &stat); if (rc < 0) { pr_err("Couldn't read FLASH_STATUS_5 rc=%d\n", rc); return rc; } pr_debug("Flash error: status=%x\n", stat); } } /* * val can be one of the following: * 1 - VREG_OK is set. * 0 - VREG_OK is 0 but no Flash error. * -EFAULT - Flash Error is set. */ *val = vreg_state; return 0; } void schgm_flashlite_torch_priority(struct smb_charger *chg, enum torch_mode mode) { int rc; u8 reg; if ((mode != TORCH_BOOST_MODE) && (mode != TORCH_BUCK_MODE)) return; reg = mode; rc = smblite_lib_masked_write(chg, SCHGM_TORCH_PRIORITY_CONTROL_REG, TORCH_PRIORITY_CONTROL_BIT, reg); if (rc < 0) pr_err("Couldn't configure Torch priority control rc=%d\n", rc); pr_debug("Torch priority changed to: %d\n", mode); } int schgm_flashlite_config_usbin_collapse(struct smb_charger *chg, bool enable) { return smblite_lib_masked_write(chg, SCHG_L_FLASH_FLASH_FAULT_CFG, CFG_FLASH_USB_COLLAPSE_BIT, enable ? CFG_FLASH_USB_COLLAPSE_BIT : 0); } int schgm_flashlite_init(struct smb_charger *chg) { int rc; u8 reg, mask; schgm_flashlite_parse_dt(chg); if (chg->wa_flags & FLASH_DIE_TEMP_DERATE_WA) { mask = TEMP_DIE_REG_L_DERATE_EN_BIT | TEMP_DIE_REG_H_DERATE_EN_BIT; rc = smblite_lib_masked_write(chg, SCHGM_FLASH_CONTROL_REG, mask, 0); if (rc < 0) pr_err("Couldn't disable die-temp derate rc=%d\n", rc); } if (chg->flash_derating_soc != -EINVAL) { rc = smblite_lib_write(chg, SCHGM_SOC_BASED_FLASH_DERATE_TH_CFG_REG, chg->flash_derating_soc); if (rc < 0) { pr_err("Couldn't configure SOC for flash derating rc=%d\n", rc); return rc; } } if (chg->flash_disable_soc != -EINVAL) { rc = smblite_lib_write(chg, SCHGM_SOC_BASED_FLASH_DISABLE_TH_CFG_REG, chg->flash_disable_soc); if (rc < 0) { pr_err("Couldn't configure SOC for flash disable rc=%d\n", rc); return rc; } } if ((chg->flash_derating_soc != -EINVAL) || (chg->flash_disable_soc != -EINVAL)) { /* Check if SOC based derating/disable is enabled */ rc = smblite_lib_read(chg, SCHGM_FLASH_CONTROL_REG, ®); if (rc < 0) { pr_err("Couldn't read flash control reg rc=%d\n", rc); return rc; } if (!(reg & SOC_LOW_FOR_FLASH_EN_BIT)) pr_warn("Soc based flash derating not enabled\n"); } chg->flash_init_done = true; return 0; }