| 1 | /// Types for the peripheral singletons. |
| 2 | #[macro_export ] |
| 3 | macro_rules! peripherals_definition { |
| 4 | ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { |
| 5 | /// Types for the peripheral singletons. |
| 6 | pub mod peripherals { |
| 7 | $( |
| 8 | $(#[$cfg])? |
| 9 | #[allow(non_camel_case_types)] |
| 10 | #[doc = concat!(stringify!($name), " peripheral" )] |
| 11 | pub struct $name { _private: () } |
| 12 | |
| 13 | $(#[$cfg])? |
| 14 | impl $name { |
| 15 | /// Unsafely create an instance of this peripheral out of thin air. |
| 16 | /// |
| 17 | /// # Safety |
| 18 | /// |
| 19 | /// You must ensure that you're only using one instance of this type at a time. |
| 20 | #[inline] |
| 21 | pub unsafe fn steal() -> Self { |
| 22 | Self{ _private: ()} |
| 23 | } |
| 24 | } |
| 25 | |
| 26 | $(#[$cfg])? |
| 27 | $crate::impl_peripheral!($name); |
| 28 | )* |
| 29 | } |
| 30 | }; |
| 31 | } |
| 32 | |
| 33 | /// Define the peripherals struct. |
| 34 | #[macro_export ] |
| 35 | macro_rules! peripherals_struct { |
| 36 | ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { |
| 37 | /// Struct containing all the peripheral singletons. |
| 38 | /// |
| 39 | /// To obtain the peripherals, you must initialize the HAL, by calling [`crate::init`]. |
| 40 | #[allow(non_snake_case)] |
| 41 | pub struct Peripherals { |
| 42 | $( |
| 43 | #[doc = concat!(stringify!($name), " peripheral" )] |
| 44 | $(#[$cfg])? |
| 45 | pub $name: peripherals::$name, |
| 46 | )* |
| 47 | } |
| 48 | |
| 49 | impl Peripherals { |
| 50 | ///Returns all the peripherals *once* |
| 51 | #[inline] |
| 52 | pub(crate) fn take() -> Self { |
| 53 | critical_section::with(Self::take_with_cs) |
| 54 | } |
| 55 | |
| 56 | ///Returns all the peripherals *once* |
| 57 | #[inline] |
| 58 | pub(crate) fn take_with_cs(_cs: critical_section::CriticalSection) -> Self { |
| 59 | #[no_mangle] |
| 60 | static mut _EMBASSY_DEVICE_PERIPHERALS: bool = false; |
| 61 | |
| 62 | // safety: OK because we're inside a CS. |
| 63 | unsafe { |
| 64 | if _EMBASSY_DEVICE_PERIPHERALS { |
| 65 | panic!("init called more than once!" ) |
| 66 | } |
| 67 | _EMBASSY_DEVICE_PERIPHERALS = true; |
| 68 | Self::steal() |
| 69 | } |
| 70 | } |
| 71 | } |
| 72 | |
| 73 | impl Peripherals { |
| 74 | /// Unsafely create an instance of this peripheral out of thin air. |
| 75 | /// |
| 76 | /// # Safety |
| 77 | /// |
| 78 | /// You must ensure that you're only using one instance of this type at a time. |
| 79 | #[inline] |
| 80 | pub unsafe fn steal() -> Self { |
| 81 | Self { |
| 82 | $( |
| 83 | $(#[$cfg])? |
| 84 | $name: peripherals::$name::steal(), |
| 85 | )* |
| 86 | } |
| 87 | } |
| 88 | } |
| 89 | }; |
| 90 | } |
| 91 | |
| 92 | /// Defining peripheral type. |
| 93 | #[macro_export ] |
| 94 | macro_rules! peripherals { |
| 95 | ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { |
| 96 | $crate::peripherals_definition!( |
| 97 | $( |
| 98 | $(#[$cfg])? |
| 99 | $name, |
| 100 | )* |
| 101 | ); |
| 102 | $crate::peripherals_struct!( |
| 103 | $( |
| 104 | $(#[$cfg])? |
| 105 | $name, |
| 106 | )* |
| 107 | ); |
| 108 | }; |
| 109 | } |
| 110 | |
| 111 | /// Convenience converting into reference. |
| 112 | #[macro_export ] |
| 113 | macro_rules! into_ref { |
| 114 | ($($name:ident),*) => { |
| 115 | $( |
| 116 | let mut $name = $name.into_ref(); |
| 117 | )* |
| 118 | } |
| 119 | } |
| 120 | |
| 121 | /// Implement the peripheral trait. |
| 122 | #[macro_export ] |
| 123 | macro_rules! impl_peripheral { |
| 124 | ($type:ident) => { |
| 125 | impl $crate::Peripheral for $type { |
| 126 | type P = $type; |
| 127 | |
| 128 | #[inline] |
| 129 | unsafe fn clone_unchecked(&self) -> Self::P { |
| 130 | #[allow(clippy::needless_update)] |
| 131 | $type { ..*self } |
| 132 | } |
| 133 | } |
| 134 | }; |
| 135 | } |
| 136 | |