2023-05-29 21:49:43 +02:00
|
|
|
#![cfg_attr(not(test), no_std)]
|
2023-11-29 17:23:48 +01:00
|
|
|
#![allow(async_fn_in_trait)]
|
2021-03-19 21:26:20 +01:00
|
|
|
|
2023-07-28 16:59:13 +02:00
|
|
|
//! ## Feature flags
|
|
|
|
#![doc = document_features::document_features!(feature_label = r#"<span class="stab portability"><code>{feature}</code></span>"#)]
|
|
|
|
|
2021-04-23 23:47:34 +02:00
|
|
|
// This must go FIRST so that all the other modules see its macros.
|
2023-08-09 18:33:20 +02:00
|
|
|
mod fmt;
|
2022-03-04 17:42:38 +01:00
|
|
|
include!(concat!(env!("OUT_DIR"), "/_macros.rs"));
|
2021-03-19 21:26:20 +01:00
|
|
|
|
2021-05-25 04:17:24 +02:00
|
|
|
// Utilities
|
|
|
|
pub mod time;
|
2022-02-10 21:38:03 +01:00
|
|
|
mod traits;
|
2021-05-25 04:17:24 +02:00
|
|
|
|
|
|
|
// Always-present hardware
|
2021-07-15 05:42:06 +02:00
|
|
|
pub mod dma;
|
2021-04-06 01:31:29 +02:00
|
|
|
pub mod gpio;
|
2021-05-25 04:17:24 +02:00
|
|
|
pub mod rcc;
|
2021-08-04 12:05:22 +02:00
|
|
|
#[cfg(feature = "_time-driver")]
|
2021-08-03 22:08:13 +02:00
|
|
|
mod time_driver;
|
2021-12-08 17:36:40 +01:00
|
|
|
pub mod timer;
|
2021-05-25 04:17:24 +02:00
|
|
|
|
|
|
|
// Sometimes-present hardware
|
2021-07-08 20:55:27 +02:00
|
|
|
|
2021-06-10 21:33:43 +02:00
|
|
|
#[cfg(adc)]
|
|
|
|
pub mod adc;
|
2021-08-06 11:59:16 +02:00
|
|
|
#[cfg(can)]
|
|
|
|
pub mod can;
|
2023-07-29 00:07:08 +02:00
|
|
|
#[cfg(crc)]
|
|
|
|
pub mod crc;
|
2021-05-28 19:31:40 +02:00
|
|
|
#[cfg(dac)]
|
|
|
|
pub mod dac;
|
2021-11-10 14:47:16 +01:00
|
|
|
#[cfg(dcmi)]
|
|
|
|
pub mod dcmi;
|
2022-05-04 20:48:37 +02:00
|
|
|
#[cfg(eth)]
|
2021-06-07 07:30:38 +02:00
|
|
|
pub mod eth;
|
2022-01-12 12:43:24 +01:00
|
|
|
#[cfg(feature = "exti")]
|
2021-06-25 00:36:42 +02:00
|
|
|
pub mod exti;
|
2023-07-29 00:07:08 +02:00
|
|
|
pub mod flash;
|
2022-02-08 14:32:18 +01:00
|
|
|
#[cfg(fmc)]
|
|
|
|
pub mod fmc;
|
2023-07-31 15:42:03 +02:00
|
|
|
#[cfg(hrtim)]
|
2023-07-29 00:07:08 +02:00
|
|
|
pub mod hrtim;
|
2021-05-25 04:17:24 +02:00
|
|
|
#[cfg(i2c)]
|
2021-05-24 17:41:37 +02:00
|
|
|
pub mod i2c;
|
2023-05-04 01:17:57 +02:00
|
|
|
#[cfg(all(spi_v1, rcc_f4))]
|
|
|
|
pub mod i2s;
|
2023-06-12 13:27:51 +02:00
|
|
|
#[cfg(stm32wb)]
|
|
|
|
pub mod ipcc;
|
2023-08-23 00:00:00 +02:00
|
|
|
#[cfg(feature = "low-power")]
|
|
|
|
pub mod low_power;
|
2023-10-03 23:34:13 +02:00
|
|
|
#[cfg(opamp)]
|
|
|
|
pub mod opamp;
|
2023-03-22 08:44:58 +01:00
|
|
|
#[cfg(quadspi)]
|
|
|
|
pub mod qspi;
|
2021-05-25 04:17:24 +02:00
|
|
|
#[cfg(rng)]
|
2021-04-26 20:11:46 +02:00
|
|
|
pub mod rng;
|
2023-04-18 02:07:58 +02:00
|
|
|
#[cfg(all(rtc, not(rtc_v1)))]
|
2022-09-29 07:49:32 +02:00
|
|
|
pub mod rtc;
|
2023-09-30 03:57:59 +02:00
|
|
|
#[cfg(sai)]
|
|
|
|
pub mod sai;
|
2021-05-25 04:17:24 +02:00
|
|
|
#[cfg(sdmmc)]
|
2021-05-17 02:04:51 +02:00
|
|
|
pub mod sdmmc;
|
2021-05-25 04:17:24 +02:00
|
|
|
#[cfg(spi)]
|
2021-05-10 21:21:57 +02:00
|
|
|
pub mod spi;
|
2023-12-03 03:05:55 +01:00
|
|
|
#[cfg(uid)]
|
|
|
|
pub mod uid;
|
2021-05-25 04:17:24 +02:00
|
|
|
#[cfg(usart)]
|
2021-05-06 03:59:16 +02:00
|
|
|
pub mod usart;
|
2023-05-19 15:20:37 +02:00
|
|
|
#[cfg(usb)]
|
2022-05-30 00:36:30 +02:00
|
|
|
pub mod usb;
|
2023-01-11 17:51:30 +01:00
|
|
|
#[cfg(otg)]
|
2022-02-09 01:02:23 +01:00
|
|
|
pub mod usb_otg;
|
2022-06-10 16:10:54 +02:00
|
|
|
#[cfg(iwdg)]
|
|
|
|
pub mod wdg;
|
|
|
|
|
2021-05-25 04:17:24 +02:00
|
|
|
// This must go last, so that it sees all the impl_foo! macros defined earlier.
|
2022-03-04 17:42:38 +01:00
|
|
|
pub(crate) mod _generated {
|
2021-05-25 04:17:24 +02:00
|
|
|
#![allow(dead_code)]
|
|
|
|
#![allow(unused_imports)]
|
|
|
|
#![allow(non_snake_case)]
|
2021-05-10 21:21:57 +02:00
|
|
|
|
2022-03-04 17:42:38 +01:00
|
|
|
include!(concat!(env!("OUT_DIR"), "/_generated.rs"));
|
2021-05-25 04:17:24 +02:00
|
|
|
}
|
2022-06-11 05:08:57 +02:00
|
|
|
|
2023-06-08 16:08:40 +02:00
|
|
|
pub use crate::_generated::interrupt;
|
2023-05-25 00:29:56 +02:00
|
|
|
|
2023-06-08 16:08:40 +02:00
|
|
|
/// Macro to bind interrupts to handlers.
|
|
|
|
///
|
|
|
|
/// This defines the right interrupt handlers, and creates a unit struct (like `struct Irqs;`)
|
|
|
|
/// and implements the right [`Binding`]s for it. You can pass this struct to drivers to
|
|
|
|
/// prove at compile-time that the right interrupts have been bound.
|
2023-12-08 21:26:28 +01:00
|
|
|
///
|
|
|
|
/// Example of how to bind one interrupt:
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// use embassy_stm32::{bind_interrupts, usb_otg, peripherals};
|
|
|
|
///
|
|
|
|
/// bind_interrupts!(struct Irqs {
|
|
|
|
/// OTG_FS => usb_otg::InterruptHandler<peripherals::USB_OTG_FS>;
|
|
|
|
/// });
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// Example of how to bind multiple interrupts, and multiple handlers to each interrupt, in a single macro invocation:
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// use embassy_stm32::{bind_interrupts, i2c, peripherals};
|
|
|
|
///
|
|
|
|
/// bind_interrupts!(struct Irqs {
|
|
|
|
/// I2C1 => i2c::EventInterruptHandler<peripherals::I2C1>, i2c::ErrorInterruptHandler<peripherals::I2C1>;
|
|
|
|
/// I2C2_3 => i2c::EventInterruptHandler<peripherals::I2C2>, i2c::ErrorInterruptHandler<peripherals::I2C2>,
|
|
|
|
/// i2c::EventInterruptHandler<peripherals::I2C3>, i2c::ErrorInterruptHandler<peripherals::I2C3>;
|
|
|
|
/// });
|
|
|
|
/// ```
|
|
|
|
|
2023-07-28 13:23:22 +02:00
|
|
|
// developer note: this macro can't be in `embassy-hal-internal` due to the use of `$crate`.
|
2023-06-08 16:08:40 +02:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! bind_interrupts {
|
|
|
|
($vis:vis struct $name:ident { $($irq:ident => $($handler:ty),*;)* }) => {
|
2023-09-25 22:34:41 +02:00
|
|
|
#[derive(Copy, Clone)]
|
2023-06-08 16:08:40 +02:00
|
|
|
$vis struct $name;
|
2023-05-25 00:29:56 +02:00
|
|
|
|
2023-06-08 16:08:40 +02:00
|
|
|
$(
|
|
|
|
#[allow(non_snake_case)]
|
|
|
|
#[no_mangle]
|
|
|
|
unsafe extern "C" fn $irq() {
|
2023-05-25 00:29:56 +02:00
|
|
|
$(
|
2023-06-08 16:08:40 +02:00
|
|
|
<$handler as $crate::interrupt::typelevel::Handler<$crate::interrupt::typelevel::$irq>>::on_interrupt();
|
2023-05-25 00:29:56 +02:00
|
|
|
)*
|
2023-06-08 16:08:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$(
|
|
|
|
unsafe impl $crate::interrupt::typelevel::Binding<$crate::interrupt::typelevel::$irq, $handler> for $name {}
|
2023-05-25 00:29:56 +02:00
|
|
|
)*
|
2023-06-08 16:08:40 +02:00
|
|
|
)*
|
|
|
|
};
|
2023-05-25 00:29:56 +02:00
|
|
|
}
|
|
|
|
|
2022-06-11 05:08:57 +02:00
|
|
|
// Reexports
|
2022-03-04 17:42:38 +01:00
|
|
|
pub use _generated::{peripherals, Peripherals};
|
2023-07-28 13:23:22 +02:00
|
|
|
pub use embassy_hal_internal::{into_ref, Peripheral, PeripheralRef};
|
2022-06-11 05:08:57 +02:00
|
|
|
#[cfg(feature = "unstable-pac")]
|
|
|
|
pub use stm32_metapac as pac;
|
|
|
|
#[cfg(not(feature = "unstable-pac"))]
|
|
|
|
pub(crate) use stm32_metapac as pac;
|
2021-05-01 03:07:17 +02:00
|
|
|
|
2023-06-09 16:14:13 +02:00
|
|
|
use crate::interrupt::Priority;
|
2023-06-08 18:00:19 +02:00
|
|
|
#[cfg(feature = "rt")]
|
2023-06-08 16:39:05 +02:00
|
|
|
pub use crate::pac::NVIC_PRIO_BITS;
|
2023-09-24 23:54:32 +02:00
|
|
|
use crate::rcc::sealed::RccPeripheral;
|
2023-06-08 16:39:05 +02:00
|
|
|
|
2021-05-01 03:07:17 +02:00
|
|
|
#[non_exhaustive]
|
|
|
|
pub struct Config {
|
2021-08-04 17:32:39 +02:00
|
|
|
pub rcc: rcc::Config,
|
2022-02-25 01:16:23 +01:00
|
|
|
#[cfg(dbgmcu)]
|
2021-08-19 23:32:22 +02:00
|
|
|
pub enable_debug_during_sleep: bool,
|
2022-11-23 10:11:19 +01:00
|
|
|
#[cfg(bdma)]
|
|
|
|
pub bdma_interrupt_priority: Priority,
|
|
|
|
#[cfg(dma)]
|
|
|
|
pub dma_interrupt_priority: Priority,
|
2023-04-17 00:04:54 +02:00
|
|
|
#[cfg(gpdma)]
|
|
|
|
pub gpdma_interrupt_priority: Priority,
|
2021-05-25 17:09:01 +02:00
|
|
|
}
|
|
|
|
|
2021-05-01 03:07:17 +02:00
|
|
|
impl Default for Config {
|
|
|
|
fn default() -> Self {
|
2021-05-25 13:30:42 +02:00
|
|
|
Self {
|
|
|
|
rcc: Default::default(),
|
2022-02-25 01:16:23 +01:00
|
|
|
#[cfg(dbgmcu)]
|
2021-08-19 23:32:22 +02:00
|
|
|
enable_debug_during_sleep: true,
|
2022-11-23 10:11:19 +01:00
|
|
|
#[cfg(bdma)]
|
|
|
|
bdma_interrupt_priority: Priority::P0,
|
|
|
|
#[cfg(dma)]
|
|
|
|
dma_interrupt_priority: Priority::P0,
|
2023-04-17 00:04:54 +02:00
|
|
|
#[cfg(gpdma)]
|
|
|
|
gpdma_interrupt_priority: Priority::P0,
|
2021-05-25 13:30:42 +02:00
|
|
|
}
|
2021-05-01 03:07:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-19 16:18:24 +01:00
|
|
|
/// Initialize embassy.
|
2021-05-25 13:30:42 +02:00
|
|
|
pub fn init(config: Config) -> Peripherals {
|
2023-10-12 00:34:47 +02:00
|
|
|
critical_section::with(|cs| {
|
|
|
|
let p = Peripherals::take_with_cs(cs);
|
|
|
|
|
|
|
|
#[cfg(dbgmcu)]
|
|
|
|
if config.enable_debug_during_sleep {
|
|
|
|
crate::pac::DBGMCU.cr().modify(|cr| {
|
|
|
|
#[cfg(any(dbgmcu_f0, dbgmcu_c0, dbgmcu_g0, dbgmcu_u5, dbgmcu_wba))]
|
|
|
|
{
|
|
|
|
cr.set_dbg_stop(true);
|
|
|
|
cr.set_dbg_standby(true);
|
|
|
|
}
|
|
|
|
#[cfg(any(
|
|
|
|
dbgmcu_f1, dbgmcu_f2, dbgmcu_f3, dbgmcu_f4, dbgmcu_f7, dbgmcu_g4, dbgmcu_f7, dbgmcu_l0, dbgmcu_l1,
|
|
|
|
dbgmcu_l4, dbgmcu_wb, dbgmcu_wl
|
|
|
|
))]
|
|
|
|
{
|
|
|
|
cr.set_dbg_sleep(true);
|
|
|
|
cr.set_dbg_stop(true);
|
|
|
|
cr.set_dbg_standby(true);
|
|
|
|
}
|
|
|
|
#[cfg(dbgmcu_h7)]
|
|
|
|
{
|
|
|
|
cr.set_d1dbgcken(true);
|
|
|
|
cr.set_d3dbgcken(true);
|
|
|
|
cr.set_dbgsleep_d1(true);
|
|
|
|
cr.set_dbgstby_d1(true);
|
|
|
|
cr.set_dbgstop_d1(true);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2021-08-03 22:08:13 +02:00
|
|
|
|
2023-10-12 00:34:47 +02:00
|
|
|
#[cfg(not(any(stm32f1, stm32wb, stm32wl)))]
|
|
|
|
peripherals::SYSCFG::enable_and_reset_with_cs(cs);
|
|
|
|
#[cfg(not(any(stm32h5, stm32h7, stm32wb, stm32wl)))]
|
|
|
|
peripherals::PWR::enable_and_reset_with_cs(cs);
|
|
|
|
#[cfg(not(any(stm32f2, stm32f4, stm32f7, stm32l0, stm32h5, stm32h7)))]
|
|
|
|
peripherals::FLASH::enable_and_reset_with_cs(cs);
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
#[cfg(feature = "_split-pins-enabled")]
|
|
|
|
crate::pac::SYSCFG.pmcr().modify(|pmcr| {
|
|
|
|
#[cfg(feature = "split-pa0")]
|
|
|
|
pmcr.set_pa0so(true);
|
|
|
|
#[cfg(feature = "split-pa1")]
|
|
|
|
pmcr.set_pa1so(true);
|
|
|
|
#[cfg(feature = "split-pc2")]
|
|
|
|
pmcr.set_pc2so(true);
|
|
|
|
#[cfg(feature = "split-pc3")]
|
|
|
|
pmcr.set_pc3so(true);
|
|
|
|
});
|
2023-08-25 02:29:11 +02:00
|
|
|
|
2023-10-12 00:34:47 +02:00
|
|
|
gpio::init(cs);
|
|
|
|
dma::init(
|
|
|
|
cs,
|
|
|
|
#[cfg(bdma)]
|
|
|
|
config.bdma_interrupt_priority,
|
|
|
|
#[cfg(dma)]
|
|
|
|
config.dma_interrupt_priority,
|
|
|
|
#[cfg(gpdma)]
|
|
|
|
config.gpdma_interrupt_priority,
|
|
|
|
);
|
|
|
|
#[cfg(feature = "exti")]
|
|
|
|
exti::init(cs);
|
|
|
|
|
|
|
|
rcc::init(config.rcc);
|
|
|
|
|
|
|
|
// must be after rcc init
|
|
|
|
#[cfg(feature = "_time-driver")]
|
|
|
|
time_driver::init(cs);
|
|
|
|
|
|
|
|
#[cfg(feature = "low-power")]
|
2023-10-26 02:07:31 +02:00
|
|
|
{
|
2023-11-04 19:49:54 +01:00
|
|
|
crate::rcc::REFCOUNT_STOP2 = 0;
|
|
|
|
crate::rcc::REFCOUNT_STOP1 = 0;
|
|
|
|
}
|
2023-08-25 02:29:11 +02:00
|
|
|
}
|
2021-05-01 03:07:17 +02:00
|
|
|
|
2023-10-12 00:34:47 +02:00
|
|
|
p
|
|
|
|
})
|
2021-05-01 03:07:17 +02:00
|
|
|
}
|