2018-09-21 18:25:58 +02:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2018 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
|
*
|
|
|
|
|
* 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_cc26x0
|
|
|
|
|
* @ingroup drivers_periph_i2c
|
|
|
|
|
* @{
|
|
|
|
|
*
|
|
|
|
|
* @file
|
|
|
|
|
* @brief Low-level I2C driver implementation
|
2019-03-28 18:57:36 +01:00
|
|
|
|
* @note This CPU has weak pullups, external pullup resistors may be
|
|
|
|
|
* required.
|
2018-09-21 18:25:58 +02:00
|
|
|
|
*
|
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
|
*
|
|
|
|
|
* @}
|
|
|
|
|
*/
|
|
|
|
|
|
2019-08-22 11:52:51 +02:00
|
|
|
|
#include <assert.h>
|
2018-09-21 18:25:58 +02:00
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
#include "mutex.h"
|
|
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
|
#include "periph/i2c.h"
|
|
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG 0
|
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
|
|
#define PREG(x) DEBUG("%s=0x%08x\n", #x, (unsigned)x);
|
|
|
|
|
|
|
|
|
|
/**
|
2019-03-28 18:57:36 +01:00
|
|
|
|
* @brief Mutex lock for the only available I2C periph
|
|
|
|
|
* @note If multiple I2C devices are added locks must be an array for each one.
|
2018-09-21 18:25:58 +02:00
|
|
|
|
*/
|
2019-03-28 18:57:36 +01:00
|
|
|
|
static mutex_t _lock;
|
|
|
|
|
|
|
|
|
|
static int _check_errors(void)
|
|
|
|
|
{
|
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
|
|
/* The reference manual (SWCU117H) is ambiguous on how to wait:
|
|
|
|
|
*
|
|
|
|
|
* 1. 21.4 8. says "wait until BUSBUSY is cleared"
|
|
|
|
|
* 2. command flow diagrams (e.g., 21.3.5.1) indicate to wait while
|
|
|
|
|
* BUSY is set
|
|
|
|
|
*
|
|
|
|
|
* (3. 21.5.1.10 says BUSY is only valid after 4 SYSBUS clock cycles)
|
|
|
|
|
*
|
|
|
|
|
* Waiting first for cleared IDLE and then for cleared BUSY works fine.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* wait for transfer to be complete, this also could be a few nops... */
|
|
|
|
|
while (I2C->MSTAT & MSTAT_IDLE) {}
|
|
|
|
|
while (I2C->MSTAT & MSTAT_BUSY) {}
|
|
|
|
|
/* check if there was an error */
|
|
|
|
|
if (I2C->MSTAT & MSTAT_ERR) {
|
|
|
|
|
DEBUG("%s\n", __FUNCTION__);
|
|
|
|
|
PREG(I2C->MSTAT);
|
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
|
if (I2C->MSTAT & MSTAT_ADRACK_N) {
|
|
|
|
|
DEBUG("ADDRESS NACK\n");
|
|
|
|
|
return -ENXIO;
|
|
|
|
|
}
|
|
|
|
|
else if (I2C->MSTAT & MSTAT_DATACK_N) {
|
|
|
|
|
DEBUG("DATA NACK\n");
|
|
|
|
|
ret = -EIO;
|
|
|
|
|
}
|
|
|
|
|
else if (I2C->MSTAT & MSTAT_ARBLST) {
|
|
|
|
|
DEBUG("ARBITRATION LOSS\n");
|
|
|
|
|
ret = -EAGAIN;
|
|
|
|
|
}
|
|
|
|
|
/*
|
|
|
|
|
* If a non-NACK error occurs we must reinit or lock up.
|
|
|
|
|
* dev = 0 since it is the only one, if more are added it should be
|
|
|
|
|
* the dev num, this is done to avoid passing in arguments and
|
|
|
|
|
* increasing code size.
|
|
|
|
|
*/
|
|
|
|
|
i2c_init(0);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
|
|
|
|
void i2c_init(i2c_t devnum)
|
|
|
|
|
{
|
|
|
|
|
(void)devnum;
|
|
|
|
|
assert(devnum < I2C_NUMOF);
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
/* Make sure everything is shut off in case of reinit */
|
|
|
|
|
PRCM->PDCTL0SERIAL = 0;
|
|
|
|
|
I2C->MCR = 0;
|
|
|
|
|
PRCM->I2CCLKGR = 0;
|
|
|
|
|
|
2018-09-21 18:25:58 +02:00
|
|
|
|
/* enable SERIAL power domain */
|
|
|
|
|
PRCM->PDCTL0SERIAL = 1;
|
|
|
|
|
while (!(PRCM->PDSTAT0 & PDSTAT0_SERIAL_ON)) {}
|
|
|
|
|
|
|
|
|
|
/* enable i2c clock in run mode */
|
|
|
|
|
PRCM->I2CCLKGR = 1;
|
|
|
|
|
PRCM->CLKLOADCTL |= CLKLOADCTL_LOAD;
|
|
|
|
|
while (!(PRCM->CLKLOADCTL & CLKLOADCTL_LOADDONE)) {}
|
|
|
|
|
|
|
|
|
|
/* configure pins */
|
|
|
|
|
IOC->CFG[I2C_SDA_PIN] = (IOCFG_PORTID_I2C_MSSDA
|
|
|
|
|
| IOCFG_INPUT_ENABLE
|
|
|
|
|
| IOCFG_IOMODE_OPEN_DRAIN
|
|
|
|
|
| IOCFG_PULLCTL_UP);
|
|
|
|
|
IOC->CFG[I2C_SCL_PIN] = (IOCFG_PORTID_I2C_MSSCL
|
|
|
|
|
| IOCFG_INPUT_ENABLE
|
|
|
|
|
| IOCFG_IOMODE_OPEN_DRAIN
|
|
|
|
|
| IOCFG_PULLCTL_UP);
|
|
|
|
|
|
|
|
|
|
/* initialize I2C master */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
I2C->MCR = MCR_MFE;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
/* configure clock speed
|
|
|
|
|
* {PERDMACLK / [2 × (SCL_LP + SCL_HP) × SCL_CLK]} – 1
|
|
|
|
|
* with SCL_LP==6 && SCL_HP==4 use 0x17 for 100kHZ with 48MHZ CPU clock */
|
|
|
|
|
I2C->MTPR = MTPR_TPR_100KHZ;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int i2c_acquire(i2c_t dev)
|
|
|
|
|
{
|
|
|
|
|
assert(dev < I2C_NUMOF);
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mutex_lock(&_lock);
|
2018-09-21 18:25:58 +02:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-22 11:52:51 +02:00
|
|
|
|
void i2c_release(i2c_t dev)
|
2018-09-21 18:25:58 +02:00
|
|
|
|
{
|
|
|
|
|
assert(dev < I2C_NUMOF);
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mutex_unlock(&_lock);
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int i2c_read_bytes(i2c_t dev, uint16_t addr,
|
|
|
|
|
void *data, size_t len, uint8_t flags)
|
|
|
|
|
{
|
|
|
|
|
(void)dev;
|
|
|
|
|
int ret = 0;
|
2019-03-28 18:57:36 +01:00
|
|
|
|
char *bufpos = data;
|
|
|
|
|
|
|
|
|
|
DEBUG("%s %u\n", __FUNCTION__, len);
|
|
|
|
|
PREG(I2C->MSTAT);
|
|
|
|
|
|
2018-09-21 18:25:58 +02:00
|
|
|
|
assert(dev < I2C_NUMOF);
|
2019-03-28 18:57:36 +01:00
|
|
|
|
assert(data != NULL);
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
|
|
|
|
/* Check for unsupported operations */
|
|
|
|
|
if (flags & I2C_ADDR10) {
|
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check for wrong arguments given */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
if (len == 0) {
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (!(I2C->MSTAT & MSTAT_BUSBSY) && (flags & I2C_NOSTART)) {
|
2018-09-21 18:25:58 +02:00
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
/* Sequence may be omitted in a single master system */
|
|
|
|
|
while (I2C->MSTAT & MSTAT_BUSY) {}
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
I2C->MSA = ((uint32_t)addr << 1) | MSA_RS;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
|
|
|
|
while (len--) {
|
|
|
|
|
DEBUG("LOOP %u\n", len);
|
|
|
|
|
/* setup transfer */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
uint32_t mctrl = MCTRL_RUN;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
if (!(flags & I2C_NOSTART)) {
|
|
|
|
|
DEBUG("START\n");
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mctrl |= MCTRL_START;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
|
|
|
|
/* make note not to generate START from second byte onwards */
|
|
|
|
|
flags |= I2C_NOSTART;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* after last byte, generate STOP unless told not to */
|
|
|
|
|
if (!len && !(flags & I2C_NOSTOP)) {
|
|
|
|
|
DEBUG("STOP\n");
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mctrl |= MCTRL_STOP;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
DEBUG("ACK\n");
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mctrl |= MCTRL_ACK;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
while (I2C->MSTAT & MSTAT_BUSY) {}
|
2018-09-21 18:25:58 +02:00
|
|
|
|
/* initiate transfer */
|
|
|
|
|
I2C->MCTRL = mctrl;
|
|
|
|
|
|
|
|
|
|
/* check if there was an error */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
ret = _check_errors();
|
|
|
|
|
if (ret != 0) {
|
|
|
|
|
return ret;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
/* copy next byte from I2C data register */
|
|
|
|
|
DEBUG("IN=0x%02x\n", (unsigned)I2C->MDR);
|
|
|
|
|
*bufpos++ = I2C->MDR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data, size_t len,
|
|
|
|
|
uint8_t flags)
|
|
|
|
|
{
|
|
|
|
|
(void)dev;
|
2019-03-28 18:57:36 +01:00
|
|
|
|
int ret = 0;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
const unsigned char *bufpos = data;
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
DEBUG("%s %u\n", __FUNCTION__, len);
|
|
|
|
|
PREG(I2C->MSTAT);
|
|
|
|
|
|
|
|
|
|
assert(dev < I2C_NUMOF);
|
|
|
|
|
assert(data != NULL);
|
|
|
|
|
|
2018-09-21 18:25:58 +02:00
|
|
|
|
/* Check for unsupported operations */
|
|
|
|
|
if (flags & I2C_ADDR10) {
|
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check for wrong arguments given */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
if (len == 0) {
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (!(I2C->MSTAT & MSTAT_BUSBSY) && (flags & I2C_NOSTART)) {
|
2018-09-21 18:25:58 +02:00
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
/* Since write is 0 we just need shift the address in */
|
2018-09-21 18:25:58 +02:00
|
|
|
|
I2C->MSA = (uint32_t)addr << 1;
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
/* Sequence may be omitted in a single master system. */
|
|
|
|
|
while (I2C->MSTAT & MSTAT_BUSY) {}
|
|
|
|
|
|
2018-09-21 18:25:58 +02:00
|
|
|
|
while (len--) {
|
|
|
|
|
DEBUG("LOOP %u 0x%2x\n", len, (unsigned)*bufpos);
|
|
|
|
|
/* copy next byte into I2C data register */
|
|
|
|
|
I2C->MDR = *bufpos++;
|
|
|
|
|
|
|
|
|
|
/* setup transfer */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
uint32_t mctrl = MCTRL_RUN;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
if (!(flags & I2C_NOSTART)) {
|
|
|
|
|
DEBUG("START\n");
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mctrl |= MCTRL_START;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
|
|
|
|
|
/* make note not to generate START from second byte onwards */
|
|
|
|
|
flags |= I2C_NOSTART;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* after last byte, generate STOP unless told not to */
|
|
|
|
|
if (!len && !(flags & I2C_NOSTOP)) {
|
|
|
|
|
DEBUG("STOP\n");
|
2019-03-28 18:57:36 +01:00
|
|
|
|
mctrl |= MCTRL_STOP;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* initiate transfer */
|
|
|
|
|
I2C->MCTRL = mctrl;
|
|
|
|
|
|
|
|
|
|
/* check if there was an error */
|
2019-03-28 18:57:36 +01:00
|
|
|
|
ret = _check_errors();
|
|
|
|
|
if (ret != 0) {
|
|
|
|
|
return ret;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 18:57:36 +01:00
|
|
|
|
return ret;
|
2018-09-21 18:25:58 +02:00
|
|
|
|
}
|