blob: 3c86eb20ab439d904690bea978ca1d7977d11df1 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0+
/*
* AXP PMIC SPL driver
* (C) Copyright 2024 Arm Ltd.
*/
#include <errno.h>
#include <linux/types.h>
#include <asm/arch/pmic_bus.h>
#include <axp_pmic.h>
struct axp_reg_desc_spl {
u8 enable_reg;
u8 enable_mask;
u8 volt_reg;
u8 volt_mask;
u16 min_mV;
u16 max_mV;
u8 step_mV;
u8 split;
};
#define NA 0xff
#if defined(CONFIG_AXP717_POWER) /* AXP717 */
static const struct axp_reg_desc_spl axp_spl_dcdc_regulators[] = {
{ 0x80, BIT(0), 0x83, 0x7f, 500, 1540, 10, 70 },
{ 0x80, BIT(1), 0x84, 0x7f, 500, 1540, 10, 70 },
{ 0x80, BIT(2), 0x85, 0x7f, 500, 1840, 10, 70 },
};
#define AXP_CHIP_VERSION 0x0
#define AXP_CHIP_VERSION_MASK 0x0
#define AXP_CHIP_ID 0x0
#define AXP_SHUTDOWN_REG 0x27
#define AXP_SHUTDOWN_MASK BIT(0)
#elif defined(CONFIG_AXP313_POWER) /* AXP313 */
static const struct axp_reg_desc_spl axp_spl_dcdc_regulators[] = {
{ 0x10, BIT(0), 0x13, 0x7f, 500, 1540, 10, 70 },
{ 0x10, BIT(1), 0x14, 0x7f, 500, 1540, 10, 70 },
{ 0x10, BIT(2), 0x15, 0x7f, 500, 1840, 10, 70 },
};
#define AXP_CHIP_VERSION 0x3
#define AXP_CHIP_VERSION_MASK 0xc8
#define AXP_CHIP_ID 0x48
#define AXP_SHUTDOWN_REG 0x1a
#define AXP_SHUTDOWN_MASK BIT(7)
#elif defined(CONFIG_AXP305_POWER) /* AXP305 */
static const struct axp_reg_desc_spl axp_spl_dcdc_regulators[] = {
{ 0x10, BIT(0), 0x12, 0x7f, 600, 1520, 10, 50 },
{ 0x10, BIT(1), 0x13, 0x1f, 1000, 2550, 50, NA },
{ 0x10, BIT(2), 0x14, 0x7f, 600, 1520, 10, 50 },
{ 0x10, BIT(3), 0x15, 0x3f, 600, 1500, 20, NA },
{ 0x10, BIT(4), 0x16, 0x1f, 1100, 3400, 100, NA },
};
#define AXP_CHIP_VERSION 0x3
#define AXP_CHIP_VERSION_MASK 0xcf
#define AXP_CHIP_ID 0x40
#define AXP_SHUTDOWN_REG 0x32
#define AXP_SHUTDOWN_MASK BIT(7)
#else
#error "Please define the regulator registers in axp_spl_regulators[]."
#endif
static u8 axp_mvolt_to_cfg(int mvolt, const struct axp_reg_desc_spl *reg)
{
if (mvolt < reg->min_mV)
mvolt = reg->min_mV;
else if (mvolt > reg->max_mV)
mvolt = reg->max_mV;
mvolt -= reg->min_mV;
/* voltage in the first range ? */
if (mvolt <= reg->split * reg->step_mV)
return mvolt / reg->step_mV;
mvolt -= reg->split * reg->step_mV;
return reg->split + mvolt / (reg->step_mV * 2);
}
static int axp_set_dcdc(int dcdc_num, unsigned int mvolt)
{
const struct axp_reg_desc_spl *reg;
int ret;
if (dcdc_num < 1 || dcdc_num > ARRAY_SIZE(axp_spl_dcdc_regulators))
return -EINVAL;
reg = &axp_spl_dcdc_regulators[dcdc_num - 1];
if (mvolt == 0)
return pmic_bus_clrbits(reg->enable_reg, reg->enable_mask);
ret = pmic_bus_write(reg->volt_reg, axp_mvolt_to_cfg(mvolt, reg));
if (ret)
return ret;
return pmic_bus_setbits(reg->enable_reg, reg->enable_mask);
}
int axp_set_dcdc1(unsigned int mvolt)
{
return axp_set_dcdc(1, mvolt);
}
int axp_set_dcdc2(unsigned int mvolt)
{
return axp_set_dcdc(2, mvolt);
}
int axp_set_dcdc3(unsigned int mvolt)
{
return axp_set_dcdc(3, mvolt);
}
int axp_set_dcdc4(unsigned int mvolt)
{
return axp_set_dcdc(4, mvolt);
}
int axp_set_dcdc5(unsigned int mvolt)
{
return axp_set_dcdc(5, mvolt);
}
int axp_init(void)
{
int ret = pmic_bus_init();
if (ret)
return ret;
if (AXP_CHIP_VERSION_MASK) {
u8 axp_chip_id;
ret = pmic_bus_read(AXP_CHIP_VERSION, &axp_chip_id);
if (ret)
return ret;
if ((axp_chip_id & AXP_CHIP_VERSION_MASK) != AXP_CHIP_ID) {
debug("unknown PMIC: 0x%x\n", axp_chip_id);
return -EINVAL;
}
}
return 0;
}
#if !CONFIG_IS_ENABLED(ARM_PSCI_FW) && !IS_ENABLED(CONFIG_SYSRESET_CMD_POWEROFF)
int do_poweroff(struct cmd_tbl *cmdtp, int flag, int argc, char *const argv[])
{
pmic_bus_setbits(AXP_SHUTDOWN_REG, AXP_SHUTDOWN_MASK);
/* infinite loop during shutdown */
while (1)
;
/* not reached */
return 0;
}
#endif