diff --git a/esp-hal/src/lib.rs b/esp-hal/src/lib.rs index df92a6d6f4d..2a9bd2b428a 100644 --- a/esp-hal/src/lib.rs +++ b/esp-hal/src/lib.rs @@ -216,6 +216,8 @@ pub mod rom; pub mod rsa; #[cfg(any(lp_clkrst, rtc_cntl))] pub mod rtc_cntl; +#[cfg(sdm)] +pub mod sdm; #[cfg(sha)] pub mod sha; #[cfg(any(spi0, spi1, spi2, spi3))] diff --git a/esp-hal/src/sdm.rs b/esp-hal/src/sdm.rs new file mode 100644 index 00000000000..88960e28e73 --- /dev/null +++ b/esp-hal/src/sdm.rs @@ -0,0 +1,257 @@ +//! Sigma-Delta modulation peripheral driver. + +use fugit::HertzU32; + +use crate::{ + clock::Clocks, + gpio::{OutputSignal, PeripheralOutput}, + peripheral::{Peripheral, PeripheralRef}, + peripherals, + private, +}; + +/// Sigma-Delta modulation peripheral driver. +pub struct Sdm<'d, SD> { + _sd: PeripheralRef<'d, SD>, + channels_usage: u8, +} + +/// Channel errors +#[derive(Debug, Clone, Copy, PartialEq)] +#[cfg_attr(feature = "defmt", derive(defmt::Format))] +pub enum Error { + /// Prescale out of range + PrescaleRange, + /// No free channels to use + NoChannels, +} + +impl<'d, SD> Sdm<'d, SD> +where + SD: RegisterAccess, +{ + /// Initialize driver using a given SD instance. + pub fn new(sd_instance: impl crate::peripheral::Peripheral

+ 'd) -> Self { + Self { + _sd: sd_instance.into_ref(), + channels_usage: 0, + } + } + + /// Configure and acquire channel. + pub fn enable_pin( + &mut self, + pin: impl Peripheral

+ 'd, + frequency: HertzU32, + ) -> Result, Error> { + crate::into_ref!(pin); + + let chs = self.channels_usage; + let chidx = self.alloc_channel()?; + let signal = CHANNELS[chidx as usize]; + + if chs == 0 { + SD::enable_clock(true); + } + + pin.connect_peripheral_to_output(signal, private::Internal); + + let mut channel = Channel { + _sdm: self, + pin, + chidx, + }; + + channel.set_frequency(frequency)?; + + Ok(channel) + } + + /// Deconfigure and release channel. + pub fn disable_pin(&mut self, channel: Channel<'d, SD, PIN>) { + let Channel { mut pin, chidx, .. } = channel; + + let signal = CHANNELS[chidx as usize]; + + pin.disconnect_from_peripheral_output(signal, private::Internal); + + self.dealloc_channel(chidx); + + if self.channels_usage == 0 { + SD::enable_clock(false); + } + } + + fn alloc_channel(&mut self) -> Result { + let mut usage = self.channels_usage; + let mut chidx: u8 = 0; + while usage & 1 != 0 && chidx < CHANNELS.len() as u8 { + usage >>= 1; + chidx += 1; + } + if chidx < CHANNELS.len() as u8 { + self.channels_usage |= 1 << chidx; + Ok(chidx) + } else { + Err(Error::NoChannels) + } + } + + fn dealloc_channel(&mut self, chidx: u8) { + self.channels_usage &= !(1 << chidx); + } +} + +/// Sigma-Delta modulation channel handle. +pub struct Channel<'d, SD, PIN> { + _sdm: &'d Sdm<'d, SD>, + pin: PeripheralRef<'d, PIN>, + chidx: u8, +} + +impl<'d, SD, PIN> Channel<'d, SD, PIN> +where + SD: RegisterAccess, +{ + /// Set raw pulse density + /// + /// Sigma-delta quantized density of one channel, the value ranges from -128 + /// to 127, recommended range is -90 ~ 90. The waveform is more like a + /// random one in this range. + pub fn set_pulse_density(&mut self, density: i8) { + SD::set_pulse_density(self.chidx, density); + } + + /// Set duty cycle + pub fn set_duty(&mut self, duty: u8) { + let density = duty as i16 - 128; + self.set_pulse_density(density as i8) + } + + /// Set raw prescale + /// + /// The divider of source clock, ranges from 1 to 256 + pub fn set_prescale(&mut self, prescale: u16) -> Result<(), Error> { + if (1..=256).contains(&prescale) { + SD::set_prescale(self.chidx, prescale); + Ok(()) + } else { + Err(Error::PrescaleRange) + } + } + + /// Set prescale using frequency + pub fn set_frequency(&mut self, frequency: HertzU32) -> Result<(), Error> { + let clocks = Clocks::get(); + let clock_frequency = clocks.apb_clock.to_Hz(); + let frequency = frequency.to_Hz(); + + let prescale = prescale_from_frequency(clock_frequency, frequency); + + self.set_prescale(prescale) + } +} + +mod ehal1 { + use embedded_hal::pwm::{Error as PwmError, ErrorKind, ErrorType, SetDutyCycle}; + + use super::{Channel, Error, RegisterAccess}; + use crate::gpio::OutputPin; + + impl PwmError for Error { + fn kind(&self) -> ErrorKind { + ErrorKind::Other + } + } + + impl<'d, SD: RegisterAccess, PIN: OutputPin> ErrorType for Channel<'d, SD, PIN> { + type Error = Error; + } + + impl<'d, SD: RegisterAccess, PIN: OutputPin> SetDutyCycle for Channel<'d, SD, PIN> { + fn max_duty_cycle(&self) -> u16 { + 255 + } + + fn set_duty_cycle(&mut self, mut duty: u16) -> Result<(), Self::Error> { + let max = self.max_duty_cycle(); + duty = if duty > max { max } else { duty }; + self.set_duty(duty as u8); + Ok(()) + } + } +} + +#[cfg(any(esp32, esp32s2, esp32s3))] +const CHANNELS: [OutputSignal; 8] = [ + OutputSignal::GPIO_SD0, + OutputSignal::GPIO_SD1, + OutputSignal::GPIO_SD2, + OutputSignal::GPIO_SD3, + OutputSignal::GPIO_SD4, + OutputSignal::GPIO_SD5, + OutputSignal::GPIO_SD6, + OutputSignal::GPIO_SD7, +]; + +#[cfg(any(esp32c3, esp32c6, esp32h2))] +const CHANNELS: [OutputSignal; 4] = [ + OutputSignal::GPIO_SD0, + OutputSignal::GPIO_SD1, + OutputSignal::GPIO_SD2, + OutputSignal::GPIO_SD3, +]; + +#[doc(hidden)] +pub trait RegisterAccess { + /// Enable/disable sigma/delta clock + fn enable_clock(en: bool); + + /// Set channel pulse density + fn set_pulse_density(ch: u8, density: i8); + + /// Set channel clock pre-scale + fn set_prescale(ch: u8, prescale: u16); +} + +impl RegisterAccess for peripherals::GPIO_SD { + fn enable_clock(_en: bool) { + // The clk enable register does not exist on ESP32. + #[cfg(not(esp32))] + { + let sd = unsafe { &*Self::PTR }; + + sd.sigmadelta_misc() + .modify(|_, w| w.function_clk_en().bit(_en)); + } + } + + fn set_pulse_density(ch: u8, density: i8) { + let sd = unsafe { &*Self::PTR }; + + sd.sigmadelta(ch as _) + .modify(|_, w| unsafe { w.in_().bits(density as _) }); + } + + fn set_prescale(ch: u8, prescale: u16) { + let sd = unsafe { &*Self::PTR }; + + sd.sigmadelta(ch as _) + .modify(|_, w| unsafe { w.prescale().bits((prescale - 1) as _) }); + } +} + +fn prescale_from_frequency(clk_freq: u32, req_freq: u32) -> u16 { + let pre = clk_freq / req_freq; + let err = clk_freq % req_freq; + + // Do the normal rounding and error >= (src/n + src/(n+1)) / 2, + // then carry the bit + let pre = if err >= clk_freq / (2 * pre * (pre + 1)) { + pre + 1 + } else { + pre + }; + + pre as _ +} diff --git a/esp-metadata/devices/esp32.toml b/esp-metadata/devices/esp32.toml index e5a70a666f4..6b885acf57a 100644 --- a/esp-metadata/devices/esp32.toml +++ b/esp-metadata/devices/esp32.toml @@ -33,6 +33,7 @@ peripherals = [ "rtc_io", "sdhost", "sens", + "sdm", "sha", "slc", "slchost", diff --git a/esp-metadata/devices/esp32c3.toml b/esp-metadata/devices/esp32c3.toml index 08a62907cb3..f385b633119 100644 --- a/esp-metadata/devices/esp32c3.toml +++ b/esp-metadata/devices/esp32c3.toml @@ -26,6 +26,7 @@ peripherals = [ "rsa", "rtc_cntl", "sensitive", + "sdm", "sha", "spi0", "spi1", diff --git a/esp-metadata/devices/esp32c6.toml b/esp-metadata/devices/esp32c6.toml index 128e8a06c24..4c57a6a917a 100644 --- a/esp-metadata/devices/esp32c6.toml +++ b/esp-metadata/devices/esp32c6.toml @@ -50,6 +50,7 @@ peripherals = [ "rmt", "rng", "rsa", + "sdm", "sha", "slchost", "soc_etm", diff --git a/esp-metadata/devices/esp32h2.toml b/esp-metadata/devices/esp32h2.toml index 901f4bd8ef9..803a1ad4bf7 100644 --- a/esp-metadata/devices/esp32h2.toml +++ b/esp-metadata/devices/esp32h2.toml @@ -44,6 +44,7 @@ peripherals = [ "rmt", "rng", "rsa", + "sdm", "sha", "soc_etm", "spi0", diff --git a/esp-metadata/devices/esp32p4.toml b/esp-metadata/devices/esp32p4.toml index 9d745becd85..4766b8efb7c 100644 --- a/esp-metadata/devices/esp32p4.toml +++ b/esp-metadata/devices/esp32p4.toml @@ -71,6 +71,7 @@ peripherals = [ # "rmt", # "rsa", # "sdhost", + # "sdm", # "sha", # "soc_etm", # "spi0", diff --git a/esp-metadata/devices/esp32s2.toml b/esp-metadata/devices/esp32s2.toml index 2b556cd00cf..0505462981f 100644 --- a/esp-metadata/devices/esp32s2.toml +++ b/esp-metadata/devices/esp32s2.toml @@ -29,6 +29,7 @@ peripherals = [ "rtc_i2c", "rtc_io", "sens", + "sdm", "sha", "spi0", "spi1", diff --git a/esp-metadata/devices/esp32s3.toml b/esp-metadata/devices/esp32s3.toml index 96fbe57278a..b9bd0d29658 100644 --- a/esp-metadata/devices/esp32s3.toml +++ b/esp-metadata/devices/esp32s3.toml @@ -37,6 +37,7 @@ peripherals = [ "rtc_io", "sens", "sensitive", + "sdm", "sha", "spi0", "spi1",