2023-03-31 20:43:51 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2023 Cirrus Logic, Inc.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*
|
|
|
|
* Emulator for SBS 1.1 compliant smart battery charger.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define DT_DRV_COMPAT sbs_sbs_charger
|
|
|
|
|
|
|
|
#include <zephyr/device.h>
|
|
|
|
#include <zephyr/drivers/emul.h>
|
|
|
|
#include <zephyr/drivers/i2c.h>
|
|
|
|
#include <zephyr/drivers/i2c_emul.h>
|
|
|
|
#include <zephyr/logging/log.h>
|
|
|
|
#include <zephyr/sys/byteorder.h>
|
|
|
|
|
|
|
|
#include "sbs_charger.h"
|
|
|
|
|
|
|
|
LOG_MODULE_REGISTER(sbs_sbs_charger);
|
|
|
|
|
|
|
|
/** Static configuration for the emulator */
|
|
|
|
struct sbs_charger_emul_cfg {
|
|
|
|
/** I2C address of emulator */
|
|
|
|
uint16_t addr;
|
|
|
|
};
|
|
|
|
|
2023-12-05 00:03:42 +01:00
|
|
|
/** Run-time data used by the emulator */
|
|
|
|
struct sbs_charger_emul_data {
|
|
|
|
uint16_t reg_charger_mode;
|
|
|
|
};
|
|
|
|
|
2023-03-31 20:43:51 +02:00
|
|
|
static int emul_sbs_charger_reg_write(const struct emul *target, int reg, int val)
|
|
|
|
{
|
2023-12-05 00:03:42 +01:00
|
|
|
struct sbs_charger_emul_data *data = target->data;
|
|
|
|
|
2023-03-31 20:43:51 +02:00
|
|
|
LOG_INF("write %x = %x", reg, val);
|
|
|
|
switch (reg) {
|
2023-12-05 00:03:42 +01:00
|
|
|
case SBS_CHARGER_REG_CHARGER_MODE:
|
|
|
|
data->reg_charger_mode = val;
|
|
|
|
break;
|
2023-03-31 20:43:51 +02:00
|
|
|
default:
|
|
|
|
LOG_ERR("Unknown write %x", reg);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int emul_sbs_charger_reg_read(const struct emul *target, int reg, int *val)
|
|
|
|
{
|
|
|
|
switch (reg) {
|
|
|
|
case SBS_CHARGER_REG_SPEC_INFO:
|
|
|
|
case SBS_CHARGER_REG_CHARGER_MODE:
|
|
|
|
case SBS_CHARGER_REG_STATUS:
|
|
|
|
case SBS_CHARGER_REG_ALARM_WARNING:
|
|
|
|
/* Arbitrary stub value. */
|
|
|
|
*val = 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LOG_ERR("Unknown register 0x%x read", reg);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
LOG_INF("read 0x%x = 0x%x", reg, *val);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sbs_charger_emul_transfer_i2c(const struct emul *target, struct i2c_msg *msgs,
|
|
|
|
int num_msgs, int addr)
|
|
|
|
{
|
|
|
|
/* Largely copied from emul_sbs_gauge.c */
|
|
|
|
struct sbs_charger_emul_data *data;
|
|
|
|
unsigned int val;
|
|
|
|
int reg;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
data = target->data;
|
|
|
|
|
2023-09-07 20:35:27 +02:00
|
|
|
i2c_dump_msgs_rw(target->dev, msgs, num_msgs, addr, false);
|
2023-03-31 20:43:51 +02:00
|
|
|
switch (num_msgs) {
|
|
|
|
case 2:
|
|
|
|
if (msgs->flags & I2C_MSG_READ) {
|
|
|
|
LOG_ERR("Unexpected read");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
if (msgs->len != 1) {
|
|
|
|
LOG_ERR("Unexpected msg0 length %d", msgs->len);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
reg = msgs->buf[0];
|
|
|
|
|
|
|
|
/* Now process the 'read' part of the message */
|
|
|
|
msgs++;
|
|
|
|
if (msgs->flags & I2C_MSG_READ) {
|
|
|
|
switch (msgs->len - 1) {
|
|
|
|
case 1:
|
|
|
|
rc = emul_sbs_charger_reg_read(target, reg, &val);
|
|
|
|
if (rc) {
|
|
|
|
/* Return before writing bad value to message buffer */
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* SBS uses SMBus, which sends data in little-endian format. */
|
|
|
|
sys_put_le16(val, msgs->buf);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LOG_ERR("Unexpected msg1 length %d", msgs->len);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* We write a word (2 bytes by the SBS spec) */
|
|
|
|
if (msgs->len != 2) {
|
|
|
|
LOG_ERR("Unexpected msg1 length %d", msgs->len);
|
|
|
|
}
|
|
|
|
uint16_t value = sys_get_le16(msgs->buf);
|
|
|
|
|
|
|
|
rc = emul_sbs_charger_reg_write(target, reg, value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LOG_ERR("Invalid number of messages: %d", num_msgs);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct i2c_emul_api sbs_charger_emul_api_i2c = {
|
|
|
|
.transfer = sbs_charger_emul_transfer_i2c,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int emul_sbs_sbs_charger_init(const struct emul *target, const struct device *parent)
|
|
|
|
{
|
|
|
|
ARG_UNUSED(target);
|
|
|
|
ARG_UNUSED(parent);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Main instantiation macro. SBS Charger Emulator only implemented for I2C
|
|
|
|
*/
|
|
|
|
#define SBS_CHARGER_EMUL(n) \
|
2023-12-05 00:03:42 +01:00
|
|
|
static struct sbs_charger_emul_data sbs_charger_emul_data_##n; \
|
|
|
|
\
|
2023-03-31 20:43:51 +02:00
|
|
|
static const struct sbs_charger_emul_cfg sbs_charger_emul_cfg_##n = { \
|
|
|
|
.addr = DT_INST_REG_ADDR(n), \
|
|
|
|
}; \
|
2023-12-05 00:03:42 +01:00
|
|
|
EMUL_DT_INST_DEFINE(n, emul_sbs_sbs_charger_init, &sbs_charger_emul_data_##n, \
|
|
|
|
&sbs_charger_emul_cfg_##n, &sbs_charger_emul_api_i2c, NULL)
|
2023-03-31 20:43:51 +02:00
|
|
|
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(SBS_CHARGER_EMUL)
|