blob: 64ea3482b28afea63c7b50c395bed56ac944d99a [file] [log] [blame]
/*
* Copyright (c), 2023 Basalte bv
*
* SPDX-License-Identifier: Apache-2.0
*/
#define DT_DRV_COMPAT nxp_sc18im704_i2c
#include <errno.h>
#include <zephyr/kernel.h>
#include <zephyr/device.h>
#include <zephyr/init.h>
#include <zephyr/drivers/gpio.h>
#include <zephyr/drivers/i2c.h>
#include <zephyr/drivers/uart.h>
#include <zephyr/logging/log.h>
LOG_MODULE_REGISTER(i2c_sc18im, CONFIG_I2C_LOG_LEVEL);
#include "i2c_sc18im704.h"
struct i2c_sc18im_config {
const struct device *bus;
uint32_t bus_speed;
const struct gpio_dt_spec reset_gpios;
};
struct i2c_sc18im_data {
struct k_mutex lock;
uint32_t i2c_config;
};
int sc18im704_claim(const struct device *dev)
{
struct i2c_sc18im_data *data = dev->data;
return k_mutex_lock(&data->lock, K_FOREVER);
}
int sc18im704_release(const struct device *dev)
{
struct i2c_sc18im_data *data = dev->data;
return k_mutex_unlock(&data->lock);
}
int sc18im704_transfer(const struct device *dev,
const uint8_t *tx_data, uint8_t tx_len,
uint8_t *rx_data, uint8_t rx_len)
{
const struct i2c_sc18im_config *cfg = dev->config;
struct i2c_sc18im_data *data = dev->data;
int ret = 0;
ret = k_mutex_lock(&data->lock, K_FOREVER);
if (ret < 0) {
return ret;
}
if (tx_data != NULL) {
for (uint8_t i = 0; i < tx_len; ++i) {
uart_poll_out(cfg->bus, tx_data[i]);
}
}
if (rx_data != NULL) {
k_timepoint_t end;
for (uint8_t i = 0; i < rx_len && ret == 0; ++i) {
/* Make sure we don't wait forever */
end = sys_timepoint_calc(K_SECONDS(1));
do {
ret = uart_poll_in(cfg->bus, &rx_data[i]);
} while (ret == -1 && !sys_timepoint_expired(end));
}
/* -1 indicates we timed out */
ret = ret == -1 ? -EAGAIN : ret;
if (ret < 0) {
LOG_ERR("Failed to read data (%d)", ret);
}
}
k_mutex_unlock(&data->lock);
return ret;
}
static int i2c_sc18im_configure(const struct device *dev, uint32_t config)
{
struct i2c_sc18im_data *data = dev->data;
if (!(I2C_MODE_CONTROLLER & config)) {
return -EINVAL;
}
if (I2C_ADDR_10_BITS & config) {
return -EINVAL;
}
if (I2C_SPEED_GET(config) != I2C_SPEED_GET(data->i2c_config)) {
uint8_t buf[] = {
SC18IM704_CMD_WRITE_REG,
SC18IM704_REG_I2C_CLK_L,
0,
SC18IM704_CMD_STOP,
};
int ret;
/* CLK value is calculated as 15000000 / (8 * freq), see datasheet */
switch (I2C_SPEED_GET(config)) {
case I2C_SPEED_STANDARD:
buf[2] = 0x13; /* 99 kHz */
break;
case I2C_SPEED_FAST:
buf[2] = 0x05; /* 375 kHz */
break;
default:
return -EINVAL;
}
ret = sc18im704_transfer(dev, buf, sizeof(buf), NULL, 0);
if (ret < 0) {
LOG_ERR("Failed to set I2C speed (%d)", ret);
return -EIO;
}
}
data->i2c_config = config;
return 0;
}
static int i2c_sc18im_get_config(const struct device *dev, uint32_t *config)
{
struct i2c_sc18im_data *data = dev->data;
*config = data->i2c_config;
return 0;
}
static int i2c_sc18im_transfer_msg(const struct device *dev,
struct i2c_msg *msg,
uint16_t addr)
{
uint8_t start[] = {
SC18IM704_CMD_I2C_START,
0x00,
0x00,
};
uint8_t stop = SC18IM704_CMD_STOP;
int ret;
if (msg->flags & I2C_MSG_ADDR_10_BITS || msg->len > UINT8_MAX) {
return -EINVAL;
}
start[1] = addr | (msg->flags & I2C_MSG_RW_MASK);
start[2] = msg->len;
ret = sc18im704_transfer(dev, start, sizeof(start), NULL, 0);
if (ret < 0) {
return ret;
}
if (msg->flags & I2C_MSG_READ) {
/* Send the stop character before reading */
ret = sc18im704_transfer(dev, &stop, 1, msg->buf, msg->len);
if (ret < 0) {
return ret;
}
} else {
ret = sc18im704_transfer(dev, msg->buf, msg->len, NULL, 0);
if (ret < 0) {
return ret;
}
if (msg->flags & I2C_MSG_STOP) {
ret = sc18im704_transfer(dev, &stop, 1, NULL, 0);
if (ret < 0) {
return ret;
}
}
}
return 0;
}
static int i2c_sc18im_transfer(const struct device *dev,
struct i2c_msg *msgs,
uint8_t num_msgs, uint16_t addr)
{
int ret;
if (num_msgs == 0) {
return 0;
}
ret = sc18im704_claim(dev);
if (ret < 0) {
LOG_ERR("Failed to claim I2C bridge (%d)", ret);
return ret;
}
for (uint8_t i = 0; i < num_msgs && ret == 0; ++i) {
ret = i2c_sc18im_transfer_msg(dev, &msgs[i], addr);
}
#ifdef CONFIG_I2C_SC18IM704_VERIFY
if (ret == 0) {
uint8_t buf[] = {
SC18IM704_CMD_READ_REG,
SC18IM704_REG_I2C_STAT,
SC18IM704_CMD_STOP,
};
uint8_t data;
ret = sc18im704_transfer(dev, buf, sizeof(buf), &data, 1);
if (ret == 0 && data != SC18IM704_I2C_STAT_OK) {
ret = -EIO;
}
}
#endif /* CONFIG_I2C_SC18IM704_VERIFY */
sc18im704_release(dev);
return ret;
}
static int i2c_sc18im_init(const struct device *dev)
{
const struct i2c_sc18im_config *cfg = dev->config;
struct i2c_sc18im_data *data = dev->data;
int ret;
/* The device baudrate after reset is 9600 */
struct uart_config uart_cfg = {
.baudrate = 9600,
.parity = UART_CFG_PARITY_NONE,
.stop_bits = UART_CFG_STOP_BITS_1,
.data_bits = UART_CFG_DATA_BITS_8,
.flow_ctrl = UART_CFG_FLOW_CTRL_NONE,
};
k_mutex_init(&data->lock);
if (!device_is_ready(cfg->bus)) {
LOG_ERR("UART bus not ready");
return -ENODEV;
}
ret = uart_configure(cfg->bus, &uart_cfg);
if (ret < 0) {
LOG_ERR("Failed to configure UART (%d)", ret);
return ret;
}
if (cfg->reset_gpios.port) {
uint8_t buf[2];
if (!gpio_is_ready_dt(&cfg->reset_gpios)) {
LOG_ERR("Reset GPIO device not ready");
return -ENODEV;
}
ret = gpio_pin_configure_dt(&cfg->reset_gpios, GPIO_OUTPUT_ACTIVE);
if (ret < 0) {
LOG_ERR("Failed to configure reset GPIO (%d)", ret);
return ret;
}
ret = gpio_pin_set_dt(&cfg->reset_gpios, 0);
if (ret < 0) {
LOG_ERR("Failed to set reset GPIO (%d)", ret);
return ret;
}
/* The device sends "OK" */
ret = sc18im704_transfer(dev, NULL, 0, buf, sizeof(buf));
if (ret < 0) {
LOG_ERR("Failed to get OK (%d)", ret);
return ret;
}
}
if (cfg->bus_speed != 9600) {
uint16_t brg = (7372800 / cfg->bus_speed) - 16;
uint8_t buf[] = {
SC18IM704_CMD_WRITE_REG,
SC18IM704_REG_BRG0,
brg & 0xff,
SC18IM704_REG_BRG1,
brg >> 8,
SC18IM704_CMD_STOP,
};
ret = sc18im704_transfer(dev, buf, sizeof(buf), NULL, 0);
if (ret < 0) {
LOG_ERR("Failed to set baudrate (%d)", ret);
return ret;
}
/* Make sure UART buffer is sent */
k_msleep(1);
/* Re-configure the UART controller with the new baudrate */
uart_cfg.baudrate = cfg->bus_speed;
ret = uart_configure(cfg->bus, &uart_cfg);
if (ret < 0) {
LOG_ERR("Failed to re-configure UART (%d)", ret);
return ret;
}
}
return 0;
}
static const struct i2c_driver_api i2c_sc18im_driver_api = {
.configure = i2c_sc18im_configure,
.get_config = i2c_sc18im_get_config,
.transfer = i2c_sc18im_transfer,
#ifdef CONFIG_I2C_RTIO
.iodev_submit = i2c_iodev_submit_fallback,
#endif
};
#define I2C_SC18IM_DEFINE(n) \
\
static const struct i2c_sc18im_config i2c_sc18im_config_##n = { \
.bus = DEVICE_DT_GET(DT_BUS(DT_INST_PARENT(n))), \
.bus_speed = DT_PROP_OR(DT_INST_PARENT(n), target_speed, 9600), \
.reset_gpios = GPIO_DT_SPEC_GET_OR(DT_INST_PARENT(n), reset_gpios, {0}), \
}; \
static struct i2c_sc18im_data i2c_sc18im_data_##n = { \
.i2c_config = I2C_MODE_CONTROLLER | (I2C_SPEED_STANDARD << I2C_SPEED_SHIFT), \
}; \
\
I2C_DEVICE_DT_INST_DEFINE(n, i2c_sc18im_init, NULL, \
&i2c_sc18im_data_##n, &i2c_sc18im_config_##n, \
POST_KERNEL, CONFIG_I2C_SC18IM704_INIT_PRIORITY, \
&i2c_sc18im_driver_api);
DT_INST_FOREACH_STATUS_OKAY(I2C_SC18IM_DEFINE)