use kernel::hil;
use kernel::utilities::cells::OptionalCell;
use kernel::utilities::cells::TakeCell;
use kernel::utilities::cells::VolatileCell;
use kernel::utilities::registers::interfaces::{ReadWriteable, Readable, Writeable};
use kernel::utilities::registers::{register_bitfields, register_structs, ReadWrite, WriteOnly};
use kernel::utilities::StaticRef;
use nrf5x::pinmux::Pinmux;
const INSTANCES: [StaticRef<TwiRegisters>; 2] = unsafe {
[
StaticRef::new(0x40003000 as *const TwiRegisters),
StaticRef::new(0x40004000 as *const TwiRegisters),
]
};
pub struct TWI<'a> {
registers: StaticRef<TwiRegisters>,
client: OptionalCell<&'a dyn hil::i2c::I2CHwMasterClient>,
slave_client: OptionalCell<&'a dyn hil::i2c::I2CHwSlaveClient>,
buf: TakeCell<'static, [u8]>,
slave_read_buf: TakeCell<'static, [u8]>,
}
#[repr(u32)]
pub enum Speed {
K100 = 0x01980000,
K250 = 0x04000000,
K400 = 0x06400000,
}
impl TWI<'_> {
const fn new(registers: StaticRef<TwiRegisters>) -> Self {
Self {
registers,
client: OptionalCell::empty(),
slave_client: OptionalCell::empty(),
buf: TakeCell::empty(),
slave_read_buf: TakeCell::empty(),
}
}
pub const fn new_twi0() -> Self {
TWI::new(INSTANCES[0])
}
pub const fn new_twi1() -> Self {
TWI::new(INSTANCES[1])
}
pub fn configure(&self, scl: Pinmux, sda: Pinmux) {
self.registers.psel_scl.set(scl);
self.registers.psel_sda.set(sda);
}
pub fn set_speed(&self, speed: Speed) {
self.registers.frequency.set(speed as u32);
}
pub fn clear_events(&self) {
self.registers.events_stopped.write(EVENT::EVENT::CLEAR);
self.registers.events_error.write(EVENT::EVENT::CLEAR);
self.registers.events_rxstarted.write(EVENT::EVENT::CLEAR);
self.registers.events_txstarted.write(EVENT::EVENT::CLEAR);
self.registers.events_write.write(EVENT::EVENT::CLEAR);
self.registers.events_read.write(EVENT::EVENT::CLEAR);
self.registers.events_suspended.write(EVENT::EVENT::CLEAR);
self.registers.events_lastrx.write(EVENT::EVENT::CLEAR);
self.registers.events_lasttx.write(EVENT::EVENT::CLEAR);
}
pub fn disable_interrupts(&self) {
self.registers.inten.set(0x00);
self.registers.intenclr.set(0xFFFF_FFFF);
}
fn enable_master(&self) {
self.clear_events();
self.registers.enable.write(ENABLE::ENABLE::EnableMaster);
}
fn enable_slave(&self) {
self.clear_events();
self.registers.enable.write(ENABLE::ENABLE::EnableSlave);
}
fn disable(&self) {
self.clear_events();
self.disable_interrupts();
self.registers.enable.write(ENABLE::ENABLE::Disable);
}
pub fn handle_interrupt(&self) {
if self.is_master_enabled() {
if self.registers.events_stopped.is_set(EVENT::EVENT) {
self.registers.events_stopped.write(EVENT::EVENT::CLEAR);
self.client.map(|client| match self.buf.take() {
None => (),
Some(buf) => {
self.clear_events();
client.command_complete(buf, Ok(()));
}
});
}
if self.registers.events_error.is_set(EVENT::EVENT) {
self.registers.events_error.write(EVENT::EVENT::CLEAR);
let errorsrc = self.registers.errorsrc_master.extract();
self.registers
.errorsrc_master
.write(ERRORSRC::ANACK::ErrorDidNotOccur + ERRORSRC::DNACK::ErrorDidNotOccur);
self.client.map(|client| match self.buf.take() {
None => (),
Some(buf) => {
let status = if errorsrc.is_set(ERRORSRC::ANACK) {
Err(hil::i2c::Error::AddressNak)
} else if errorsrc.is_set(ERRORSRC::DNACK) {
Err(hil::i2c::Error::DataNak)
} else {
Ok(())
};
self.clear_events();
client.command_complete(buf, status);
}
});
}
} else {
self.registers.events_stopped.write(EVENT::EVENT::CLEAR);
if self.registers.events_rxstarted.is_set(EVENT::EVENT) {
self.registers.events_rxstarted.write(EVENT::EVENT::CLEAR);
self.slave_client.map(|client| {
if self.buf.is_none() {
client.write_expected();
}
});
}
if self.registers.events_txstarted.is_set(EVENT::EVENT) {
self.registers.events_txstarted.write(EVENT::EVENT::CLEAR);
self.slave_client.map(|client| {
if self.slave_read_buf.is_none() {
client.read_expected();
}
});
}
if self.registers.events_write.is_set(EVENT::EVENT) {
self.registers.events_write.write(EVENT::EVENT::CLEAR);
let length = self.registers.rxd_amount.read(AMOUNT::AMOUNT) as usize;
self.slave_client.map(|client| match self.buf.take() {
None => (),
Some(buf) => {
self.clear_events();
client.command_complete(
buf,
length,
hil::i2c::SlaveTransmissionType::Write,
);
}
});
}
if self.registers.events_read.is_set(EVENT::EVENT) {
self.registers.events_read.write(EVENT::EVENT::CLEAR);
let length = self.registers.txd_amount.read(AMOUNT::AMOUNT) as usize;
self.slave_client
.map(|client| match self.slave_read_buf.take() {
None => (),
Some(buf) => {
self.clear_events();
client.command_complete(
buf,
length,
hil::i2c::SlaveTransmissionType::Read,
);
}
});
}
}
self.registers.events_suspended.write(EVENT::EVENT::CLEAR);
self.registers.events_lastrx.write(EVENT::EVENT::CLEAR);
self.registers.events_lasttx.write(EVENT::EVENT::CLEAR);
}
pub fn is_enabled(&self) -> bool {
self.is_master_enabled() || self.is_slave_enabled()
}
fn is_master_enabled(&self) -> bool {
self.registers
.enable
.matches_all(ENABLE::ENABLE::EnableMaster)
}
fn is_slave_enabled(&self) -> bool {
self.registers
.enable
.matches_all(ENABLE::ENABLE::EnableSlave)
}
}
impl<'a> hil::i2c::I2CMaster<'a> for TWI<'a> {
fn set_master_client(&self, client: &'a dyn hil::i2c::I2CHwMasterClient) {
self.client.set(client);
}
fn enable(&self) {
self.enable_master();
}
fn disable(&self) {
self.disable();
}
fn write_read(
&self,
addr: u8,
data: &'static mut [u8],
write_len: usize,
read_len: usize,
) -> Result<(), (hil::i2c::Error, &'static mut [u8])> {
self.registers
.address_0
.write(ADDRESS::ADDRESS.val(addr as u32));
self.registers.txd_ptr.set(data.as_mut_ptr() as u32);
self.registers
.txd_maxcnt
.write(MAXCNT::MAXCNT.val(write_len as u32));
self.registers.rxd_ptr.set(data.as_mut_ptr() as u32);
self.registers
.rxd_maxcnt
.write(MAXCNT::MAXCNT.val(read_len as u32));
self.registers
.shorts
.write(SHORTS::LASTTX_STARTRX::EnableShortcut + SHORTS::LASTRX_STOP::EnableShortcut);
self.registers
.intenset
.write(INTE::STOPPED::Enable + INTE::ERROR::Enable);
self.registers.tasks_starttx.write(TASK::TASK::SET);
self.buf.replace(data);
Ok(())
}
fn write(
&self,
addr: u8,
data: &'static mut [u8],
len: usize,
) -> Result<(), (hil::i2c::Error, &'static mut [u8])> {
self.registers
.address_0
.write(ADDRESS::ADDRESS.val(addr as u32));
self.registers.txd_ptr.set(data.as_mut_ptr() as u32);
self.registers
.txd_maxcnt
.write(MAXCNT::MAXCNT.val(len as u32));
self.registers
.shorts
.write(SHORTS::LASTTX_STOP::EnableShortcut);
self.registers
.intenset
.write(INTE::STOPPED::Enable + INTE::ERROR::Enable);
self.registers.tasks_starttx.write(TASK::TASK::SET);
self.buf.replace(data);
Ok(())
}
fn read(
&self,
addr: u8,
buffer: &'static mut [u8],
len: usize,
) -> Result<(), (hil::i2c::Error, &'static mut [u8])> {
self.registers
.address_0
.write(ADDRESS::ADDRESS.val(addr as u32));
self.registers.rxd_ptr.set(buffer.as_mut_ptr() as u32);
self.registers
.rxd_maxcnt
.write(MAXCNT::MAXCNT.val(len as u32));
self.registers
.shorts
.write(SHORTS::LASTRX_STOP::EnableShortcut);
self.registers
.intenset
.write(INTE::STOPPED::Enable + INTE::ERROR::Enable);
self.registers.tasks_startrx.write(TASK::TASK::SET);
self.buf.replace(buffer);
Ok(())
}
}
impl<'a> hil::i2c::I2CSlave<'a> for TWI<'a> {
fn set_slave_client(&self, client: &'a dyn hil::i2c::I2CHwSlaveClient) {
self.slave_client.set(client);
}
fn enable(&self) {
self.enable_slave();
}
fn disable(&self) {
self.disable();
}
fn set_address(&self, addr: u8) -> Result<(), hil::i2c::Error> {
self.registers
.address_0
.write(ADDRESS::ADDRESS.val(addr as u32));
self.registers.config.modify(CONFIG::ADDRESS0::Enable);
Ok(())
}
fn write_receive(
&self,
data: &'static mut [u8],
max_len: usize,
) -> Result<(), (hil::i2c::Error, &'static mut [u8])> {
self.registers.rxd_ptr.set(data.as_mut_ptr() as u32);
self.registers
.rxd_maxcnt
.write(MAXCNT::MAXCNT.val(max_len as u32));
self.registers
.intenset
.modify(INTE::STOPPED::Enable + INTE::ERROR::Enable);
self.buf.replace(data);
self.registers.tasks_preparerx.write(TASK::TASK::SET);
Ok(())
}
fn read_send(
&self,
data: &'static mut [u8],
max_len: usize,
) -> Result<(), (hil::i2c::Error, &'static mut [u8])> {
self.registers.txd_ptr.set(data.as_mut_ptr() as u32);
self.registers
.txd_maxcnt
.write(MAXCNT::MAXCNT.val(max_len as u32));
self.registers
.intenset
.modify(INTE::STOPPED::Enable + INTE::ERROR::Enable + INTE::READ::Enable);
self.slave_read_buf.replace(data);
self.registers.tasks_preparetx.write(TASK::TASK::SET);
Ok(())
}
fn listen(&self) {
self.registers.tasks_preparerx.write(TASK::TASK::SET);
}
}
impl<'a> hil::i2c::I2CMasterSlave<'a> for TWI<'a> {}
register_structs! {
pub TwiRegisters {
(0x00 => tasks_startrx: WriteOnly<u32, TASK::Register>),
(0x04 => _reserved0),
(0x08 => tasks_starttx: WriteOnly<u32, TASK::Register>),
(0x0C => _reserved1),
(0x14 => tasks_stop: WriteOnly<u32, TASK::Register>),
(0x18 => _reserved2),
(0x1C => tasks_suspend: WriteOnly<u32, TASK::Register>),
(0x20 => tasks_resume: WriteOnly<u32, TASK::Register>),
(0x24 => _reserved3),
(0x30 => tasks_preparerx: WriteOnly<u32, TASK::Register>),
(0x34 => tasks_preparetx: WriteOnly<u32, TASK::Register>),
(0x38 => _reserved4),
(0x104 => events_stopped: ReadWrite<u32, EVENT::Register>),
(0x108 => _reserved5),
(0x124 => events_error: ReadWrite<u32, EVENT::Register>),
(0x128 => _reserved6),
(0x148 => events_suspended: ReadWrite<u32, EVENT::Register>),
(0x14C => events_rxstarted: ReadWrite<u32, EVENT::Register>),
(0x150 => events_txstarted: ReadWrite<u32, EVENT::Register>),
(0x154 => _reserved7),
(0x15C => events_lastrx: ReadWrite<u32, EVENT::Register>),
(0x160 => events_lasttx: ReadWrite<u32, EVENT::Register>),
(0x164 => events_write: ReadWrite<u32, EVENT::Register>),
(0x168 => events_read: ReadWrite<u32, EVENT::Register>),
(0x16C => _reserved8),
(0x200 => shorts: ReadWrite<u32, SHORTS::Register>),
(0x204 => _reserved9),
(0x300 => inten: ReadWrite<u32, INTE::Register>),
(0x304 => intenset: ReadWrite<u32, INTE::Register>),
(0x308 => intenclr: ReadWrite<u32, INTE::Register>),
(0x30C => _reserved10),
(0x4C4 => errorsrc_master: ReadWrite<u32, ERRORSRC::Register>),
(0x4C8 => _reserved11),
(0x4D0 => errorsrc_slave: ReadWrite<u32, ERRORSRC::Register>),
(0x4D4 => match_reg: ReadWrite<u32>),
(0x4D8 => _reserved12),
(0x500 => enable: ReadWrite<u32, ENABLE::Register>),
(0x504 => _reserved13),
(0x508 => psel_scl: VolatileCell<Pinmux>),
(0x50C => psel_sda: VolatileCell<Pinmux>),
(0x510 => _reserved_14),
(0x524 => frequency: ReadWrite<u32>),
(0x528 => _reserved15),
(0x534 => rxd_ptr: ReadWrite<u32>),
(0x538 => rxd_maxcnt: ReadWrite<u32, MAXCNT::Register>),
(0x53C => rxd_amount: ReadWrite<u32, AMOUNT::Register>),
(0x540 => rxd_list: ReadWrite<u32>),
(0x544 => txd_ptr: ReadWrite<u32>),
(0x548 => txd_maxcnt: ReadWrite<u32, MAXCNT::Register>),
(0x54C => txd_amount: ReadWrite<u32, AMOUNT::Register>),
(0x550 => txd_list: ReadWrite<u32>),
(0x554 => _reserved_16),
(0x588 => address_0: ReadWrite<u32, ADDRESS::Register>),
(0x58C => address_1: ReadWrite<u32, ADDRESS::Register>),
(0x590 => _reserved_17),
(0x594 => config: ReadWrite<u32, CONFIG::Register>),
(0x598 => _reserved_18),
(0x5C0 => orc: ReadWrite<u32>),
(0x5C4 => @END),
}
}
register_bitfields![u32,
SHORTS [
LASTTX_STARTRX OFFSET(7) NUMBITS(1) [
DisableShortcut = 0,
EnableShortcut = 1
],
LASTTX_SUSPEND OFFSET(8) NUMBITS(1) [
DisableShortcut = 0,
EnableShortcut = 1
],
LASTTX_STOP OFFSET(9) NUMBITS(1) [
DisableShortcut = 0,
EnableShortcut = 1
],
LASTRX_STARTTX OFFSET(10) NUMBITS(1) [
DisableShortcut = 0,
EnableShortcut = 1
],
LASTRX_STOP OFFSET(12) NUMBITS(1) [
DisableShortcut = 0,
EnableShortcut = 1
]
],
INTE [
STOPPED OFFSET(1) NUMBITS(1) [
Disable = 0,
Enable = 1
],
ERROR OFFSET(9) NUMBITS(1) [
Disable = 0,
Enable = 1
],
RXSTARTED OFFSET(19) NUMBITS(1) [
Disable = 0,
Enable = 1
],
TXSTARTED OFFSET(20) NUMBITS(1) [
Disable = 0,
Enable = 1
],
LASTRX OFFSET(23) NUMBITS(1) [
Disable = 0,
Enable = 1
],
LASTTX OFFSET(24) NUMBITS(1) [
Disable = 0,
Enable = 1
],
WRITE OFFSET(25) NUMBITS(1) [
Disable = 0,
Enable = 1
],
READ OFFSET(26) NUMBITS(1) [
Disable = 0,
Enable = 1
],
],
ERRORSRC [
ANACK OFFSET(1) NUMBITS(1) [
ErrorDidNotOccur = 0,
ErrorOccurred = 1
],
DNACK OFFSET(2) NUMBITS(1) [
ErrorDidNotOccur = 0,
ErrorOccurred = 1
]
],
EVENT [
EVENT 0
],
TASK [
TASK 0
],
ENABLE [
ENABLE OFFSET(0) NUMBITS(4) [
Disable = 0,
EnableMaster = 6,
EnableSlave = 9,
]
],
MAXCNT [
MAXCNT OFFSET(0) NUMBITS(16)
],
AMOUNT [
AMOUNT OFFSET(0) NUMBITS(16),
],
ADDRESS [
ADDRESS OFFSET(0) NUMBITS(7)
],
CONFIG [
ADDRESS0 OFFSET(0) NUMBITS(1) [
Disable = 0,
Enable = 1,
],
ADDRESS1 OFFSET(1) NUMBITS(1) [
Disable = 0,
Enable = 1,
]
],
];