1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/cpu/cc2538/periph/i2c.c
2018-07-25 12:01:38 +02:00

361 lines
9.0 KiB
C

/*
* Copyright (C) 2015 Loci Controls Inc.
* 2017 HAW Hamburg
*
* 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_cc2538
* @ingroup drivers_periph_i2c
* @{
*
* @file
* @brief Low-level I2C driver implementation
*
* @author Ian Martin <ian@locicontrols.com>
* @author Sebastian Meiling <s@mlng.net>
* @}
*/
#include <assert.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdint.h>
#include <errno.h>
#include "mutex.h"
#include "cpu.h"
#include "periph/gpio.h"
#include "periph/i2c.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
/* short cuts macros */
#define SCL_PIN(x) (i2c_config[x].scl_pin)
#define SDA_PIN(x) (i2c_config[x].sda_pin)
#define SPEED(x) (i2c_config[x].speed)
/* I2CM_CTRL Bits */
#define ACK (0x08)
#define STOP (0x04)
#define START (0x02)
#define RUN (0x01)
/* I2CM_STAT Bits */
#define BUSBSY (0x40)
#define IDLE (0x20)
#define ARBLST (0x10)
#define DATACK (0x08)
#define ADRACK (0x04)
#define ERROR (0x02)
#define BUSY (0x01)
/* I2CM_CR Bits */
#define SFE (0x20) /**< I2C slave function enable */
#define MFE (0x10) /**< I2C master function enable */
#define LPBK (0x01) /**< I2C loopback */
#define SCL_LP (6U) /**< SCL Low Period (fixed at 6). */
#define SCL_HP (4U) /**< SCL High Period (fixed at 4). */
#define CMD_WAIT (16U) /**< small delay to wait for I2C command */
#define RST_WAIT (32U) /**< reset delay */
#define INVALID_SPEED_MASK (0x0f)
static mutex_t lock = MUTEX_INIT;
void isr_i2c(void)
{
/* Clear the interrupt flag */
I2CM_ICR = 0x1;
I2CM_MIS = 0x1;
cortexm_isr_end();
}
static inline void _i2c_reset(void)
{
DEBUG("%s\n", __FUNCTION__);
/* Reset I2C peripheral */
SYS_CTRL_SRI2C |= 1;
/* wait shortly for reset */
for (unsigned delay = 0; delay < RST_WAIT; ++delay) {}
/* clear periph reset trigger */
SYS_CTRL_SRI2C &= ~1;
}
static inline void _i2c_clock_enable(bool enable)
{
DEBUG("%s\n", __FUNCTION__);
if (enable) {
SYS_CTRL_RCGCI2C |= 1;
SYS_CTRL_SCGCI2C |= 1;
SYS_CTRL_DCGCI2C |= 1;
}
else {
SYS_CTRL_RCGCI2C &= ~1;
SYS_CTRL_SCGCI2C &= ~1;
SYS_CTRL_DCGCI2C &= ~1;
}
}
static inline void _i2c_master_enable(bool enable)
{
DEBUG("%s: %s\n", __FUNCTION__, (enable ? "yes" : "no"));
if (enable) {
/* enable I2C master function */
I2CM_CR |= MFE;
/* Enable I2C master interrupts */
NVIC_SetPriority(I2C_IRQn, I2C_IRQ_PRIO);
NVIC_EnableIRQ(I2C_IRQn);
/* Enable I2C master interrupts */
I2CM_IMR = 1;
}
else {
/* Disable I2C master interrupts */
I2CM_IMR = 0;
NVIC_DisableIRQ(I2C_IRQn);
/* disable master function */
I2CM_CR &= ~(MFE);
}
}
static inline void _i2c_master_frequency(i2c_speed_t speed)
{
assert ((speed == I2C_SPEED_NORMAL) || (speed == I2C_SPEED_FAST));
DEBUG("%s (%" PRIu32 ")\n", __FUNCTION__, (uint32_t)speed);
/* if selected speed is not applicable fall back to normal */
if (speed & INVALID_SPEED_MASK) {
DEBUG("! invalid speed setting, fall back to normal !\n");
speed = I2C_SPEED_NORMAL;
}
/* Set the SCL clock speed */
uint32_t denom = 2 * (SCL_LP + SCL_HP) * (uint32_t)speed;
uint32_t ps = (sys_clock_freq() + denom - 1) / denom;
I2CM_TPR = (ps - 1);
}
static uint_fast8_t _i2c_master_stat_get(void)
{
return I2CM_STAT;
}
static bool _i2c_master_busy(void)
{
return ((I2CM_STAT & BUSY) ? true : false);
}
static bool _i2c_master_busbusy(void)
{
return ((I2CM_STAT & BUSBSY) ? true : false);
}
static void _i2c_master_slave_addr(uint16_t addr, bool receive)
{
DEBUG("%s (%" PRIx16 ", %d)\n", __FUNCTION__, addr, (int)receive);
assert(!(addr & 0x80));
I2CM_SA = (addr << 1) | receive;
}
static void _i2c_master_data_put(uint8_t data)
{
DEBUG("%s (0x%x)\n", __FUNCTION__, data);
I2CM_DR = data;
}
static uint_fast8_t _i2c_master_data_get(void)
{
uint_fast8_t data = I2CM_DR;
DEBUG("%s (0x%x)\n", __FUNCTION__, data);
return data;
}
static inline void _i2c_master_ctrl(uint_fast8_t cmd)
{
DEBUG("%s (%" PRIx32 ")\n", __FUNCTION__, (uint32_t)cmd);
I2CM_CTRL = cmd;
}
static inline int _i2c_master_status(void)
{
DEBUG("%s\n", __FUNCTION__);
uint_fast8_t stat = _i2c_master_stat_get();
DEBUG(" - I2C master status (0x%x): ", stat);
if (stat & BUSY) {
DEBUG("busy!\n");
return 0;
}
else if (stat & (ERROR | ARBLST)) {
_i2c_master_ctrl(STOP | RUN);
unsigned cw = CMD_WAIT;
while (_i2c_master_busy() || (cw--)) {}
if (stat & ADRACK) {
DEBUG("addr ack lost!\n");
return -ENXIO;
}
if (stat & DATACK) {
DEBUG("data ack lost!\n");
return -EIO;
}
if (stat & ARBLST) {
DEBUG("lost bus arbitration!\n");
return -EAGAIN;
}
DEBUG("unknown!\n");
return -EAGAIN;
}
DEBUG("okay.\n");
return 0;
}
void i2c_init(i2c_t dev)
{
DEBUG("%s (%i)\n", __FUNCTION__, (int)dev);
assert(dev < I2C_NUMOF);
/* enable i2c clock */
_i2c_clock_enable(true);
/* reset i2c periph */
_i2c_reset();
/* init pins */
gpio_init_mux(SCL_PIN(dev), OVERRIDE_PULLUP, I2C_SCL_OUT, I2C_SCL_IN);
gpio_init_mux(SDA_PIN(dev), OVERRIDE_PULLUP, I2C_SDA_OUT, I2C_SDA_IN);
/* enable master mode */
_i2c_master_enable(true);
/* set bus frequency */
_i2c_master_frequency(SPEED(dev));
DEBUG(" - I2C master status (0x%x).\n", _i2c_master_stat_get());
}
int i2c_acquire(i2c_t dev)
{
DEBUG("%s\n", __FUNCTION__);
if (dev < I2C_NUMOF) {
mutex_lock(&lock);
return 0;
}
return -1;
}
int i2c_release(i2c_t dev)
{
DEBUG("%s\n", __FUNCTION__);
if (dev < I2C_NUMOF) {
mutex_unlock(&lock);
return 0;
}
return -1;
}
int i2c_read_bytes(i2c_t dev, uint16_t addr,
void *data, size_t len, uint8_t flags)
{
DEBUG("%s\n", __FUNCTION__);
DEBUG(" - I2C master status (0x%x).\n", _i2c_master_stat_get());
if ((dev >= I2C_NUMOF) || (data == NULL) || (len == 0)) {
return -EINVAL;
}
if (flags & (I2C_REG16 | I2C_ADDR10)) {
return -EOPNOTSUPP;
}
if (_i2c_master_busy()) {
DEBUG("i2c_read_bytes: device busy!\n");
return -EAGAIN;
}
if (!(_i2c_master_busbusy()) && (flags & I2C_NOSTART)) {
DEBUG("i2c_read_bytes: bus not busy!\n");
return -EAGAIN;
}
/* set slave address for receive */
_i2c_master_slave_addr(addr, true);
int rc = 0;
uint8_t *buf = data;
for (size_t n = 0; n < len; n++) {
uint_fast8_t cmd = RUN;
if ((n == 0) && !(flags & I2C_NOSTART)) {
cmd |= START;
}
if (((len - n) == 1) && !(flags & I2C_NOSTOP)) {
cmd |= STOP;
}
else {
cmd |= ACK;
}
/* run command */
_i2c_master_ctrl(cmd);
/* wait until master is done transferring */
DEBUG ("%s: wait for master...\n", __FUNCTION__);
unsigned cw = CMD_WAIT;
while (_i2c_master_busy() || (cw--)) {}
/* check master status */
if ((rc = _i2c_master_status()) != 0) {
break;
}
/* read data into buffer */
buf[n] = _i2c_master_data_get();
}
return rc;
}
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data,
size_t len, uint8_t flags)
{
DEBUG("%s\n", __FUNCTION__);
DEBUG(" - I2C master status (0x%x).\n", _i2c_master_stat_get());
if ((dev >= I2C_NUMOF) || (data == NULL) || (len == 0)) {
return -EINVAL;
}
if (flags & (I2C_REG16 | I2C_ADDR10)) {
return -EOPNOTSUPP;
}
if (_i2c_master_busy()) {
DEBUG("i2c_write_bytes: device busy!\n");
return -EAGAIN;
}
if (!(_i2c_master_busbusy()) && (flags & I2C_NOSTART)) {
DEBUG("i2c_read_bytes: bus not busy!\n");
return -EAGAIN;
}
/* set slave address for write */
_i2c_master_slave_addr(addr, false);
int rc = 0;
const uint8_t *buf = data;
for (size_t n = 0; n < len; n++) {
uint_fast8_t cmd = RUN;
if ((n == 0) && !(flags & I2C_NOSTART)) {
cmd |= START;
}
if (((len - n) == 1) && !(flags & I2C_NOSTOP)) {
cmd |= STOP;
}
/* write byte to mem */
_i2c_master_data_put(buf[n]);
/* run command */
_i2c_master_ctrl(cmd);
/* wait until master is done transferring */
DEBUG ("%s: wait for master...\n", __FUNCTION__);
unsigned cw = CMD_WAIT;
while (_i2c_master_busy() || (cw--)) {}
/* check master status */
if ((rc = _i2c_master_status()) != 0) {
break;
}
}
return rc;
}