embassy/embassy-stm32/src/dma/mod.rs

198 lines
4.5 KiB
Rust
Raw Normal View History

2021-07-15 05:42:06 +02:00
#[cfg(bdma)]
mod bdma;
#[cfg(dma)]
mod dma;
#[cfg(dmamux)]
mod dmamux;
2021-05-16 02:57:46 +02:00
#[cfg(dmamux)]
pub use dmamux::*;
use core::future::Future;
use core::marker::PhantomData;
use core::pin::Pin;
use core::task::Waker;
use core::task::{Context, Poll};
2021-07-15 05:42:06 +02:00
use embassy::util::Unborrow;
use embassy_hal_common::unborrow;
2021-11-19 19:15:55 +01:00
#[cfg(feature = "unstable-pac")]
pub use transfers::*;
#[cfg(not(feature = "unstable-pac"))]
pub(crate) use transfers::*;
2021-07-15 05:42:06 +02:00
#[cfg(any(bdma_v2, dma_v2, dmamux))]
pub type Request = u8;
#[cfg(not(any(bdma_v2, dma_v2, dmamux)))]
pub type Request = ();
pub(crate) mod sealed {
2021-11-19 19:15:55 +01:00
use super::*;
2021-11-19 19:15:55 +01:00
pub trait Channel {
/// Starts this channel for writing a stream of words.
unsafe fn start_write<W: Word>(&mut self, request: Request, buf: &[W], reg_addr: *mut u32);
/// Starts this channel for writing a word repeatedly.
unsafe fn start_write_repeated<W: Word>(
&mut self,
request: Request,
repeated: W,
count: usize,
reg_addr: *mut u32,
);
/// Starts this channel for reading a stream of words.
unsafe fn start_read<W: Word>(
&mut self,
request: Request,
reg_addr: *mut u32,
buf: &mut [W],
);
/// Stops this channel.
fn request_stop(&mut self);
/// Returns whether this channel is active or stopped.
fn is_running(&self) -> bool;
2021-11-19 19:15:55 +01:00
/// Returns the total number of remaining transfers.
fn remaining_transfers(&mut self) -> u16;
/// Sets the waker that is called when this channel completes.
fn set_waker(&mut self, waker: &Waker);
}
2021-07-15 05:42:06 +02:00
}
2021-11-19 19:15:55 +01:00
pub enum WordSize {
OneByte,
TwoBytes,
FourBytes,
}
pub trait Word {
fn bits() -> WordSize;
}
impl Word for u8 {
fn bits() -> WordSize {
WordSize::OneByte
}
}
2021-07-15 05:42:06 +02:00
2021-11-19 19:15:55 +01:00
impl Word for u16 {
fn bits() -> WordSize {
WordSize::TwoBytes
}
}
impl Word for u32 {
fn bits() -> WordSize {
WordSize::FourBytes
}
}
2021-07-15 05:42:06 +02:00
2021-11-19 19:15:55 +01:00
mod transfers {
use super::*;
#[allow(unused)]
pub fn read<'a, W: Word>(
channel: impl Unborrow<Target = impl Channel> + 'a,
2021-07-15 05:42:06 +02:00
request: Request,
2021-11-19 19:15:55 +01:00
reg_addr: *mut u32,
buf: &'a mut [W],
) -> impl Future<Output = ()> + 'a {
2021-11-19 19:15:55 +01:00
assert!(buf.len() <= 0xFFFF);
unborrow!(channel);
2021-07-15 05:42:06 +02:00
2021-11-19 19:15:55 +01:00
unsafe { channel.start_read::<W>(request, reg_addr, buf) };
Transfer {
channel,
_phantom: PhantomData,
}
2021-11-19 19:15:55 +01:00
}
#[allow(unused)]
pub fn write<'a, W: Word>(
channel: impl Unborrow<Target = impl Channel> + 'a,
2021-07-15 05:42:06 +02:00
request: Request,
2021-11-19 19:15:55 +01:00
buf: &'a [W],
reg_addr: *mut u32,
) -> impl Future<Output = ()> + 'a {
2021-11-19 19:15:55 +01:00
assert!(buf.len() <= 0xFFFF);
unborrow!(channel);
unsafe { channel.start_write::<W>(request, buf, reg_addr) };
Transfer {
channel,
_phantom: PhantomData,
}
2021-11-19 19:15:55 +01:00
}
2021-11-19 19:15:55 +01:00
#[allow(unused)]
pub fn write_repeated<'a, W: Word>(
channel: impl Unborrow<Target = impl Channel> + 'a,
request: Request,
2021-11-19 19:15:55 +01:00
repeated: W,
count: usize,
reg_addr: *mut u32,
) -> impl Future<Output = ()> + 'a {
2021-11-19 19:15:55 +01:00
unborrow!(channel);
unsafe { channel.start_write_repeated::<W>(request, repeated, count, reg_addr) };
Transfer {
channel,
_phantom: PhantomData,
}
2021-11-19 19:15:55 +01:00
}
struct Transfer<'a, C: Channel> {
channel: C,
_phantom: PhantomData<&'a mut C>,
}
2021-11-19 19:15:55 +01:00
impl<'a, C: Channel> Drop for Transfer<'a, C> {
fn drop(&mut self) {
self.channel.request_stop();
while self.channel.is_running() {}
}
}
2021-11-19 19:15:55 +01:00
impl<'a, C: Channel> Unpin for Transfer<'a, C> {}
impl<'a, C: Channel> Future for Transfer<'a, C> {
type Output = ();
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
self.channel.set_waker(cx.waker());
if self.channel.is_running() {
2021-11-19 19:15:55 +01:00
Poll::Pending
} else {
Poll::Ready(())
2021-11-19 19:15:55 +01:00
}
}
2021-11-19 19:15:55 +01:00
}
2021-07-15 05:42:06 +02:00
}
pub trait Channel: sealed::Channel + Unborrow<Target = Self> + 'static {}
2021-11-19 19:15:55 +01:00
2021-07-15 05:42:06 +02:00
pub struct NoDma;
unsafe impl Unborrow for NoDma {
type Target = NoDma;
unsafe fn unborrow(self) -> Self::Target {
self
}
}
// safety: must be called only once at startup
pub(crate) unsafe fn init() {
#[cfg(bdma)]
bdma::init();
#[cfg(dma)]
dma::init();
#[cfg(dmamux)]
dmamux::init();
}