use core::marker::PhantomData; use embassy::util::Unborrow; use embassy_hal_common::unborrow; use super::*; #[allow(unused_imports)] use crate::gpio::sealed::{AFType, Pin}; use crate::time::Hertz; pub struct SimplePwm<'d, T> { phantom: PhantomData<&'d mut T>, inner: T, } macro_rules! config_pins { ($($pin:ident),*) => { unborrow!($($pin),*); // NOTE(unsafe) Exclusive access to the registers critical_section::with(|_| unsafe { $( $pin.set_low(); $pin.set_as_af($pin.af_num(), AFType::OutputPushPull); #[cfg(gpio_v2)] $pin.set_speed(crate::gpio::Speed::VeryHigh); )* }) }; } impl<'d, T: CaptureCompare16bitInstance> SimplePwm<'d, T> { pub fn new_1ch>( tim: impl Unborrow + 'd, ch1: impl Unborrow> + 'd, freq: F, ) -> Self { Self::new_inner(tim, freq, move || { config_pins!(ch1); }) } pub fn new_2ch>( tim: impl Unborrow + 'd, ch1: impl Unborrow> + 'd, ch2: impl Unborrow> + 'd, freq: F, ) -> Self { Self::new_inner(tim, freq, move || { config_pins!(ch1, ch2); }) } pub fn new_3ch>( tim: impl Unborrow + 'd, ch1: impl Unborrow> + 'd, ch2: impl Unborrow> + 'd, ch3: impl Unborrow> + 'd, freq: F, ) -> Self { Self::new_inner(tim, freq, move || { config_pins!(ch1, ch2, ch3); }) } pub fn new_4ch>( tim: impl Unborrow + 'd, ch1: impl Unborrow> + 'd, ch2: impl Unborrow> + 'd, ch3: impl Unborrow> + 'd, ch4: impl Unborrow> + 'd, freq: F, ) -> Self { Self::new_inner(tim, freq, move || { config_pins!(ch1, ch2, ch3, ch4); }) } fn new_inner>( tim: impl Unborrow + 'd, freq: F, configure_pins: impl FnOnce(), ) -> Self { unborrow!(tim); T::enable(); ::reset(); configure_pins(); let mut this = Self { inner: tim, phantom: PhantomData, }; this.inner.set_frequency(freq); this.inner.start(); unsafe { this.inner .set_output_compare_mode(Channel::Ch1, OutputCompareMode::PwmMode1); this.inner .set_output_compare_mode(Channel::Ch2, OutputCompareMode::PwmMode1); this.inner .set_output_compare_mode(Channel::Ch3, OutputCompareMode::PwmMode1); this.inner .set_output_compare_mode(Channel::Ch4, OutputCompareMode::PwmMode1); } this } pub fn enable(&mut self, channel: Channel) { unsafe { self.inner.enable_channel(channel, true); } } pub fn disable(&mut self, channel: Channel) { unsafe { self.inner.enable_channel(channel, false); } } pub fn set_freq>(&mut self, freq: F) { self.inner.set_frequency(freq); } pub fn get_max_duty(&self) -> u16 { unsafe { self.inner.get_max_compare_value() } } pub fn set_duty(&mut self, channel: Channel, duty: u16) { assert!(duty < self.get_max_duty()); unsafe { self.inner.set_compare_value(channel, duty) } } }