atsamd_hal/sercom/i2c/flags.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
//! Flag definitions
#![allow(clippy::identity_op)]
#![allow(unused_braces)]
use bitflags::bitflags;
use modular_bitfield::specifiers::{B1, B5};
use modular_bitfield::*;
bitflags! {
/// Interrupt bitflags for I2C transactions
///
/// The available interrupt flags are `MB`, `SB`, and `ERROR`. The binary format of the underlying bits exactly matches the
/// INTFLAG bits.
#[derive(Clone, Copy)]
pub struct Flags: u8 {
/// Master on bus interrupt
const MB = 0x01;
/// Slave on bus interrupt
const SB = 0x02;
/// Error interrupt
const ERROR = 0x80;
}
}
/// Type representing the current bus state
#[derive(BitfieldSpecifier, PartialEq)]
pub enum BusState {
Unknown = 0x00,
Idle = 0x01,
Owner = 0x02,
Busy = 0x03,
}
impl Default for Status {
fn default() -> Self {
Self::new()
}
}
/// Status flags for I2C transactions
///
/// The available status flags are `BUSERR`, `ARBLOST`, `RXNACK`,
/// `BUSSTATE`, `LOWTOUT` and `CLKHOLD`, `MEXTTOUT`, `SEXTTOUT` and
/// `LENERR`. The binary format of the underlying bits exactly matches
/// the STATUS bits.
#[bitfield]
#[repr(u16)]
pub struct Status {
pub buserr: bool,
pub arblost: bool,
#[skip(setters)]
pub rxnack: bool,
#[skip]
_reserved: B1,
pub busstate: BusState,
pub lowtout: bool,
#[skip(setters)]
pub clkhold: bool,
pub mexttout: bool,
pub sexttout: bool,
pub lenerr: bool,
#[skip]
_reserved: B5,
}
impl Status {
/// Check whether [`Self`] originates from an error.
///
/// # Errors
///
/// Returns an error if `STATUS` contains:
///
/// * `BUSERR` - Bus Error
/// * `ARBLOST` - Arbitration lost
/// * `LENERR` - Length error
/// * `RXNACK` - Receive not acknowledged
pub fn check_bus_error(self) -> Result<(), Error> {
if self.buserr() {
Err(Error::BusError)
} else if self.arblost() {
Err(Error::ArbitrationLost)
} else if self.lenerr() {
Err(Error::LengthError)
} else if self.rxnack() {
Err(Error::Nack)
} else {
Ok(())
}
}
pub fn is_idle(self) -> bool {
self.busstate() == BusState::Idle
}
}
/// Errors available for I2C transactions
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub enum Error {
BusError,
ArbitrationLost,
LengthError,
Nack,
Timeout,
#[cfg(feature = "dma")]
Dma(crate::dmac::Error),
}