nrf/spis: switch to new interrupt binding.
This commit is contained in:
parent
a32e82029a
commit
9f5762d365
2 changed files with 43 additions and 51 deletions
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
#![macro_use]
|
#![macro_use]
|
||||||
use core::future::poll_fn;
|
use core::future::poll_fn;
|
||||||
|
use core::marker::PhantomData;
|
||||||
use core::sync::atomic::{compiler_fence, Ordering};
|
use core::sync::atomic::{compiler_fence, Ordering};
|
||||||
use core::task::Poll;
|
use core::task::Poll;
|
||||||
|
|
||||||
|
@ -12,7 +13,7 @@ pub use embedded_hal_02::spi::{Mode, Phase, Polarity, MODE_0, MODE_1, MODE_2, MO
|
||||||
use crate::chip::FORCE_COPY_BUFFER_SIZE;
|
use crate::chip::FORCE_COPY_BUFFER_SIZE;
|
||||||
use crate::gpio::sealed::Pin as _;
|
use crate::gpio::sealed::Pin as _;
|
||||||
use crate::gpio::{self, AnyPin, Pin as GpioPin};
|
use crate::gpio::{self, AnyPin, Pin as GpioPin};
|
||||||
use crate::interrupt::{Interrupt, InterruptExt};
|
use crate::interrupt::{self, Interrupt, InterruptExt};
|
||||||
use crate::util::{slice_in_ram_or, slice_ptr_parts, slice_ptr_parts_mut};
|
use crate::util::{slice_in_ram_or, slice_ptr_parts, slice_ptr_parts_mut};
|
||||||
use crate::{pac, Peripheral};
|
use crate::{pac, Peripheral};
|
||||||
|
|
||||||
|
@ -29,11 +30,6 @@ pub enum Error {
|
||||||
BufferNotInRAM,
|
BufferNotInRAM,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// SPIS driver.
|
|
||||||
pub struct Spis<'d, T: Instance> {
|
|
||||||
_p: PeripheralRef<'d, T>,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// SPIS configuration.
|
/// SPIS configuration.
|
||||||
#[non_exhaustive]
|
#[non_exhaustive]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
|
@ -67,11 +63,38 @@ impl Default for Config {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Interrupt handler.
|
||||||
|
pub struct InterruptHandler<T: Instance> {
|
||||||
|
_phantom: PhantomData<T>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Instance> interrupt::Handler<T::Interrupt> for InterruptHandler<T> {
|
||||||
|
unsafe fn on_interrupt() {
|
||||||
|
let r = T::regs();
|
||||||
|
let s = T::state();
|
||||||
|
|
||||||
|
if r.events_end.read().bits() != 0 {
|
||||||
|
s.waker.wake();
|
||||||
|
r.intenclr.write(|w| w.end().clear());
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.events_acquired.read().bits() != 0 {
|
||||||
|
s.waker.wake();
|
||||||
|
r.intenclr.write(|w| w.acquired().clear());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// SPIS driver.
|
||||||
|
pub struct Spis<'d, T: Instance> {
|
||||||
|
_p: PeripheralRef<'d, T>,
|
||||||
|
}
|
||||||
|
|
||||||
impl<'d, T: Instance> Spis<'d, T> {
|
impl<'d, T: Instance> Spis<'d, T> {
|
||||||
/// Create a new SPIS driver.
|
/// Create a new SPIS driver.
|
||||||
pub fn new(
|
pub fn new(
|
||||||
spis: impl Peripheral<P = T> + 'd,
|
spis: impl Peripheral<P = T> + 'd,
|
||||||
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
_irq: impl interrupt::Binding<T::Interrupt, InterruptHandler<T>> + 'd,
|
||||||
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
miso: impl Peripheral<P = impl GpioPin> + 'd,
|
miso: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
|
@ -81,7 +104,6 @@ impl<'d, T: Instance> Spis<'d, T> {
|
||||||
into_ref!(cs, sck, miso, mosi);
|
into_ref!(cs, sck, miso, mosi);
|
||||||
Self::new_inner(
|
Self::new_inner(
|
||||||
spis,
|
spis,
|
||||||
irq,
|
|
||||||
cs.map_into(),
|
cs.map_into(),
|
||||||
sck.map_into(),
|
sck.map_into(),
|
||||||
Some(miso.map_into()),
|
Some(miso.map_into()),
|
||||||
|
@ -93,48 +115,31 @@ impl<'d, T: Instance> Spis<'d, T> {
|
||||||
/// Create a new SPIS driver, capable of TX only (MISO only).
|
/// Create a new SPIS driver, capable of TX only (MISO only).
|
||||||
pub fn new_txonly(
|
pub fn new_txonly(
|
||||||
spis: impl Peripheral<P = T> + 'd,
|
spis: impl Peripheral<P = T> + 'd,
|
||||||
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
_irq: impl interrupt::Binding<T::Interrupt, InterruptHandler<T>> + 'd,
|
||||||
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
miso: impl Peripheral<P = impl GpioPin> + 'd,
|
miso: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
config: Config,
|
config: Config,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
into_ref!(cs, sck, miso);
|
into_ref!(cs, sck, miso);
|
||||||
Self::new_inner(
|
Self::new_inner(spis, cs.map_into(), sck.map_into(), Some(miso.map_into()), None, config)
|
||||||
spis,
|
|
||||||
irq,
|
|
||||||
cs.map_into(),
|
|
||||||
sck.map_into(),
|
|
||||||
Some(miso.map_into()),
|
|
||||||
None,
|
|
||||||
config,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create a new SPIS driver, capable of RX only (MOSI only).
|
/// Create a new SPIS driver, capable of RX only (MOSI only).
|
||||||
pub fn new_rxonly(
|
pub fn new_rxonly(
|
||||||
spis: impl Peripheral<P = T> + 'd,
|
spis: impl Peripheral<P = T> + 'd,
|
||||||
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
_irq: impl interrupt::Binding<T::Interrupt, InterruptHandler<T>> + 'd,
|
||||||
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
cs: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
sck: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
mosi: impl Peripheral<P = impl GpioPin> + 'd,
|
mosi: impl Peripheral<P = impl GpioPin> + 'd,
|
||||||
config: Config,
|
config: Config,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
into_ref!(cs, sck, mosi);
|
into_ref!(cs, sck, mosi);
|
||||||
Self::new_inner(
|
Self::new_inner(spis, cs.map_into(), sck.map_into(), None, Some(mosi.map_into()), config)
|
||||||
spis,
|
|
||||||
irq,
|
|
||||||
cs.map_into(),
|
|
||||||
sck.map_into(),
|
|
||||||
None,
|
|
||||||
Some(mosi.map_into()),
|
|
||||||
config,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn new_inner(
|
fn new_inner(
|
||||||
spis: impl Peripheral<P = T> + 'd,
|
spis: impl Peripheral<P = T> + 'd,
|
||||||
irq: impl Peripheral<P = T::Interrupt> + 'd,
|
|
||||||
cs: PeripheralRef<'d, AnyPin>,
|
cs: PeripheralRef<'d, AnyPin>,
|
||||||
sck: PeripheralRef<'d, AnyPin>,
|
sck: PeripheralRef<'d, AnyPin>,
|
||||||
miso: Option<PeripheralRef<'d, AnyPin>>,
|
miso: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
|
@ -143,7 +148,7 @@ impl<'d, T: Instance> Spis<'d, T> {
|
||||||
) -> Self {
|
) -> Self {
|
||||||
compiler_fence(Ordering::SeqCst);
|
compiler_fence(Ordering::SeqCst);
|
||||||
|
|
||||||
into_ref!(spis, irq, cs, sck);
|
into_ref!(spis, cs, sck);
|
||||||
|
|
||||||
let r = T::regs();
|
let r = T::regs();
|
||||||
|
|
||||||
|
@ -209,28 +214,12 @@ impl<'d, T: Instance> Spis<'d, T> {
|
||||||
// Disable all events interrupts.
|
// Disable all events interrupts.
|
||||||
r.intenclr.write(|w| unsafe { w.bits(0xFFFF_FFFF) });
|
r.intenclr.write(|w| unsafe { w.bits(0xFFFF_FFFF) });
|
||||||
|
|
||||||
irq.set_handler(Self::on_interrupt);
|
unsafe { T::Interrupt::steal() }.unpend();
|
||||||
irq.unpend();
|
unsafe { T::Interrupt::steal() }.enable();
|
||||||
irq.enable();
|
|
||||||
|
|
||||||
Self { _p: spis }
|
Self { _p: spis }
|
||||||
}
|
}
|
||||||
|
|
||||||
fn on_interrupt(_: *mut ()) {
|
|
||||||
let r = T::regs();
|
|
||||||
let s = T::state();
|
|
||||||
|
|
||||||
if r.events_end.read().bits() != 0 {
|
|
||||||
s.waker.wake();
|
|
||||||
r.intenclr.write(|w| w.end().clear());
|
|
||||||
}
|
|
||||||
|
|
||||||
if r.events_acquired.read().bits() != 0 {
|
|
||||||
s.waker.wake();
|
|
||||||
r.intenclr.write(|w| w.acquired().clear());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prepare(&mut self, rx: *mut [u8], tx: *const [u8]) -> Result<(), Error> {
|
fn prepare(&mut self, rx: *mut [u8], tx: *const [u8]) -> Result<(), Error> {
|
||||||
slice_in_ram_or(tx, Error::BufferNotInRAM)?;
|
slice_in_ram_or(tx, Error::BufferNotInRAM)?;
|
||||||
// NOTE: RAM slice check for rx is not necessary, as a mutable
|
// NOTE: RAM slice check for rx is not necessary, as a mutable
|
||||||
|
|
|
@ -4,17 +4,20 @@
|
||||||
|
|
||||||
use defmt::info;
|
use defmt::info;
|
||||||
use embassy_executor::Spawner;
|
use embassy_executor::Spawner;
|
||||||
use embassy_nrf::interrupt;
|
|
||||||
use embassy_nrf::spis::{Config, Spis};
|
use embassy_nrf::spis::{Config, Spis};
|
||||||
|
use embassy_nrf::{bind_interrupts, peripherals, spis};
|
||||||
use {defmt_rtt as _, panic_probe as _};
|
use {defmt_rtt as _, panic_probe as _};
|
||||||
|
|
||||||
|
bind_interrupts!(struct Irqs {
|
||||||
|
SPIM2_SPIS2_SPI2 => spis::InterruptHandler<peripherals::SPI2>;
|
||||||
|
});
|
||||||
|
|
||||||
#[embassy_executor::main]
|
#[embassy_executor::main]
|
||||||
async fn main(_spawner: Spawner) {
|
async fn main(_spawner: Spawner) {
|
||||||
let p = embassy_nrf::init(Default::default());
|
let p = embassy_nrf::init(Default::default());
|
||||||
info!("Running!");
|
info!("Running!");
|
||||||
|
|
||||||
let irq = interrupt::take!(SPIM2_SPIS2_SPI2);
|
let mut spis = Spis::new(p.SPI2, Irqs, p.P0_31, p.P0_29, p.P0_28, p.P0_30, Config::default());
|
||||||
let mut spis = Spis::new(p.SPI2, irq, p.P0_31, p.P0_29, p.P0_28, p.P0_30, Config::default());
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let mut rx_buf = [0_u8; 64];
|
let mut rx_buf = [0_u8; 64];
|
||||||
|
|
Loading…
Reference in a new issue