Skip to content

Commit

Permalink
Add sigma-delta modulator driver
Browse files Browse the repository at this point in the history
Solution for issue esp-rs#2370
  • Loading branch information
katyo committed Oct 20, 2024
1 parent d32a733 commit db8bab0
Showing 1 changed file with 264 additions and 0 deletions.
264 changes: 264 additions & 0 deletions esp-hal/src/sdm.rs
Original file line number Diff line number Diff line change
@@ -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<P = SD> + 'd) -> Self {
Self {
_sd: sd_instance.into_ref(),
channels_usage: 0,
}
}

/// Configure and acquire channel.
pub fn enable_pin<PIN: PeripheralOutput>(
&mut self,
pin: impl Peripheral<P = PIN> + 'd,
frequency: HertzU32,
) -> Result<Channel<'_, SD, PIN>, 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<PIN: PeripheralOutput>(&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<u8, Error> {
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 _
}

0 comments on commit db8bab0

Please sign in to comment.