nuttx/arch/xtensa/src/esp32s2/esp32s2_i2c.c
anjiahao e1ca516488 use SEM_INITIALIZER inside of NXSEM_INITIALIZER
Signed-off-by: anjiahao <anjiahao@xiaomi.com>
2022-10-22 14:50:48 +08:00

1617 lines
44 KiB
C

/****************************************************************************
* arch/xtensa/src/esp32s2/esp32s2_i2c.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#ifdef CONFIG_ESP32S2_I2C
#include <assert.h>
#include <debug.h>
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h>
#include <sys/types.h>
#include <time.h>
#include <nuttx/arch.h>
#include <nuttx/clock.h>
#include <nuttx/irq.h>
#include <nuttx/i2c/i2c_master.h>
#include <nuttx/mutex.h>
#include <nuttx/semaphore.h>
#include <arch/board/board.h>
#include "esp32s2_gpio.h"
#include "esp32s2_i2c.h"
#include "esp32s2_irq.h"
#include "xtensa.h"
#include "hardware/esp32s2_gpio_sigmap.h"
#include "hardware/esp32s2_i2c.h"
#include "hardware/esp32s2_soc.h"
#include "hardware/esp32s2_system.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/* Command format */
#define I2C_BASE_CMD(_cmd, _check_ack) (((_cmd) << 11) | \
((_check_ack) << 8))
#define I2C_SEND_CMD(_cmd, _check_ack, _bytes) (((_cmd) << 11) | \
((_check_ack) << 8) | \
(_bytes))
#define I2C_RECV_CMD(_cmd, _ack_val, _bytes) (((_cmd) << 11) | \
((_ack_val) << 10) | \
(_bytes))
/* Helper */
#ifdef CONFIG_I2C_POLLED
#define TIMESPEC_TO_US(sec, nano) ((sec) * USEC_PER_SEC + (nano) / NSEC_PER_USEC)
#endif
#define ESP32S2_I2CTIMEOTICKS \
(SEC2TICK(CONFIG_ESP32S2_I2CTIMEOSEC) + MSEC2TICK(CONFIG_ESP32S2_I2CTIMEOMS))
/* Default option */
#define I2C_FIFO_SIZE (32)
#define I2C_FILTER_CYC_NUM_DEF (7)
#define I2C_CLK_FREQ_DEF (100 * 1000)
#define I2C_INT_ERR_MASK (I2C_NACK_INT_ENA | \
I2C_TIME_OUT_INT_ENA | \
I2C_ARBITRATION_LOST_INT_ENA)
#define I2C_SCL_CYC_NUM_DEF 9
/* Access I2C FIFO Data registers via Peripheral Bus 2 (PeriBus2).
* Refer to ESP32-S2 Technical Reference Manual, section 3.3.5 for further
* details.
*/
#define FIFO_DATA_REG(i) (0x60013000 + (i) * 0x14000 + 0x001c)
/* I2C event trace logic.
* NOTE: trace uses the internal, non-standard, low-level debug interface
* syslog() but does not require that any other debug is enabled.
*/
#ifndef CONFIG_I2C_TRACE
# define i2c_tracereset(p)
# define i2c_tracenew(p,s)
# define i2c_traceevent(p,e,a,s)
# define i2c_tracedump(p)
#endif
#ifndef CONFIG_I2C_NTRACE
# define CONFIG_I2C_NTRACE 32
#endif
/****************************************************************************
* Private Types
****************************************************************************/
/* I2C state */
enum esp32s2_i2cstate_e
{
I2CSTATE_IDLE = 0,
I2CSTATE_PROC,
I2CSTATE_STOP,
#ifndef CONFIG_I2C_POLLED
I2CSTATE_FINISH,
#endif
I2CSTATE_ERROR
};
/* I2C hardware command */
enum i2c_opmode_e
{
I2C_CMD_RESTART = 0, /* I2C restart command */
I2C_CMD_WRITE, /* I2C write command */
I2C_CMD_READ, /* I2C read command */
I2C_CMD_STOP, /* I2C stop command */
I2C_CMD_END /* I2C end command */
};
#ifdef CONFIG_I2C_TRACE
/* Trace events */
enum esp32s2_trace_e
{
I2CEVENT_NONE = 0, /* No events have occurred with this status */
I2CEVENT_SENDADDR, /* Start/Master bit set and address sent, param = addr */
I2CEVENT_SENDBYTE, /* Send byte, param = bytes */
I2CEVENT_RCVMODEEN, /* Receive mode enabled, param = 0 */
I2CEVENT_RCVBYTE, /* Read more dta, param = bytes */
I2CEVENT_STOP, /* Last byte sten, send stop, param = length */
I2CEVENT_ERROR /* Error occurred, param = 0 */
};
/* Trace data */
struct esp32s2_trace_s
{
uint32_t status; /* I2C 32-bit SR status */
uint32_t count; /* Interrupt count when status change */
enum esp32s2_trace_e event; /* Last event that occurred with this status */
uint32_t parm; /* Parameter associated with the event */
clock_t time; /* First of event or first status */
};
#endif /* CONFIG_I2C_TRACE */
/* I2C Device hardware configuration */
struct esp32s2_i2c_config_s
{
uint32_t clk_freq; /* Clock frequency */
uint8_t scl_pin; /* GPIO configuration for SCL as SCL */
uint8_t sda_pin; /* GPIO configuration for SDA as SDA */
#ifndef CONFIG_I2C_POLLED
uint8_t periph; /* Peripheral ID */
uint8_t irq; /* Interrupt ID */
#endif
uint32_t clk_bit; /* Clock enable bit */
uint32_t rst_bit; /* I2C reset bit */
uint32_t scl_insig; /* I2C SCL input signal index */
uint32_t scl_outsig; /* I2C SCL output signal index */
uint32_t sda_insig; /* I2C SDA input signal index */
uint32_t sda_outsig; /* I2C SDA output signal index */
};
/* I2C Device Private Data */
struct esp32s2_i2c_priv_s
{
const struct i2c_ops_s *ops; /* Standard I2C operations */
uint32_t id; /* I2C instance */
/* Port configuration */
const struct esp32s2_i2c_config_s *config;
int refs; /* Reference count */
mutex_t lock; /* Mutual exclusion mutex */
#ifndef CONFIG_I2C_POLLED
sem_t sem_isr; /* Interrupt wait semaphore */
int cpuint; /* CPU interrupt assigned to this I2C */
#endif
/* I2C work state (see enum esp32s2_i2cstate_e) */
volatile enum esp32s2_i2cstate_e i2cstate;
struct i2c_msg_s *msgv; /* Message list */
uint8_t msgid; /* Current message ID */
ssize_t bytes; /* Processed data bytes */
uint32_t error; /* I2C transform error */
bool ready_read; /* If I2C has read data */
uint32_t clk_freq; /* Current I2C Clock frequency */
#ifdef CONFIG_I2C_TRACE
/* I2C trace support */
int tndx; /* Trace array index */
clock_t start_time; /* Time when the trace was started */
/* The actual trace data */
struct esp32s2_trace_s trace[CONFIG_I2C_NTRACE];
#endif
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static void i2c_init_clock(struct esp32s2_i2c_priv_s *priv,
uint32_t clock);
static void i2c_init(struct esp32s2_i2c_priv_s *priv);
static void i2c_deinit(struct esp32s2_i2c_priv_s *priv);
static int i2c_transfer(struct i2c_master_s *dev, struct i2c_msg_s *msgs,
int count);
static inline void i2c_process(struct esp32s2_i2c_priv_s *priv,
uint32_t status);
#ifndef CONFIG_I2C_POLLED
static int i2c_sem_waitdone(struct esp32s2_i2c_priv_s *priv);
#endif
#ifdef CONFIG_I2C_POLLED
static int i2c_polling_waitdone(struct esp32s2_i2c_priv_s *priv);
#endif
static void i2c_clear_bus(struct esp32s2_i2c_priv_s *priv);
static void i2c_reset_fsmc(struct esp32s2_i2c_priv_s *priv);
#ifdef CONFIG_I2C_RESET
static int i2c_reset(struct i2c_master_s *dev);
#endif
#ifdef CONFIG_I2C_TRACE
static void i2c_tracereset(struct esp32s2_i2c_priv_s *priv);
static void i2c_tracenew(struct esp32s2_i2c_priv_s *priv, uint32_t status);
static void i2c_traceevent(struct esp32s2_i2c_priv_s *priv,
enum esp32s2_trace_e event,
uint32_t parm,
uint32_t status);
static void i2c_tracedump(struct esp32s2_i2c_priv_s *priv);
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Private Data
****************************************************************************/
/* I2C interface */
static const struct i2c_ops_s g_esp32s2_i2c_ops =
{
.transfer = i2c_transfer
#ifdef CONFIG_I2C_RESET
, .reset = i2c_reset
#endif
};
/* I2C device structures */
#ifdef CONFIG_ESP32S2_I2C0
static const struct esp32s2_i2c_config_s g_esp32s2_i2c0_config =
{
.clk_freq = I2C_CLK_FREQ_DEF,
.scl_pin = CONFIG_ESP32S2_I2C0_SCLPIN,
.sda_pin = CONFIG_ESP32S2_I2C0_SDAPIN,
#ifndef CONFIG_I2C_POLLED
.periph = ESP32S2_PERIPH_I2C_EXT0,
.irq = ESP32S2_IRQ_I2C_EXT0,
#endif
.clk_bit = SYSTEM_I2C_EXT0_CLK_EN,
.rst_bit = SYSTEM_I2C_EXT0_RST,
.scl_insig = I2CEXT0_SCL_IN_IDX,
.scl_outsig = I2CEXT0_SCL_OUT_IDX,
.sda_insig = I2CEXT0_SDA_IN_IDX,
.sda_outsig = I2CEXT0_SDA_OUT_IDX
};
static struct esp32s2_i2c_priv_s g_esp32s2_i2c0_priv =
{
.ops = &g_esp32s2_i2c_ops,
.id = ESP32S2_I2C0,
.config = &g_esp32s2_i2c0_config,
.refs = 0,
.lock = NXMUTEX_INITIALIZER,
#ifndef CONFIG_I2C_POLLED
.sem_isr = SEM_INITIALIZER(0),
#endif
.i2cstate = I2CSTATE_IDLE,
.msgv = NULL,
.msgid = 0,
.bytes = 0,
.ready_read = false
};
#endif /* CONFIG_ESP32S2_I2C0 */
#ifdef CONFIG_ESP32S2_I2C1
static const struct esp32s2_i2c_config_s g_esp32s2_i2c1_config =
{
.clk_freq = I2C_CLK_FREQ_DEF,
.scl_pin = CONFIG_ESP32S2_I2C1_SCLPIN,
.sda_pin = CONFIG_ESP32S2_I2C1_SDAPIN,
#ifndef CONFIG_I2C_POLLED
.periph = ESP32S2_PERIPH_I2C_EXT1,
.irq = ESP32S2_IRQ_I2C_EXT1,
#endif
.clk_bit = SYSTEM_I2C_EXT1_CLK_EN,
.rst_bit = SYSTEM_I2C_EXT1_RST,
.scl_insig = I2CEXT1_SCL_IN_IDX,
.scl_outsig = I2CEXT1_SCL_OUT_IDX,
.sda_insig = I2CEXT1_SDA_IN_IDX,
.sda_outsig = I2CEXT1_SDA_OUT_IDX
};
static struct esp32s2_i2c_priv_s g_esp32s2_i2c1_priv =
{
.ops = &g_esp32s2_i2c_ops,
.id = ESP32S2_I2C1,
.config = &g_esp32s2_i2c1_config,
.refs = 0,
.lock = NXMUTEX_INITIALIZER,
#ifndef CONFIG_I2C_POLLED
.sem_isr = SEM_INITIALIZER(0),
#endif
.i2cstate = I2CSTATE_IDLE,
.msgv = NULL,
.msgid = 0,
.bytes = 0,
.ready_read = false
};
#endif /* CONFIG_ESP32S2_I2C1 */
/* Trace events strings */
#ifdef CONFIG_I2C_TRACE
static const char *g_trace_names[] =
{
"NONE ",
"SENDADDR ",
"SENDBYTE ",
"RCVMODEEN ",
"RCVBYTE ",
"STOP ",
"ERROR "
};
#endif
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: i2c_reset_fifo
*
* Description:
* Reset I2C RX and TX hardware FIFO.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_reset_fifo(struct esp32s2_i2c_priv_s *priv)
{
uint32_t bits = I2C_TX_FIFO_RST | I2C_RX_FIFO_RST;
modifyreg32(I2C_FIFO_CONF_REG(priv->id), 0, bits);
modifyreg32(I2C_FIFO_CONF_REG(priv->id), bits, 0);
}
/****************************************************************************
* Name: i2c_intr_enable
*
* Description:
* Enable I2C interrupts.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_intr_enable(struct esp32s2_i2c_priv_s *priv)
{
putreg32(UINT32_MAX, I2C_INT_CLR_REG(priv->id));
putreg32(I2C_TRANS_COMPLETE_INT_ENA | I2C_END_DETECT_INT_ENA |
I2C_INT_ERR_MASK, I2C_INT_ENA_REG(priv->id));
}
/****************************************************************************
* Name: i2c_intr_disable
*
* Description:
* Disable I2C interrupts.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_intr_disable(struct esp32s2_i2c_priv_s *priv)
{
putreg32(0, I2C_INT_ENA_REG(priv->id));
putreg32(UINT32_MAX, I2C_INT_CLR_REG(priv->id));
}
/****************************************************************************
* Name: i2c_sendstart
*
* Description:
* Send I2C start signal.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_sendstart(struct esp32s2_i2c_priv_s *priv)
{
struct i2c_msg_s *msg = &priv->msgv[priv->msgid];
/* Write I2C command registers */
putreg32(I2C_BASE_CMD(I2C_CMD_RESTART, 0), I2C_COMD0_REG(priv->id));
putreg32(I2C_SEND_CMD(I2C_CMD_WRITE, 1, 1), I2C_COMD1_REG(priv->id));
putreg32(I2C_BASE_CMD(I2C_CMD_END, 0), I2C_COMD2_REG(priv->id));
/* Write data to FIFO register */
if ((msg->flags & I2C_M_READ) == 0)
{
putreg32(I2C_WRITEADDR8(msg->addr), FIFO_DATA_REG(priv->id));
}
else
{
putreg32(I2C_READADDR8(msg->addr), FIFO_DATA_REG(priv->id));
}
/* Enable I2C master TX interrupt */
i2c_intr_enable(priv);
/* Configure the I2C to trigger a transaction */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_TRANS_START);
}
/****************************************************************************
* Name: i2c_senddata
*
* Description:
* Send I2C data.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_senddata(struct esp32s2_i2c_priv_s *priv)
{
struct i2c_msg_s *msg = &priv->msgv[priv->msgid];
int n = msg->length - priv->bytes;
n = n < I2C_FIFO_SIZE ? n : I2C_FIFO_SIZE;
putreg32(I2C_SEND_CMD(I2C_CMD_WRITE, 1, n), I2C_COMD0_REG(priv->id));
putreg32(I2C_BASE_CMD(I2C_CMD_END, 0), I2C_COMD1_REG(priv->id));
for (int i = 0; i < n; i++)
{
putreg32(msg->buffer[priv->bytes + i], FIFO_DATA_REG(priv->id));
}
priv->bytes += n;
/* Enable I2C master TX interrupt */
i2c_intr_enable(priv);
/* Configure the I2C to trigger a transaction */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_TRANS_START);
}
/****************************************************************************
* Name: i2c_recvdata
*
* Description:
* Transfer data from the FIFO to the driver buffer.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_recvdata(struct esp32s2_i2c_priv_s *priv)
{
struct i2c_msg_s *msg = &priv->msgv[priv->msgid];
uint32_t cmd = getreg32(I2C_COMD0_REG(priv->id));
uint8_t n = cmd & 0xff;
uint32_t data = 0;
for (int i = 0; i < n; i++)
{
data = getreg32(FIFO_DATA_REG(priv->id));
msg->buffer[priv->bytes + i] = data & 0xff;
}
priv->bytes += n;
}
/****************************************************************************
* Name: i2c_startrecv
*
* Description:
* Configure I2C to prepare receiving data and it will create an interrupt
* to receive real data.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_startrecv(struct esp32s2_i2c_priv_s *priv)
{
int ack_value;
struct i2c_msg_s *msg = &priv->msgv[priv->msgid];
int n = msg->length - priv->bytes;
if (n > 1)
{
n -= 1;
n = n < I2C_FIFO_SIZE ? n : I2C_FIFO_SIZE;
ack_value = 0;
}
else
{
ack_value = 1;
}
putreg32(I2C_RECV_CMD(I2C_CMD_READ, ack_value, n),
I2C_COMD0_REG(priv->id));
putreg32(I2C_BASE_CMD(I2C_CMD_END, 0), I2C_COMD1_REG(priv->id));
/* Enable I2C master RX interrupt */
i2c_intr_enable(priv);
/* Configure the I2C to trigger a transaction */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_TRANS_START);
}
/****************************************************************************
* Name: i2c_sendstop
*
* Description:
* Send I2C STOP signal.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_sendstop(struct esp32s2_i2c_priv_s *priv)
{
putreg32(I2C_BASE_CMD(I2C_CMD_STOP, 0), I2C_COMD0_REG(priv->id));
/* Enable I2C master TX interrupt */
i2c_intr_enable(priv);
/* Configure the I2C to trigger a transaction */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_TRANS_START);
}
/****************************************************************************
* Name: i2c_init_clock
*
* Description:
* Initialize I2C hardware clock.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
* bus_freq - Clock frequency of the I2C bus in Hz.
*
****************************************************************************/
static void i2c_init_clock(struct esp32s2_i2c_priv_s *priv,
uint32_t bus_freq)
{
if (bus_freq == priv->clk_freq)
{
return;
}
uint32_t reg_value = 0;
uint32_t scl_low = 0;
uint32_t scl_high = 0;
uint32_t scl_wait_high = 0;
uint32_t sda_hold = 0;
uint32_t sda_sample = 0;
uint32_t setup = 0;
uint32_t hold = 0;
uint32_t timeout = 0;
uint32_t source_clk = APB_CLK_FREQ;
uint32_t half_cycle = source_clk / bus_freq / 2;
scl_low = half_cycle - 1;
putreg32(scl_low, I2C_SCL_LOW_PERIOD_REG(priv->id));
/* By default, scl_wait_high must be less than scl_high */
scl_high = half_cycle / 2 + 2;
scl_wait_high = half_cycle - scl_high;
reg_value = VALUE_TO_FIELD(scl_high, I2C_SCL_HIGH_PERIOD);
reg_value |= VALUE_TO_FIELD(scl_wait_high, I2C_SCL_WAIT_HIGH_PERIOD);
putreg32(reg_value, I2C_SCL_HIGH_PERIOD_REG(priv->id));
sda_hold = half_cycle / 2;
putreg32(sda_hold, I2C_SDA_HOLD_REG(priv->id));
/* scl_wait_high < sda_sample <= scl_high */
sda_sample = half_cycle / 2 - 1;
putreg32(sda_sample, I2C_SDA_SAMPLE_REG(priv->id));
setup = half_cycle;
putreg32(setup, I2C_SCL_RSTART_SETUP_REG(priv->id));
putreg32(setup, I2C_SCL_STOP_SETUP_REG(priv->id));
hold = half_cycle;
putreg32(hold - 1, I2C_SCL_START_HOLD_REG(priv->id));
putreg32(hold, I2C_SCL_STOP_HOLD_REG(priv->id));
/* By default, we set the timeout value to 10 bus cycles */
timeout = half_cycle * 20;
reg_value = I2C_TIME_OUT_EN;
reg_value |= VALUE_TO_FIELD(timeout, I2C_TIME_OUT_VALUE);
putreg32(reg_value, I2C_TO_REG(priv->id));
/* Set current bus clock frequency */
priv->clk_freq = bus_freq;
}
/****************************************************************************
* Name: i2c_init
*
* Description:
* Initialize I2C hardware.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_init(struct esp32s2_i2c_priv_s *priv)
{
const struct esp32s2_i2c_config_s *config = priv->config;
esp32s2_gpiowrite(config->scl_pin, 1);
esp32s2_configgpio(config->scl_pin, INPUT_PULLUP | OUTPUT_OPEN_DRAIN);
esp32s2_gpio_matrix_out(config->scl_pin, config->scl_outsig, 0, 0);
esp32s2_gpio_matrix_in(config->scl_pin, config->scl_insig, 0);
esp32s2_gpiowrite(config->sda_pin, 1);
esp32s2_configgpio(config->sda_pin, INPUT_PULLUP | OUTPUT_OPEN_DRAIN);
esp32s2_gpio_matrix_out(config->sda_pin, config->sda_outsig, 0, 0);
esp32s2_gpio_matrix_in(config->sda_pin, config->sda_insig, 0);
/* Enable I2C hardware */
modifyreg32(SYSTEM_PERIP_CLK_EN0_REG, 0, config->clk_bit);
modifyreg32(SYSTEM_PERIP_RST_EN0_REG, config->rst_bit, 0);
/* Disable I2C interrupts */
i2c_intr_disable(priv);
/* Initialize I2C Master */
putreg32(I2C_MS_MODE | I2C_CLK_EN | I2C_SCL_FORCE_OUT | I2C_SDA_FORCE_OUT,
I2C_CTR_REG(priv->id));
/* Set FIFO mode */
modifyreg32(I2C_FIFO_CONF_REG(priv->id), I2C_NONFIFO_EN, 0);
/* Ensure I2C data mode is set to MSB */
modifyreg32(I2C_CTR_REG(priv->id), I2C_TX_LSB_FIRST | I2C_RX_LSB_FIRST, 0);
i2c_reset_fifo(priv);
/* Configure the hardware filter function */
putreg32(I2C_SCL_FILTER_EN |
VALUE_TO_FIELD(I2C_FILTER_CYC_NUM_DEF, I2C_SCL_FILTER_THRES),
I2C_SCL_FILTER_CFG_REG(priv->id));
putreg32(I2C_SDA_FILTER_EN |
VALUE_TO_FIELD(I2C_FILTER_CYC_NUM_DEF, I2C_SDA_FILTER_THRES),
I2C_SDA_FILTER_CFG_REG(priv->id));
/* Set I2C source clock */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_REF_ALWAYS_ON);
/* Configure I2C bus frequency */
i2c_init_clock(priv, config->clk_freq);
}
/****************************************************************************
* Name: i2c_deinit
*
* Description:
* Disable I2C hardware.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_deinit(struct esp32s2_i2c_priv_s *priv)
{
const struct esp32s2_i2c_config_s *config = priv->config;
priv->clk_freq = 0;
modifyreg32(SYSTEM_PERIP_RST_EN0_REG, 0, config->rst_bit);
modifyreg32(SYSTEM_PERIP_CLK_EN0_REG, config->clk_bit, 0);
}
/****************************************************************************
* Name: i2c_reset_fsmc
*
* Description:
* Reset I2C hardware state machine and registers.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_reset_fsmc(struct esp32s2_i2c_priv_s *priv)
{
/* Reset FSM machine */
modifyreg32(I2C_CTR_REG(priv->id), 0, I2C_FSM_RST);
i2c_clear_bus(priv);
}
/****************************************************************************
* Name: i2c_sem_waitdone
*
* Description:
* Wait for a transfer to complete.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
* Returned Value:
* Zero (OK) is returned on success. A negated errno value is returned on
* failure.
*
****************************************************************************/
#ifndef CONFIG_I2C_POLLED
static int i2c_sem_waitdone(struct esp32s2_i2c_priv_s *priv)
{
/* Wait on ISR semaphore */
return nxsem_tickwait_uninterruptible(&priv->sem_isr,
ESP32S2_I2CTIMEOTICKS);
}
#endif
/****************************************************************************
* Name: i2c_polling_waitdone
*
* Description:
* Wait for a transfer to complete by polling status interrupt registers,
* which indicates the status of the I2C operations. This function is only
* used in polling driven mode.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
* Returned Values:
* Zero (OK) is returned on successfull transfer. -ETIMEDOUT is returned
* in case a transfer didn't finish within the timeout interval. And ERROR
* is returned in case of any I2C error during the transfer has happened.
*
****************************************************************************/
#ifdef CONFIG_I2C_POLLED
static int i2c_polling_waitdone(struct esp32s2_i2c_priv_s *priv)
{
int ret;
clock_t current;
clock_t timeout;
uint32_t status = 0;
/* Get the current absolute time and add an offset as timeout.
* Preferable to use monotonic, so in case the time changes,
* the time reference is kept, i.e., current time can't jump
* forward and backwards.
*/
current = clock_systime_ticks();
timeout = current + SEC2TICK(10);
/* Loop while a transfer is in progress
* and an error didn't occur within the timeout
*/
while ((current < timeout) && (priv->error == 0))
{
/* Check if any interrupt triggered, clear them
* process the operation.
*/
status = getreg32(I2C_INT_STATUS_REG(priv->id));
if (status != 0)
{
/* Check if the stop operation ended. Don't use
* I2CSTATE_FINISH, because it is set before the stop
* signal really ends. This works for interrupts because
* the i2c_state is checked in the next interrupt when
* stop signal has concluded. This is not the case of
* polling.
*/
if ((status & I2C_TRANS_COMPLETE_INT_ST) != 0)
{
putreg32(status, I2C_INT_CLR_REG(priv->id));
break;
}
putreg32(status, I2C_INT_CLR_REG(priv->id));
i2c_process(priv, status);
}
/* Update current time */
current = clock_systime_ticks();
}
/* Return a negated value in case of timeout, and in the other scenarios
* return a positive value.
* The transfer function will check the status of priv to check the other
* scenarios.
*/
if (current >= timeout)
{
ret = -ETIMEDOUT;
}
else if (priv->error != 0)
{
ret = ERROR;
}
else
{
ret = OK;
}
/* Disable all interrupts */
i2c_intr_disable(priv);
return ret;
}
#endif
/****************************************************************************
* Device Driver Operations
****************************************************************************/
/****************************************************************************
* Name: i2c_transfer
*
* Description:
* Generic I2C transfer function.
*
* Parameters:
* dev - Device-specific state data
* msgs - A pointer to a set of message descriptors
* count - The number of transfers to perform
*
* Returned Value:
* Zero (OK) is returned on success. A negated errno value is returned on
* failure.
*
****************************************************************************/
static int i2c_transfer(struct i2c_master_s *dev, struct i2c_msg_s *msgs,
int count)
{
int ret = OK;
struct esp32s2_i2c_priv_s *priv = (struct esp32s2_i2c_priv_s *)dev;
DEBUGASSERT(count > 0);
ret = nxmutex_lock(&priv->lock);
if (ret < 0)
{
return ret;
}
/* If previous state is different than idle,
* reset the FSMC to the idle state.
*/
if (priv->i2cstate != I2CSTATE_IDLE)
{
i2c_reset_fsmc(priv);
priv->i2cstate = I2CSTATE_IDLE;
}
/* Transfer the messages to the internal struct
* and loop count times to make all transfers.
*/
priv->msgv = msgs;
for (int i = 0; i < count; i++)
{
/* Clear TX and RX FIFOs. */
i2c_reset_fifo(priv);
priv->bytes = 0;
priv->msgid = i;
priv->ready_read = false;
priv->error = 0;
priv->i2cstate = I2CSTATE_PROC;
/* Set the SCLK frequency for the current msg. */
i2c_init_clock(priv, msgs[i].frequency);
if ((msgs[i].flags & I2C_M_NOSTART) != 0)
{
i2c_traceevent(priv, I2CEVENT_SENDBYTE, priv->bytes,
getreg32(I2C_SR_REG(priv->id)));
i2c_senddata(priv);
if (priv->bytes == msgs[i].length)
{
if ((msgs[i].flags & I2C_M_NOSTOP) == 0)
{
priv->i2cstate = I2CSTATE_STOP;
}
#ifndef CONFIG_I2C_POLLED
else
{
priv->i2cstate = I2CSTATE_FINISH;
}
#endif
}
}
else
{
/* Reset I2C trace logic */
i2c_tracereset(priv);
i2c_traceevent(priv, I2CEVENT_SENDADDR, msgs[i].addr,
getreg32(I2C_SR_REG(priv->id)));
i2c_sendstart(priv);
}
#ifndef CONFIG_I2C_POLLED
if (i2c_sem_waitdone(priv) < 0)
{
/* Timed out - transfer was not completed within the timeout */
i2cerr("Message %" PRIu8 " timed out.\n", priv->msgid);
ret = -ETIMEDOUT;
break;
}
else
{
if (priv->error != 0)
{
/* An error occurred */
i2cerr("Transfer error %" PRIu32 "\n", priv->error);
ret = -EIO;
break;
}
else
{
/* Successful transfer, update the I2C state to idle */
priv->i2cstate = I2CSTATE_IDLE;
ret = OK;
}
}
#else
ret = i2c_polling_waitdone(priv);
if (ret < 0)
{
if (ret == -ETIMEDOUT)
{
break;
}
else
{
ret = -EIO;
break;
}
}
else
{
/* Successful transfer, update the I2C state to idle */
priv->i2cstate = I2CSTATE_IDLE;
ret = OK;
}
#endif
i2cinfo("Message %" PRIu8 " transfer complete.\n", priv->msgid);
}
/* Dump the trace result */
i2c_tracedump(priv);
nxmutex_unlock(&priv->lock);
return ret;
}
/****************************************************************************
* Name: i2c_clear_bus
*
* Description:
* Clear I2C bus, when the slave is stuck in a deadlock and keeps pulling
* the bus low, master can control the SCL bus to generate 9 CLKs.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
static void i2c_clear_bus(struct esp32s2_i2c_priv_s *priv)
{
modifyreg32(I2C_SCL_SP_CONF_REG(priv->id),
I2C_SCL_RST_SLV_EN | I2C_SCL_RST_SLV_NUM_M,
VALUE_TO_FIELD(I2C_SCL_CYC_NUM_DEF, I2C_SCL_RST_SLV_NUM));
modifyreg32(I2C_SCL_SP_CONF_REG(priv->id), 0, I2C_SCL_RST_SLV_EN);
}
/****************************************************************************
* Name: i2c_reset
*
* Description:
* Perform an I2C bus reset in an attempt to break loose stuck I2C devices.
*
* Input Parameters:
* dev - Device-specific state data
*
* Returned Value:
* Zero (OK) on success; a negated errno value on failure.
*
****************************************************************************/
#ifdef CONFIG_I2C_RESET
static int i2c_reset(struct i2c_master_s *dev)
{
irqstate_t flags;
struct esp32s2_i2c_priv_s *priv = (struct esp32s2_i2c_priv_s *)dev;
DEBUGASSERT(dev != NULL);
DEBUGASSERT(priv->refs > 0);
flags = enter_critical_section();
i2c_reset_fsmc(priv);
/* Clear bus */
i2c_clear_bus(priv);
priv->i2cstate = I2CSTATE_IDLE;
priv->msgid = 0;
priv->bytes = 0;
priv->ready_read = false;
leave_critical_section(flags);
return OK;
}
#endif
/****************************************************************************
* Name: i2c_traceclear
*
* Description:
* Set I2C trace fields to default value.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
#ifdef CONFIG_I2C_TRACE
static void i2c_traceclear(struct esp32s2_i2c_priv_s *priv)
{
struct esp32s2_trace_s *trace = &priv->trace[priv->tndx];
trace->status = 0;
trace->count = 0;
trace->event = I2CEVENT_NONE;
trace->parm = 0;
trace->time = 0;
}
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Name: i2c_tracereset
*
* Description:
* Reset the trace info for a new data collection.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
#ifdef CONFIG_I2C_TRACE
static void i2c_tracereset(struct esp32s2_i2c_priv_s *priv)
{
priv->tndx = 0;
priv->start_time = clock_systime_ticks();
i2c_traceclear(priv);
}
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Name: i2c_tracenew
*
* Description:
* Create a new trace entry.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
* status - Current value of I2C status register.
*
****************************************************************************/
#ifdef CONFIG_I2C_TRACE
static void i2c_tracenew(struct esp32s2_i2c_priv_s *priv, uint32_t status)
{
struct esp32s2_trace_s *trace = &priv->trace[priv->tndx];
/* Check if the current entry is already initialized or if its status had
* already changed
*/
if (trace->count == 0 || status != trace->status)
{
/* Check whether the status changed */
if (trace->count != 0)
{
/* Bump up the trace index (unless we are out of trace entries) */
if (priv->tndx >= (CONFIG_I2C_NTRACE - 1))
{
i2cerr("ERROR: Trace table overflow\n");
return;
}
priv->tndx++;
trace = &priv->trace[priv->tndx];
}
/* Initialize the new trace entry */
i2c_traceclear(priv);
trace->status = status;
trace->count = 1;
trace->time = clock_systime_ticks();
}
else
{
/* Just increment the count of times that we have seen this status */
trace->count++;
}
}
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Name: i2c_traceevent
*
* Description:
* Record a new trace event.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
* event - Event to be recorded on the trace.
* parm - Parameter associated with the event.
* status - Current value of I2C status register.
*
****************************************************************************/
#ifdef CONFIG_I2C_TRACE
static void i2c_traceevent(struct esp32s2_i2c_priv_s *priv,
enum esp32s2_trace_e event,
uint32_t parm,
uint32_t status)
{
/* Check for new trace setup */
i2c_tracenew(priv, status);
if (event != I2CEVENT_NONE)
{
struct esp32s2_trace_s *trace = &priv->trace[priv->tndx];
/* Initialize the new trace entry */
trace->event = event;
trace->parm = parm;
/* Bump up the trace index (unless we are out of trace entries) */
if (priv->tndx >= (CONFIG_I2C_NTRACE - 1))
{
i2cerr("ERROR: Trace table overflow\n");
return;
}
priv->tndx++;
i2c_traceclear(priv);
}
}
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Name: i2c_tracedump
*
* Description:
* Dump the trace results.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
*
****************************************************************************/
#ifdef CONFIG_I2C_TRACE
static void i2c_tracedump(struct esp32s2_i2c_priv_s *priv)
{
syslog(LOG_DEBUG, "Elapsed time: %" PRIu32 "\n",
clock_systime_ticks() - priv->start_time);
for (int i = 0; i < priv->tndx; i++)
{
struct esp32s2_trace_s *trace = &priv->trace[i];
syslog(LOG_DEBUG,
"%2d. STATUS: %08" PRIx32 " COUNT: %3" PRIu32 " EVENT: %s(%2d)"
" PARM: %08" PRIx32 " TIME: %" PRIu32 "\n",
i + 1, trace->status, trace->count, g_trace_names[trace->event],
trace->event, trace->parm, trace->time - priv->start_time);
}
}
#endif /* CONFIG_I2C_TRACE */
/****************************************************************************
* Name: i2c_irq
*
* Description:
* This is the common I2C interrupt handler. It will be invoked when an
* interrupt is received on the device.
*
* Parameters:
* cpuint - CPU interrupt index
* context - Context data from the ISR
* arg - Opaque pointer to the internal driver state structure.
*
* Returned Value:
* Zero (OK) is returned on success. A negated errno value is returned on
* failure.
*
****************************************************************************/
#ifndef CONFIG_I2C_POLLED
static int i2c_irq(int cpuint, void *context, void *arg)
{
struct esp32s2_i2c_priv_s *priv = (struct esp32s2_i2c_priv_s *)arg;
/* Get the interrupt status and clear the interrupts that
* triggered.
*/
uint32_t irq_status = getreg32(I2C_INT_STATUS_REG(priv->id));
putreg32(irq_status, I2C_INT_CLR_REG(priv->id));
i2c_process(priv, irq_status);
return OK;
}
#endif
/****************************************************************************
* Name: i2c_process
*
* Description:
* This routine manages the transfer. It's called after some specific
* commands from the I2C controller are executed or in case of errors.
* It's responsible for writing/reading operations and transferring data
* from/to FIFO.
* It's called in the interrupt and polled driven mode.
*
* Parameters:
* priv - Pointer to the internal driver state structure.
* status - The current interrupt status register.
*
****************************************************************************/
static void i2c_process(struct esp32s2_i2c_priv_s *priv, uint32_t status)
{
struct i2c_msg_s *msg = &priv->msgv[priv->msgid];
/* Check for any errors */
if ((I2C_INT_ERR_MASK & status) != 0)
{
/* Save the errors, register the error event, disable interrupts
* and release the semaphore to conclude the transfer.
*/
priv->error = status & I2C_INT_ERR_MASK;
priv->i2cstate = I2CSTATE_ERROR;
i2c_traceevent(priv, I2CEVENT_ERROR, priv->error,
getreg32(I2C_SR_REG(priv->id)));
i2c_intr_disable(priv);
#ifndef CONFIG_I2C_POLLED
nxsem_post(&priv->sem_isr);
#endif
}
else /* If no error */
{
/* If a transfer has just initialized */
if (priv->i2cstate == I2CSTATE_PROC)
{
/* Check the flags to perform a read or write operation */
if ((msg->flags & I2C_M_READ) != 0)
{
/* RX FIFO has available data */
if (priv->ready_read)
{
i2c_traceevent(priv, I2CEVENT_RCVBYTE, priv->bytes,
getreg32(I2C_SR_REG(priv->id)));
i2c_recvdata(priv);
priv->ready_read = false;
}
/* Received all data. Finish the transaction
* and update the I2C state.
*/
if (priv->bytes == msg->length)
{
i2c_traceevent(priv, I2CEVENT_STOP, msg->length,
getreg32(I2C_SR_REG(priv->id)));
i2c_sendstop(priv);
#ifndef CONFIG_I2C_POLLED
priv->i2cstate = I2CSTATE_FINISH;
#endif
}
else /* Start a receive operation */
{
i2c_traceevent(priv, I2CEVENT_RCVMODEEN, 0,
getreg32(I2C_SR_REG(priv->id)));
i2c_startrecv(priv);
priv->ready_read = true;
}
}
else /* Write operation */
{
i2c_traceevent(priv, I2CEVENT_SENDBYTE, priv->bytes,
getreg32(I2C_SR_REG(priv->id)));
i2c_senddata(priv);
/* Finally sent the entire message. Update the I2C state to
* send a stop signal in the next interrupt.
*/
if (priv->bytes == msg->length)
{
if ((msg->flags & I2C_M_NOSTOP) == 0)
{
priv->i2cstate = I2CSTATE_STOP;
}
#ifndef CONFIG_I2C_POLLED
else
{
priv->i2cstate = I2CSTATE_FINISH;
}
#endif
}
}
}
else if (priv->i2cstate == I2CSTATE_STOP)
{
/* Transmitted all data. Finish the transaction sending a stop
* and update the I2C state.
*/
i2c_traceevent(priv, I2CEVENT_STOP, msg->length,
getreg32(I2C_SR_REG(priv->id)));
i2c_sendstop(priv);
#ifndef CONFIG_I2C_POLLED
priv->i2cstate = I2CSTATE_FINISH;
#endif
}
#ifndef CONFIG_I2C_POLLED
else if (priv->i2cstate == I2CSTATE_FINISH)
{
/* Disable all interrupts and release the semaphore */
i2c_intr_disable(priv);
nxsem_post(&priv->sem_isr);
}
#endif
}
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: esp32s2_i2cbus_initialize
*
* Description:
* Initialize the selected I2C port. And return a pointer to an unique
* instance of struct i2c_master_s. This function may be called to obtain
* multiple instances of the interface, each of which may be set up with a
* different frequency and slave address.
*
* Parameters:
* port - Port number of the I2C interface to be initialized.
*
* Returned Value:
* Pointer to valid I2C device structure is returned on success.
* A NULL pointer is returned on failure.
*
****************************************************************************/
struct i2c_master_s *esp32s2_i2cbus_initialize(int port)
{
struct esp32s2_i2c_priv_s *priv;
#ifndef CONFIG_I2C_POLLED
const struct esp32s2_i2c_config_s *config;
int ret;
#endif
switch (port)
{
#ifdef CONFIG_ESP32S2_I2C0
case ESP32S2_I2C0:
priv = &g_esp32s2_i2c0_priv;
break;
#endif
#ifdef CONFIG_ESP32S2_I2C1
case ESP32S2_I2C1:
priv = &g_esp32s2_i2c1_priv;
break;
#endif
default:
return NULL;
}
nxmutex_lock(&priv->lock);
if (priv->refs++ != 0)
{
nxmutex_unlock(&priv->lock);
return (struct i2c_master_s *)priv;
}
#ifndef CONFIG_I2C_POLLED
config = priv->config;
/* Set up to receive peripheral interrupts on the current CPU */
priv->cpuint = esp32s2_setup_irq(config->periph, 1, ESP32S2_CPUINT_LEVEL);
if (priv->cpuint < 0)
{
/* Failed to allocate a CPU interrupt of this type */
priv->refs--;
nxmutex_unlock(&priv->lock);
return NULL;
}
ret = irq_attach(config->irq, i2c_irq, priv);
if (ret != OK)
{
esp32s2_teardown_irq(config->periph, priv->cpuint);
priv->refs--;
nxmutex_unlock(&priv->lock);
return NULL;
}
up_enable_irq(config->irq);
#endif
i2c_init(priv);
nxmutex_unlock(&priv->lock);
return (struct i2c_master_s *)priv;
}
/****************************************************************************
* Name: esp32s2_i2cbus_uninitialize
*
* Description:
* De-initialize the selected I2C port and power down the device.
*
* Parameters:
* dev - Device structure as returned by
* esp32s2_i2cbus_initialize()
*
* Returned Value:
* OK is returned on success. ERROR is returned when internal reference
* count mismatches or dev points to invalid hardware device.
*
****************************************************************************/
int esp32s2_i2cbus_uninitialize(struct i2c_master_s *dev)
{
struct esp32s2_i2c_priv_s *priv = (struct esp32s2_i2c_priv_s *)dev;
DEBUGASSERT(dev != NULL);
if (priv->refs == 0)
{
return ERROR;
}
nxmutex_lock(&priv->lock);
if (--priv->refs)
{
nxmutex_unlock(&priv->lock);
return OK;
}
#ifndef CONFIG_I2C_POLLED
up_disable_irq(priv->config->irq);
esp32s2_teardown_irq(priv->config->periph, priv->cpuint);
#endif
i2c_deinit(priv);
nxmutex_unlock(&priv->lock);
return OK;
}
#endif /* CONFIG_ESP32S2_I2C */