0addb90ff7
By default N_VBUSEN signal is used by the host to define if VBUS signal should be used for power supply (VBUS-IPSOUT). This feature enables the possbibility to change this behaviour. When N_VBUSEN detection is disabled, axp192 will automatically decide to use VBUS (REG10H[7] = 1). Please refer to datasheet for details. Signed-off-by: Martin Kiepfer <mrmarteng@teleschirm.org>
621 lines
16 KiB
C
621 lines
16 KiB
C
/*
|
|
* Copyright (c) 2023 Martin Kiepfer <mrmarteng@teleschirm.org>
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#define DT_DRV_COMPAT x_powers_axp192
|
|
|
|
#include <errno.h>
|
|
#include <stdbool.h>
|
|
|
|
#include <zephyr/drivers/mfd/axp192.h>
|
|
#include <zephyr/drivers/i2c.h>
|
|
#include <zephyr/sys/util.h>
|
|
#include <zephyr/logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(mfd_axp192, CONFIG_MFD_LOG_LEVEL);
|
|
|
|
/* Chip ID value */
|
|
#define AXP192_CHIP_ID 0x03U
|
|
|
|
/* Registers definitions */
|
|
#define AXP192_REG_CHIP_ID 0x03U
|
|
|
|
/* AXP192 GPIO register addresses */
|
|
#define AXP192_EXTEN_DCDC2_CONTROL_REG 0x10U
|
|
#define AXP192_VBUS_CFG_REG 0x30U
|
|
#define AXP192_GPIO0_FUNC_REG 0x90U
|
|
#define AXP192_GPIO1_FUNC_REG 0x92U
|
|
#define AXP192_GPIO2_FUNC_REG 0x93U
|
|
#define AXP192_GPIO34_FUNC_REG 0x95U
|
|
#define AXP192_GPIO012_PINVAL_REG 0x94U
|
|
#define AXP192_GPIO34_PINVAL_REG 0x96U
|
|
#define AXP192_GPIO012_PULLDOWN_REG 0x97U
|
|
|
|
/* VBUS control reg values */
|
|
#define AXP192_VBUS_CFG_VAL_VBUSEN_DISABLE 0x80U
|
|
|
|
/* GPIO function control parameters */
|
|
#define AXP192_GPIO012_FUNC_VAL_OUTPUT_OD 0x00U
|
|
#define AXP192_GPIO012_FUNC_VAL_INPUT 0x01U
|
|
#define AXP192_GPIO012_FUNC_VAL_LDO 0x02U /* only applicable for GPIO0 */
|
|
#define AXP192_GPIO012_FUNC_VAL_ADC 0x04U
|
|
#define AXP192_GPIO012_FUNC_VAL_OUTPUT_LOW 0x05U
|
|
#define AXP192_GPIO012_FUNC_VAL_FLOAT 0x06U
|
|
#define AXP192_GPIO012_FUNC_MASK \
|
|
(AXP192_GPIO012_FUNC_VAL_OUTPUT_OD | AXP192_GPIO012_FUNC_VAL_INPUT | \
|
|
AXP192_GPIO012_FUNC_VAL_LDO | AXP192_GPIO012_FUNC_VAL_ADC | \
|
|
AXP192_GPIO012_FUNC_VAL_OUTPUT_LOW | AXP192_GPIO012_FUNC_VAL_FLOAT)
|
|
|
|
#define AXP192_GPIO34_FUNC_ENA 0x80U
|
|
#define AXP192_GPIO3_FUNC_VAL_CHARGE_CTL 0x00U
|
|
#define AXP192_GPIO3_FUNC_VAL_OUTPUT_OD 0x01U
|
|
#define AXP192_GPIO3_FUNC_VAL_INPUT 0x02U
|
|
#define AXP192_GPIO3_FUNC_MASK \
|
|
(AXP192_GPIO34_FUNC_ENA | AXP192_GPIO3_FUNC_VAL_CHARGE_CTL | \
|
|
AXP192_GPIO3_FUNC_VAL_OUTPUT_OD | AXP192_GPIO3_FUNC_VAL_INPUT)
|
|
|
|
#define AXP192_GPIO4_FUNC_VAL_CHARGE_CTL 0x00U
|
|
#define AXP192_GPIO4_FUNC_VAL_OUTPUT_OD 0x04U
|
|
#define AXP192_GPIO4_FUNC_VAL_INPUT 0x08U
|
|
#define AXP192_GPIO4_FUNC_VAL_ADC 0x0CU
|
|
#define AXP192_GPIO4_FUNC_MASK \
|
|
(AXP192_GPIO34_FUNC_ENA | AXP192_GPIO4_FUNC_VAL_CHARGE_CTL | \
|
|
AXP192_GPIO4_FUNC_VAL_OUTPUT_OD | AXP192_GPIO4_FUNC_VAL_INPUT)
|
|
|
|
#define AXP192_EXTEN_ENA 0x04U
|
|
#define AXP192_EXTEN_MASK 0x04U
|
|
|
|
/* Pull-Down enable parameters */
|
|
#define AXP192_GPIO0_PULLDOWN_ENABLE 0x01U
|
|
#define AXP192_GPIO1_PULLDOWN_ENABLE 0x02U
|
|
#define AXP192_GPIO2_PULLDOWN_ENABLE 0x04U
|
|
|
|
/* GPIO Value parameters */
|
|
#define AXP192_GPIO0_INPUT_VAL 0x10U
|
|
#define AXP192_GPIO1_INPUT_VAL 0x20U
|
|
#define AXP192_GPIO2_INPUT_VAL 0x40U
|
|
#define AXP192_GPIO012_INTPUT_SHIFT 4U
|
|
#define AXP192_GPIO012_INTPUT_MASK \
|
|
(AXP192_GPIO0_INPUT_VAL | AXP192_GPIO1_INPUT_VAL | AXP192_GPIO2_INPUT_VAL)
|
|
#define AXP192_GPIO3_INPUT_VAL 0x10U
|
|
#define AXP192_GPIO4_INPUT_VAL 0x20U
|
|
#define AXP192_GPIO34_INTPUT_SHIFT 4U
|
|
#define AXP192_GPIO34_INTPUT_MASK (AXP192_GPIO3_INPUT_VAL | AXP192_GPIO4_INPUT_VAL)
|
|
|
|
#define AXP192_GPIO0_OUTPUT_VAL 0x01U
|
|
#define AXP192_GPIO1_OUTPUT_VAL 0x02U
|
|
#define AXP192_GPIO2_OUTPUT_VAL 0x04U
|
|
#define AXP192_GPIO012_OUTPUT_MASK \
|
|
(AXP192_GPIO0_OUTPUT_VAL | AXP192_GPIO1_OUTPUT_VAL | AXP192_GPIO2_OUTPUT_VAL)
|
|
|
|
#define AXP192_GPIO3_OUTPUT_VAL 0x01U
|
|
#define AXP192_GPIO4_OUTPUT_VAL 0x02U
|
|
#define AXP192_GPIO34_OUTPUT_MASK (AXP192_GPIO3_OUTPUT_VAL | AXP192_GPIO4_OUTPUT_VAL)
|
|
|
|
#define AXP192_GPIO5_OUTPUT_MASK 0x04U
|
|
#define AXP192_GPIO5_OUTPUT_VAL 0x04U
|
|
#define AXP192_GPIO5_OUTPUT_SHIFT 3U
|
|
|
|
struct mfd_axp192_config {
|
|
struct i2c_dt_spec i2c;
|
|
bool vbusen_disable;
|
|
};
|
|
|
|
struct mfd_axp192_data {
|
|
const struct device *gpio_mask_used[AXP192_GPIO_MAX_NUM];
|
|
uint8_t gpio_mask_output;
|
|
};
|
|
|
|
struct mfd_axp192_func_reg_desc {
|
|
uint8_t reg;
|
|
uint8_t mask;
|
|
};
|
|
|
|
const struct mfd_axp192_func_reg_desc gpio_reg_desc[AXP192_GPIO_MAX_NUM] = {
|
|
{
|
|
/* GPIO0 */
|
|
.reg = AXP192_GPIO0_FUNC_REG,
|
|
.mask = AXP192_GPIO012_FUNC_MASK,
|
|
},
|
|
{
|
|
/* GPIO1 */
|
|
.reg = AXP192_GPIO1_FUNC_REG,
|
|
.mask = AXP192_GPIO012_FUNC_MASK,
|
|
},
|
|
{
|
|
/* GPIO2 */
|
|
.reg = AXP192_GPIO2_FUNC_REG,
|
|
.mask = AXP192_GPIO012_FUNC_MASK,
|
|
},
|
|
{
|
|
/* GPIO3 */
|
|
.reg = AXP192_GPIO34_FUNC_REG,
|
|
.mask = AXP192_GPIO3_FUNC_MASK,
|
|
},
|
|
{
|
|
/* GPIO4 */
|
|
.reg = AXP192_GPIO34_FUNC_REG,
|
|
.mask = AXP192_GPIO4_FUNC_MASK,
|
|
},
|
|
};
|
|
|
|
static int mfd_axp192_init(const struct device *dev)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
uint8_t chip_id;
|
|
uint8_t vbus_val;
|
|
int ret;
|
|
|
|
LOG_DBG("Initializing instance");
|
|
|
|
if (!i2c_is_ready_dt(&config->i2c)) {
|
|
LOG_ERR("I2C bus not ready");
|
|
return -ENODEV;
|
|
}
|
|
|
|
/* Check if axp192 chip is available */
|
|
ret = i2c_reg_read_byte_dt(&config->i2c, AXP192_REG_CHIP_ID, &chip_id);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
if (chip_id != AXP192_CHIP_ID) {
|
|
LOG_ERR("Invalid Chip detected (%d)", chip_id);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Disable N_VBUSEN */
|
|
vbus_val = 0;
|
|
if (config->vbusen_disable) {
|
|
vbus_val = AXP192_VBUS_CFG_VAL_VBUSEN_DISABLE;
|
|
}
|
|
ret = i2c_reg_update_byte_dt(&config->i2c, AXP192_VBUS_CFG_REG,
|
|
AXP192_VBUS_CFG_VAL_VBUSEN_DISABLE, vbus_val);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mfd_axp192_gpio_func_get(const struct device *dev, uint8_t gpio, enum axp192_gpio_func *func)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
int ret;
|
|
uint8_t reg_fnc;
|
|
|
|
if (gpio >= AXP192_GPIO_MAX_NUM) {
|
|
LOG_ERR("Invalid gpio (%d)", gpio);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (gpio < ARRAY_SIZE(gpio_reg_desc)) {
|
|
ret = i2c_reg_read_byte_dt(&(config->i2c), gpio_reg_desc[gpio].reg, ®_fnc);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
switch (gpio) {
|
|
case 0U:
|
|
__fallthrough;
|
|
case 1U:
|
|
__fallthrough;
|
|
case 2U:
|
|
/* GPIO 0-2*/
|
|
switch (reg_fnc) {
|
|
case AXP192_GPIO012_FUNC_VAL_INPUT:
|
|
*func = AXP192_GPIO_FUNC_INPUT;
|
|
break;
|
|
case AXP192_GPIO012_FUNC_VAL_OUTPUT_OD:
|
|
*func = AXP192_GPIO_FUNC_OUTPUT_OD;
|
|
break;
|
|
case AXP192_GPIO012_FUNC_VAL_OUTPUT_LOW:
|
|
*func = AXP192_GPIO_FUNC_OUTPUT_LOW;
|
|
break;
|
|
case AXP192_GPIO012_FUNC_VAL_LDO:
|
|
if (gpio == 0) {
|
|
/* LDO is only applicable on GPIO0 */
|
|
*func = AXP192_GPIO_FUNC_LDO;
|
|
} else {
|
|
ret = -ENOTSUP;
|
|
}
|
|
break;
|
|
case AXP192_GPIO012_FUNC_VAL_ADC:
|
|
*func = AXP192_GPIO_FUNC_ADC;
|
|
break;
|
|
case AXP192_GPIO012_FUNC_VAL_FLOAT:
|
|
*func = AXP192_GPIO_FUNC_FLOAT;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 3U:
|
|
/* GPIO3 */
|
|
switch (reg_fnc) {
|
|
case (AXP192_GPIO3_FUNC_VAL_INPUT | AXP192_GPIO34_FUNC_ENA):
|
|
*func = AXP192_GPIO_FUNC_INPUT;
|
|
break;
|
|
case (AXP192_GPIO3_FUNC_VAL_OUTPUT_OD | AXP192_GPIO34_FUNC_ENA):
|
|
*func = AXP192_GPIO_FUNC_OUTPUT_OD;
|
|
break;
|
|
case AXP192_GPIO3_FUNC_VAL_CHARGE_CTL:
|
|
*func = AXP192_GPIO_FUNC_CHARGE_CTL;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 4U:
|
|
/* GPIO4 */
|
|
switch (reg_fnc) {
|
|
case (AXP192_GPIO4_FUNC_VAL_INPUT | AXP192_GPIO34_FUNC_ENA):
|
|
*func = AXP192_GPIO_FUNC_INPUT;
|
|
break;
|
|
case (AXP192_GPIO4_FUNC_VAL_OUTPUT_OD | AXP192_GPIO34_FUNC_ENA):
|
|
*func = AXP192_GPIO_FUNC_OUTPUT_OD;
|
|
break;
|
|
case (AXP192_GPIO4_FUNC_VAL_ADC | AXP192_GPIO34_FUNC_ENA):
|
|
*func = AXP192_GPIO_FUNC_ADC;
|
|
break;
|
|
case AXP192_GPIO4_FUNC_VAL_CHARGE_CTL:
|
|
*func = AXP192_GPIO_FUNC_CHARGE_CTL;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 5U:
|
|
/* EXTEN is an output only pin */
|
|
*func = AXP192_GPIO_FUNC_OUTPUT_LOW;
|
|
break;
|
|
|
|
default:
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int mfd_axp192_gpio_func_ctrl(const struct device *dev, const struct device *client_dev,
|
|
uint8_t gpio, enum axp192_gpio_func func)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
struct mfd_axp192_data *data = dev->data;
|
|
bool is_output = false;
|
|
int ret = 0;
|
|
uint8_t reg_cfg = 0;
|
|
|
|
if (!AXP192_GPIO_FUNC_VALID(func)) {
|
|
LOG_ERR("Invalid function");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (gpio >= AXP192_GPIO_MAX_NUM) {
|
|
LOG_ERR("Invalid gpio (%d)", gpio);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if ((data->gpio_mask_used[gpio] != 0) && (data->gpio_mask_used[gpio] != client_dev)) {
|
|
LOG_INF("Warning: Pin already configured. Please check dt configuration");
|
|
}
|
|
|
|
switch (gpio) {
|
|
case 0U:
|
|
__fallthrough;
|
|
case 1U:
|
|
__fallthrough;
|
|
case 2U:
|
|
/* GPIO 0-2*/
|
|
switch (func) {
|
|
case AXP192_GPIO_FUNC_INPUT:
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_INPUT;
|
|
break;
|
|
case AXP192_GPIO_FUNC_OUTPUT_OD:
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_OUTPUT_OD;
|
|
is_output = true;
|
|
break;
|
|
case AXP192_GPIO_FUNC_OUTPUT_LOW:
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_OUTPUT_LOW;
|
|
is_output = true;
|
|
break;
|
|
case AXP192_GPIO_FUNC_LDO:
|
|
if (gpio == 0) {
|
|
/* LDO is only applicable on GPIO0 */
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_LDO;
|
|
} else {
|
|
ret = -ENOTSUP;
|
|
}
|
|
break;
|
|
case AXP192_GPIO_FUNC_ADC:
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_ADC;
|
|
break;
|
|
case AXP192_GPIO_FUNC_FLOAT:
|
|
reg_cfg = AXP192_GPIO012_FUNC_VAL_FLOAT;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 3U:
|
|
/* GPIO3 */
|
|
switch (func) {
|
|
case AXP192_GPIO_FUNC_INPUT:
|
|
reg_cfg = AXP192_GPIO3_FUNC_VAL_INPUT | AXP192_GPIO34_FUNC_ENA;
|
|
break;
|
|
case AXP192_GPIO_FUNC_OUTPUT_OD:
|
|
reg_cfg = AXP192_GPIO3_FUNC_VAL_OUTPUT_OD | AXP192_GPIO34_FUNC_ENA;
|
|
is_output = true;
|
|
break;
|
|
case AXP192_GPIO_FUNC_CHARGE_CTL:
|
|
reg_cfg = AXP192_GPIO3_FUNC_VAL_CHARGE_CTL;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 4U:
|
|
/* GPIO4 */
|
|
switch (func) {
|
|
case AXP192_GPIO_FUNC_INPUT:
|
|
reg_cfg = AXP192_GPIO4_FUNC_VAL_INPUT | AXP192_GPIO34_FUNC_ENA;
|
|
break;
|
|
case AXP192_GPIO_FUNC_OUTPUT_OD:
|
|
reg_cfg = AXP192_GPIO4_FUNC_VAL_OUTPUT_OD | AXP192_GPIO34_FUNC_ENA;
|
|
is_output = true;
|
|
break;
|
|
case AXP192_GPIO_FUNC_ADC:
|
|
reg_cfg = AXP192_GPIO4_FUNC_VAL_ADC | AXP192_GPIO34_FUNC_ENA;
|
|
break;
|
|
case AXP192_GPIO_FUNC_CHARGE_CTL:
|
|
reg_cfg = AXP192_GPIO4_FUNC_VAL_CHARGE_CTL;
|
|
break;
|
|
default:
|
|
ret = -ENOTSUP;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 5U:
|
|
/* EXTEN is an output only pin */
|
|
break;
|
|
|
|
default:
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
if (ret != 0) {
|
|
LOG_ERR("Invalid function (0x%x) for gpio %d", func, gpio);
|
|
return ret;
|
|
}
|
|
|
|
if (gpio < ARRAY_SIZE(gpio_reg_desc)) {
|
|
ret = i2c_reg_update_byte_dt(&(config->i2c), gpio_reg_desc[gpio].reg,
|
|
gpio_reg_desc[gpio].mask, reg_cfg);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
/* Save gpio configuration state */
|
|
data->gpio_mask_used[gpio] = client_dev;
|
|
if (is_output) {
|
|
data->gpio_mask_output |= (1u << gpio);
|
|
} else {
|
|
data->gpio_mask_output &= ~(1u << gpio);
|
|
}
|
|
LOG_DBG("GPIO %d configured successfully (func=0x%x)", gpio, reg_cfg);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mfd_axp192_gpio_pd_get(const struct device *dev, uint8_t gpio, bool *enabled)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
uint8_t gpio_val;
|
|
uint8_t pd_reg_mask = 0;
|
|
int ret = 0;
|
|
|
|
switch (gpio) {
|
|
case 0U:
|
|
pd_reg_mask = AXP192_GPIO0_PULLDOWN_ENABLE;
|
|
break;
|
|
case 1U:
|
|
pd_reg_mask = AXP192_GPIO1_PULLDOWN_ENABLE;
|
|
break;
|
|
case 2U:
|
|
pd_reg_mask = AXP192_GPIO2_PULLDOWN_ENABLE;
|
|
break;
|
|
|
|
case 3U:
|
|
__fallthrough;
|
|
case 4U:
|
|
__fallthrough;
|
|
case 5U:
|
|
LOG_DBG("Pull-Down not support on gpio %d", gpio);
|
|
return -ENOTSUP;
|
|
|
|
default:
|
|
LOG_ERR("Invalid gpio (%d)", gpio);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = i2c_reg_read_byte_dt(&(config->i2c), AXP192_GPIO012_PULLDOWN_REG, &gpio_val);
|
|
|
|
if (ret == 0) {
|
|
*enabled = ((gpio_val & pd_reg_mask) != 0);
|
|
LOG_DBG("Pull-Down stats of gpio %d: %d", gpio, *enabled);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mfd_axp192_gpio_pd_ctrl(const struct device *dev, uint8_t gpio, bool enable)
|
|
{
|
|
uint8_t reg_pd_val = 0;
|
|
uint8_t reg_pd_mask = 0;
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
int ret = 0;
|
|
|
|
/* Configure pull-down. Pull-down is only supported by GPIO3 and GPIO4 */
|
|
switch (gpio) {
|
|
case 0U:
|
|
reg_pd_mask = AXP192_GPIO0_PULLDOWN_ENABLE;
|
|
if (enable) {
|
|
reg_pd_val = AXP192_GPIO0_PULLDOWN_ENABLE;
|
|
}
|
|
break;
|
|
|
|
case 1U:
|
|
reg_pd_mask = AXP192_GPIO1_PULLDOWN_ENABLE;
|
|
if (enable) {
|
|
reg_pd_val = AXP192_GPIO1_PULLDOWN_ENABLE;
|
|
}
|
|
break;
|
|
|
|
case 2U:
|
|
reg_pd_mask = AXP192_GPIO2_PULLDOWN_ENABLE;
|
|
if (enable) {
|
|
reg_pd_val = AXP192_GPIO2_PULLDOWN_ENABLE;
|
|
}
|
|
break;
|
|
|
|
case 3U:
|
|
__fallthrough;
|
|
case 4U:
|
|
__fallthrough;
|
|
case 5U:
|
|
LOG_ERR("Pull-Down not support on gpio %d", gpio);
|
|
return -ENOTSUP;
|
|
|
|
default:
|
|
LOG_ERR("Invalid gpio (%d)", gpio);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = i2c_reg_update_byte_dt(&(config->i2c), AXP192_GPIO012_PULLDOWN_REG, reg_pd_mask,
|
|
reg_pd_val);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int mfd_axp192_gpio_read_port(const struct device *dev, uint8_t *value)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
const struct mfd_axp192_data *data = dev->data;
|
|
int ret;
|
|
uint8_t gpio012_val;
|
|
uint8_t gpio34_val;
|
|
uint8_t gpio5_val;
|
|
uint8_t gpio_input_val;
|
|
uint8_t gpio_output_val;
|
|
|
|
/* read gpio0-2 */
|
|
ret = i2c_reg_read_byte_dt(&(config->i2c), AXP192_GPIO012_PINVAL_REG, &gpio012_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* read gpio3-4 */
|
|
ret = i2c_reg_read_byte_dt(&(config->i2c), AXP192_GPIO34_PINVAL_REG, &gpio34_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* read gpio5 */
|
|
ret = i2c_reg_read_byte_dt(&(config->i2c), AXP192_EXTEN_DCDC2_CONTROL_REG, &gpio5_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
LOG_DBG("GPIO012 pinval-reg=0x%x", gpio012_val);
|
|
LOG_DBG("GPIO34 pinval-reg =0x%x", gpio34_val);
|
|
LOG_DBG("GPIO5 pinval-reg =0x%x", gpio5_val);
|
|
LOG_DBG("Output-Mask =0x%x", data->gpio_mask_output);
|
|
|
|
gpio_input_val =
|
|
((gpio012_val & AXP192_GPIO012_INTPUT_MASK) >> AXP192_GPIO012_INTPUT_SHIFT);
|
|
gpio_input_val |=
|
|
(((gpio34_val & AXP192_GPIO34_INTPUT_MASK) >> AXP192_GPIO34_INTPUT_SHIFT) << 3u);
|
|
|
|
gpio_output_val = (gpio012_val & AXP192_GPIO012_OUTPUT_MASK);
|
|
gpio_output_val |= ((gpio34_val & AXP192_GPIO34_OUTPUT_MASK) << 3u);
|
|
gpio_output_val |=
|
|
(((gpio5_val & AXP192_GPIO5_OUTPUT_MASK) >> AXP192_GPIO5_OUTPUT_SHIFT) << 5u);
|
|
|
|
*value = gpio_input_val & ~(data->gpio_mask_output);
|
|
*value |= (gpio_output_val & data->gpio_mask_output);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mfd_axp192_gpio_write_port(const struct device *dev, uint8_t value, uint8_t mask)
|
|
{
|
|
const struct mfd_axp192_config *config = dev->config;
|
|
int ret;
|
|
uint8_t gpio_reg_val;
|
|
uint8_t gpio_reg_mask;
|
|
|
|
/* Write gpio0-2. Mask out other port pins */
|
|
gpio_reg_val = (value & AXP192_GPIO012_OUTPUT_MASK);
|
|
gpio_reg_mask = (mask & AXP192_GPIO012_OUTPUT_MASK);
|
|
if (gpio_reg_mask != 0) {
|
|
ret = i2c_reg_update_byte_dt(&(config->i2c), AXP192_GPIO012_PINVAL_REG,
|
|
gpio_reg_mask, gpio_reg_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
LOG_DBG("GPIO012 pinval-reg=0x%x mask=0x%x", gpio_reg_val, gpio_reg_mask);
|
|
}
|
|
|
|
/* Write gpio3-4. Mask out other port pins */
|
|
gpio_reg_val = value >> 3U;
|
|
gpio_reg_mask = (mask >> 3U) & AXP192_GPIO34_OUTPUT_MASK;
|
|
if (gpio_reg_mask != 0) {
|
|
ret = i2c_reg_update_byte_dt(&(config->i2c), AXP192_GPIO34_PINVAL_REG,
|
|
gpio_reg_mask, gpio_reg_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
LOG_DBG("GPIO34 pinval-reg =0x%x mask=0x%x", gpio_reg_val, gpio_reg_mask);
|
|
}
|
|
|
|
/* Write gpio5. Mask out other port pins */
|
|
if ((mask & BIT(5)) != 0) {
|
|
gpio_reg_mask = AXP192_EXTEN_MASK;
|
|
gpio_reg_val = (value & BIT(5)) ? AXP192_EXTEN_ENA : 0U;
|
|
ret = i2c_reg_update_byte_dt(&(config->i2c), AXP192_EXTEN_DCDC2_CONTROL_REG,
|
|
gpio_reg_mask, gpio_reg_val);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
LOG_DBG("GPIO5 pinval-reg =0x%x mask=0x%x\n", gpio_reg_val, gpio_reg_mask);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define MFD_AXP192_DEFINE(inst) \
|
|
static const struct mfd_axp192_config config##inst = { \
|
|
.i2c = I2C_DT_SPEC_INST_GET(inst), \
|
|
.vbusen_disable = DT_INST_PROP_OR(inst, vbusen_disable, false), \
|
|
}; \
|
|
\
|
|
static struct mfd_axp192_data data##inst; \
|
|
\
|
|
DEVICE_DT_INST_DEFINE(inst, mfd_axp192_init, NULL, &data##inst, &config##inst, \
|
|
POST_KERNEL, CONFIG_MFD_INIT_PRIORITY, NULL);
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MFD_AXP192_DEFINE);
|