| /* |
| * Copyright 2024 Arduino SA |
| * |
| * SPDX-License-Identifier: Apache-2.0 |
| */ |
| |
| #define DT_DRV_COMPAT nxp_pf1550_charger |
| |
| #include <zephyr/device.h> |
| #include <zephyr/drivers/charger.h> |
| #include <zephyr/drivers/gpio.h> |
| #include <zephyr/drivers/i2c.h> |
| #include <zephyr/kernel.h> |
| #include <zephyr/sys/util.h> |
| #include <zephyr/sys/linear_range.h> |
| |
| #include <zephyr/logging/log.h> |
| LOG_MODULE_REGISTER(pf1550_charger, CONFIG_CHARGER_LOG_LEVEL); |
| |
| #define INT_ENABLE_DELAY K_MSEC(500) |
| |
| #define CHARGER_CHG_INT (0x80 + 0x00) |
| #define CHARGER_CHG_INT_MASK (0x80 + 0x02) |
| #define CHARGER_CHG_INT_OK (0x80 + 0x04) |
| #define CHARGER_VBUS_SNS (0x80 + 0x06) |
| #define CHARGER_CHG_SNS (0x80 + 0x07) |
| #define CHARGER_BATT_SNS (0x80 + 0x08) |
| #define CHARGER_CHG_OPER (0x80 + 0x09) |
| #define CHARGER_CHG_TMR (0x80 + 0x0A) |
| #define CHARGER_CHG_EOC_CNFG (0x80 + 0x0D) |
| #define CHARGER_CHG_CURR_CNFG (0x80 + 0x0E) |
| #define CHARGER_BATT_REG (0x80 + 0x0F) |
| #define CHARGER_BATFET_CNFG (0x80 + 0x11) |
| #define CHARGER_THM_REG_CNFG (0x80 + 0x12) |
| #define CHARGER_VBUS_INLIM_CNFG (0x80 + 0x14) |
| #define CHARGER_VBUS_LIN_DPM (0x80 + 0x15) |
| #define CHARGER_USB_PHY_LDO_CNFG (0x80 + 0x16) |
| #define CHARGER_DBNC_DELAY_TIME (0x80 + 0x18) |
| #define CHARGER_CHG_INT_CNFG (0x80 + 0x19) |
| #define CHARGER_THM_ADJ_SETTING (0x80 + 0x1A) |
| #define CHARGER_VBUS2SYS_CNFG (0x80 + 0x1B) |
| #define CHARGER_LED_PWM (0x80 + 0x1C) |
| #define CHARGER_FAULT_BATFET_CNFG (0x80 + 0x1D) |
| #define CHARGER_LED_CNFG (0x80 + 0x1E) |
| #define CHARGER_CHGR_KEY2 (0x80 + 0x1F) |
| |
| #define PF1550_BAT_IRQ BIT(2) |
| #define PF1550_CHG_IRQ BIT(3) |
| #define PF1550_VBUS_IRQ BIT(5) |
| #define PF1550_VBUS_DPM_IRQ BIT(5) |
| #define CHG_INT_ENABLE_ALL (0xFF) |
| |
| #define LED_PWM_LED_EN BIT(7) |
| #define LED_PWM_FULL_ON BIT(5) |
| |
| #define LED_CNFG_LED_CFG BIT(4) |
| #define LED_CNFG_LEDOVRD BIT(5) |
| |
| #define CHG_OPER_CHG_OPER_MASK GENMASK(1, 0) |
| #define CHG_CURR_CNFG_CHG_CC_MASK GENMASK(4, 0) |
| #define CHG_SNS_CHG_SNS_MASK GENMASK(3, 0) |
| #define VBUS_INLIM_CNFG_VBUS_INLIM_MASK GENMASK(7, 3) |
| #define BATT_REG_CHGCV_MASK GENMASK(5, 0) |
| #define BATT_REG_VSYSMIN_MASK GENMASK(7, 6) |
| #define THM_REG_CNFG_THM_CNFG_MASK GENMASK(1, 0) |
| |
| #define CHG_OPER_CHARGER_OFF_LINEAR_OFF 0 |
| #define CHG_OPER_CHARGER_OFF_LINEAR_ON 1 |
| #define CHG_OPER_CHARGER_ON_LINEAR_ON 2 |
| |
| enum charger_pf1550_therm_mode { |
| PF1550_THERM_MODE_DISABLED, |
| PF1550_THERM_MODE_THERMISTOR, |
| PF1550_THERM_MODE_JEITA_1, |
| PF1550_THERM_MODE_JEITA_2, |
| PF1550_THERM_MODE_UNKNOWN, |
| }; |
| |
| /* synced with YAML binding */ |
| enum charger_pf1550_led_behaviour { |
| PF1550_LED_ON_IN_CHARGING_FLASH_IN_FAULT, |
| PF1550_LED_FLASH_IN_CHARGING_ON_IN_FAULT, |
| PF1550_LED_MANUAL_OFF |
| }; |
| |
| struct charger_pf1550_led_config { |
| bool enabled; |
| bool manual; |
| enum charger_pf1550_led_behaviour behaviour; |
| }; |
| |
| struct charger_pf1550_config { |
| struct i2c_dt_spec bus; |
| struct gpio_dt_spec int_gpio; |
| char *therm_mon_mode; |
| uint32_t charge_current_ua; |
| uint32_t vbus_ilim_ua; |
| uint32_t charge_voltage_max_uv; |
| uint32_t vsys_min_uv; |
| }; |
| |
| struct charger_pf1550_data { |
| const struct device *dev; |
| struct gpio_callback gpio_cb; |
| struct k_work int_routine_work; |
| struct k_work_delayable int_enable_work; |
| enum charger_status charger_status; |
| enum charger_online charger_online; |
| charger_status_notifier_t charger_status_notifier; |
| charger_online_notifier_t charger_online_notifier; |
| bool charger_enabled; |
| uint32_t charge_current_ua; |
| uint32_t vbus_ilim_ua; |
| struct charger_pf1550_led_config *led_config; |
| }; |
| |
| static const struct linear_range charger_vbus_ilim_range[] = { |
| LINEAR_RANGE_INIT(10000, 5000, 0, 8), |
| LINEAR_RANGE_INIT(100000, 50000, 9, 10), |
| LINEAR_RANGE_INIT(200000, 100000, 11, 19), |
| LINEAR_RANGE_INIT(1500000, 0, 20, 20), |
| }; |
| |
| static const struct linear_range charger_fast_charge_ua_range[] = { |
| LINEAR_RANGE_INIT(100000, 50000, 0, 18), |
| }; |
| |
| static const struct linear_range charger_battery_termination_uv_range[] = { |
| LINEAR_RANGE_INIT(3500000, 20000, 8, 55), |
| }; |
| |
| static const struct linear_range charger_vsysmin_uv[] = { |
| LINEAR_RANGE_INIT(3500000, 0, 0, 0), |
| LINEAR_RANGE_INIT(3700000, 0, 1, 1), |
| LINEAR_RANGE_INIT(4300000, 0, 2, 2), |
| }; |
| |
| static int pf1550_get_charger_status(const struct device *dev, enum charger_status *status) |
| { |
| enum chg_sns { |
| PF1550_CHARGER_PRECHARGE, |
| PF1550_FAST_CHARGE_CONSTANT_CURRENT, |
| PF1550_FAST_CHARGE_CONSTANT_VOLTAGE, |
| PF1550_END_OF_CHARGE, |
| PF1550_CHARGE_DONE, |
| PF1550_TIMER_FAULT = 6, |
| PF1550_THERMISTOR_SUSPEND, |
| PF1550_CHARGER_OFF_INVALID_INPUT, |
| PF1550_BATTERY_OVERVOLTAGE, |
| PF1550_BATTERY_OVERTEMPERATURE, |
| PF1550_CHARGER_OFF_LINEAR_MODE = 12, |
| }; |
| |
| const struct charger_pf1550_config *const config = dev->config; |
| uint8_t val; |
| int ret; |
| |
| ret = i2c_reg_read_byte_dt(&config->bus, CHARGER_CHG_SNS, &val); |
| if (ret) { |
| return ret; |
| } |
| |
| val = FIELD_GET(CHG_SNS_CHG_SNS_MASK, val); |
| |
| if (val == PF1550_CHARGE_DONE) { |
| *status = CHARGER_STATUS_FULL; |
| } else if (val < PF1550_CHARGE_DONE) { |
| *status = CHARGER_STATUS_CHARGING; |
| } else { |
| *status = CHARGER_STATUS_NOT_CHARGING; |
| } |
| |
| return 0; |
| } |
| |
| static int pf1550_get_charger_online(const struct device *dev, enum charger_online *online) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint8_t val; |
| int ret; |
| |
| ret = i2c_reg_read_byte_dt(&config->bus, CHARGER_CHG_OPER, &val); |
| if (ret) { |
| return ret; |
| } |
| |
| val = FIELD_GET(CHG_OPER_CHG_OPER_MASK, val); |
| |
| switch (val) { |
| case CHG_OPER_CHARGER_ON_LINEAR_ON: |
| *online = CHARGER_ONLINE_FIXED; |
| break; |
| default: |
| *online = CHARGER_ONLINE_OFFLINE; |
| break; |
| }; |
| |
| return 0; |
| } |
| |
| static int pf1550_set_constant_charge_current(const struct device *dev, uint32_t current_ua) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint16_t idx; |
| uint8_t val; |
| int ret; |
| |
| ret = linear_range_group_get_index(charger_fast_charge_ua_range, |
| ARRAY_SIZE(charger_fast_charge_ua_range), current_ua, |
| &idx); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| val = FIELD_PREP(CHG_CURR_CNFG_CHG_CC_MASK, idx); |
| |
| return i2c_reg_update_byte_dt(&config->bus, CHARGER_CHG_CURR_CNFG, |
| CHG_CURR_CNFG_CHG_CC_MASK, val); |
| } |
| |
| static int pf1550_set_vbus_ilim(const struct device *dev, uint32_t current_ua) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint16_t idx; |
| uint8_t val; |
| int ret; |
| |
| ret = linear_range_group_get_index(charger_vbus_ilim_range, |
| ARRAY_SIZE(charger_vbus_ilim_range), current_ua, &idx); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| val = FIELD_PREP(VBUS_INLIM_CNFG_VBUS_INLIM_MASK, idx); |
| |
| return i2c_reg_update_byte_dt(&config->bus, CHARGER_VBUS_INLIM_CNFG, |
| VBUS_INLIM_CNFG_VBUS_INLIM_MASK, val); |
| } |
| |
| static int pf1550_set_vsys_min(const struct device *dev, uint32_t voltage_uv) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint16_t idx; |
| uint8_t val; |
| int ret; |
| |
| ret = linear_range_group_get_index(charger_vsysmin_uv, ARRAY_SIZE(charger_vsysmin_uv), |
| voltage_uv, &idx); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| val = FIELD_PREP(BATT_REG_VSYSMIN_MASK, idx); |
| |
| return i2c_reg_update_byte_dt(&config->bus, CHARGER_BATT_REG, BATT_REG_VSYSMIN_MASK, val); |
| } |
| |
| static int pf1550_set_charge_termination_uv(const struct device *dev, uint32_t voltage_uv) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint16_t idx; |
| uint8_t val; |
| int ret; |
| |
| ret = linear_range_group_get_index(charger_battery_termination_uv_range, |
| ARRAY_SIZE(charger_battery_termination_uv_range), |
| voltage_uv, &idx); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| val = FIELD_PREP(BATT_REG_CHGCV_MASK, idx); |
| |
| return i2c_reg_update_byte_dt(&config->bus, CHARGER_BATT_REG, BATT_REG_CHGCV_MASK, val); |
| } |
| |
| static int pf1550_set_thermistor_mode(const struct device *dev, enum charger_pf1550_therm_mode mode) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| uint8_t val; |
| |
| val = FIELD_PREP(THM_REG_CNFG_THM_CNFG_MASK, mode); |
| |
| return i2c_reg_update_byte_dt(&config->bus, CHARGER_THM_REG_CNFG, |
| THM_REG_CNFG_THM_CNFG_MASK, val); |
| } |
| |
| static int pf1550_set_enabled(const struct device *dev, bool enable) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *const config = dev->config; |
| |
| int ret = i2c_reg_update_byte_dt(&config->bus, CHARGER_CHG_OPER, CHG_OPER_CHG_OPER_MASK, |
| enable ? 2 : 0); |
| |
| if (ret == 0) { |
| data->charger_enabled = enable; |
| } |
| |
| return ret; |
| } |
| |
| static int pf1550_get_interrupt_source(const struct device *dev, uint8_t *int_a) |
| { |
| const struct charger_pf1550_config *config = dev->config; |
| uint8_t buf = 0; |
| int ret; |
| |
| ret = i2c_reg_read_byte_dt(&config->bus, CHARGER_CHG_INT, &buf); |
| |
| if (int_a) { |
| *int_a = buf; |
| } |
| return ret; |
| } |
| |
| static int pf1550_enable_interrupts(const struct device *dev) |
| { |
| const struct charger_pf1550_config *config = dev->config; |
| int ret; |
| |
| ret = pf1550_get_interrupt_source(dev, NULL); |
| if (ret < 0) { |
| LOG_WRN("Failed to clear pending interrupts: %d", ret); |
| return ret; |
| } |
| |
| return i2c_reg_write_byte_dt(&config->bus, CHARGER_CHG_INT_MASK, CHG_INT_ENABLE_ALL); |
| } |
| |
| static int pf1550_led_config(const struct device *dev) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *config = dev->config; |
| struct charger_pf1550_led_config *cfg = data->led_config; |
| int ret; |
| uint8_t val; |
| |
| cfg->enabled = true; |
| |
| if (cfg->behaviour == PF1550_LED_MANUAL_OFF) { |
| cfg->manual = true; |
| cfg->enabled = false; |
| } |
| |
| val = (cfg->enabled ? LED_PWM_LED_EN : 0) | LED_PWM_FULL_ON; |
| |
| ret = i2c_reg_write_byte_dt(&config->bus, CHARGER_LED_PWM, val); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| val = (cfg->manual ? LED_CNFG_LEDOVRD : 0) | |
| (cfg->behaviour == PF1550_LED_FLASH_IN_CHARGING_ON_IN_FAULT ? |
| LED_CNFG_LED_CFG : 0); |
| |
| return i2c_reg_write_byte_dt(&config->bus, CHARGER_LED_CNFG, val); |
| } |
| |
| static int pf1550_init_properties(const struct device *dev) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *config = dev->config; |
| int ret; |
| |
| data->charger_enabled = true; |
| data->charge_current_ua = config->charge_current_ua; |
| data->vbus_ilim_ua = config->vbus_ilim_ua; |
| |
| ret = pf1550_get_charger_status(dev, &data->charger_status); |
| if (ret < 0) { |
| LOG_ERR("Failed to read charger status: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_get_charger_online(dev, &data->charger_online); |
| if (ret < 0) { |
| LOG_ERR("Failed to read charger online: %d", ret); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| enum charger_pf1550_therm_mode pf1550_string_to_therm_mode(const char *mode_string) |
| { |
| static const char *const modes[] = { |
| [PF1550_THERM_MODE_DISABLED] = "disabled", |
| [PF1550_THERM_MODE_THERMISTOR] = "thermistor", |
| [PF1550_THERM_MODE_JEITA_1] = "JEITA-1", |
| [PF1550_THERM_MODE_JEITA_2] = "JEITA-2", |
| }; |
| enum charger_pf1550_therm_mode i; |
| |
| for (i = PF1550_THERM_MODE_DISABLED; i < ARRAY_SIZE(modes); i++) { |
| if (strncmp(mode_string, modes[i], strlen(modes[i])) == 0) { |
| return i; |
| } |
| } |
| |
| return PF1550_THERM_MODE_UNKNOWN; |
| } |
| |
| static int pf1550_update_properties(const struct device *dev) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *config = dev->config; |
| enum charger_pf1550_therm_mode therm_mode; |
| int ret; |
| |
| ret = pf1550_set_vbus_ilim(dev, config->vbus_ilim_ua); |
| if (ret < 0) { |
| LOG_ERR("Failed to set vbus current limit: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_set_vsys_min(dev, config->vsys_min_uv); |
| if (ret < 0) { |
| LOG_ERR("Failed to set minimum system voltage threshold: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_set_charge_termination_uv(dev, config->charge_voltage_max_uv); |
| if (ret < 0) { |
| LOG_ERR("Failed to set recharge threshold: %d", ret); |
| return ret; |
| } |
| |
| therm_mode = pf1550_string_to_therm_mode(config->therm_mon_mode); |
| ret = pf1550_set_thermistor_mode(dev, therm_mode); |
| if (ret < 0) { |
| LOG_ERR("Failed to set thermistor mode: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_set_constant_charge_current(dev, data->charge_current_ua); |
| if (ret < 0) { |
| LOG_ERR("Failed to set charge voltage: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_set_enabled(dev, data->charger_enabled); |
| if (ret < 0) { |
| LOG_ERR("Failed to set enabled: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_led_config(dev); |
| if (ret < 0) { |
| LOG_ERR("Failed to configure led: %d", ret); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int pf1550_get_prop(const struct device *dev, charger_prop_t prop, |
| union charger_propval *val) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| |
| switch (prop) { |
| case CHARGER_PROP_ONLINE: |
| val->online = data->charger_online; |
| return 0; |
| case CHARGER_PROP_STATUS: |
| val->status = data->charger_status; |
| return 0; |
| case CHARGER_PROP_CONSTANT_CHARGE_CURRENT_UA: |
| val->const_charge_current_ua = data->charge_current_ua; |
| return 0; |
| default: |
| return -ENOTSUP; |
| } |
| } |
| |
| static int pf1550_set_prop(const struct device *dev, charger_prop_t prop, |
| const union charger_propval *val) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| int ret; |
| |
| switch (prop) { |
| case CHARGER_PROP_CONSTANT_CHARGE_CURRENT_UA: |
| ret = pf1550_set_constant_charge_current(dev, val->const_charge_current_ua); |
| if (ret == 0) { |
| data->charge_current_ua = val->const_charge_current_ua; |
| } |
| return ret; |
| case CHARGER_PROP_INPUT_REGULATION_CURRENT_UA: |
| ret = pf1550_set_vbus_ilim(dev, val->input_current_regulation_current_ua); |
| if (ret == 0) { |
| data->vbus_ilim_ua = val->input_current_regulation_current_ua; |
| } |
| return ret; |
| case CHARGER_PROP_STATUS_NOTIFICATION: |
| data->charger_status_notifier = val->status_notification; |
| return 0; |
| case CHARGER_PROP_ONLINE_NOTIFICATION: |
| data->charger_online_notifier = val->online_notification; |
| return 0; |
| default: |
| return -ENOTSUP; |
| } |
| } |
| |
| static int pf1550_enable_interrupt_pin(const struct device *dev, bool enabled) |
| { |
| const struct charger_pf1550_config *const config = dev->config; |
| gpio_flags_t flags; |
| int ret; |
| |
| flags = enabled ? GPIO_INT_EDGE_TO_ACTIVE : GPIO_INT_DISABLE; |
| |
| ret = gpio_pin_interrupt_configure_dt(&config->int_gpio, flags); |
| if (ret < 0) { |
| LOG_ERR("Could not %s interrupt GPIO callback: %d", enabled ? "enable" : "disable", |
| ret); |
| } |
| |
| return ret; |
| } |
| |
| static void pf1550_gpio_callback(const struct device *dev, struct gpio_callback *cb, uint32_t pins) |
| { |
| struct charger_pf1550_data *data = CONTAINER_OF(cb, struct charger_pf1550_data, gpio_cb); |
| int ret; |
| |
| (void)pf1550_enable_interrupt_pin(data->dev, false); |
| |
| ret = k_work_submit(&data->int_routine_work); |
| if (ret < 0) { |
| LOG_WRN("Could not submit int work: %d", ret); |
| } |
| } |
| |
| static void pf1550_int_routine_work_handler(struct k_work *work) |
| { |
| struct charger_pf1550_data *data = |
| CONTAINER_OF(work, struct charger_pf1550_data, int_routine_work); |
| uint8_t int_src; |
| int ret; |
| |
| ret = pf1550_get_interrupt_source(data->dev, &int_src); |
| if (ret < 0) { |
| LOG_WRN("Failed to read interrupt source: %d", ret); |
| return; |
| } |
| |
| LOG_DBG("Interrupt received: %x", int_src); |
| |
| ret = pf1550_get_charger_status(data->dev, &data->charger_status); |
| if (ret < 0) { |
| LOG_WRN("Failed to read charger status: %d", ret); |
| return; |
| } |
| |
| ret = pf1550_get_charger_online(data->dev, &data->charger_online); |
| if (ret < 0) { |
| LOG_WRN("Failed to read charger online %d", ret); |
| return; |
| } |
| |
| if (data->charger_status_notifier != NULL) { |
| data->charger_status_notifier(data->charger_status); |
| } |
| if (data->charger_online_notifier != NULL) { |
| data->charger_online_notifier(data->charger_online); |
| } |
| |
| if (data->charger_online != CHARGER_ONLINE_OFFLINE) { |
| (void)pf1550_update_properties(data->dev); |
| } |
| |
| ret = k_work_reschedule(&data->int_enable_work, INT_ENABLE_DELAY); |
| if (ret < 0) { |
| LOG_WRN("Could not reschedule int_enable_work: %d", ret); |
| } |
| } |
| |
| static void pf1550_int_enable_work_handler(struct k_work *work) |
| { |
| struct k_work_delayable *dwork = k_work_delayable_from_work(work); |
| struct charger_pf1550_data *data = |
| CONTAINER_OF(dwork, struct charger_pf1550_data, int_enable_work); |
| |
| (void)pf1550_enable_interrupt_pin(data->dev, true); |
| } |
| |
| static int pf1550_configure_interrupt_pin(const struct device *dev) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *config = dev->config; |
| int ret; |
| |
| ret = gpio_is_ready_dt(&config->int_gpio) ? 0 : -ENODEV; |
| if (ret < 0) { |
| LOG_ERR("Interrupt GPIO device not ready: %d", ret); |
| return ret; |
| } |
| |
| ret = gpio_pin_configure_dt(&config->int_gpio, GPIO_INPUT); |
| if (ret < 0) { |
| LOG_ERR("Could not configure interrupt GPIO: %d", ret); |
| return ret; |
| } |
| |
| gpio_init_callback(&data->gpio_cb, pf1550_gpio_callback, BIT(config->int_gpio.pin)); |
| ret = gpio_add_callback_dt(&config->int_gpio, &data->gpio_cb); |
| if (ret < 0) { |
| LOG_ERR("Could not add interrupt GPIO callback: %d", ret); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int pf1550_init(const struct device *dev) |
| { |
| struct charger_pf1550_data *data = dev->data; |
| const struct charger_pf1550_config *config = dev->config; |
| int ret; |
| |
| if (!i2c_is_ready_dt(&config->bus)) { |
| return -ENODEV; |
| } |
| |
| data->dev = dev; |
| |
| ret = pf1550_init_properties(dev); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| k_work_init(&data->int_routine_work, pf1550_int_routine_work_handler); |
| k_work_init_delayable(&data->int_enable_work, pf1550_int_enable_work_handler); |
| |
| ret = pf1550_configure_interrupt_pin(dev); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = pf1550_enable_interrupt_pin(dev, true); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = pf1550_enable_interrupts(dev); |
| if (ret < 0) { |
| LOG_ERR("Failed to enable interrupts: %d", ret); |
| return ret; |
| } |
| |
| ret = pf1550_update_properties(dev); |
| if (ret < 0) { |
| LOG_ERR("Failed to setup charger: %d", ret); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static DEVICE_API(charger, pf1550_driver_api) = { |
| .get_property = pf1550_get_prop, |
| .set_property = pf1550_set_prop, |
| .charge_enable = pf1550_set_enabled, |
| }; |
| |
| #define PF1550_DEFINE(inst) \ |
| static struct charger_pf1550_led_config charger_pf1550_led_config_##inst = { \ |
| .behaviour = DT_INST_ENUM_IDX(inst, pf1550_led_behaviour), \ |
| }; \ |
| static struct charger_pf1550_data charger_pf1550_data_##inst = { \ |
| .led_config = &charger_pf1550_led_config_##inst, \ |
| }; \ |
| static const struct charger_pf1550_config charger_pf1550_config_##inst = { \ |
| .bus = I2C_DT_SPEC_GET(DT_INST_PARENT(inst)), \ |
| .int_gpio = GPIO_DT_SPEC_INST_GET(inst, pf1550_int_gpios), \ |
| .charge_current_ua = DT_INST_PROP(inst, constant_charge_current_max_microamp), \ |
| .vsys_min_uv = DT_INST_PROP(inst, pf1550_system_voltage_min_threshold_microvolt), \ |
| .therm_mon_mode = DT_INST_PROP(inst, pf1550_thermistor_monitoring_mode), \ |
| .vbus_ilim_ua = DT_INST_PROP(inst, pf1550_vbus_current_limit_microamp), \ |
| .charge_voltage_max_uv = \ |
| DT_INST_PROP(inst, constant_charge_voltage_max_microvolt), \ |
| }; \ |
| \ |
| DEVICE_DT_INST_DEFINE(inst, &pf1550_init, NULL, &charger_pf1550_data_##inst, \ |
| &charger_pf1550_config_##inst, POST_KERNEL, \ |
| CONFIG_MFD_INIT_PRIORITY, &pf1550_driver_api); |
| |
| DT_INST_FOREACH_STATUS_OKAY(PF1550_DEFINE) |