1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00

cpu/stm32_common: various improvements in i2c_1 driver

- dont send stop if bus is busy and before any action is done on bus
- and bus check that looks for bus error, arbitration and nack events
- check for ACK (TXIS) before writing bytes on bus and not between each write
This commit is contained in:
Alexandre Abadie 2018-06-22 17:14:41 +02:00 committed by dylad
parent 11356f602f
commit 5f1d5e0d83

View File

@ -61,6 +61,7 @@ static inline int _start(I2C_TypeDef *dev, uint16_t address, size_t length,
static inline int _read(I2C_TypeDef *dev, uint8_t *data, size_t length); static inline int _read(I2C_TypeDef *dev, uint8_t *data, size_t length);
static inline int _write(I2C_TypeDef *i2c, const uint8_t *data, size_t length); static inline int _write(I2C_TypeDef *i2c, const uint8_t *data, size_t length);
static inline int _stop(I2C_TypeDef *i2c); static inline int _stop(I2C_TypeDef *i2c);
static inline int _check_bus(I2C_TypeDef *i2c);
/** /**
* @brief Array holding one pre-initialized mutex for each I2C device * @brief Array holding one pre-initialized mutex for each I2C device
@ -169,10 +170,11 @@ int i2c_read_bytes(i2c_t dev, uint16_t address, void *data,
int ret = 0; int ret = 0;
if (!(flags & I2C_NOSTART)) { if (!(flags & I2C_NOSTART)) {
DEBUG("[i2c] read_bytes: start transmission\n"); DEBUG("[i2c] read_bytes: send start condition\n");
/* start reception and send slave address */ /* start reception and send slave address */
ret = _start(i2c, address, length, I2C_FLAG_READ, flags); ret = _start(i2c, address, length, I2C_FLAG_READ, flags);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
return ret; return ret;
} }
} }
@ -181,6 +183,7 @@ int i2c_read_bytes(i2c_t dev, uint16_t address, void *data,
/* read the data bytes */ /* read the data bytes */
ret = _read(i2c, data, length); ret = _read(i2c, data, length);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
DEBUG("[i2c] read_bytes: error while reading\n"); DEBUG("[i2c] read_bytes: error while reading\n");
return ret; return ret;
} }
@ -212,20 +215,17 @@ int i2c_read_regs(i2c_t dev, uint16_t address, uint16_t reg, void *data,
I2C_TypeDef *i2c = i2c_config[dev].dev; I2C_TypeDef *i2c = i2c_config[dev].dev;
/* Check to see if the bus is busy */ /* Check to see if the bus is busy */
while ((i2c->ISR & I2C_ISR_BUSY) && tick--) { while ((i2c->ISR & I2C_ISR_BUSY) && tick--) {}
if ((i2c->ISR & ERROR_FLAG) || !tick) { if ((i2c->ISR & ERROR_FLAG) || !tick) {
/* end transmission */
_stop(i2c);
return -ETIMEDOUT; return -ETIMEDOUT;
} }
}
DEBUG("[i2c] read_regs: send start sequence\n");
if (!(flags & I2C_NOSTART)) { if (!(flags & I2C_NOSTART)) {
DEBUG("[i2c] read_regs: send start sequence\n");
/* send start sequence and slave address */ /* send start sequence and slave address */
int ret = _start(i2c, address, 1, 0, flags); int ret = _start(i2c, address, 1, 0, flags);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
return ret; return ret;
} }
} }
@ -236,7 +236,7 @@ int i2c_read_regs(i2c_t dev, uint16_t address, uint16_t reg, void *data,
if ((i2c->ISR & ERROR_FLAG) || !tick) { if ((i2c->ISR & ERROR_FLAG) || !tick) {
/* end transmission */ /* end transmission */
_stop(i2c); _stop(i2c);
return -ETIMEDOUT; return -ENXIO;
} }
} }
@ -266,14 +266,16 @@ int i2c_write_bytes(i2c_t dev, uint16_t address, const void *data,
/* start transmission and send slave address */ /* start transmission and send slave address */
ret = _start(i2c, address, length, I2C_FLAG_WRITE, flags); ret = _start(i2c, address, length, I2C_FLAG_WRITE, flags);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
return ret; return ret;
} }
} }
DEBUG("[i2c] write_bytes: write the data\n"); DEBUG("[i2c] write_bytes: write the data (%d bytes)\n", length);
/* send out data bytes */ /* send out data bytes */
ret = _write(i2c, data, length); ret = _write(i2c, data, length);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
DEBUG("[i2c] write_bytes: nothing was written\n"); DEBUG("[i2c] write_bytes: nothing was written\n");
return ret; return ret;
} }
@ -305,17 +307,17 @@ int i2c_write_regs(i2c_t dev, uint16_t address, uint16_t reg, const void *data,
} }
/* Check to see if the bus is busy */ /* Check to see if the bus is busy */
while ((i2c->ISR & I2C_ISR_BUSY) && tick--) { while ((i2c->ISR & I2C_ISR_BUSY) && tick--) {}
if ((i2c->ISR & ERROR_FLAG) || !tick) { if ((i2c->ISR & ERROR_FLAG) || !tick) {
return -ETIMEDOUT; return -ETIMEDOUT;
} }
}
if (!(flags & I2C_NOSTART)) { if (!(flags & I2C_NOSTART)) {
/* start transmission and send slave address */ /* start transmission and send slave address */
/* increase length because our data is register+data */ /* increase length because our data is register+data */
ret = _start(i2c, address, length + 1, I2C_FLAG_WRITE, flags); ret = _start(i2c, address, length + 1, I2C_FLAG_WRITE, flags);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
return ret; return ret;
} }
} }
@ -327,6 +329,7 @@ int i2c_write_regs(i2c_t dev, uint16_t address, uint16_t reg, const void *data,
/* write out data bytes */ /* write out data bytes */
ret = _write(i2c, data, length); ret = _write(i2c, data, length);
if (ret < 0) { if (ret < 0) {
_stop(i2c);
return ret; return ret;
} }
@ -351,8 +354,6 @@ static inline int _start(I2C_TypeDef *i2c, uint16_t address,
assert(i2c != NULL); assert(i2c != NULL);
uint16_t tick = TICK_TIMEOUT;
i2c->CR2 = 0; i2c->CR2 = 0;
DEBUG("[i2c] start: set address mode\n"); DEBUG("[i2c] start: set address mode\n");
@ -385,10 +386,15 @@ static inline int _start(I2C_TypeDef *i2c, uint16_t address,
i2c->CR2 |= I2C_CR2_START; i2c->CR2 |= I2C_CR2_START;
/* Wait for the start followed by the address to be sent */ /* Wait for the start followed by the address to be sent */
while (!(i2c->CR2 & I2C_CR2_START) && tick--) { uint16_t tick = TICK_TIMEOUT;
while (!(i2c->CR2 & I2C_CR2_START) && tick--) {}
if (!tick) { if (!tick) {
return -ETIMEDOUT; return -ETIMEDOUT;
} }
int ret = _check_bus(i2c);
if (ret < 0) {
return ret;
} }
return 0; return 0;
@ -402,11 +408,10 @@ static inline int _read(I2C_TypeDef *i2c, uint8_t *data, size_t length)
/* wait for transfer to finish */ /* wait for transfer to finish */
DEBUG("[i2c] read: Waiting for DR to be full\n"); DEBUG("[i2c] read: Waiting for DR to be full\n");
uint16_t tick = TICK_TIMEOUT; uint16_t tick = TICK_TIMEOUT;
while (!(i2c->ISR & I2C_ISR_RXNE) && tick--) { while (!(i2c->ISR & I2C_ISR_RXNE) && tick--) {}
if (i2c->ISR & ERROR_FLAG || !tick) { if (i2c->ISR & ERROR_FLAG || !tick) {
return -ETIMEDOUT; return -ETIMEDOUT;
} }
}
DEBUG("[i2c] read: DR is now full\n"); DEBUG("[i2c] read: DR is now full\n");
@ -415,6 +420,11 @@ static inline int _read(I2C_TypeDef *i2c, uint8_t *data, size_t length)
DEBUG("[i2c] read: Read byte %i from DR\n", i); DEBUG("[i2c] read: Read byte %i from DR\n", i);
} }
int ret = _check_bus(i2c);
if (ret < 0) {
return ret;
}
return 0; return 0;
} }
@ -426,16 +436,28 @@ static inline int _write(I2C_TypeDef *i2c, const uint8_t *data, size_t length)
/* wait for ack */ /* wait for ack */
DEBUG("[i2c] write: Waiting for ACK\n"); DEBUG("[i2c] write: Waiting for ACK\n");
uint16_t tick = TICK_TIMEOUT; uint16_t tick = TICK_TIMEOUT;
while (!(i2c->ISR & I2C_ISR_TXIS) && tick--) { while (!(i2c->ISR & I2C_ISR_TXIS) && tick--) {}
if (i2c->ISR & ERROR_FLAG || !tick) { if (i2c->ISR & ERROR_FLAG || !tick) {
return -ETIMEDOUT; DEBUG("[i2c] write: TXIS timeout\n");
return -ENXIO;
}
/* write data to data register */
DEBUG("[i2c] write: Write byte %02X to DR\n", data[i]);
i2c->TXDR = data[i];
DEBUG("[i2c] write: Sending data\n");
int ret = _check_bus(i2c);
if (ret < 0) {
return ret;
} }
} }
/* write data to data register */ DEBUG("[i2c] write: Waiting for write to complete\n");
DEBUG("[i2c] write: Write byte %i to DR\n", i); uint16_t tick = TICK_TIMEOUT;
i2c->TXDR = data[i]; while (!(i2c->ISR & I2C_ISR_TC) && tick--) {}
DEBUG("[i2c] write: Sending data\n"); if (i2c->ISR & ERROR_FLAG || !tick) {
DEBUG("[i2c] write: write didn't complete\n");
return -ENXIO;
} }
return 0; return 0;
@ -449,19 +471,53 @@ static inline int _stop(I2C_TypeDef *i2c)
/* make sure transfer is complete */ /* make sure transfer is complete */
DEBUG("[i2c] stop: Wait for transfer to be complete\n"); DEBUG("[i2c] stop: Wait for transfer to be complete\n");
while (!(i2c->ISR & I2C_ISR_TC) && tick--) { while (!(i2c->ISR & I2C_ISR_TC) && tick--) {}
if (i2c->ISR & ERROR_FLAG || !tick) { if (i2c->ISR & ERROR_FLAG || !tick) {
return -ETIMEDOUT; return -EIO;
}
} }
/* send STOP condition */ /* send STOP condition */
DEBUG("[i2c] stop: Generate stop condition\n"); DEBUG("[i2c] stop: Generate stop condition\n");
i2c->CR2 |= I2C_CR2_STOP; i2c->CR2 |= I2C_CR2_STOP;
/* Wait for the stop to complete */
tick = TICK_TIMEOUT;
while (!(i2c->CR2 & I2C_CR2_STOP) && tick--) {}
if (!tick) {
return -ETIMEDOUT;
}
return 0; return 0;
} }
static inline int _check_bus(I2C_TypeDef *i2c)
{
assert(i2c != NULL);
int ret = 0;
/* wait a bit for any potential error to arrive */
uint16_t tick = TICK_TIMEOUT;
while (tick--) {}
if (i2c->ISR & I2C_ISR_NACKF) {
DEBUG("[i2c] check_bus: NACK received\n");
return -ENXIO;
}
if (i2c->ISR & I2C_ISR_ARLO) {
DEBUG("[i2c] check_bus: arbitration lost\n");
return -EAGAIN;
}
if (i2c->ISR & I2C_ISR_BERR) {
DEBUG("[i2c] check_bus: bus error\n");
return -EIO;
}
return ret;
}
static inline void irq_handler(i2c_t dev) static inline void irq_handler(i2c_t dev)
{ {
assert(dev < I2C_NUMOF); assert(dev < I2C_NUMOF);