#![macro_use] use crate::peripherals; use crate::time::Hertz; use core::mem::MaybeUninit; #[cfg_attr(any(rcc_f0, rcc_f0x0), path = "f0.rs")] #[cfg_attr(rcc_f1, path = "f1.rs")] #[cfg_attr(rcc_f3, path = "f3.rs")] #[cfg_attr(any(rcc_f4, rcc_f410), path = "f4.rs")] #[cfg_attr(rcc_f7, path = "f7.rs")] #[cfg_attr(rcc_g0, path = "g0.rs")] #[cfg_attr(rcc_g4, path = "g4.rs")] #[cfg_attr(any(rcc_h7, rcc_h7ab), path = "h7.rs")] #[cfg_attr(rcc_l0, path = "l0.rs")] #[cfg_attr(rcc_l1, path = "l1.rs")] #[cfg_attr(rcc_l4, path = "l4.rs")] #[cfg_attr(rcc_u5, path = "u5.rs")] #[cfg_attr(rcc_wb, path = "wb.rs")] #[cfg_attr(rcc_wl5, path = "wl5.rs")] mod _version; pub use _version::*; #[derive(Clone, Copy)] pub struct Clocks { pub sys: Hertz, #[cfg(rcc_g0)] pub apb: Hertz, #[cfg(rcc_g0)] pub apb_tim: Hertz, #[cfg(not(rcc_g0))] pub apb1: Hertz, #[cfg(not(rcc_g0))] pub apb1_tim: Hertz, #[cfg(not(rcc_g0))] pub apb2: Hertz, #[cfg(not(rcc_g0))] pub apb2_tim: Hertz, #[cfg(any(rcc_wl5, rcc_u5))] pub apb3: Hertz, #[cfg(any(rcc_l0, rcc_l1, rcc_f0, rcc_f1, rcc_f3, rcc_f0x0, rcc_g0))] pub ahb: Hertz, #[cfg(any(rcc_l4, rcc_f4, rcc_f7, rcc_h7, rcc_g4, rcc_u5, rcc_wb, rcc_wl5))] pub ahb1: Hertz, #[cfg(any(rcc_l4, rcc_f4, rcc_f7, rcc_h7, rcc_g4, rcc_u5, rcc_wb, rcc_wl5))] pub ahb2: Hertz, #[cfg(any(rcc_l4, rcc_f4, rcc_f7, rcc_h7, rcc_u5, rcc_wb, rcc_wl5))] pub ahb3: Hertz, #[cfg(any(rcc_h7))] pub ahb4: Hertz, #[cfg(any(rcc_h7))] pub apb4: Hertz, #[cfg(any(rcc_f4, rcc_f7))] pub pll48: Option, #[cfg(rcc_f1)] pub adc: Hertz, } /// Frozen clock frequencies /// /// The existence of this value indicates that the clock configuration can no longer be changed static mut CLOCK_FREQS: MaybeUninit = MaybeUninit::uninit(); /// Sets the clock frequencies /// /// Safety: Sets a mutable global. pub(crate) unsafe fn set_freqs(freqs: Clocks) { CLOCK_FREQS.as_mut_ptr().write(freqs); } /// Safety: Reads a mutable global. pub(crate) unsafe fn get_freqs() -> &'static Clocks { &*CLOCK_FREQS.as_ptr() } #[cfg(feature = "unstable-pac")] pub mod low_level { pub use super::sealed::*; } pub(crate) mod sealed { pub trait RccPeripheral { fn frequency() -> crate::time::Hertz; fn reset(); fn enable(); fn disable(); } } pub trait RccPeripheral: sealed::RccPeripheral + 'static {} crate::pac::peripheral_rcc!( ($inst:ident, gpio, GPIO, $clk:ident, $en:tt, $rst:tt) => {}; ($inst:ident, $module:ident, $block:ident, $clk:ident, ($en_reg:ident, $en_field:ident, $en_set_field:ident), ($rst_reg:ident, $rst_field:ident, $rst_set_field:ident)) => { impl sealed::RccPeripheral for peripherals::$inst { fn frequency() -> crate::time::Hertz { critical_section::with(|_| { unsafe { get_freqs().$clk } }) } fn enable() { critical_section::with(|_| { unsafe { crate::pac::RCC.$en_reg().modify(|w| w.$en_set_field(true)); } }) } fn disable() { critical_section::with(|_| { unsafe { crate::pac::RCC.$en_reg().modify(|w| w.$en_set_field(false)); } }) } fn reset() { critical_section::with(|_| { unsafe { crate::pac::RCC.$rst_reg().modify(|w| w.$rst_set_field(true)); crate::pac::RCC.$rst_reg().modify(|w| w.$rst_set_field(false)); } }) } } impl RccPeripheral for peripherals::$inst {} }; ($inst:ident, $module:ident, $block:ident, $clk:ident, ($en_reg:ident, $en_field:ident, $en_set_field:ident), _) => { impl sealed::RccPeripheral for peripherals::$inst { fn frequency() -> crate::time::Hertz { critical_section::with(|_| { unsafe { get_freqs().$clk } }) } fn enable() { critical_section::with(|_| { unsafe { crate::pac::RCC.$en_reg().modify(|w| w.$en_set_field(true)); } }) } fn disable() { critical_section::with(|_| { unsafe { crate::pac::RCC.$en_reg().modify(|w| w.$en_set_field(false)); } }) } fn reset() {} } impl RccPeripheral for peripherals::$inst {} }; );