1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-17 05:32:45 +01:00

cpu/qn908x: Implement blocking I2C support

This initial I2C support allows to use the I2C bus in controller mode to
interact with multiple peripherals in blocking mode. The CPU will
perform a busy wait when transferring data over I2C.
This commit is contained in:
iosabi 2020-12-13 21:04:26 +00:00
parent 5406297176
commit 70113b5fd3
6 changed files with 438 additions and 5 deletions

View File

@ -13,6 +13,7 @@ config CPU_FAM_QN908X
select HAS_PERIPH_CPUID
select HAS_PERIPH_GPIO
select HAS_PERIPH_GPIO_IRQ
select HAS_PERIPH_I2C_RECONFIGURE
select HAS_PERIPH_RTC
select HAS_PERIPH_WDT
select HAS_PERIPH_WDT_CB

View File

@ -10,7 +10,7 @@ USEMODULE += vendor_fsl_clock
# All peripherals use gpio_mux.h
USEMODULE += periph_gpio_mux
ifneq (,$(filter periph_uart,$(USEMODULE)))
ifneq (,$(filter periph_uart periph_i2c,$(USEMODULE)))
USEMODULE += periph_flexcomm
endif

View File

@ -4,6 +4,7 @@ CPU_FAM = qn908x
FEATURES_PROVIDED += cortexm_mpu
FEATURES_PROVIDED += periph_cpuid
FEATURES_PROVIDED += periph_gpio periph_gpio_irq
FEATURES_PROVIDED += periph_i2c_reconfigure
FEATURES_PROVIDED += periph_rtc
FEATURES_PROVIDED += periph_wdt periph_wdt_cb

View File

@ -31,6 +31,35 @@ The GPIO driver uses the @ref GPIO_PIN(port, pin) macro to declare pins.
No configuration is necessary.
@defgroup cpu_qn908x_i2c NXP QN908x I2C
@ingroup cpu_qn908x
@brief NXP QN908x I2C driver
There are several FLEXCOMM interfaces in this chip, but only two of these
support I2C (FLEXCOMM1 and FLEXCOMM2) which are mapped as I2C0 and I2C1,
respectively. A single FLEXCOMM interface can only be used for one of the I2C,
UART or SPI interfaces, so for example USART1 and I2C0 can't be used at the
same time since they are both the same FLEXCOMM1 interface.
### I2C configuration example (for periph_conf.h) ###
static const i2c_conf_t i2c_config[] = {
{
.dev = I2C0,
.pin_scl = GPIO_PIN(PORT_A, 6), // or A8, A12, A20
.pin_sda = GPIO_PIN(PORT_A, 7), // or A9, A13, A21
.speed = I2C_SPEED_FAST,
},
{
.dev = I2C1,
.pin_scl = GPIO_PIN(PORT_A, 2), // or A5, A23, A27
.pin_sda = GPIO_PIN(PORT_A, 3), // or A4, A22, A26
.speed = I2C_SPEED_FAST,
},
};
#define I2C_NUMOF ARRAY_SIZE(i2c_config)
@defgroup cpu_qn908x_timer NXP QN908x Standard counter/timers (CTIMER)
@ingroup cpu_qn908x
@brief NXP QN908x timer driver
@ -85,13 +114,13 @@ other pin can be set to GPIO_UNDEF.
static const uart_conf_t uart_config[] = {
{
.dev = USART0,
.rx_pin = GPIO_PIN(PORT_A, 17), /* or 5 */
.tx_pin = GPIO_PIN(PORT_A, 16), /* or 4 */
.rx_pin = GPIO_PIN(PORT_A, 17), // or A5
.tx_pin = GPIO_PIN(PORT_A, 16), // or A4
},
{
.dev = USART1,
.rx_pin = GPIO_PIN(PORT_A, 9), /* or 13 */
.tx_pin = GPIO_PIN(PORT_A, 8), /* or 12 */
.rx_pin = GPIO_PIN(PORT_A, 9), // or A13
.tx_pin = GPIO_PIN(PORT_A, 8), // or A12
},
};
#define UART_NUMOF ARRAY_SIZE(uart_config)

View File

@ -149,6 +149,51 @@ enum {
#define TIMER_MAX_VALUE (0xffffffff)
/** @} */
/**
* @brief I2C bus speed values in kbit/s.
*
* @note We support arbitrary speed values up to 400 kbit/s.
*
* @{
*/
#define HAVE_I2C_SPEED_T
typedef enum {
I2C_SPEED_LOW = 10000u, /**< low speed mode: ~10 kbit/s */
I2C_SPEED_NORMAL = 100000u, /**< normal mode: ~100 kbit/s */
I2C_SPEED_FAST = 400000u, /**< fast mode: ~400 kbit/s */
I2C_SPEED_FAST_PLUS = 400000u, /**< not supported, capped at 400 kbit/s */
I2C_SPEED_HIGH = 400000u, /**< not supported, capped at 400 kbit/s */
} i2c_speed_t;
/** @} */
/**
* @brief I2C configuration options
*/
typedef struct {
I2C_Type *dev; /**< hardware device */
gpio_t pin_scl; /**< SCL pin */
gpio_t pin_sda; /**< SDA pin */
uint32_t speed; /**< bus speed in bit/s */
} i2c_conf_t;
/**
* @name Use shared I2C functions
* @{
*/
#define PERIPH_I2C_NEED_READ_REG
#define PERIPH_I2C_NEED_READ_REGS
#define PERIPH_I2C_NEED_WRITE_REG
#define PERIPH_I2C_NEED_WRITE_REGS
/** @} */
/**
* @name Define macros for sda and scl pins.
* @{
*/
#define i2c_pin_sda(dev) i2c_config[dev].pin_sda
#define i2c_pin_scl(dev) i2c_config[dev].pin_scl
/** @} */
/**
* @brief UART module configuration options
*

357
cpu/qn908x/periph/i2c.c Normal file
View File

@ -0,0 +1,357 @@
/*
* Copyright (C) 2020 iosabi
*
* 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_qn908x
* @ingroup drivers_periph_i2c
*
* @{
*
* @file
* @brief Low-level I2C driver implementation
*
* @author iosabi <iosabi@protonmail.com>
*
* @}
*/
#include <assert.h>
#include <errno.h>
#include <stdint.h>
#include <stdlib.h>
#include "cpu.h"
#include "mutex.h"
#include "periph_conf.h"
#include "periph/i2c.h"
#include "vendor/drivers/fsl_clock.h"
#include "gpio_mux.h"
#include "flexcomm.h"
#define ENABLE_DEBUG 0
#include "debug.h"
static mutex_t locks[I2C_NUMOF];
/**
* @brief Limit value I2C CLKDIV register.
*/
#define I2C_CLKDIV_MAX 0xffffu
/**
* @brief Set the I2C controller mode clock speed.
*/
static void _i2c_controller_set_speed(I2C_Type *i2c_dev, uint32_t speed_hz)
{
/* The I2C clock source is based on the FLEXCOMM clock with the following
* formula:
* i2c freq := flexcomm freq / ((CLKDIV + 1) * (MSTTIME + 2) * 2)
* Where MSTTIME is a number between 0 and 7, and CLKDIV is between 0 and
* 0xffff.
* The MSTTIME register controls for how many cycles does the clock stay
* low and high, allowing to use different values for each one but we only
* use symmetric ones here, which is why there's a * 2 in the formula above.
*/
assert(speed_hz > 0);
uint32_t bus_freq = CLOCK_GetFreq(kCLOCK_BusClk);
uint32_t target = bus_freq / (2 * speed_hz);
uint32_t best_error = UINT_MAX;
uint32_t best_clkdiv = 0;
uint32_t best_msttime = 0;
for (uint32_t msttime_p2 = 9; msttime_p2 >= 2; msttime_p2--) {
uint32_t clkdiv_p1 = (target + msttime_p2 / 2) / msttime_p2;
if (clkdiv_p1 >= I2C_CLKDIV_MAX + 1) {
clkdiv_p1 = I2C_CLKDIV_MAX + 1;
}
uint32_t error =
abs((int32_t)target - (int32_t)(clkdiv_p1 * msttime_p2));
if (error < best_error) {
best_error = error;
best_clkdiv = clkdiv_p1 - 1;
best_msttime = msttime_p2 - 2;
}
}
i2c_dev->CLKDIV = I2C_CLKDIV_DIVVAL(best_clkdiv);
i2c_dev->MSTTIME = I2C_MSTTIME_MSTSCLLOW(best_msttime) |
I2C_MSTTIME_MSTSCLHIGH(best_msttime);
DEBUG("[i2c]: bus_fq=%" PRIu32" target_freq=%" PRIu32" msttime=%" PRIu32
" clkdiv=%" PRIu32 " error=%" PRIu32 "\n",
bus_freq, speed_hz, best_msttime, best_clkdiv, best_error);
}
static void _i2c_init_pins(i2c_t dev)
{
const i2c_conf_t *conf = &i2c_config[dev];
/* Configure SDA and SCL pins, the function value depends on the pin:
* FUNC4: A6, A7, A8, A9, A12, A13, A22, A23, A26, A27
* FUNC5: A2, A3, A4, A5, A20, A21
*/
static const uint32_t func5_mask =
(1 << 2) | (1 << 3) | (1 << 4) | (1 << 5) | (1 << 20) | (1 << 21);
/* TODO: Have a way to configure IOCON_MODE_PULLUP and IOCON_DRIVE_HIGH
* from the board. */
gpio_init_mux(conf->pin_sda,
((1u << GPIO_T_PIN(conf->pin_sda)) & func5_mask) ? 5 : 4);
gpio_init_mux(conf->pin_scl,
((1u << GPIO_T_PIN(conf->pin_scl)) & func5_mask) ? 5 : 4);
mutex_unlock(&locks[dev]);
}
void i2c_init(i2c_t dev)
{
assert(dev < I2C_NUMOF);
const i2c_conf_t *conf = &i2c_config[dev];
I2C_Type *const i2c_dev = conf->dev;
int flexcomm_num = flexcomm_init((FLEXCOMM_Type *)i2c_dev, FLEXCOMM_ID_I2C);
assert(flexcomm_num >= 0);
if (flexcomm_num == 1) {
/* Disable the FLECOMM1 MULT1/DIV1 divisor. FLEXCOMM2 doesn't have a
* fractional divisor. This divides the clock by (1 + N / 256) where
* N is set to 0 here, so the fractional divisor only divides by one. */
CLOCK_SetClkDiv(kCLOCK_DivFrg1, 0u);
}
/* Enable controller mode, no timeout, no monitor, no clock stretching. */
i2c_dev->CFG = I2C_CFG_MSTEN_MASK;
_i2c_controller_set_speed(i2c_dev, conf->speed);
locks[dev] = (mutex_t)MUTEX_INIT_LOCKED;
/* This also unlocks the mutex. */
_i2c_init_pins(dev);
}
#ifdef MODULE_PERIPH_I2C_RECONFIGURE
void i2c_init_pins(i2c_t dev)
{
assert(dev < I2C_NUMOF);
i2c_config[dev].dev->CFG |= I2C_CFG_MSTEN_MASK;
_i2c_init_pins(dev);
}
void i2c_deinit_pins(i2c_t dev)
{
assert(dev < I2C_NUMOF);
const i2c_conf_t *conf = &i2c_config[dev];
mutex_lock(&locks[dev]);
conf->dev->CFG &= ~I2C_CFG_MSTEN_MASK;
gpio_init_mux(conf->pin_sda, 0);
gpio_init_mux(conf->pin_scl, 0);
}
#endif /* MODULE_PERIPH_I2C_RECONFIGURE */
int i2c_acquire(i2c_t dev)
{
assert(dev < I2C_NUMOF);
mutex_lock(&locks[dev]);
return 0;
}
void i2c_release(i2c_t dev)
{
assert(dev < I2C_NUMOF);
mutex_unlock(&locks[dev]);
}
static uint32_t _i2c_wait_idle(I2C_Type *i2c_dev)
{
uint32_t status;
do {
status = i2c_dev->STAT;
} while ((status & I2C_STAT_MSTPENDING_MASK) == 0);
i2c_dev->STAT = I2C_STAT_MSTARBLOSS_MASK | I2C_STAT_MSTSTSTPERR_MASK;
return status;
}
static void _i2c_start(I2C_Type *i2c_dev, uint8_t addr_dir)
{
_i2c_wait_idle(i2c_dev);
i2c_dev->MSTDAT = addr_dir;
i2c_dev->MSTCTL = I2C_MSTCTL_MSTSTART_MASK; /* start */
}
static uint32_t _i2c_stop(I2C_Type *i2c_dev)
{
uint32_t status = _i2c_wait_idle(i2c_dev);
i2c_dev->MSTCTL = I2C_MSTCTL_MSTSTOP_MASK; /* stop */
return status;
}
/* Definitions for MSTSTATE bits in I2C Status register STAT */
/* Controller Idle State Code */
#define I2C_STAT_MSTSTATE_IDLE (0)
/* Controller Receive Ready State Code */
#define I2C_STAT_MSTSTATE_RXREADY (1)
/* Controller Transmit Ready State Code */
#define I2C_STAT_MSTSTATE_TXREADY (2)
/* Controller NACK by peripheral on address State Code */
#define I2C_STAT_MSTSTATE_NACKADR (3)
/* Controller NACK by peripheral on data State Code */
#define I2C_STAT_MSTSTATE_NACKDAT (4)
static int _i2c_transfer_blocking(i2c_t dev, uint32_t addr_dir, uint8_t *data,
size_t len, uint8_t flags)
{
assert((data != NULL) || (len == 0));
I2C_Type *i2c_dev = i2c_config[dev].dev;
uint32_t status = 0;
uint32_t controller_state;
if ((flags & I2C_NOSTART) == 0) {
if ((flags & I2C_ADDR10) != 0) {
_i2c_start(i2c_dev,
0xf0 | (addr_dir & 1) | ((addr_dir >> 8) & 0x6));
/* The second call to start sends a repeated start */
_i2c_start(i2c_dev, (addr_dir >> 1) & 0xff);
}
else {
_i2c_start(i2c_dev, addr_dir);
}
}
while (len) {
status = _i2c_wait_idle(i2c_dev);
if (status & (I2C_STAT_MSTARBLOSS_MASK | I2C_STAT_MSTSTSTPERR_MASK)) {
break;
}
controller_state = (status & I2C_STAT_MSTSTATE_MASK) >>
I2C_STAT_MSTSTATE_SHIFT;
switch (controller_state) {
case I2C_STAT_MSTSTATE_TXREADY:
/* I2C write case. */
if ((addr_dir & 1) != 0) {
/* This means that the direction requested in the transfer
* was to read, without a start condition and following
* a previous request to write, so the block is ready to
* write but we should be reading. */
_i2c_stop(i2c_dev);
return -EINVAL;
}
i2c_dev->MSTDAT = *(data++);
len--;
i2c_dev->MSTCTL = I2C_MSTCTL_MSTCONTINUE_MASK;
break;
case I2C_STAT_MSTSTATE_RXREADY:
/* I2C read case. */
if ((addr_dir & 1) != 1) {
/* Analog to the write case, we should be trying to writing
* in this case. */
i2c_dev->MSTDAT;
_i2c_stop(i2c_dev);
return -EINVAL;
}
*(data++) = i2c_dev->MSTDAT;
if (--len) {
i2c_dev->MSTCTL = I2C_MSTCTL_MSTCONTINUE_MASK;
}
else {
if ((flags & I2C_NOSTOP) == 0) {
/* initiate NAK and stop */
i2c_dev->MSTCTL = I2C_MSTCTL_MSTSTOP_MASK;
status = _i2c_wait_idle(i2c_dev);
}
}
break;
case I2C_STAT_MSTSTATE_NACKADR:
/* NACK on the address means no device. */
i2c_dev->MSTCTL = I2C_MSTCTL_MSTSTOP_MASK;
return -ENXIO;
case I2C_STAT_MSTSTATE_NACKDAT:
/* NACK on the data means that the device didn't ACK the last
* byte. */
i2c_dev->MSTCTL = I2C_MSTCTL_MSTSTOP_MASK;
return -EIO;
default:
return -EINVAL;
}
}
/* In the write case we didn't send a stop condition yet. */
if ((flags & I2C_NOSTOP) == 0 && (addr_dir & 1) == 0) {
status = _i2c_stop(i2c_dev);
/* The status will contain the last ACK/NACK */
controller_state = (status & I2C_STAT_MSTSTATE_MASK) >>
I2C_STAT_MSTSTATE_SHIFT;
if (controller_state == I2C_STAT_MSTSTATE_NACKDAT) {
return -EIO;
}
}
if (status & I2C_STAT_MSTARBLOSS_MASK) {
DEBUG("[i2c] Controller arbitration loss error flag, status 0x%.2"
PRIx32 "\n", status);
return -EAGAIN;
}
if (status & I2C_STAT_MSTSTSTPERR_MASK) {
/* Controller start/stop error flag. */
DEBUG("[i2c] Controller start/stop error flag\n");
return -EINVAL;
}
return 0;
}
#define I2C_ADDR_READ(addr) (((uint32_t)(addr) << 1u) | 1u)
#define I2C_ADDR_WRITE(addr) ((uint32_t)(addr) << 1u)
int i2c_read_bytes(i2c_t dev, uint16_t addr, void *data,
size_t len, uint8_t flags)
{
DEBUG("[i2c] R a=%.2x len=%2u f=%.2x, data:", addr, (unsigned)len, flags);
int ret =
_i2c_transfer_blocking(dev, I2C_ADDR_READ(addr), data, len, flags);
#if ENABLE_DEBUG
for (size_t i = 0; i < len; i++) {
DEBUG(" %.2x", ((uint8_t *)data)[i]);
}
DEBUG(", ret=%d\n", ret);
#endif
return ret;
}
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data, size_t len,
uint8_t flags)
{
/* The write transfer should technically only use a const uint8_t* buffer
* but we are re-using the same function to save on code here. It will not
* be written to in the I2C write case. */
#if ENABLE_DEBUG
DEBUG("[i2c] W a=%.2x len=%2u f=%.2x, data:", addr, (unsigned)len, flags);
for (size_t i = 0; i < len; i++) {
DEBUG(" %.2x", ((uint8_t *)data)[i]);
}
#endif
int ret = _i2c_transfer_blocking(
dev, I2C_ADDR_WRITE(addr), (uint8_t *)data, len, flags);
DEBUG(", ret=%d\n", ret);
return ret;
}
void isr_flexcomm_i2c(USART_Type *dev, uint32_t flexcomm_num)
{
// TODO: Set up async mode with interrupts.
(void)dev;
(void)flexcomm_num;
cortexm_isr_end();
}