mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
485 lines
11 KiB
C
485 lines
11 KiB
C
|
/*
|
||
|
* Copyright (C) 2020 Beuth Hochschule für Technik Berlin
|
||
|
*
|
||
|
* 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_lpc2387
|
||
|
* @ingroup drivers_periph_i2c
|
||
|
* @{
|
||
|
*
|
||
|
* @file
|
||
|
* @brief Low-level I2C driver implementation for lpc23xx
|
||
|
*
|
||
|
* @author Benjamin Valentin <benpicco@beuth-hochschule.de>
|
||
|
*
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <stdint.h>
|
||
|
#include <errno.h>
|
||
|
|
||
|
#include "cpu.h"
|
||
|
#include "board.h"
|
||
|
#include "byteorder.h"
|
||
|
#include "periph_conf.h"
|
||
|
#include "periph/i2c.h"
|
||
|
|
||
|
#include "sched.h"
|
||
|
#include "thread.h"
|
||
|
#include "mutex.h"
|
||
|
|
||
|
#define ENABLE_DEBUG (0)
|
||
|
#include "debug.h"
|
||
|
|
||
|
#if I2C_NUMOF > 0
|
||
|
static void I2C0_IRQHandler(void) __attribute__((interrupt("IRQ")));
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 1
|
||
|
static void I2C1_IRQHandler(void) __attribute__((interrupt("IRQ")));
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 2
|
||
|
static void I2C2_IRQHandler(void) __attribute__((interrupt("IRQ")));
|
||
|
#endif
|
||
|
|
||
|
/**
|
||
|
* We only ever need two buffers for read/write reg
|
||
|
*/
|
||
|
#define TRX_BUFS_MAX (2)
|
||
|
static struct i2c_ctx {
|
||
|
mutex_t lock;
|
||
|
mutex_t tx_done;
|
||
|
uint8_t *buf[TRX_BUFS_MAX];
|
||
|
uint8_t *buf_end[TRX_BUFS_MAX];
|
||
|
uint8_t *cur;
|
||
|
uint8_t *end;
|
||
|
int res;
|
||
|
uint8_t addr[TRX_BUFS_MAX];
|
||
|
uint8_t buf_num;
|
||
|
uint8_t buf_cur;
|
||
|
} ctx[I2C_NUMOF] = {
|
||
|
#if I2C_NUMOF > 0
|
||
|
{ .lock = MUTEX_INIT, .tx_done = MUTEX_INIT_LOCKED },
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 1
|
||
|
{ .lock = MUTEX_INIT, .tx_done = MUTEX_INIT_LOCKED },
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 2
|
||
|
{ .lock = MUTEX_INIT, .tx_done = MUTEX_INIT_LOCKED },
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
static void poweron(lpc23xx_i2c_t *i2c)
|
||
|
{
|
||
|
switch ((uint32_t)i2c) {
|
||
|
case I2C0_BASE_ADDR:
|
||
|
PCONP |= PCI2C0;
|
||
|
break;
|
||
|
case I2C1_BASE_ADDR:
|
||
|
PCONP |= PCI2C1;
|
||
|
break;
|
||
|
case I2C2_BASE_ADDR:
|
||
|
PCONP |= PCI2C2;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void poweroff(lpc23xx_i2c_t *i2c)
|
||
|
{
|
||
|
switch ((uint32_t)i2c) {
|
||
|
case I2C0_BASE_ADDR:
|
||
|
PCONP &= ~PCI2C0;
|
||
|
break;
|
||
|
case I2C1_BASE_ADDR:
|
||
|
PCONP &= ~PCI2C1;
|
||
|
break;
|
||
|
case I2C2_BASE_ADDR:
|
||
|
PCONP &= ~PCI2C2;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int i2c_acquire(i2c_t dev)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
|
||
|
mutex_lock(&ctx[dev].lock);
|
||
|
poweron(i2c_config[dev].dev);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void i2c_release(i2c_t dev)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
|
||
|
poweroff(i2c_config[dev].dev);
|
||
|
mutex_unlock(&ctx[dev].lock);
|
||
|
}
|
||
|
|
||
|
static void _set_baud(lpc23xx_i2c_t *i2c, uint32_t baud)
|
||
|
{
|
||
|
uint32_t pclksel, prescale;
|
||
|
lpc2387_pclk_scale(CLOCK_CORECLOCK, baud, &pclksel, &prescale);
|
||
|
|
||
|
switch ((uint32_t)i2c) {
|
||
|
case I2C0_BASE_ADDR:
|
||
|
PCLKSEL0 &= ~(BIT14 | BIT15);
|
||
|
PCLKSEL0 |= pclksel << 14;
|
||
|
I20SCLL = prescale / 2;
|
||
|
I20SCLH = prescale / 2;
|
||
|
break;
|
||
|
case I2C1_BASE_ADDR:
|
||
|
PCLKSEL1 &= ~(BIT6 | BIT7);
|
||
|
PCLKSEL1 |= pclksel << 6;
|
||
|
I21SCLL = prescale / 2;
|
||
|
I21SCLH = prescale / 2;
|
||
|
break;
|
||
|
case I2C2_BASE_ADDR:
|
||
|
PCLKSEL1 &= ~(BIT20 | BIT21);
|
||
|
PCLKSEL1 |= pclksel << 20;
|
||
|
I22SCLL = prescale / 2;
|
||
|
I22SCLH = prescale / 2;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static unsigned _get_irq(i2c_t dev)
|
||
|
{
|
||
|
switch ((uint32_t)i2c_config[dev].dev) {
|
||
|
case I2C0_BASE_ADDR:
|
||
|
return I2C0_INT;
|
||
|
case I2C1_BASE_ADDR:
|
||
|
return I2C1_INT;
|
||
|
case I2C2_BASE_ADDR:
|
||
|
return I2C2_INT;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void _install_irq(i2c_t dev)
|
||
|
{
|
||
|
switch (dev) {
|
||
|
#if I2C_NUMOF > 0
|
||
|
case 0:
|
||
|
install_irq(_get_irq(dev), I2C0_IRQHandler, i2c_config[dev].irq_prio);
|
||
|
break;
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 1
|
||
|
case 1:
|
||
|
install_irq(_get_irq(dev), I2C1_IRQHandler, i2c_config[dev].irq_prio);
|
||
|
break;
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 2
|
||
|
case 2:
|
||
|
install_irq(_get_irq(dev), I2C2_IRQHandler, i2c_config[dev].irq_prio);
|
||
|
break;
|
||
|
#endif
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void i2c_init(i2c_t dev)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
|
||
|
const i2c_conf_t *cfg = &i2c_config[dev];
|
||
|
lpc23xx_i2c_t *i2c = cfg->dev;
|
||
|
|
||
|
poweron(i2c);
|
||
|
|
||
|
/* configure SDA & SCL pins */
|
||
|
*(&PINSEL0 + cfg->pinsel_sda) |= cfg->pinsel_msk_sda;
|
||
|
*(&PINSEL0 + cfg->pinsel_scl) |= cfg->pinsel_msk_scl;
|
||
|
|
||
|
/* clear control register */
|
||
|
i2c->CONCLR = I2CONCLR_AAC | I2CONCLR_SIC | I2CONCLR_STAC
|
||
|
| I2CONCLR_I2ENC;
|
||
|
|
||
|
_set_baud(i2c, cfg->speed);
|
||
|
|
||
|
_install_irq(dev);
|
||
|
|
||
|
/* enable the interface */
|
||
|
i2c->CONSET = I2CONSET_I2EN;
|
||
|
|
||
|
poweroff(i2c);
|
||
|
}
|
||
|
|
||
|
static void _end_tx(i2c_t dev, unsigned res)
|
||
|
{
|
||
|
ctx[dev].res = res;
|
||
|
mutex_unlock(&ctx[dev].tx_done);
|
||
|
}
|
||
|
|
||
|
static void _next_buffer(i2c_t dev)
|
||
|
{
|
||
|
/* We only need two buffers max.
|
||
|
This can only be called for the first buffer
|
||
|
as there is no next buffer for the second buffer */
|
||
|
assert(ctx[dev].buf_cur == 0);
|
||
|
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
/* if mode (read/write) changed, send START again */
|
||
|
if (ctx[dev].addr[0] != ctx[dev].addr[1]) {
|
||
|
i2c->CONSET = I2CONSET_STA;
|
||
|
}
|
||
|
|
||
|
ctx[dev].cur = ctx[dev].buf[1];
|
||
|
ctx[dev].end = ctx[dev].buf_end[1];
|
||
|
ctx[dev].buf_cur = 1;
|
||
|
}
|
||
|
|
||
|
static void irq_handler(i2c_t dev)
|
||
|
{
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
unsigned stat = i2c->STAT;
|
||
|
|
||
|
DEBUG("[i2c] STAT: %x\n", stat);
|
||
|
|
||
|
switch (stat) {
|
||
|
case 0x00: /* Bus Error */
|
||
|
DEBUG("[i2c] Bus Error\n");
|
||
|
_end_tx(dev, -EIO);
|
||
|
break;
|
||
|
|
||
|
case 0x08: /* A Start Condition is issued. */
|
||
|
case 0x10: /* A repeated Start Condition is issued */
|
||
|
ctx[dev].cur = ctx[dev].buf[ctx[dev].buf_cur];
|
||
|
i2c->DAT = ctx[dev].addr[ctx[dev].buf_cur];
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
i2c->CONCLR = I2CONCLR_STAC;
|
||
|
break;
|
||
|
|
||
|
case 0x20: /* Address NACK (write) */
|
||
|
case 0x48: /* Address NACK (read) */
|
||
|
/* slave did not ACK address - send STOP */
|
||
|
i2c->CONSET = I2CONSET_STO | I2CONSET_AA;
|
||
|
_end_tx(dev, -ENXIO);
|
||
|
break;
|
||
|
|
||
|
case 0x18: /* Master Transmit, SLA_R has been sent */
|
||
|
i2c->DAT = *ctx[dev].cur++;
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
break;
|
||
|
|
||
|
case 0x28: /* Data byte has been transmitted */
|
||
|
|
||
|
if (ctx[dev].cur == ctx[dev].end) {
|
||
|
/* we transmitted all buffers */
|
||
|
if (ctx[dev].buf_cur == ctx[dev].buf_num) {
|
||
|
i2c->CONSET = I2CONSET_STO | I2CONSET_AA;
|
||
|
_end_tx(dev, 0);
|
||
|
break;
|
||
|
} else {
|
||
|
_next_buffer(dev);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
i2c->DAT = *ctx[dev].cur++;
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
break;
|
||
|
|
||
|
case 0x30: /* Data NACK */
|
||
|
i2c->CONSET = I2CONSET_STO | I2CONSET_AA;
|
||
|
_end_tx(dev, 0);
|
||
|
break;
|
||
|
|
||
|
case 0x38: /* Arbitration has been lost */
|
||
|
i2c->CONSET = I2CONSET_STA | I2CONSET_AA;
|
||
|
break;
|
||
|
|
||
|
case 0x40: /* Master Receive, SLA_R has been sent */
|
||
|
|
||
|
/* if we only want to read one byte, send NACK already */
|
||
|
if (ctx[dev].cur + 1 == ctx[dev].end) {
|
||
|
i2c->CONCLR = I2CONCLR_AAC;
|
||
|
} else {
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case 0x50: /* Data byte has been received */
|
||
|
|
||
|
*ctx[dev].cur++ = i2c->DAT;
|
||
|
|
||
|
if (ctx[dev].cur == ctx[dev].end) {
|
||
|
i2c->CONCLR = I2CONCLR_AAC;
|
||
|
} else {
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
|
||
|
case 0x58: /* Data received, NACK */
|
||
|
*ctx[dev].cur = i2c->DAT;
|
||
|
|
||
|
if (ctx[dev].buf_cur != ctx[dev].buf_num) {
|
||
|
i2c->CONSET = I2CONSET_AA;
|
||
|
_next_buffer(dev);
|
||
|
} else {
|
||
|
i2c->CONSET = I2CONSET_AA | I2CONSET_STO;
|
||
|
_end_tx(dev, 0);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
|
||
|
/* clear interrupt flag */
|
||
|
i2c->CONCLR = I2CONCLR_SIC;
|
||
|
}
|
||
|
|
||
|
static void _init_buffer(i2c_t dev, uint8_t idx, uint8_t addr,
|
||
|
uint8_t *data, size_t len)
|
||
|
{
|
||
|
ctx[dev].addr[idx] = addr;
|
||
|
ctx[dev].buf[idx] = data;
|
||
|
ctx[dev].buf_end[idx] = data + len;
|
||
|
|
||
|
ctx[dev].buf_num = idx;
|
||
|
|
||
|
ctx[dev].buf_cur = 0;
|
||
|
ctx[dev].cur = ctx[dev].buf[0];
|
||
|
ctx[dev].end = ctx[dev].buf_end[0];
|
||
|
}
|
||
|
|
||
|
int i2c_read_bytes(i2c_t dev, uint16_t addr,
|
||
|
void *data, size_t len, uint8_t flags)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
/* Check for wrong arguments given */
|
||
|
if (data == NULL || len == 0) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* TODO: 10 bit addresses */
|
||
|
if (flags) {
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
|
||
|
_init_buffer(dev, 0, 1 | (addr << 1), (void*)data, len);
|
||
|
|
||
|
/* set Start flag */
|
||
|
i2c->CONSET = I2CONSET_STA;
|
||
|
|
||
|
mutex_lock(&ctx[dev].tx_done);
|
||
|
return ctx[dev].res;
|
||
|
}
|
||
|
|
||
|
int i2c_write_bytes(i2c_t dev, uint16_t addr, const void *data, size_t len,
|
||
|
uint8_t flags)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
/* Check for wrong arguments given */
|
||
|
if (data == NULL || len == 0) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* TODO: 10 bit addresses */
|
||
|
if (flags) {
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
|
||
|
_init_buffer(dev, 0, addr << 1, (void*)data, len);
|
||
|
|
||
|
/* set Start flag */
|
||
|
i2c->CONSET = I2CONSET_STA;
|
||
|
|
||
|
mutex_lock(&ctx[dev].tx_done);
|
||
|
return ctx[dev].res;
|
||
|
}
|
||
|
|
||
|
int i2c_read_regs(i2c_t dev, uint16_t addr, uint16_t reg,
|
||
|
void *data, size_t len, uint8_t flags)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
/* Check for wrong arguments given */
|
||
|
if (data == NULL || len == 0) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* TODO: 10 bit addresses */
|
||
|
if (flags & ~I2C_REG16) {
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
|
||
|
/* Handle endianness of register if 16 bit */
|
||
|
if (flags & I2C_REG16) {
|
||
|
reg = htons(reg); /* Make sure register is in big-endian on I2C bus */
|
||
|
}
|
||
|
|
||
|
_init_buffer(dev, 0, addr << 1, (void*)®, (flags & I2C_REG16) ? 2 : 1);
|
||
|
_init_buffer(dev, 1, 1 | (addr << 1), (void*)data, len);
|
||
|
|
||
|
/* set Start flag */
|
||
|
i2c->CONSET = I2CONSET_STA;
|
||
|
|
||
|
mutex_lock(&ctx[dev].tx_done);
|
||
|
return ctx[dev].res;
|
||
|
}
|
||
|
|
||
|
int i2c_write_regs(i2c_t dev, uint16_t addr, uint16_t reg,
|
||
|
const void *data, size_t len, uint8_t flags)
|
||
|
{
|
||
|
assert(dev < I2C_NUMOF);
|
||
|
lpc23xx_i2c_t *i2c = i2c_config[dev].dev;
|
||
|
|
||
|
/* Check for wrong arguments given */
|
||
|
if (data == NULL || len == 0) {
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
/* TODO: 10 bit addresses */
|
||
|
if (flags & ~I2C_REG16) {
|
||
|
return -ENOTSUP;
|
||
|
}
|
||
|
|
||
|
/* Handle endianness of register if 16 bit */
|
||
|
if (flags & I2C_REG16) {
|
||
|
reg = htons(reg); /* Make sure register is in big-endian on I2C bus */
|
||
|
}
|
||
|
|
||
|
_init_buffer(dev, 0, addr << 1, (void*)®, (flags & I2C_REG16) ? 2 : 1);
|
||
|
_init_buffer(dev, 1, addr << 1, (void*)data, len);
|
||
|
|
||
|
/* set Start flag */
|
||
|
i2c->CONSET = I2CONSET_STA;
|
||
|
|
||
|
mutex_lock(&ctx[dev].tx_done);
|
||
|
return ctx[dev].res;
|
||
|
}
|
||
|
|
||
|
#if I2C_NUMOF > 0
|
||
|
static void I2C0_IRQHandler(void)
|
||
|
{
|
||
|
irq_handler(0);
|
||
|
VICVectAddr = 0; /* Acknowledge Interrupt */
|
||
|
}
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 1
|
||
|
static void I2C1_IRQHandler(void)
|
||
|
{
|
||
|
irq_handler(1);
|
||
|
VICVectAddr = 0; /* Acknowledge Interrupt */
|
||
|
}
|
||
|
#endif
|
||
|
#if I2C_NUMOF > 2
|
||
|
static void I2C2_IRQHandler(void)
|
||
|
{
|
||
|
irq_handler(2);
|
||
|
VICVectAddr = 0; /* Acknowledge Interrupt */
|
||
|
}
|
||
|
#endif
|