1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/esp32/periph/i2c_hw.c
2022-07-17 18:48:23 +02:00

648 lines
20 KiB
C

/*
* Copyright (C) 2018 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup cpu_esp32
* @ingroup drivers_periph_i2c
* @{
*
* @file
* @brief Low-level I2C driver implementation for ESP32 SDK
*
* @note The hardware implementation seems to be very poor and faulty.
* I2C commands in the I2C command pipeline are not executed
* sporadically. A number of ACK errors and timeouts caused by
* protocol errors are the result. You should use the hardware
* implementation only if they can be tolerated.
*
* @author Gunar Schorcht <gunar@schorcht.net>
*
* @}
*/
/*
* The I2C peripheral driver uses a mixture of the high level functions and
* the HAL driver. While complex functions like the configuration of the I2C
* interface use the high level interface, the HAL interface is used by data
* transfer functions for better performance and better control.
*/
#include <errno.h>
#include "gpio_arch.h"
#include "irq_arch.h"
#include "log.h"
#include "macros/units.h"
#include "mutex.h"
#include "periph_conf.h"
#include "periph/i2c.h"
#include "ztimer.h"
#include "esp_attr.h"
#include "driver/i2c.h"
#include "hal/i2c_hal.h"
#include "hal/interrupt_controller_types.h"
#include "hal/interrupt_controller_ll.h"
#include "rom/ets_sys.h"
#include "soc/i2c_reg.h"
#define ENABLE_DEBUG 0
#include "debug.h"
typedef struct {
i2c_speed_t clk_freq; /* clock freuency in Hz according to bus speed */
uint8_t cmd; /* command index of a transfer*/
uint8_t cmd_op; /* main command op code of a transfer */
uint8_t len; /* data length of a transfer */
uint32_t status; /* results of a transfer */
mutex_t cmd_lock; /* command execution locking */
mutex_t dev_lock; /* device locking */
} _i2c_bus_t;
static i2c_hal_context_t _i2c_hw[] = {
#if defined(I2C0_SPEED)
{ .dev = I2C_LL_GET_HW(0) },
#endif
#if defined(I2C1_SPEED)
{ .dev = I2C_LL_GET_HW(1) },
#endif
};
static _i2c_bus_t _i2c_bus[I2C_NUMOF] = {
#if defined(I2C0_SPEED)
{
.cmd_lock = MUTEX_INIT_LOCKED,
.dev_lock = MUTEX_INIT,
},
#endif
#if defined(I2C1_SPEED)
{
.cmd_lock = MUTEX_INIT_LOCKED,
.dev_lock = MUTEX_INIT,
},
#endif
};
/* functions used from ESP-IDF driver that are not exposed in API */
extern esp_err_t i2c_hw_fsm_reset(i2c_port_t i2c_num);
/* forward declaration of internal functions */
static void _i2c_start_cmd(i2c_t dev);
static void _i2c_stop_cmd(i2c_t dev);
static void _i2c_end_cmd(i2c_t dev);
static void _i2c_write_cmd(i2c_t dev, const uint8_t* data, uint8_t len);
static void _i2c_read_cmd(i2c_t dev, uint8_t len, bool last);
static void _i2c_transfer(i2c_t dev);
static void _i2c_intr_handler(void *arg);
static int _i2c_status_to_errno(i2c_t dev);
void i2c_init(i2c_t dev)
{
_Static_assert(I2C_NUMOF <= I2C_NUMOF_MAX, "Too many I2C devices defined");
_Static_assert(I2C_NUMOF == ARRAY_SIZE(_i2c_hw),
"Number of hardware descriptors doesn't match the I2C_NUMOF");
assert(dev < I2C_NUMOF);
/* According to the Technical Reference Manual, only FAST mode is supported,
* but FAST PLUS mode seems to work also. */
assert(i2c_config[dev].speed <= I2C_SPEED_FAST_PLUS);
/* GPIOs for SCL and SDA signals must not already be used for peripherals */
if (((gpio_get_pin_usage(i2c_config[dev].scl) != _I2C) &&
(gpio_get_pin_usage(i2c_config[dev].scl) != _GPIO)) ||
((gpio_get_pin_usage(i2c_config[dev].sda) != _I2C) &&
(gpio_get_pin_usage(i2c_config[dev].sda) != _GPIO))) {
LOG_TAG_ERROR("i2c", "GPIO%u and/or GPIO%u are used for %s/%s and "
"cannot be used as I2C interface\n",
i2c_config[dev].scl, i2c_config[dev].sda,
gpio_get_pin_usage_str(i2c_config[dev].scl),
gpio_get_pin_usage_str(i2c_config[dev].sda));
assert(0);
}
_i2c_bus[dev].cmd = 0;
_i2c_bus[dev].len = 0;
i2c_acquire(dev);
i2c_config_t cfg = {};
cfg.mode = I2C_MODE_MASTER;
cfg.sda_io_num = i2c_config[dev].sda;
cfg.scl_io_num = i2c_config[dev].scl;
cfg.sda_pullup_en = i2c_config[dev].sda_pullup;
cfg.scl_pullup_en = i2c_config[dev].scl_pullup;
#if SOC_I2C_SUPPORT_RTC
cfg.clk_flags = I2C_SCLK_SRC_FLAG_LIGHT_SLEEP;
#endif
switch (i2c_config[dev].speed) {
case I2C_SPEED_LOW:
cfg.master.clk_speed = 10 * KHZ(1);
break;
case I2C_SPEED_NORMAL:
cfg.master.clk_speed = 100 * KHZ(1);
break;
case I2C_SPEED_FAST:
cfg.master.clk_speed = 400 * KHZ(1);
break;
case I2C_SPEED_FAST_PLUS:
cfg.master.clk_speed = 1000 * KHZ(1);
break;
case I2C_SPEED_HIGH:
cfg.master.clk_speed = 3400 * KHZ(1);
break;
default:
LOG_TAG_ERROR("i2c", "Invalid speed value in %s\n", __func__);
assert(0);
}
_i2c_bus[dev].clk_freq = cfg.master.clk_speed;
/* configures the GPIOs, sets the bus timing and enables the periphery */
i2c_param_config(dev, &cfg);
#if defined(SOC_I2C_SUPPORT_APB)
/* If I2C clock is derived from APB clock, the bus timing parameters
* have to be corrected if the APB clock is less than 80 MHz */
extern uint32_t rtc_clk_apb_freq_get(void);
uint32_t apb_clk = rtc_clk_apb_freq_get();
if (apb_clk < MHZ(80)) {
i2c_clk_cal_t clk_cfg;
i2c_ll_cal_bus_clk(apb_clk, cfg.master.clk_speed, &clk_cfg);
i2c_ll_set_bus_timing(_i2c_hw[dev].dev, &clk_cfg);
}
#endif
/* store the usage type in GPIO table */
gpio_set_pin_usage(i2c_config[dev].scl, _I2C);
gpio_set_pin_usage(i2c_config[dev].sda, _I2C);
/* route all I2C interrupt sources to same the CPU interrupt */
intr_matrix_set(PRO_CPU_NUM, i2c_periph_signal[dev].irq, CPU_INUM_I2C);
/* set interrupt handler and enable the CPU interrupt */
intr_cntrl_ll_set_int_handler(CPU_INUM_I2C, _i2c_intr_handler, NULL);
intr_cntrl_ll_enable_interrupts(BIT(CPU_INUM_I2C));
i2c_release(dev);
}
void i2c_acquire(i2c_t dev)
{
DEBUG ("%s\n", __func__);
assert(dev < I2C_NUMOF);
mutex_lock(&_i2c_bus[dev].dev_lock);
}
void i2c_release(i2c_t dev)
{
DEBUG ("%s\n", __func__);
assert(dev < I2C_NUMOF);
mutex_unlock(&_i2c_bus[dev].dev_lock);
}
int i2c_read_bytes(i2c_t dev, uint16_t addr, void *data, size_t len, uint8_t flags)
{
DEBUG ("%s dev=%u addr=%02x data=%p len=%d flags=%01x\n",
__func__, dev, addr, data, len, flags);
assert(dev < I2C_NUMOF);
assert(len > 0);
assert(data != NULL);
int res;
_i2c_bus[dev].cmd_op = I2C_LL_CMD_READ;
_i2c_bus[dev].cmd = 0;
/* reset TX/RX FIFO queue */
i2c_hal_txfifo_rst(&_i2c_hw[dev]);
i2c_hal_rxfifo_rst(&_i2c_hw[dev]);
/* if I2C_NOSTART is not set, START condition and ADDR is used */
if (!(flags & I2C_NOSTART)) {
/* send START condition */
_i2c_start_cmd(dev);
/* address handling */
if (flags & I2C_ADDR10) {
/* prepare 10 bit address bytes */
uint8_t addr10[2];
addr10[0] = 0xf0 | (addr & 0x0300) >> 7 | I2C_MASTER_READ;
addr10[1] = addr & 0xff;
/* send ADDR with read flag */
_i2c_write_cmd(dev, addr10, 2);
}
else {
/* send ADDR with read flag */
uint8_t addr7 = (addr << 1 | I2C_READ);
_i2c_write_cmd(dev, &addr7, 1);
}
}
/* read data bytes in blocks of SOC_I2C_FIFO_LEN bytes */
uint32_t off = 0;
/* if len > SOC_I2C_FIFO_LEN read SOC_I2C_FIFO_LEN bytes at a time */
while (len > SOC_I2C_FIFO_LEN) {
/* read one block of data bytes command */
_i2c_bus[dev].len = SOC_I2C_FIFO_LEN;
_i2c_read_cmd(dev, SOC_I2C_FIFO_LEN, false);
_i2c_end_cmd(dev);
_i2c_transfer(dev);
res = _i2c_status_to_errno(dev);
if (res) {
return res;
}
/* if transfer was successful, fetch the data from I2C RAM */
i2c_hal_read_rxfifo(&_i2c_hw[dev], data + off, len);
/* reset RX FIFO queue */
i2c_hal_rxfifo_rst(&_i2c_hw[dev]);
len -= SOC_I2C_FIFO_LEN;
off += SOC_I2C_FIFO_LEN;
}
/* read remaining data bytes command with a final NAK */
_i2c_bus[dev].len = len;
_i2c_read_cmd(dev, len, true);
/* if I2C_NOSTOP flag is not set, send STOP condition is used */
if (!(flags & I2C_NOSTOP)) {
/* send STOP condition */
_i2c_stop_cmd(dev);
}
else {
/* otherwise place end command in pipeline */
_i2c_end_cmd(dev);
}
/* finish operation by executing the command pipeline */
_i2c_transfer(dev);
if ((res = _i2c_status_to_errno(dev))) {
return res;
}
/* fetch the data from RX FIFO */
i2c_hal_read_rxfifo(&_i2c_hw[dev], data + off, len);
/* return 0 on success */
return 0;
}
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data, size_t len, uint8_t flags)
{
DEBUG ("%s dev=%u addr=%02x data=%p len=%d flags=%01x\n",
__func__, dev, addr, data, len, flags);
assert(dev < I2C_NUMOF);
assert(len > 0);
assert(data != NULL);
int res;
_i2c_bus[dev].cmd_op = I2C_LL_CMD_WRITE;
_i2c_bus[dev].cmd = 0;
/* reset TX FIFO queue */
i2c_hal_txfifo_rst(&_i2c_hw[dev]);
/* if I2C_NOSTART is not set, START condition and ADDR is used */
if (!(flags & I2C_NOSTART)) {
/* send START condition */
_i2c_start_cmd(dev);
/* address handling */
if (flags & I2C_ADDR10) {
/* prepare 10 bit address bytes */
uint8_t addr10[2];
addr10[0] = 0xf0 | (addr & 0x0300) >> 7 | I2C_MASTER_WRITE;
addr10[1] = addr & 0xff;
/* send ADDR without read flag */
_i2c_write_cmd(dev, addr10, 2);
}
else {
/* send ADDR without read flag */
uint8_t addr7 = addr << 1;
_i2c_write_cmd(dev, &addr7, 1);
}
}
/* send data bytes in blocks of SOC_I2C_FIFO_LEN bytes */
uint32_t off = 0;
uint32_t tx_fifo_free;
/* get available TX FIFO space */
i2c_hal_get_txfifo_cnt(&_i2c_hw[dev], &tx_fifo_free);
/* if len > SOC_I2C_FIFO_LEN write SOC_I2C_FIFO_LEN bytes at a time */
while (len > tx_fifo_free) {
/* send one block of data bytes */
_i2c_bus[dev].len = tx_fifo_free;
_i2c_write_cmd(dev, (uint8_t*)data + off, tx_fifo_free);
_i2c_end_cmd(dev);
_i2c_transfer(dev);
res = _i2c_status_to_errno(dev);
if (res) {
return res;
}
len -= tx_fifo_free;
off += tx_fifo_free;
/* reset TX FIFO queue */
i2c_hal_txfifo_rst(&_i2c_hw[dev]);
/* update available TX FIFO space */
i2c_hal_get_txfifo_cnt(&_i2c_hw[dev], &tx_fifo_free);
}
/* write remaining data bytes command */
_i2c_bus[dev].len = len;
_i2c_write_cmd(dev, (uint8_t*)data + off, len);
/* if I2C_NOSTOP flag is not set, send STOP condition is used */
if (!(flags & I2C_NOSTOP)) {
/* send STOP condition */
_i2c_stop_cmd(dev);
}
else {
/* otherwise place end command in pipeline */
_i2c_end_cmd(dev);
}
/* finish operation by executing the command pipeline */
_i2c_transfer(dev);
return _i2c_status_to_errno(dev);
}
#if defined(MCU_ESP32)
#define I2C_NACK_INT_ENA_M I2C_ACK_ERR_INT_ENA_M
#endif
/*
* @note:
* It is a known problem that the ESP32x I2C hardware implementation checks
* the ACK when the next command in the pipeline is executed. That is,
* for a segmented write operation of type `START ADDR DATA0 STOP` the
* `I2C_END_DETECT_INT` and the `I2C_NACK_INT` are triggered simultaneously
* in case of acknowledge errors. In this case it is not clear from the
* interrupt status whether the acknolegdement error has occurred for the
* ADDR or the DATA0 byte. The only way to find out is to check the remaining
* TX FIFO length. If the number of bytes in the TX-FIFO is still higher
* (for two-byte addresses) or equal (for one-byte addresses) to the length of
* the data, the acknolegdement error occurred for the ADDR field.
*/
static int _i2c_status_to_errno(i2c_t dev)
{
if (_i2c_bus[dev].status & I2C_ARBITRATION_LOST_INT_ENA_M) {
LOG_TAG_DEBUG("i2c", "arbitration lost dev=%u\n", dev);
return -EAGAIN;
}
if (_i2c_bus[dev].status & I2C_NACK_INT_ENA_M) {
LOG_TAG_DEBUG("i2c", "ack error dev=%u\n", dev);
if (_i2c_bus[dev].cmd_op == I2C_LL_CMD_WRITE) {
/*
* @note: In a write transfer, an ACK error can happen for the
* address field as well as for data. If the FIFO still contains
* all data bytes, the ACK error happened in address field and we
* have to return -ENXIO. Otherwise, the ACK error happened in data
* field and we have to return -EIO.
*/
uint32_t cnt;
i2c_hal_get_txfifo_cnt(&_i2c_hw[dev], &cnt);
return ((SOC_I2C_FIFO_LEN - cnt) >= _i2c_bus[dev].len) ? -ENXIO : -EIO;
}
else {
/*
* @note: In a read transfer, an ACK is only expected for the
* address field. Thus, an ACK error can only happen for the address
* field. Therefore, we always return -ENXIO in case of an ACK
* error.
*/
return -ENXIO;
}
}
if (_i2c_bus[dev].status & I2C_TIME_OUT_INT_ENA_M) {
LOG_TAG_DEBUG("i2c", "bus timeout dev=%u\n", dev);
i2c_hw_fsm_reset(dev);
return -ETIMEDOUT;
}
return 0;
}
static void _i2c_start_cmd(i2c_t dev)
{
DEBUG ("%s\n", __func__);
/* place START condition command in command queue */
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_RESTART };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
static void _i2c_stop_cmd(i2c_t dev)
{
DEBUG ("%s\n", __func__);
/* place STOP condition command in command queue */
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_STOP };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
static void _i2c_end_cmd(i2c_t dev)
{
DEBUG ("%s\n", __func__);
/* place END command for continues data transmission in command queue */
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_END };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
static void _i2c_write_cmd(i2c_t dev, const uint8_t* data, uint8_t len)
{
DEBUG ("%s dev=%u data=%p len=%d\n", __func__, dev, data, len);
if (len > SOC_I2C_FIFO_LEN) {
LOG_TAG_ERROR("i2c", "Maximum number of bytes (32 bytes) that can be "
"sent with on transfer reached\n");
return;
}
/* store the data in TX FIFO */
i2c_hal_write_txfifo(&_i2c_hw[dev], (uint8_t *)data, len);
/* place WRITE command for multiple bytes in command queue */
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_WRITE,
.byte_num = len,
.ack_en = 1,
.ack_exp = 0,
.ack_val = 0 };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
static void _i2c_read_cmd(i2c_t dev, uint8_t len, bool last)
{
DEBUG ("%s dev=%u len=%d\n", __func__, dev, len);
if (len < 1 || len > SOC_I2C_FIFO_LEN) {
/* at least one byte has to be read */
LOG_TAG_ERROR("i2c", "At least one byte has to be read\n");
return;
}
if (len > 1)
{
/* place READ command for len-1 bytes with positive ack in command queue*/
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_READ,
.byte_num = len-1,
.ack_en = 0,
.ack_exp = 0,
.ack_val = 0 };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
/* place READ command for last byte with negative ack in last segment in command queue*/
i2c_hw_cmd_t cmd = { .op_code = I2C_LL_CMD_READ,
.byte_num = 1,
.ack_en = 0,
.ack_exp = 0,
.ack_val = last ? 1 : 0 };
i2c_hal_write_cmd_reg(&_i2c_hw[dev], cmd, _i2c_bus[dev].cmd);
/* increment the command counter */
_i2c_bus[dev].cmd++;
}
/* a maximum transfer 33 byte * 9 clock cycles * 1/100000 s takes at
* I2C_SPEED_NORMAL about 450 us, at I2C_SPEED_NORMAL_LOW about 4.5 ms */
#define I2C_TRANSFER_TIMEOUT 640
/* combination of all transfer interrupts */
#define I2C_LL_MASTER_INT (I2C_LL_MASTER_TX_INT | I2C_LL_MASTER_RX_INT)
void _i2c_transfer_timeout(void *arg)
{
i2c_t dev = (i2c_t)(uintptr_t)arg;
/* reset the hardware if I2C got stuck */
i2c_hw_fsm_reset(dev);
/* set result to timeout */
_i2c_bus[dev].status = I2C_TIME_OUT_INT_ENA_M;
/* wake up the thread that is waiting for the results */
mutex_unlock(&_i2c_bus[dev].cmd_lock);
}
/* Transfer of commands in I2C controller command pipeline */
static void _i2c_transfer(i2c_t dev)
{
DEBUG("%s cmd=%d\n", __func__, _i2c_bus[dev].cmd);
/* disable and enable all transmission interrupts and clear current status */
i2c_hal_clr_intsts_mask(&_i2c_hw[dev], I2C_LL_MASTER_INT);
i2c_hal_enable_intr_mask(&_i2c_hw[dev], I2C_LL_MASTER_INT);
/* set a timer for the case the I2C hardware gets stuck */
#if defined(MODULE_ZTIMER_MSEC)
uint32_t timeout = ((I2C_TRANSFER_TIMEOUT * KHZ(1)) / _i2c_bus[dev].clk_freq) + 1;
ztimer_t timer = { .callback = _i2c_transfer_timeout,
.arg = (void*)(uintptr_t)dev };
ztimer_set(ZTIMER_MSEC, &timer, timeout);
#endif
/* start the execution of commands in command pipeline */
_i2c_bus[dev].status = 0;
i2c_hal_update_config(&_i2c_hw[dev]);
i2c_hal_trans_start(&_i2c_hw[dev]);
/* wait for transfer results and remove timeout timer*/
mutex_lock(&_i2c_bus[dev].cmd_lock);
#if defined(MODULE_ZTIMER_MSEC)
ztimer_remove(ZTIMER_MSEC, &timer);
#endif
/* returned from transmission */
DEBUG("%s status=%08"PRIx32"\n", __func__, _i2c_bus[dev].status);
/* reset command and data index */
_i2c_bus[dev].cmd = 0;
}
static void IRAM_ATTR _i2c_intr_handler(void *arg)
{
/* to satisfy the compiler */
(void)arg;
irq_isr_enter();
/* all I2C peripheral interrupt sources are routed to the same interrupt,
so we have to use the status register to distinguish interruptees */
for (unsigned dev = 0; dev < I2C_NUMOF; dev++) {
uint32_t mask = i2c_ll_get_intsts_mask(_i2c_hw[dev].dev);
/* test for transfer related interrupts */
if (mask) {
_i2c_bus[dev].status = mask;
/* disable all interrupts and clear pending interrupts */
i2c_hal_clr_intsts_mask(&_i2c_hw[dev], I2C_LL_MASTER_INT);
i2c_hal_disable_intr_mask(&_i2c_hw[dev], I2C_LL_MASTER_INT);
/* wake up the thread that is waiting for the results */
mutex_unlock(&_i2c_bus[dev].cmd_lock);
}
}
irq_isr_exit();
}
void i2c_print_config(void)
{
if (I2C_NUMOF) {
for (unsigned dev = 0; dev < I2C_NUMOF; dev++) {
printf("\tI2C_DEV(%u)\tscl=%d sda=%d\n",
dev, i2c_config[dev].scl, i2c_config[dev].sda);
}
}
else {
LOG_TAG_INFO("i2c", "no I2C devices\n");
}
}