diff --git a/esp-hal/src/lib.rs b/esp-hal/src/lib.rs index 755f02b1e7b..ceb7c8ef576 100644 --- a/esp-hal/src/lib.rs +++ b/esp-hal/src/lib.rs @@ -224,6 +224,7 @@ pub mod rtc_cntl; pub mod sha; #[cfg(any(spi0, spi1, spi2, spi3))] pub mod spi; +#[doc(hidden)] #[cfg(any(dport, hp_sys, pcr, system))] pub mod system; pub mod time; @@ -255,6 +256,9 @@ pub mod trapframe { pub use xtensa_lx_rt::exception::Context as TrapFrame; } +#[cfg(phy)] +pub mod radio_clock_ctrl; + // The `soc` module contains chip-specific implementation details and should not // be directly exposed. mod soc; diff --git a/esp-hal/src/radio_clock_ctrl.rs b/esp-hal/src/radio_clock_ctrl.rs new file mode 100644 index 00000000000..ebc20a4e6c8 --- /dev/null +++ b/esp-hal/src/radio_clock_ctrl.rs @@ -0,0 +1,175 @@ +//! Control over the radio clocs. +use core::cell::RefCell; + +use critical_section::Mutex; +use portable_atomic::{AtomicBool, Ordering}; + +use crate::peripherals::RADIO_CLK; + +/// Enumeration of the available radio peripherals for this chip. +#[derive(Clone, Copy, Debug, PartialEq, Eq)] +#[cfg_attr(feature = "defmt", derive(defmt::Format))] +pub enum RadioPeripherals { + /// Represents the PHY (Physical Layer) peripheral. + Phy, + /// Represents the Bluetooth peripheral. + #[cfg(bt)] + Bt, + /// Represents the WiFi peripheral. + #[cfg(wifi)] + Wifi, + /// Represents the IEEE 802.15.4 peripheral. + #[cfg(ieee802154)] + Ieee802154, +} + +/// Control the radio peripheral clocks +/// +/// NOTE: We don't really want the user messing with the radio clocks, so this +/// is hidden. In the future, we could just make it private. +#[doc(hidden)] +pub trait RadioClockController { + /// Enable the peripheral + fn enable(&mut self, peripheral: RadioPeripherals); + + /// Disable the peripheral + fn disable(&mut self, peripheral: RadioPeripherals); + + /// Reset the MAC + fn reset_mac(&mut self); + + /// Do any common initial initialization needed + fn init_clocks(&mut self); + + /// Initialize BLE RTC clocks + fn ble_rtc_clk_init(&mut self); + + /// Reset the Resolvable Private Address (RPA). + fn reset_rpa(&mut self); +} + +static PHY_CLOCK_ENABLED: AtomicBool = AtomicBool::new(false); +/// The PHY clock couldn't be disabled, because some modem clocks are still +/// active. +pub struct ModemClocksEnabledError; + +/// This struct allows shared access to the radio clocks. +pub struct SharedRadioClockControl { + radio_clock: Mutex>, + + #[cfg(bt)] + bt_clock_enabled: AtomicBool, + #[cfg(wifi)] + wifi_clock_enabled: AtomicBool, + #[cfg(ieee802154)] + ieee802154_clock_enabled: AtomicBool, +} +impl SharedRadioClockControl { + /// Create a new [SharedRadioClockControl]. + pub const fn new(radio_clock: RADIO_CLK) -> Self { + Self { + radio_clock: Mutex::new(RefCell::new(radio_clock)), + #[cfg(bt)] + bt_clock_enabled: AtomicBool::new(false), + #[cfg(wifi)] + wifi_clock_enabled: AtomicBool::new(false), + #[cfg(ieee802154)] + ieee802154_clock_enabled: AtomicBool::new(false), + } + } + /// Check if any modem clocks are enabled. + pub fn any_modem_clocks_enabled(&self) -> bool { + let mut any_clocks_enabled = false; + + #[cfg(bt)] + { + any_clocks_enabled |= self.bt_clock_enabled.load(Ordering::Relaxed); + } + #[cfg(wifi)] + { + any_clocks_enabled |= self.wifi_clock_enabled.load(Ordering::Relaxed); + } + #[cfg(ieee802154)] + { + any_clocks_enabled |= self.ieee802154_clock_enabled.load(Ordering::Relaxed); + } + + any_clocks_enabled + } + /// Enable or disable the clock, without changing the PHY clock status. + fn set_clock_status_raw(&self, enabled: bool, radio_peripheral: RadioPeripherals) { + critical_section::with(|cs| { + let mut radio_clock = self.radio_clock.borrow_ref_mut(cs); + if enabled { + radio_clock.enable(radio_peripheral) + } else { + radio_clock.disable(radio_peripheral) + } + }) + } + /// Enable or disable the PHY clock. + /// + /// If the PHY clock state was successfully changed, or the current state + /// already matched the specified one `Ok(())` is returned. + /// If the PHY clock is being disabled, but other modem clocks are still + /// active `Err(ModemClocksEnabledError)` is returned. + fn set_phy_clock_status(&self, enabled: bool) -> Result<(), ModemClocksEnabledError> { + // We shouldn't disable the PHY clock, if other clocks are still active. + if !enabled && self.any_modem_clocks_enabled() { + return Err(ModemClocksEnabledError); + } + // If the current clock status is already the same as the user provided one, we + // return. Otherwise we store the new value. + if PHY_CLOCK_ENABLED + .compare_exchange(!enabled, enabled, Ordering::Relaxed, Ordering::Relaxed) + .is_err() + { + return Ok(()); + } + self.set_clock_status_raw(enabled, RadioPeripherals::Phy); + Ok(()) + } + /// Set the status of the modem clock, enabling or disabling the PHY if + /// needed. + fn set_modem_clock_status_internal(&self, enabled: bool, radio_peripheral: RadioPeripherals) { + // Depending on if we're enabling or disabling, we need to set the PHY and modem + // clock status in a different order. + if enabled { + let _ = self.set_phy_clock_status(true); + self.set_clock_status_raw(true, radio_peripheral); + } else { + self.set_clock_status_raw(false, radio_peripheral); + let _ = self.set_phy_clock_status(false); + } + } + /// Enable or disable a modem clock. + /// + /// If required, this will also enabled or disable the PHY clock. + /// # Panics + /// This panics, if [RadioPeripherals::Phy] is passed in, since the PHY + /// clock is controlled internally. + pub fn set_modem_clock_status(&self, enabled: bool, radio_peripheral: RadioPeripherals) { + // We first assert, that the clock we're touching isn't the PHY clock, since + // that's special. + assert_ne!(radio_peripheral, RadioPeripherals::Phy); + + // We obtain a reference to the AtomicBool tracking the state of the clock. + let modem_clock_enabled = match radio_peripheral { + #[cfg(bt)] + RadioPeripherals::Bt => &self.bt_clock_enabled, + #[cfg(wifi)] + RadioPeripherals::Wifi => &self.wifi_clock_enabled, + #[cfg(ieee802154)] + RadioPeripherals::Ieee802154 => &self.ieee802154_clock_enabled, + _ => unreachable!(), + }; + // If the states already match, we return. Otherwise the new state is stored. + if modem_clock_enabled + .compare_exchange(!enabled, enabled, Ordering::Relaxed, Ordering::Relaxed) + .is_err() + { + return; + } + self.set_modem_clock_status_internal(enabled, radio_peripheral); + } +} diff --git a/esp-hal/src/rtc_cntl/rtc/esp32c6.rs b/esp-hal/src/rtc_cntl/rtc/esp32c6.rs index 66d72abf62c..ee31d9e4b87 100644 --- a/esp-hal/src/rtc_cntl/rtc/esp32c6.rs +++ b/esp-hal/src/rtc_cntl/rtc/esp32c6.rs @@ -21,9 +21,9 @@ use crate::{ XtalClock, }, peripherals::TIMG0, + radio_clock_ctrl::RadioPeripherals, rtc_cntl::RtcClock, soc::efuse::Efuse, - system::RadioPeripherals, }; const I2C_DIG_REG: u8 = 0x6d; diff --git a/esp-hal/src/soc/esp32/radio_clocks.rs b/esp-hal/src/soc/esp32/radio_clocks.rs index 828d8015860..6740421cfc8 100644 --- a/esp-hal/src/soc/esp32/radio_clocks.rs +++ b/esp-hal/src/soc/esp32/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; const DPORT_WIFI_CLK_WIFI_BT_COMMON_M: u32 = 0x000003c9; const DPORT_WIFI_CLK_WIFI_EN_M: u32 = 0x00000406; diff --git a/esp-hal/src/soc/esp32c2/radio_clocks.rs b/esp-hal/src/soc/esp32c2/radio_clocks.rs index 62b5fea634c..d94bb77ff59 100644 --- a/esp-hal/src/soc/esp32c2/radio_clocks.rs +++ b/esp-hal/src/soc/esp32c2/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; // Mask for clock bits used by both WIFI and Bluetooth, 0, 1, 2, 3, 7, 8, 9, 10, // 19, 20, 21, 22, 23 diff --git a/esp-hal/src/soc/esp32c3/radio_clocks.rs b/esp-hal/src/soc/esp32c3/radio_clocks.rs index 5ec9cf9e3c8..98a25add1ae 100644 --- a/esp-hal/src/soc/esp32c3/radio_clocks.rs +++ b/esp-hal/src/soc/esp32c3/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; // Mask for clock bits used by both WIFI and Bluetooth, 0, 1, 2, 3, 7, 8, 9, 10, // 19, 20, 21, 22, 23 diff --git a/esp-hal/src/soc/esp32c6/radio_clocks.rs b/esp-hal/src/soc/esp32c6/radio_clocks.rs index f9319969bcc..66e58b046a5 100644 --- a/esp-hal/src/soc/esp32c6/radio_clocks.rs +++ b/esp-hal/src/soc/esp32c6/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; impl RadioClockController for crate::peripherals::RADIO_CLK { fn enable(&mut self, peripheral: RadioPeripherals) { diff --git a/esp-hal/src/soc/esp32h2/radio_clocks.rs b/esp-hal/src/soc/esp32h2/radio_clocks.rs index 101e7ee20ad..2d4d0c55ead 100644 --- a/esp-hal/src/soc/esp32h2/radio_clocks.rs +++ b/esp-hal/src/soc/esp32h2/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; impl RadioClockController for crate::peripherals::RADIO_CLK { fn enable(&mut self, peripheral: RadioPeripherals) { diff --git a/esp-hal/src/soc/esp32s2/radio_clocks.rs b/esp-hal/src/soc/esp32s2/radio_clocks.rs index 32d942d2f24..6f23d46954c 100644 --- a/esp-hal/src/soc/esp32s2/radio_clocks.rs +++ b/esp-hal/src/soc/esp32s2/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; // Mask for clock bits used by both WIFI and Bluetooth, bit 0, 3, 6, 7, 8, 9 const DPORT_WIFI_CLK_WIFI_BT_COMMON_M: u32 = 0x000003c9; diff --git a/esp-hal/src/soc/esp32s3/radio_clocks.rs b/esp-hal/src/soc/esp32s3/radio_clocks.rs index f14a0c09a1e..9d1e6e53822 100644 --- a/esp-hal/src/soc/esp32s3/radio_clocks.rs +++ b/esp-hal/src/soc/esp32s3/radio_clocks.rs @@ -11,7 +11,7 @@ //! `RadioClockControl` struct. This trait provides methods to enable, disable, //! reset the MAC, initialize clocks and perform other related operations. -use crate::system::{RadioClockController, RadioPeripherals}; +use crate::radio_clock_ctrl::{RadioClockController, RadioPeripherals}; // Note: this comment has been copied from esp-idf, including the mistake. // Mask for clock bits used by both WIFI and Bluetooth, 0, 1, 2, 3, 7, diff --git a/esp-hal/src/system.rs b/esp-hal/src/system.rs index 0406d89d0f4..21f2f3c3249 100755 --- a/esp-hal/src/system.rs +++ b/esp-hal/src/system.rs @@ -1,9 +1,4 @@ //! # System Control -//! -//! ## Overview -//! -//! This `system` module defines the available radio peripherals and provides an -//! interface to control and configure radio clocks. use crate::peripherals::SYSTEM; @@ -985,42 +980,3 @@ impl PeripheralClockControl { } } } - -/// Enumeration of the available radio peripherals for this chip. -#[cfg(any(bt, ieee802154, wifi))] -pub enum RadioPeripherals { - /// Represents the PHY (Physical Layer) peripheral. - #[cfg(phy)] - Phy, - /// Represents the Bluetooth peripheral. - #[cfg(bt)] - Bt, - /// Represents the WiFi peripheral. - #[cfg(wifi)] - Wifi, - /// Represents the IEEE 802.15.4 peripheral. - #[cfg(ieee802154)] - Ieee802154, -} - -/// Control the radio peripheral clocks -#[cfg(any(bt, ieee802154, wifi))] -pub trait RadioClockController { - /// Enable the peripheral - fn enable(&mut self, peripheral: RadioPeripherals); - - /// Disable the peripheral - fn disable(&mut self, peripheral: RadioPeripherals); - - /// Reset the MAC - fn reset_mac(&mut self); - - /// Do any common initial initialization needed - fn init_clocks(&mut self); - - /// Initialize BLE RTC clocks - fn ble_rtc_clk_init(&mut self); - - /// Reset the Resolvable Private Address (RPA). - fn reset_rpa(&mut self); -}