mirror of
https://github.com/italicsjenga/rp-hal-boards.git
synced 2025-01-11 21:11:31 +11:00
i2c lockup fix (#94)
* Fix check for empty tx-fifo * Add read trait * Move from asserts to Err for read/write errors
This commit is contained in:
parent
ffa39f65f5
commit
301281cba8
|
@ -14,7 +14,7 @@ use crate::{
|
||||||
typelevel::Sealed,
|
typelevel::Sealed,
|
||||||
};
|
};
|
||||||
use embedded_time::rate::Hertz;
|
use embedded_time::rate::Hertz;
|
||||||
use hal::blocking::i2c::{Write, WriteRead};
|
use hal::blocking::i2c::{Read, Write, WriteRead};
|
||||||
use rp2040_pac::{I2C0, I2C1, RESETS};
|
use rp2040_pac::{I2C0, I2C1, RESETS};
|
||||||
|
|
||||||
/// I2C error
|
/// I2C error
|
||||||
|
@ -23,6 +23,14 @@ use rp2040_pac::{I2C0, I2C1, RESETS};
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
/// I2C abort with error
|
/// I2C abort with error
|
||||||
Abort(u32),
|
Abort(u32),
|
||||||
|
/// User passed in a read buffer that was 0 or >255 length
|
||||||
|
InvalidReadBufferLength(usize),
|
||||||
|
/// User passed in a write buffer that was 0 or >255 length
|
||||||
|
InvalidWriteBufferLength(usize),
|
||||||
|
/// Target i2c address is out of range
|
||||||
|
AddressOutOfRange(u8),
|
||||||
|
/// Target i2c address is reserved
|
||||||
|
AddressReserved(u8),
|
||||||
}
|
}
|
||||||
|
|
||||||
/// SCL pin
|
/// SCL pin
|
||||||
|
@ -73,6 +81,8 @@ pub struct I2C<I2C, Pins> {
|
||||||
pins: Pins,
|
pins: Pins,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const TX_FIFO_SIZE: u8 = 16;
|
||||||
|
|
||||||
fn i2c_reserved_addr(addr: u8) -> bool {
|
fn i2c_reserved_addr(addr: u8) -> bool {
|
||||||
(addr & 0x78) == 0 || (addr & 0x78) == 0x78
|
(addr & 0x78) == 0 || (addr & 0x78) == 0x78
|
||||||
}
|
}
|
||||||
|
@ -176,15 +186,38 @@ macro_rules! hal {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<PINS> I2C<$I2CX, PINS> {
|
||||||
|
/// Number of bytes currently in the TX FIFO
|
||||||
|
#[inline]
|
||||||
|
fn tx_fifo_used(&self) -> u8 {
|
||||||
|
self.i2c.ic_txflr.read().txflr().bits()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Remaining capacity in the TX FIFO
|
||||||
|
#[inline]
|
||||||
|
fn tx_fifo_free(&self) -> u8 {
|
||||||
|
TX_FIFO_SIZE - self.tx_fifo_used()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// TX FIFO is at capacity
|
||||||
|
#[inline]
|
||||||
|
fn tx_fifo_full(&self) -> bool {
|
||||||
|
self.tx_fifo_free() == 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<PINS> Write for I2C<$I2CX, PINS> {
|
impl<PINS> Write for I2C<$I2CX, PINS> {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Error> {
|
fn write(&mut self, addr: u8, bytes: &[u8]) -> Result<(), Error> {
|
||||||
// TODO support transfers of more than 255 bytes
|
// TODO support transfers of more than 255 bytes
|
||||||
assert!(bytes.len() < 256 && bytes.len() > 0);
|
if (bytes.len() > 255 || bytes.len() == 0) {
|
||||||
|
return Err(Error::InvalidWriteBufferLength(bytes.len()));
|
||||||
assert!(addr < 0x80);
|
} else if addr >= 0x80 {
|
||||||
assert!(!i2c_reserved_addr(addr));
|
return Err(Error::AddressOutOfRange(addr));
|
||||||
|
} else if i2c_reserved_addr(addr) {
|
||||||
|
return Err(Error::AddressReserved(addr));
|
||||||
|
}
|
||||||
|
|
||||||
self.i2c.ic_enable.write(|w| w.enable().disabled());
|
self.i2c.ic_enable.write(|w| w.enable().disabled());
|
||||||
self.i2c
|
self.i2c
|
||||||
|
@ -252,11 +285,15 @@ macro_rules! hal {
|
||||||
|
|
||||||
fn write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> {
|
fn write_read(&mut self, addr: u8, bytes: &[u8], buffer: &mut [u8]) -> Result<(), Error> {
|
||||||
// TODO support transfers of more than 255 bytes
|
// TODO support transfers of more than 255 bytes
|
||||||
assert!(bytes.len() < 256 && bytes.len() > 0);
|
if (bytes.len() > 255 || bytes.len() == 0) {
|
||||||
assert!(buffer.len() < 256 && buffer.len() > 0);
|
return Err(Error::InvalidWriteBufferLength(bytes.len()));
|
||||||
|
} else if (buffer.len() > 255 || buffer.len() == 0) {
|
||||||
assert!(addr < 0x80);
|
return Err(Error::InvalidReadBufferLength(buffer.len()));
|
||||||
assert!(!i2c_reserved_addr(addr));
|
} else if addr >= 0x80 {
|
||||||
|
return Err(Error::AddressOutOfRange(addr));
|
||||||
|
} else if i2c_reserved_addr(addr) {
|
||||||
|
return Err(Error::AddressReserved(addr));
|
||||||
|
}
|
||||||
|
|
||||||
self.i2c.ic_enable.write(|w| w.enable().disabled());
|
self.i2c.ic_enable.write(|w| w.enable().disabled());
|
||||||
self.i2c
|
self.i2c
|
||||||
|
@ -309,7 +346,8 @@ macro_rules! hal {
|
||||||
let first = i == 0;
|
let first = i == 0;
|
||||||
let last = i == bytes.len() - 1;
|
let last = i == bytes.len() - 1;
|
||||||
|
|
||||||
while 16 - self.i2c.ic_txflr.read().txflr().bits() > 0 {}
|
// wait until there is space in the FIFO to write the next byte
|
||||||
|
while self.tx_fifo_full() {}
|
||||||
|
|
||||||
self.i2c.ic_data_cmd.write(|w| {
|
self.i2c.ic_data_cmd.write(|w| {
|
||||||
if first {
|
if first {
|
||||||
|
@ -346,6 +384,73 @@ macro_rules! hal {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<PINS> Read for I2C<$I2CX, PINS> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn read(&mut self, addr: u8, buffer: &mut [u8]) -> Result<(), Error> {
|
||||||
|
// TODO support transfers of more than 255 bytes
|
||||||
|
if (buffer.len() > 255 || buffer.len() == 0) {
|
||||||
|
return Err(Error::InvalidReadBufferLength(buffer.len()));
|
||||||
|
} else if addr >= 0x80 {
|
||||||
|
return Err(Error::AddressOutOfRange(addr));
|
||||||
|
} else if i2c_reserved_addr(addr) {
|
||||||
|
return Err(Error::AddressReserved(addr));
|
||||||
|
}
|
||||||
|
|
||||||
|
self.i2c.ic_enable.write(|w| w.enable().disabled());
|
||||||
|
self.i2c
|
||||||
|
.ic_tar
|
||||||
|
.write(|w| unsafe { w.ic_tar().bits(addr as u16) });
|
||||||
|
self.i2c.ic_enable.write(|w| w.enable().enabled());
|
||||||
|
|
||||||
|
let mut abort = false;
|
||||||
|
let mut abort_reason = 0;
|
||||||
|
|
||||||
|
let lastindex = buffer.len() -1;
|
||||||
|
for (i, byte) in buffer.iter_mut().enumerate() {
|
||||||
|
let first = i == 0;
|
||||||
|
let last = i == lastindex;
|
||||||
|
|
||||||
|
// wait until there is space in the FIFO to write the next byte
|
||||||
|
while self.tx_fifo_full() {}
|
||||||
|
|
||||||
|
self.i2c.ic_data_cmd.write(|w| {
|
||||||
|
if first {
|
||||||
|
w.restart().enable();
|
||||||
|
} else {
|
||||||
|
w.restart().disable();
|
||||||
|
}
|
||||||
|
|
||||||
|
if last {
|
||||||
|
w.stop().enable();
|
||||||
|
} else {
|
||||||
|
w.stop().disable();
|
||||||
|
}
|
||||||
|
|
||||||
|
w.cmd().read()
|
||||||
|
});
|
||||||
|
|
||||||
|
while !abort && self.i2c.ic_rxflr.read().bits() == 0 {
|
||||||
|
abort_reason = self.i2c.ic_tx_abrt_source.read().bits();
|
||||||
|
abort = self.i2c.ic_clr_tx_abrt.read().bits() > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if abort {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
*byte = self.i2c.ic_data_cmd.read().dat().bits();
|
||||||
|
}
|
||||||
|
|
||||||
|
if abort {
|
||||||
|
Err(Error::Abort(abort_reason))
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
)+
|
)+
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue