From db8bab08af293184f073783afc9b475f261129c9 Mon Sep 17 00:00:00 2001 From: "K." Date: Mon, 21 Oct 2024 02:59:19 +0500 Subject: [PATCH] Add sigma-delta modulator driver Solution for issue #2370 --- esp-hal/src/sdm.rs | 264 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 264 insertions(+) create mode 100644 esp-hal/src/sdm.rs diff --git a/esp-hal/src/sdm.rs b/esp-hal/src/sdm.rs new file mode 100644 index 00000000000..4351ec254d2 --- /dev/null +++ b/esp-hal/src/sdm.rs @@ -0,0 +1,264 @@ +//! Sigma-Delta modulation peripheral driver. +//! + +use crate::{ + clock::Clocks, + gpio::{OutputSignal, PeripheralOutput}, + peripheral::{Peripheral, PeripheralRef}, + peripherals, + private, +}; +use fugit::HertzU32; + +/// 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 prescale < 1 || prescale > 256 { + Err(Error::PrescaleRange) + } else { + SD::set_prescale(self.chidx, prescale); + Ok(()) + } + } + + /// 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 crate::gpio::OutputPin; + use super::{Channel, RegisterAccess, Error}; + + 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 as u16 + } + + 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.max(1).min(256) as _ + + /*if pre < 1 || pre > 256 { + Err(Error::PrescaleRange) + } else { + pre as _ + }*/ + + pre as _ +}