mirror of
https://github.com/italicsjenga/rp-hal-boards.git
synced 2025-01-10 04:21:32 +11:00
commit
3a3280575f
|
@ -26,6 +26,11 @@ pub struct SioGpioBank0 {
|
|||
_private: (),
|
||||
}
|
||||
|
||||
/// Marker struct for ownership of SIO FIFO
|
||||
pub struct SioFifo {
|
||||
_private: (),
|
||||
}
|
||||
|
||||
/// Marker struct for ownership of SIO gpio qspi
|
||||
pub struct SioGpioQspi {
|
||||
_private: (),
|
||||
|
@ -53,24 +58,110 @@ pub struct Sio {
|
|||
pub gpio_qspi: SioGpioQspi,
|
||||
/// 8-cycle hardware divide/modulo module
|
||||
pub hwdivider: HwDivider,
|
||||
/// Inter-core FIFO
|
||||
pub fifo: SioFifo,
|
||||
// we can hand out other things here, for example:
|
||||
// interp0
|
||||
// interp1
|
||||
}
|
||||
|
||||
impl Sio {
|
||||
/// Create `Sio` from the PAC.
|
||||
pub fn new(sio: pac::SIO) -> Self {
|
||||
Self {
|
||||
_sio: sio,
|
||||
|
||||
gpio_bank0: SioGpioBank0 { _private: () },
|
||||
gpio_qspi: SioGpioQspi { _private: () },
|
||||
|
||||
fifo: SioFifo { _private: () },
|
||||
hwdivider: HwDivider { _private: () },
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl SioFifo {
|
||||
/// Check if the inter-core FIFO has valid data for reading.
|
||||
///
|
||||
/// Returning `true` means there is valid data, `false` means it is empty
|
||||
/// and you must not read from it.
|
||||
pub fn is_read_ready(&mut self) -> bool {
|
||||
let sio = unsafe { &(*pac::SIO::ptr()) };
|
||||
sio.fifo_st.read().vld().bit_is_set()
|
||||
}
|
||||
|
||||
/// Check if the inter-core FIFO is ready to receive data.
|
||||
///
|
||||
/// Returning `true` means there is room, `false` means it is full and you
|
||||
/// must not write to it.
|
||||
pub fn is_write_ready(&mut self) -> bool {
|
||||
let sio = unsafe { &(*pac::SIO::ptr()) };
|
||||
sio.fifo_st.read().rdy().bit_is_set()
|
||||
}
|
||||
|
||||
/// Return the FIFO status, as an integer.
|
||||
pub fn status(&self) -> u32 {
|
||||
let sio = unsafe { &(*pac::SIO::ptr()) };
|
||||
sio.fifo_st.read().bits()
|
||||
}
|
||||
|
||||
/// Write to the inter-core FIFO.
|
||||
///
|
||||
/// You must ensure the FIFO has space by calling `is_write_ready`
|
||||
pub fn write(&mut self, value: u32) {
|
||||
let sio = unsafe { &(*pac::SIO::ptr()) };
|
||||
sio.fifo_wr.write(|w| unsafe { w.bits(value) });
|
||||
}
|
||||
|
||||
/// Read from the inter-core FIFO.
|
||||
///
|
||||
/// Will return `Some(data)`, or `None` if the FIFO is empty.
|
||||
pub fn read(&mut self) -> Option<u32> {
|
||||
if self.is_read_ready() {
|
||||
let sio = unsafe { &(*pac::SIO::ptr()) };
|
||||
Some(sio.fifo_rd.read().bits())
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
/// Read from the FIFO until it is empty, throwing the contents away.
|
||||
pub fn drain(&mut self) {
|
||||
while self.read().is_some() {
|
||||
// Retry until FIFO empty
|
||||
}
|
||||
}
|
||||
|
||||
/// Push to the FIFO, spinning if there's no space.
|
||||
pub fn write_blocking(&mut self, value: u32) {
|
||||
// We busy-wait for the FIFO to have some space
|
||||
while !self.is_write_ready() {
|
||||
cortex_m::asm::nop();
|
||||
}
|
||||
|
||||
// Write the value to the FIFO - the other core will now be able to
|
||||
// pop it off its end of the FIFO.
|
||||
self.write(value as u32);
|
||||
|
||||
// Fire off an event to the other core
|
||||
cortex_m::asm::sev();
|
||||
}
|
||||
|
||||
/// Pop from the FIFO, spinning if there's currently no data.
|
||||
pub fn read_blocking(&mut self) -> u32 {
|
||||
// Keep trying until FIFO has data
|
||||
loop {
|
||||
// Have we got something?
|
||||
if let Some(data) = self.read() {
|
||||
// Yes, return it right away
|
||||
return data;
|
||||
} else {
|
||||
// No, so sleep the CPU. We expect the sending core to `sev`
|
||||
// on write.
|
||||
cortex_m::asm::wfe();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl HwDivider {
|
||||
/// Perform hardware unsigned divide/modulo operation
|
||||
pub fn unsigned(&self, dividend: u32, divisor: u32) -> DivResult<u32> {
|
||||
|
|
Loading…
Reference in a new issue