mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
540 lines
14 KiB
C
540 lines
14 KiB
C
/*
|
|
* Copyright (C) 2014 CLENET Baptiste
|
|
*
|
|
* 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_sam0_common
|
|
* @ingroup drivers_periph_i2c
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level I2C driver implementation
|
|
*
|
|
* @author Baptiste Clenet <bapclenet@gmail.com>
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "cpu.h"
|
|
#include "board.h"
|
|
#include "mutex.h"
|
|
#include "periph_conf.h"
|
|
#include "periph/i2c.h"
|
|
|
|
#include "sched.h"
|
|
#include "thread.h"
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#include "debug.h"
|
|
|
|
#define SAMD21_I2C_TIMEOUT (65535)
|
|
|
|
#define BUSSTATE_UNKNOWN SERCOM_I2CM_STATUS_BUSSTATE(0)
|
|
#define BUSSTATE_IDLE SERCOM_I2CM_STATUS_BUSSTATE(1)
|
|
#define BUSSTATE_OWNER SERCOM_I2CM_STATUS_BUSSTATE(2)
|
|
#define BUSSTATE_BUSY SERCOM_I2CM_STATUS_BUSSTATE(3)
|
|
|
|
#if CPU_FAM_SAML21
|
|
#define SERCOM_I2CM_CTRLA_MODE_I2C_MASTER SERCOM_I2CM_CTRLA_MODE(5)
|
|
#endif
|
|
|
|
/* static function definitions */
|
|
static void _i2c_poweron(SercomI2cm *sercom);
|
|
static void _i2c_poweroff(SercomI2cm *sercom);
|
|
|
|
static int _start(SercomI2cm *dev, uint8_t address, uint8_t rw_flag);
|
|
static inline int _write(SercomI2cm *dev, const uint8_t *data, int length);
|
|
static inline int _read(SercomI2cm *dev, uint8_t *data, int length);
|
|
static inline void _stop(SercomI2cm *dev);
|
|
static inline int _wait_for_response(SercomI2cm *dev, uint32_t max_timeout_counter);
|
|
|
|
/**
|
|
* @brief Array holding one pre-initialized mutex for each I2C device
|
|
*/
|
|
static mutex_t locks[] = {
|
|
#if I2C_0_EN
|
|
[I2C_0] = MUTEX_INIT,
|
|
#endif
|
|
#if I2C_1_EN
|
|
[I2C_1] = MUTEX_INIT,
|
|
#endif
|
|
#if I2C_2_EN
|
|
[I2C_2] = MUTEX_INIT
|
|
#endif
|
|
#if I2C_3_EN
|
|
[I2C_3] = MUTEX_INIT
|
|
#endif
|
|
};
|
|
|
|
int i2c_init_master(i2c_t dev, i2c_speed_t speed)
|
|
{
|
|
SercomI2cm *I2CSercom = 0;
|
|
gpio_t pin_scl = 0;
|
|
gpio_t pin_sda = 0;
|
|
gpio_mux_t mux;
|
|
uint32_t clock_source_speed = 0;
|
|
uint8_t sercom_gclk_id = 0;
|
|
uint8_t sercom_gclk_id_slow = 0;
|
|
|
|
uint32_t timeout_counter = 0;
|
|
int32_t tmp_baud;
|
|
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
I2CSercom = &I2C_0_DEV;
|
|
pin_sda = I2C_0_SDA;
|
|
pin_scl = I2C_0_SCL;
|
|
mux = I2C_0_MUX;
|
|
clock_source_speed = CLOCK_CORECLOCK;
|
|
sercom_gclk_id = I2C_0_GCLK_ID;
|
|
sercom_gclk_id_slow = I2C_0_GCLK_ID_SLOW;
|
|
break;
|
|
#endif
|
|
default:
|
|
DEBUG("I2C FALSE VALUE\n");
|
|
return -1;
|
|
}
|
|
|
|
/* HACK: fixes cppcheck issue until #7588 is merged. */
|
|
if (I2CSercom == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
/* DISABLE I2C MASTER */
|
|
i2c_poweroff(dev);
|
|
|
|
/* Reset I2C */
|
|
I2CSercom->CTRLA.reg = SERCOM_I2CS_CTRLA_SWRST;
|
|
while (I2CSercom->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
|
|
/* Turn on power manager for sercom */
|
|
#if CPU_FAM_SAML21
|
|
/* OK for SERCOM0-4 */
|
|
MCLK->APBCMASK.reg |= (MCLK_APBCMASK_SERCOM0 << (sercom_gclk_id - SERCOM0_GCLK_ID_CORE));
|
|
#else
|
|
PM->APBCMASK.reg |= (PM_APBCMASK_SERCOM0 << (sercom_gclk_id - GCLK_CLKCTRL_ID_SERCOM0_CORE_Val));
|
|
#endif
|
|
|
|
/* I2C using CLK GEN 0 */
|
|
#if CPU_FAM_SAML21
|
|
GCLK->PCHCTRL[sercom_gclk_id].reg = (GCLK_PCHCTRL_CHEN |
|
|
GCLK_PCHCTRL_GEN_GCLK0 );
|
|
while (GCLK->SYNCBUSY.bit.GENCTRL) {}
|
|
|
|
GCLK->PCHCTRL[sercom_gclk_id_slow].reg = (GCLK_PCHCTRL_CHEN |
|
|
GCLK_PCHCTRL_GEN_GCLK0 );
|
|
while (GCLK->SYNCBUSY.bit.GENCTRL) {}
|
|
#else
|
|
GCLK->CLKCTRL.reg = (GCLK_CLKCTRL_CLKEN |
|
|
GCLK_CLKCTRL_GEN_GCLK0 |
|
|
GCLK_CLKCTRL_ID(sercom_gclk_id));
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
|
|
|
GCLK->CLKCTRL.reg = (GCLK_CLKCTRL_CLKEN |
|
|
GCLK_CLKCTRL_GEN_GCLK0 |
|
|
GCLK_CLKCTRL_ID(sercom_gclk_id_slow));
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
|
#endif
|
|
|
|
|
|
/* Check if module is enabled. */
|
|
if (I2CSercom->CTRLA.reg & SERCOM_I2CM_CTRLA_ENABLE) {
|
|
DEBUG("STATUS_ERR_DENIED\n");
|
|
return -3;
|
|
}
|
|
/* Check if reset is in progress. */
|
|
if (I2CSercom->CTRLA.reg & SERCOM_I2CM_CTRLA_SWRST) {
|
|
DEBUG("STATUS_BUSY\n");
|
|
return -3;
|
|
}
|
|
|
|
/************ SERCOM PAD0 - SDA and SERCOM PAD1 - SCL *************/
|
|
gpio_init_mux(pin_sda, mux);
|
|
gpio_init_mux(pin_scl, mux);
|
|
|
|
/* I2C CONFIGURATION */
|
|
while (I2CSercom->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
|
|
/* Set sercom module to operate in I2C master mode. */
|
|
I2CSercom->CTRLA.reg = SERCOM_I2CM_CTRLA_MODE_I2C_MASTER;
|
|
|
|
/* Enable Smart Mode (ACK is sent when DATA.DATA is read) */
|
|
I2CSercom->CTRLB.reg = SERCOM_I2CM_CTRLB_SMEN;
|
|
|
|
/* Find and set baudrate. Read speed configuration. Set transfer
|
|
* speed: SERCOM_I2CM_CTRLA_SPEED(0): Standard-mode (Sm) up to 100
|
|
* kHz and Fast-mode (Fm) up to 400 kHz */
|
|
switch (speed) {
|
|
case I2C_SPEED_NORMAL:
|
|
tmp_baud = (int32_t)(((clock_source_speed + (2 * (100000)) - 1) / (2 * (100000))) - 5);
|
|
if (tmp_baud < 255 && tmp_baud > 0) {
|
|
I2CSercom->CTRLA.reg |= SERCOM_I2CM_CTRLA_SPEED(0);
|
|
I2CSercom->BAUD.reg = SERCOM_I2CM_BAUD_BAUD(tmp_baud);
|
|
}
|
|
break;
|
|
case I2C_SPEED_FAST:
|
|
tmp_baud = (int32_t)(((clock_source_speed + (2 * (400000)) - 1) / (2 * (400000))) - 5);
|
|
if (tmp_baud < 255 && tmp_baud > 0) {
|
|
I2CSercom->CTRLA.reg |= SERCOM_I2CM_CTRLA_SPEED(0);
|
|
I2CSercom->BAUD.reg = SERCOM_I2CM_BAUD_BAUD(tmp_baud);
|
|
}
|
|
break;
|
|
case I2C_SPEED_HIGH:
|
|
tmp_baud = (int32_t)(((clock_source_speed + (2 * (3400000)) - 1) / (2 * (3400000))) - 1);
|
|
if (tmp_baud < 255 && tmp_baud > 0) {
|
|
I2CSercom->CTRLA.reg |= SERCOM_I2CM_CTRLA_SPEED(2);
|
|
I2CSercom->BAUD.reg = SERCOM_I2CM_BAUD_HSBAUD(tmp_baud);
|
|
}
|
|
break;
|
|
default:
|
|
DEBUG("BAD BAUDRATE\n");
|
|
return -2;
|
|
}
|
|
|
|
/* ENABLE I2C MASTER */
|
|
i2c_poweron(dev);
|
|
|
|
/* Start timeout if bus state is unknown. */
|
|
while ((I2CSercom->STATUS.reg & SERCOM_I2CM_STATUS_BUSSTATE_Msk) == BUSSTATE_UNKNOWN) {
|
|
if (timeout_counter++ >= SAMD21_I2C_TIMEOUT) {
|
|
/* Timeout, force bus state to idle. */
|
|
I2CSercom->STATUS.reg = BUSSTATE_IDLE;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int i2c_acquire(i2c_t dev)
|
|
{
|
|
if (dev >= I2C_NUMOF) {
|
|
return -1;
|
|
}
|
|
mutex_lock(&locks[dev]);
|
|
return 0;
|
|
}
|
|
|
|
int i2c_release(i2c_t dev)
|
|
{
|
|
if (dev >= I2C_NUMOF) {
|
|
return -1;
|
|
}
|
|
mutex_unlock(&locks[dev]);
|
|
return 0;
|
|
}
|
|
|
|
int i2c_read_byte(i2c_t dev, uint8_t address, void *data)
|
|
{
|
|
return i2c_read_bytes(dev, address, data, 1);
|
|
}
|
|
|
|
int i2c_read_bytes(i2c_t dev, uint8_t address, void *data, int length)
|
|
{
|
|
SercomI2cm *i2c;
|
|
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
i2c = &I2C_0_DEV;
|
|
break;
|
|
#endif
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
/* start transmission and send slave address */
|
|
if (_start(i2c, address, I2C_FLAG_READ) < 0) {
|
|
return 0;
|
|
}
|
|
/* read data to register */
|
|
if (_read(i2c, data, length) < 0) {
|
|
return 0;
|
|
}
|
|
_stop(i2c);
|
|
/* return number of bytes sent */
|
|
return length;
|
|
}
|
|
|
|
int i2c_read_reg(i2c_t dev, uint8_t address, uint8_t reg, void *data)
|
|
{
|
|
return i2c_read_regs(dev, address, reg, data, 1);
|
|
}
|
|
|
|
int i2c_read_regs(i2c_t dev, uint8_t address, uint8_t reg, void *data, int length)
|
|
{
|
|
SercomI2cm *i2c;
|
|
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
i2c = &I2C_0_DEV;
|
|
break;
|
|
#endif
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
/* start transmission and send slave address */
|
|
if (_start(i2c, address, I2C_FLAG_WRITE) < 0) {
|
|
return 0;
|
|
}
|
|
/* send register address/command and wait for complete transfer to
|
|
* be finished */
|
|
if (_write(i2c, ®, 1) < 0) {
|
|
return 0;
|
|
}
|
|
return i2c_read_bytes(dev, address, data, length);
|
|
}
|
|
|
|
int i2c_write_byte(i2c_t dev, uint8_t address, uint8_t data)
|
|
{
|
|
return i2c_write_bytes(dev, address, &data, 1);
|
|
}
|
|
|
|
int i2c_write_bytes(i2c_t dev, uint8_t address, const void *data, int length)
|
|
{
|
|
SercomI2cm *I2CSercom;
|
|
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
I2CSercom = &I2C_0_DEV;
|
|
break;
|
|
#endif
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (_start(I2CSercom, address, I2C_FLAG_WRITE) < 0) {
|
|
return 0;
|
|
}
|
|
if (_write(I2CSercom, data, length) < 0) {
|
|
return 0;
|
|
}
|
|
_stop(I2CSercom);
|
|
return length;
|
|
}
|
|
|
|
|
|
int i2c_write_reg(i2c_t dev, uint8_t address, uint8_t reg, uint8_t data)
|
|
{
|
|
return i2c_write_regs(dev, address, reg, &data, 1);
|
|
}
|
|
|
|
int i2c_write_regs(i2c_t dev, uint8_t address, uint8_t reg, const void *data, int length)
|
|
{
|
|
SercomI2cm *i2c;
|
|
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
i2c = &I2C_0_DEV;
|
|
break;
|
|
#endif
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
/* start transmission and send slave address */
|
|
if (_start(i2c, address, I2C_FLAG_WRITE) < 0) {
|
|
return 0;
|
|
}
|
|
/* send register address and wait for complete transfer to be finished */
|
|
if (_write(i2c, ®, 1) < 0) {
|
|
return 0;
|
|
}
|
|
/* write data to register */
|
|
if (_write(i2c, data, length) < 0) {
|
|
return 0;
|
|
}
|
|
/* finish transfer */
|
|
_stop(i2c);
|
|
return length;
|
|
}
|
|
|
|
static void _i2c_poweron(SercomI2cm *sercom)
|
|
{
|
|
if (sercom == NULL) {
|
|
return;
|
|
}
|
|
sercom->CTRLA.bit.ENABLE = 1;
|
|
while (sercom->SYNCBUSY.bit.ENABLE) {}
|
|
}
|
|
|
|
void i2c_poweron(i2c_t dev)
|
|
{
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
_i2c_poweron(&I2C_0_DEV);
|
|
break;
|
|
#endif
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void _i2c_poweroff(SercomI2cm *sercom)
|
|
{
|
|
if (sercom == NULL) {
|
|
return;
|
|
}
|
|
sercom->CTRLA.bit.ENABLE = 0;
|
|
while (sercom->SYNCBUSY.bit.ENABLE) {}
|
|
}
|
|
|
|
void i2c_poweroff(i2c_t dev)
|
|
{
|
|
switch (dev) {
|
|
#if I2C_0_EN
|
|
case I2C_0:
|
|
_i2c_poweroff(&I2C_0_DEV);
|
|
break;
|
|
#endif
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
static int _start(SercomI2cm *dev, uint8_t address, uint8_t rw_flag)
|
|
{
|
|
/* Wait for hardware module to sync */
|
|
DEBUG("Wait for device to be ready\n");
|
|
while (dev->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
|
|
/* Set action to ACK. */
|
|
dev->CTRLB.reg &= ~SERCOM_I2CM_CTRLB_ACKACT;
|
|
|
|
/* Send Start | Address | Write/Read */
|
|
DEBUG("Generate start condition by sending address\n");
|
|
dev->ADDR.reg = (address << 1) | rw_flag | (0 << SERCOM_I2CM_ADDR_HS_Pos);
|
|
|
|
/* Wait for response on bus. */
|
|
if (rw_flag == I2C_FLAG_READ) {
|
|
/* Some devices (e.g. SHT2x) can hold the bus while preparing the reply */
|
|
if (_wait_for_response(dev, 100 * SAMD21_I2C_TIMEOUT) < 0)
|
|
return -1;
|
|
}
|
|
else {
|
|
if (_wait_for_response(dev, SAMD21_I2C_TIMEOUT) < 0)
|
|
return -1;
|
|
}
|
|
|
|
/* Check for address response error unless previous error is detected. */
|
|
/* Check for error and ignore bus-error; workaround for BUSSTATE
|
|
* stuck in BUSY */
|
|
if (dev->INTFLAG.reg & SERCOM_I2CM_INTFLAG_SB) {
|
|
/* Clear write interrupt flag */
|
|
dev->INTFLAG.reg = SERCOM_I2CM_INTFLAG_SB;
|
|
/* Check arbitration. */
|
|
if (dev->STATUS.reg & SERCOM_I2CM_STATUS_ARBLOST) {
|
|
DEBUG("STATUS_ERR_PACKET_COLLISION\n");
|
|
return -2;
|
|
}
|
|
}
|
|
/* Check that slave responded with ack. */
|
|
else if (dev->STATUS.reg & SERCOM_I2CM_STATUS_RXNACK) {
|
|
/* Slave busy. Issue ack and stop command. */
|
|
dev->CTRLB.reg |= SERCOM_I2CM_CTRLB_CMD(3);
|
|
DEBUG("STATUS_ERR_BAD_ADDRESS\n");
|
|
return -3;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int _write(SercomI2cm *dev, const uint8_t *data, int length)
|
|
{
|
|
uint16_t tmp_data_length = length;
|
|
uint16_t buffer_counter = 0;
|
|
|
|
/* Write data buffer until the end. */
|
|
DEBUG("Looping through bytes\n");
|
|
while (tmp_data_length--) {
|
|
/* Check that bus ownership is not lost. */
|
|
if ((dev->STATUS.reg & SERCOM_I2CM_STATUS_BUSSTATE_Msk) != BUSSTATE_OWNER) {
|
|
DEBUG("STATUS_ERR_PACKET_COLLISION\n");
|
|
return -2;
|
|
}
|
|
|
|
/* Wait for hardware module to sync */
|
|
while (dev->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
|
|
DEBUG("Written byte #%i to data reg, now waiting for DR to be empty again\n", buffer_counter);
|
|
dev->DATA.reg = data[buffer_counter++];
|
|
|
|
/* Wait for response on bus. */
|
|
if (_wait_for_response(dev, SAMD21_I2C_TIMEOUT) < 0)
|
|
return -1;
|
|
|
|
/* Check for NACK from slave. */
|
|
if (dev->STATUS.reg & SERCOM_I2CM_STATUS_RXNACK) {
|
|
DEBUG("STATUS_ERR_OVERFLOW\n");
|
|
return -4;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static inline int _read(SercomI2cm *dev, uint8_t *data, int length)
|
|
{
|
|
uint8_t count = 0;
|
|
|
|
/* Set action to ack. */
|
|
dev->CTRLB.reg &= ~SERCOM_I2CM_CTRLB_ACKACT;
|
|
|
|
/* Read data buffer. */
|
|
while (count != length) {
|
|
/* Check that bus ownership is not lost. */
|
|
if ((dev->STATUS.reg & SERCOM_I2CM_STATUS_BUSSTATE_Msk) != BUSSTATE_OWNER) {
|
|
DEBUG("STATUS_ERR_PACKET_COLLISION\n");
|
|
return -2;
|
|
}
|
|
|
|
/* Wait for hardware module to sync */
|
|
while (dev->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
/* Save data to buffer. */
|
|
data[count] = dev->DATA.reg;
|
|
|
|
/* Wait for response on bus. */
|
|
if (_wait_for_response(dev, SAMD21_I2C_TIMEOUT) < 0)
|
|
return -1;
|
|
|
|
count++;
|
|
}
|
|
/* Send NACK before STOP */
|
|
dev->CTRLB.reg |= SERCOM_I2CM_CTRLB_ACKACT;
|
|
return 0;
|
|
}
|
|
|
|
static inline void _stop(SercomI2cm *dev)
|
|
{
|
|
/* Wait for hardware module to sync */
|
|
while (dev->SYNCBUSY.reg & SERCOM_I2CM_SYNCBUSY_MASK) {}
|
|
/* Stop command */
|
|
dev->CTRLB.reg |= SERCOM_I2CM_CTRLB_CMD(3);
|
|
/* Wait for bus to be idle again */
|
|
while ((dev->STATUS.reg & SERCOM_I2CM_STATUS_BUSSTATE_Msk) != BUSSTATE_IDLE) {}
|
|
DEBUG("Stop sent\n");
|
|
}
|
|
|
|
static inline int _wait_for_response(SercomI2cm *dev, uint32_t max_timeout_counter)
|
|
{
|
|
uint32_t timeout_counter = 0;
|
|
DEBUG("Wait for response.\n");
|
|
while (!(dev->INTFLAG.reg & SERCOM_I2CM_INTFLAG_MB)
|
|
&& !(dev->INTFLAG.reg & SERCOM_I2CM_INTFLAG_SB)) {
|
|
if (++timeout_counter >= max_timeout_counter) {
|
|
DEBUG("STATUS_ERR_TIMEOUT\n");
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|