blob: 919b2930c635346bcb0d4b018afa180d0e119278 [file] [log] [blame]
// SPDX-License-Identifier: BSD-3-Clause
/*
* Copyright (c) 2016-2018, STMicroelectronics - All Rights Reserved
*/
#include <assert.h>
#include <drivers/stpmic1.h>
#include <kernel/panic.h>
#include <platform_config.h>
#include <stdint.h>
#include <string.h>
#include <trace.h>
struct regul_struct {
const char *dt_node_name;
const uint16_t *voltage_table;
uint8_t voltage_table_size;
uint8_t control_reg;
uint8_t low_power_reg;
uint8_t pull_down_reg;
uint8_t pull_down_pos;
uint8_t mask_reset_reg;
uint8_t mask_reset_pos;
};
static struct i2c_handle_s *pmic_i2c_handle;
static uint16_t pmic_i2c_addr;
/* Voltage tables in mV */
static const uint16_t buck1_voltage_table[] = {
725,
725,
725,
725,
725,
725,
750,
775,
800,
825,
850,
875,
900,
925,
950,
975,
1000,
1025,
1050,
1075,
1100,
1125,
1150,
1175,
1200,
1225,
1250,
1275,
1300,
1325,
1350,
1375,
1400,
1425,
1450,
1475,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
1500,
};
static const uint16_t buck2_voltage_table[] = {
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1050,
1050,
1100,
1100,
1150,
1150,
1200,
1200,
1250,
1250,
1300,
1300,
1350,
1350,
1400,
1400,
1450,
1450,
1500,
};
static const uint16_t buck3_voltage_table[] = {
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1000,
1100,
1100,
1100,
1100,
1200,
1200,
1200,
1200,
1300,
1300,
1300,
1300,
1400,
1400,
1400,
1400,
1500,
1600,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
3400,
};
static const uint16_t buck4_voltage_table[] = {
600,
625,
650,
675,
700,
725,
750,
775,
800,
825,
850,
875,
900,
925,
950,
975,
1000,
1025,
1050,
1075,
1100,
1125,
1150,
1175,
1200,
1225,
1250,
1275,
1300,
1300,
1350,
1350,
1400,
1400,
1450,
1450,
1500,
1600,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
3400,
3500,
3600,
3700,
3800,
3900,
};
static const uint16_t ldo1_voltage_table[] = {
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
};
static const uint16_t ldo2_voltage_table[] = {
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
};
static const uint16_t ldo3_voltage_table[] = {
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
3300,
3300,
3300,
3300,
3300,
3300,
0xFFFF, /* VREFDDR */
};
static const uint16_t ldo5_voltage_table[] = {
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
3400,
3500,
3600,
3700,
3800,
3900,
};
static const uint16_t ldo6_voltage_table[] = {
900,
1000,
1100,
1200,
1300,
1400,
1500,
1600,
1700,
1800,
1900,
2000,
2100,
2200,
2300,
2400,
2500,
2600,
2700,
2800,
2900,
3000,
3100,
3200,
3300,
};
static const uint16_t ldo4_voltage_table[] = {
3300,
};
static const uint16_t vref_ddr_voltage_table[] = {
3300,
};
/* Table of Regulators in PMIC SoC */
static const struct regul_struct regulators_table[] = {
{
.dt_node_name = "buck1",
.voltage_table = buck1_voltage_table,
.voltage_table_size = ARRAY_SIZE(buck1_voltage_table),
.control_reg = BUCK1_CONTROL_REG,
.low_power_reg = BUCK1_PWRCTRL_REG,
.pull_down_reg = BUCK_PULL_DOWN_REG,
.pull_down_pos = BUCK1_PULL_DOWN_SHIFT,
.mask_reset_reg = MASK_RESET_BUCK_REG,
.mask_reset_pos = BUCK1_MASK_RESET_SHIFT,
},
{
.dt_node_name = "buck2",
.voltage_table = buck2_voltage_table,
.voltage_table_size = ARRAY_SIZE(buck2_voltage_table),
.control_reg = BUCK2_CONTROL_REG,
.low_power_reg = BUCK2_PWRCTRL_REG,
.pull_down_reg = BUCK_PULL_DOWN_REG,
.pull_down_pos = BUCK2_PULL_DOWN_SHIFT,
.mask_reset_reg = MASK_RESET_BUCK_REG,
.mask_reset_pos = BUCK2_MASK_RESET_SHIFT,
},
{
.dt_node_name = "buck3",
.voltage_table = buck3_voltage_table,
.voltage_table_size = ARRAY_SIZE(buck3_voltage_table),
.control_reg = BUCK3_CONTROL_REG,
.low_power_reg = BUCK3_PWRCTRL_REG,
.pull_down_reg = BUCK_PULL_DOWN_REG,
.pull_down_pos = BUCK3_PULL_DOWN_SHIFT,
.mask_reset_reg = MASK_RESET_BUCK_REG,
.mask_reset_pos = BUCK3_MASK_RESET_SHIFT,
},
{
.dt_node_name = "buck4",
.voltage_table = buck4_voltage_table,
.voltage_table_size = ARRAY_SIZE(buck4_voltage_table),
.control_reg = BUCK4_CONTROL_REG,
.low_power_reg = BUCK4_PWRCTRL_REG,
.pull_down_reg = BUCK_PULL_DOWN_REG,
.pull_down_pos = BUCK4_PULL_DOWN_SHIFT,
.mask_reset_reg = MASK_RESET_BUCK_REG,
.mask_reset_pos = BUCK4_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo1",
.voltage_table = ldo1_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo1_voltage_table),
.control_reg = LDO1_CONTROL_REG,
.low_power_reg = LDO1_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO1_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo2",
.voltage_table = ldo2_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo2_voltage_table),
.control_reg = LDO2_CONTROL_REG,
.low_power_reg = LDO2_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO2_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo3",
.voltage_table = ldo3_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo3_voltage_table),
.control_reg = LDO3_CONTROL_REG,
.low_power_reg = LDO3_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO3_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo4",
.voltage_table = ldo4_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo4_voltage_table),
.control_reg = LDO4_CONTROL_REG,
.low_power_reg = LDO4_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO4_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo5",
.voltage_table = ldo5_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo5_voltage_table),
.control_reg = LDO5_CONTROL_REG,
.low_power_reg = LDO5_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO5_MASK_RESET_SHIFT,
},
{
.dt_node_name = "ldo6",
.voltage_table = ldo6_voltage_table,
.voltage_table_size = ARRAY_SIZE(ldo6_voltage_table),
.control_reg = LDO6_CONTROL_REG,
.low_power_reg = LDO6_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = LDO6_MASK_RESET_SHIFT,
},
{
.dt_node_name = "vref_ddr",
.voltage_table = vref_ddr_voltage_table,
.voltage_table_size = ARRAY_SIZE(vref_ddr_voltage_table),
.control_reg = VREF_DDR_CONTROL_REG,
.low_power_reg = VREF_DDR_PWRCTRL_REG,
.mask_reset_reg = MASK_RESET_LDO_REG,
.mask_reset_pos = VREF_DDR_MASK_RESET_SHIFT,
},
{
.dt_node_name = "boost",
},
{
.dt_node_name = "pwr_sw1",
},
{
.dt_node_name = "pwr_sw2",
},
};
static const struct regul_struct *get_regulator_data(const char *name)
{
unsigned int i = 0;
for (i = 0; i < ARRAY_SIZE(regulators_table); i++)
if (strcmp(name, regulators_table[i].dt_node_name) == 0)
return &regulators_table[i];
/* Regulator not found */
panic(name);
}
static uint8_t voltage_to_index(const char *name, uint16_t millivolts)
{
const struct regul_struct *regul = get_regulator_data(name);
unsigned int i = 0;
assert(regul->voltage_table);
for (i = 0; i < regul->voltage_table_size; i++)
if (regul->voltage_table[i] == millivolts)
return i;
/* Voltage not found */
panic(name);
}
int stpmic1_powerctrl_on(void)
{
return stpmic1_register_update(MAIN_CONTROL_REG, PWRCTRL_PIN_VALID,
PWRCTRL_PIN_VALID);
}
int stpmic1_switch_off(void)
{
return stpmic1_register_update(MAIN_CONTROL_REG, 1,
SOFTWARE_SWITCH_OFF_ENABLED);
}
int stpmic1_regulator_enable(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
return stpmic1_register_update(regul->control_reg, BIT(0), BIT(0));
}
int stpmic1_regulator_disable(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
return stpmic1_register_update(regul->control_reg, 0, BIT(0));
}
uint8_t stpmic1_is_regulator_enabled(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
uint8_t val = 0;
if (stpmic1_register_read(regul->control_reg, &val))
panic();
return val & 0x1;
}
int stpmic1_regulator_voltage_set(const char *name, uint16_t millivolts)
{
uint8_t voltage_index = voltage_to_index(name, millivolts);
const struct regul_struct *regul = get_regulator_data(name);
uint8_t mask = 0;
/* Voltage can be set for buck<N> or ldo<N> (except ldo4) regulators */
if (!strcmp(name, "buck"))
mask = BUCK_VOLTAGE_MASK;
else if (!strcmp(name, "ldo") && strcmp(name, "ldo4"))
mask = LDO_VOLTAGE_MASK;
else
return 0;
return stpmic1_register_update(regul->control_reg,
voltage_index << LDO_BUCK_VOLTAGE_SHIFT,
mask);
}
int stpmic1_regulator_mask_reset_set(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
return stpmic1_register_update(regul->mask_reset_reg,
BIT(regul->mask_reset_pos),
LDO_BUCK_RESET_MASK <<
regul->mask_reset_pos);
}
int stpmic1_bo_enable_unpg(struct stpmic1_bo_cfg *cfg)
{
return stpmic1_register_update(cfg->ctrl_reg, BIT(0), BIT(0));
}
/* Returns 1 if no configuration are expected applied at runtime, 0 otherwise */
int stpmic1_bo_voltage_cfg(const char *name, uint16_t millivolts,
struct stpmic1_bo_cfg *cfg)
{
uint8_t voltage_index = voltage_to_index(name, millivolts);
const struct regul_struct *regul = get_regulator_data(name);
uint8_t mask = 0;
/* Voltage can be set for buck<N> or ldo<N> (except ldo4) regulators */
if (!strcmp(name, "buck"))
mask = BUCK_VOLTAGE_MASK;
else if (!strcmp(name, "ldo") && strcmp(name, "ldo4"))
mask = LDO_VOLTAGE_MASK;
else
return 1;
cfg->ctrl_reg = regul->control_reg;
cfg->value = voltage_index << LDO_BUCK_VOLTAGE_SHIFT;
cfg->mask = mask;
return 0;
}
int stpmic1_bo_voltage_unpg(struct stpmic1_bo_cfg *cfg)
{
return stpmic1_register_update(cfg->ctrl_reg, cfg->value, cfg->mask);
}
int stpmic1_bo_pull_down_cfg(const char *name, struct stpmic1_bo_cfg *cfg)
{
const struct regul_struct *regul = get_regulator_data(name);
cfg->pd_reg = regul->pull_down_reg;
cfg->pd_value = BIT(regul->pull_down_pos);
cfg->pd_mask = LDO_BUCK_PULL_DOWN_MASK << regul->pull_down_pos;
return 0;
}
int stpmic1_bo_pull_down_unpg(struct stpmic1_bo_cfg *cfg)
{
return stpmic1_register_update(cfg->pd_reg, cfg->pd_value,
cfg->pd_mask);
}
int stpmic1_bo_mask_reset_cfg(const char *name, struct stpmic1_bo_cfg *cfg)
{
const struct regul_struct *regul = get_regulator_data(name);
cfg->mrst_reg = regul->mask_reset_reg;
cfg->mrst_value = BIT(regul->mask_reset_pos);
cfg->mrst_mask = LDO_BUCK_RESET_MASK << regul->mask_reset_pos;
return 0;
}
int stpmic1_bo_mask_reset_unpg(struct stpmic1_bo_cfg *cfg)
{
return stpmic1_register_update(cfg->mrst_reg, cfg->mrst_value,
cfg->mrst_mask);
}
int stpmic1_regulator_voltage_get(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
uint8_t value = 0;
uint8_t mask = 0;
/* Voltage can be set for buck<N> or ldo<N> (except ldo4) regulators */
if (!strcmp(name, "buck"))
mask = BUCK_VOLTAGE_MASK;
else if (!strcmp(name, "ldo") && strcmp(name, "ldo4"))
mask = LDO_VOLTAGE_MASK;
else
return 0;
if (stpmic1_register_read(regul->control_reg, &value))
return -1;
value = (value & mask) >> LDO_BUCK_VOLTAGE_SHIFT;
if (value > regul->voltage_table_size)
return -1;
return regul->voltage_table[value];
}
int stpmic1_lp_copy_reg(const char *name)
{
const struct regul_struct *regul = get_regulator_data(name);
uint8_t val = 0;
int status = 0;
status = stpmic1_register_read(regul->control_reg, &val);
if (status)
return status;
return stpmic1_register_write(regul->low_power_reg, val);
}
int stpmic1_lp_cfg(const char *name, struct stpmic1_lp_cfg *cfg)
{
const struct regul_struct *regul = get_regulator_data(name);
cfg->ctrl_reg = regul->control_reg;
cfg->lp_reg = regul->low_power_reg;
return 0;
}
int stpmic1_lp_load_unpg(struct stpmic1_lp_cfg *cfg)
{
uint8_t val = 0;
int status = 0;
status = stpmic1_register_read(cfg->ctrl_reg, &val);
if (!status)
status = stpmic1_register_write(cfg->lp_reg, val);
return status;
}
int stpmic1_lp_reg_on_off(const char *name, uint8_t enable)
{
const struct regul_struct *regul = get_regulator_data(name);
return stpmic1_register_update(regul->low_power_reg, enable,
LDO_BUCK_ENABLE_MASK);
}
int stpmic1_lp_on_off_unpg(struct stpmic1_lp_cfg *cfg, int enable)
{
assert(enable == 0 || enable == 1);
return stpmic1_register_update(cfg->lp_reg, enable,
LDO_BUCK_ENABLE_MASK);
}
int stpmic1_lp_set_mode(const char *name, uint8_t hplp)
{
const struct regul_struct *regul = get_regulator_data(name);
return stpmic1_register_update(regul->low_power_reg,
hplp << LDO_BUCK_HPLP_SHIFT,
LDO_BUCK_HPLP_ENABLE_MASK);
}
int stpmic1_lp_mode_unpg(struct stpmic1_lp_cfg *cfg, unsigned int mode)
{
assert(mode == 0 || mode == 1);
return stpmic1_register_update(cfg->lp_reg,
mode << LDO_BUCK_HPLP_SHIFT,
LDO_BUCK_HPLP_ENABLE_MASK);
}
int stpmic1_lp_set_voltage(const char *name, uint16_t millivolts)
{
uint8_t voltage_index = voltage_to_index(name, millivolts);
const struct regul_struct *regul = get_regulator_data(name);
uint8_t mask = 0;
/* Voltage can be set for buck<N> or ldo<N> (except ldo4) regulators */
if (!strcmp(name, "buck"))
mask = BUCK_VOLTAGE_MASK;
else if (!strcmp(name, "ldo") && strcmp(name, "ldo4"))
mask = LDO_VOLTAGE_MASK;
else
return 0;
return stpmic1_register_update(regul->low_power_reg, voltage_index << 2,
mask);
}
/* Returns 1 if no configuration are expected applied at runtime, 0 otherwise */
int stpmic1_lp_voltage_cfg(const char *name, uint16_t millivolts,
struct stpmic1_lp_cfg *cfg)
{
uint8_t voltage_index = voltage_to_index(name, millivolts);
uint8_t mask = 0;
/* Voltage can be set for buck<N> or ldo<N> (except ldo4) regulators */
if (!strcmp(name, "buck"))
mask = BUCK_VOLTAGE_MASK;
else if (!strcmp(name, "ldo") && strcmp(name, "ldo4"))
mask = LDO_VOLTAGE_MASK;
else
return 1;
assert(cfg->lp_reg == get_regulator_data(name)->low_power_reg);
cfg->value = voltage_index << 2;
cfg->mask = mask;
return 0;
}
int stpmic1_lp_voltage_unpg(struct stpmic1_lp_cfg *cfg)
{
return stpmic1_register_update(cfg->lp_reg, cfg->value, cfg->mask);
}
int stpmic1_register_read(uint8_t register_id, uint8_t *value)
{
struct i2c_handle_s *i2c = pmic_i2c_handle;
return stm32_i2c_read_write_membyte(i2c, pmic_i2c_addr,
register_id, value,
false /* !write */);
}
int stpmic1_register_write(uint8_t register_id, uint8_t value)
{
struct i2c_handle_s *i2c = pmic_i2c_handle;
uint8_t val = value;
return stm32_i2c_read_write_membyte(i2c, pmic_i2c_addr,
register_id, &val,
true /* write */);
}
int stpmic1_register_update(uint8_t register_id, uint8_t value, uint8_t mask)
{
int status = 0;
uint8_t val = 0;
status = stpmic1_register_read(register_id, &val);
if (status)
return status;
val = (val & ~mask) | (value & mask);
return stpmic1_register_write(register_id, val);
}
void stpmic1_bind_i2c(struct i2c_handle_s *i2c_handle, uint16_t i2c_addr)
{
pmic_i2c_handle = i2c_handle;
pmic_i2c_addr = i2c_addr;
}
void stpmic1_dump_regulators(void)
{
size_t i = 0;
char __maybe_unused const *name = NULL;
for (i = 0; i < ARRAY_SIZE(regulators_table); i++) {
if (!regulators_table[i].control_reg)
continue;
name = regulators_table[i].dt_node_name;
DMSG("PMIC regul %s: %sable, %dmV",
name, stpmic1_is_regulator_enabled(name) ? "en" : "dis",
stpmic1_regulator_voltage_get(name));
}
}
int stpmic1_get_version(unsigned long *version)
{
uint8_t read_val = 0;
if (stpmic1_register_read(VERSION_STATUS_REG, &read_val))
return -1;
*version = read_val;
return 0;
}