1#![macro_use]
3
4#[cfg(any(bdma, dma))]
5mod dma_bdma;
6#[cfg(any(bdma, dma))]
7pub use dma_bdma::*;
8
9#[cfg(gpdma)]
10pub(crate) mod gpdma;
11#[cfg(gpdma)]
12pub use gpdma::*;
13
14#[cfg(dmamux)]
15mod dmamux;
16#[cfg(dmamux)]
17pub(crate) use dmamux::*;
18
19mod util;
20pub(crate) use util::*;
21
22pub(crate) mod ringbuffer;
23pub mod word;
24
25use embassy_hal_internal::{impl_peripheral, Peripheral};
26
27use crate::interrupt;
28
29#[derive(Debug, Copy, Clone, PartialEq, Eq)]
30#[cfg_attr(feature = "defmt", derive(defmt::Format))]
31enum Dir {
32 MemoryToPeripheral,
33 PeripheralToMemory,
34}
35
36#[cfg(any(dma_v2, bdma_v2, gpdma, dmamux))]
38pub type Request = u8;
39#[cfg(not(any(dma_v2, bdma_v2, gpdma, dmamux)))]
41pub type Request = ();
42
43pub(crate) trait SealedChannel {
44 fn id(&self) -> u8;
45}
46
47pub(crate) trait ChannelInterrupt {
48 #[cfg_attr(not(feature = "rt"), allow(unused))]
49 unsafe fn on_irq();
50}
51
52#[allow(private_bounds)]
54pub trait Channel: SealedChannel + Peripheral<P = Self> + Into<AnyChannel> + 'static {
55 #[inline]
61 fn degrade(self) -> AnyChannel {
62 AnyChannel { id: self.id() }
63 }
64}
65
66macro_rules! dma_channel_impl {
67 ($channel_peri:ident, $index:expr) => {
68 impl crate::dma::SealedChannel for crate::peripherals::$channel_peri {
69 fn id(&self) -> u8 {
70 $index
71 }
72 }
73 impl crate::dma::ChannelInterrupt for crate::peripherals::$channel_peri {
74 unsafe fn on_irq() {
75 crate::dma::AnyChannel { id: $index }.on_irq();
76 }
77 }
78
79 impl crate::dma::Channel for crate::peripherals::$channel_peri {}
80
81 impl From<crate::peripherals::$channel_peri> for crate::dma::AnyChannel {
82 fn from(x: crate::peripherals::$channel_peri) -> Self {
83 crate::dma::Channel::degrade(x)
84 }
85 }
86 };
87}
88
89pub struct AnyChannel {
91 pub(crate) id: u8,
92}
93impl_peripheral!(AnyChannel);
94
95impl AnyChannel {
96 fn info(&self) -> &ChannelInfo {
97 &crate::_generated::DMA_CHANNELS[self.id as usize]
98 }
99}
100
101impl SealedChannel for AnyChannel {
102 fn id(&self) -> u8 {
103 self.id
104 }
105}
106impl Channel for AnyChannel {}
107
108const CHANNEL_COUNT: usize = crate::_generated::DMA_CHANNELS.len();
109static STATE: [ChannelState; CHANNEL_COUNT] = [ChannelState::NEW; CHANNEL_COUNT];
110
111pub struct NoDma;
119
120impl_peripheral!(NoDma);
121
122pub(crate) unsafe fn init(
124 cs: critical_section::CriticalSection,
125 #[cfg(bdma)] bdma_priority: interrupt::Priority,
126 #[cfg(dma)] dma_priority: interrupt::Priority,
127 #[cfg(gpdma)] gpdma_priority: interrupt::Priority,
128) {
129 #[cfg(any(dma, bdma))]
130 dma_bdma::init(
131 cs,
132 #[cfg(dma)]
133 dma_priority,
134 #[cfg(bdma)]
135 bdma_priority,
136 );
137 #[cfg(gpdma)]
138 gpdma::init(cs, gpdma_priority);
139 #[cfg(dmamux)]
140 dmamux::init(cs);
141}