atsamd_hal/sercom/
mod.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
//! # Configure the SERCOM peripherals
//!
//! The SERCOM module is used to configure the SERCOM peripherals as USART, SPI
//! or I2C interfaces.
//! # Undocumented features
//!
//! The ATSAMx5x chips contain certain features that aren't documented in the
//! datasheet. These features are implemented in the HAL based on
//! experimentation with certain boards which have verifiably demonstrated that
//! those features work as intended.
//!
//! * [`UndocIoSet1`]: Implement an undocumented `IoSet` for PA16, PA17, PB22 &
//!   PB23 configured for [`Sercom1`]. The pygamer & feather_m4 use this
//!   combination, but it is not listed as valid in the datasheet.
//!
//! * [`UndocIoSet2`]: Implement an undocumented `IoSet` for PA00, PA01, PB22 &
//!   PB23 configured for [`Sercom1`]. The itsybitsy_m4 uses this combination,
//!   but it is not listed as valid in the datasheet.
//!
//! * [`PB02`] is I2C-capable according to metro_m4. As such, [`PB02`]
//!   implements [`IsI2cPad`].
//!
//! * [`PB03`] is I2C-capable according to metro_m4. As such, [`PB03`]
//!   implements [`IsI2cPad`].
//!
//! [`PB02`]: crate::gpio::pin::PB02
//! [`PB03`]: crate::gpio::pin::PB03
//! [`IsI2cPad`]: pad::IsI2cPad

use atsamd_hal_macros::hal_cfg;

use core::ops::Deref;

use crate::pac;
use pac::sercom0;
use pac::Peripherals;

#[hal_cfg("sercom0-d5x")]
use pac::Mclk as ApbClkCtrl;
#[hal_cfg(any("sercom0-d11", "sercom0-d21"))]
use pac::Pm as ApbClkCtrl;

#[cfg(feature = "dma")]
use crate::dmac::TriggerSource;

use crate::typelevel::Sealed;

pub mod pad;
pub use pad::*;

pub mod i2c;
pub mod spi;

#[deprecated(
    since = "0.19.0",
    note = "spi_future is deprecated and will be removed in a later version of atsamd_hal. Consider using the `async` APIs available in the `spi` module as a replacement."
)]
pub mod spi_future;
pub mod uart;

#[cfg(feature = "dma")]
pub mod dma;

//==============================================================================
//  Sercom
//==============================================================================

/// Type-level `enum` representing a Serial Communication Interface (SERCOM)
pub trait Sercom: Sealed + Deref<Target = sercom0::RegisterBlock> {
    /// SERCOM number
    const NUM: usize;

    /// RX Trigger source for DMA transactions
    #[cfg(feature = "dma")]
    const DMA_RX_TRIGGER: TriggerSource;

    /// TX trigger source for DMA transactions
    #[cfg(feature = "dma")]
    const DMA_TX_TRIGGER: TriggerSource;

    #[cfg(feature = "async")]
    type Interrupt: crate::async_hal::interrupts::InterruptSource;

    /// Enable the corresponding APB clock
    fn enable_apb_clock(&mut self, ctrl: &ApbClkCtrl);

    /// Get a reference to the sercom from a
    /// [`Peripherals`] block
    fn reg_block(peripherals: &mut Peripherals) -> &crate::pac::sercom0::RegisterBlock;

    /// Get a reference to this [`Sercom`]'s associated RX Waker
    #[cfg(feature = "async")]
    #[inline]
    fn rx_waker() -> &'static embassy_sync::waitqueue::AtomicWaker {
        &crate::sercom::async_api::RX_WAKERS[Self::NUM]
    }

    /// Get a reference to this [`Sercom`]'s associated TX Waker
    #[cfg(feature = "async")]
    #[inline]
    fn tx_waker() -> &'static embassy_sync::waitqueue::AtomicWaker {
        &crate::sercom::async_api::TX_WAKERS[Self::NUM]
    }
}

macro_rules! sercom {
    ( $apbmask:ident, $N:expr) => {
        paste::paste! {
            // use pac::$pac_type;
            /// Type alias for the corresponding SERCOM instance
            pub type [< Sercom $N >] = $crate::pac::[< Sercom $N >];
            impl Sealed for [< Sercom $N >] {}
            impl Sercom for [< Sercom $N >] {
                const NUM: usize = $N;

                #[cfg(feature = "dma")]
                const DMA_RX_TRIGGER: TriggerSource = TriggerSource::[< Sercom $N Rx >];

                #[cfg(feature = "dma")]
                const DMA_TX_TRIGGER: TriggerSource = TriggerSource::[< Sercom $N Tx >];

                #[cfg(feature = "async")]
                #[hal_cfg(any("sercom0-d11", "sercom0-d21"))]
                type Interrupt = $crate::async_hal::interrupts::[< SERCOM $N >];

                #[cfg(feature = "async")]
                #[hal_cfg("sercom0-d5x")]
                type Interrupt = $crate::async_hal::interrupts::[< SERCOM $N >];

                #[inline]
                fn enable_apb_clock(&mut self, ctrl: &ApbClkCtrl) {
                    ctrl.$apbmask().modify(|_, w| w.[< sercom $N _>]().set_bit());
                }

                #[inline]
                fn reg_block(peripherals: &mut Peripherals) -> &crate::pac::sercom0::RegisterBlock {
                    &*peripherals.[< sercom $N >]
                }
            }


        }
    };
}

// d11 and d21 families
#[hal_cfg(any("sercom0-d11", "sercom0-d21"))]
sercom!(apbcmask, 0);

#[hal_cfg(any("sercom1-d11", "sercom1-d21"))]
sercom!(apbcmask, 1);

#[hal_cfg(any("sercom2-d11", "sercom2-d21"))]
sercom!(apbcmask, 2);

#[hal_cfg("sercom3-d21")]
sercom!(apbcmask, 3);

#[hal_cfg("sercom4-d21")]
sercom!(apbcmask, 4);

#[hal_cfg("sercom5-d21")]
sercom!(apbcmask, 5);

// d5x family
#[hal_cfg("sercom0-d5x")]
sercom!(apbamask, 0);

#[hal_cfg("sercom1-d5x")]
sercom!(apbamask, 1);

#[hal_cfg("sercom2-d5x")]
sercom!(apbbmask, 2);

#[hal_cfg("sercom3-d5x")]
sercom!(apbbmask, 3);

#[hal_cfg("sercom4-d5x")]
sercom!(apbdmask, 4);

#[hal_cfg("sercom5-d5x")]
sercom!(apbdmask, 5);

#[hal_cfg("sercom6-d5x")]
sercom!(apbdmask, 6);

#[hal_cfg("sercom7-d5x")]
sercom!(apbdmask, 7);

// Reserve space for the max number of SERCOM peripherals based on chip type,
// even though some wakers may not be used on some chips if they actually don't
// exist on variant's hardware
#[hal_cfg("sercom0-d11")]
#[cfg(feature = "async")]
const NUM_SERCOM: usize = 3;

#[hal_cfg("sercom0-d21")]
#[cfg(feature = "async")]
const NUM_SERCOM: usize = 6;

#[hal_cfg("sercom0-d5x")]
#[cfg(feature = "async")]
const NUM_SERCOM: usize = 8;

#[cfg(feature = "async")]
pub(super) mod async_api {
    use embassy_sync::waitqueue::AtomicWaker;

    #[allow(clippy::declare_interior_mutable_const)]
    const NEW_WAKER: AtomicWaker = AtomicWaker::new();
    /// Waker for a RX event. By convention, if a SERCOM has only one type of
    /// event (ie, I2C), this the waker to be used.
    pub(super) static RX_WAKERS: [AtomicWaker; super::NUM_SERCOM] = [NEW_WAKER; super::NUM_SERCOM];
    /// Waker for a TX event.
    pub(super) static TX_WAKERS: [AtomicWaker; super::NUM_SERCOM] = [NEW_WAKER; super::NUM_SERCOM];
}