2021-05-12 02:56:11 +00:00
|
|
|
#![macro_use]
|
|
|
|
|
|
|
|
use core::cell::UnsafeCell;
|
|
|
|
use core::marker::PhantomData;
|
|
|
|
use core::sync::atomic::{compiler_fence, Ordering};
|
|
|
|
use embassy::util::Unborrow;
|
2021-07-29 11:44:51 +00:00
|
|
|
use embassy_hal_common::unborrow;
|
2021-05-12 02:56:11 +00:00
|
|
|
|
|
|
|
use crate::gpio::sealed::Pin as _;
|
|
|
|
use crate::gpio::OptionalPin as GpioOptionalPin;
|
|
|
|
use crate::interrupt::Interrupt;
|
2021-11-01 08:15:04 +00:00
|
|
|
use crate::pac;
|
2021-10-26 07:37:52 +00:00
|
|
|
use crate::util::slice_in_ram_or;
|
2021-05-12 02:56:11 +00:00
|
|
|
|
2021-11-01 08:15:04 +00:00
|
|
|
/// PWM Base clock is system clock (16MHz) divided by prescaler
|
2021-05-12 02:56:11 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
|
pub enum Prescaler {
|
|
|
|
Div1,
|
|
|
|
Div2,
|
|
|
|
Div4,
|
|
|
|
Div8,
|
|
|
|
Div16,
|
|
|
|
Div32,
|
|
|
|
Div64,
|
|
|
|
Div128,
|
|
|
|
}
|
|
|
|
|
2021-11-01 08:15:04 +00:00
|
|
|
/// How the sequence values are distributed across the channels
|
2021-10-26 07:37:52 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
|
pub enum SequenceLoad {
|
2021-11-01 08:20:01 +00:00
|
|
|
/// Provided sequence will be used across all channels
|
2021-10-26 07:37:52 +00:00
|
|
|
Common,
|
2021-11-01 08:20:01 +00:00
|
|
|
/// Provided sequence contains grouped values for each channel ex:
|
|
|
|
/// [ch0_0_and_ch1_0, ch2_0_and_ch3_0, ... ch0_n_and_ch1_n, ch2_n_and_ch3_n]
|
2021-10-26 07:37:52 +00:00
|
|
|
Grouped,
|
2021-11-01 08:20:01 +00:00
|
|
|
/// Provided sequence contains individual values for each channel ex:
|
|
|
|
/// [ch0_0, ch1_0, ch2_0, ch3_0... ch0_n, ch1_n, ch2_n, ch3_n]
|
2021-10-26 07:37:52 +00:00
|
|
|
Individual,
|
2021-11-01 08:20:01 +00:00
|
|
|
/// Similar to Individual mode, but only three channels are used. The fourth
|
|
|
|
/// value is loaded into the pulse generator counter as its top value.
|
2021-10-26 07:37:52 +00:00
|
|
|
Waveform,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
|
pub enum CounterMode {
|
2021-11-01 08:15:04 +00:00
|
|
|
/// Up counter (edge-aligned PWM duty cycle)
|
2021-10-26 07:37:52 +00:00
|
|
|
Up,
|
2021-11-01 08:15:04 +00:00
|
|
|
/// Up and down counter (center-aligned PWM duty cycle)
|
2021-10-26 07:37:52 +00:00
|
|
|
UpAndDown,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Interface to the PWM peripheral
|
2021-05-12 02:56:11 +00:00
|
|
|
pub struct Pwm<'d, T: Instance> {
|
|
|
|
phantom: PhantomData<&'d mut T>,
|
|
|
|
}
|
|
|
|
|
2021-11-02 02:11:37 +00:00
|
|
|
pub struct PwmSeq<'d, T: Instance> {
|
|
|
|
phantom: PhantomData<&'d mut T>,
|
2021-10-26 07:37:52 +00:00
|
|
|
}
|
|
|
|
|
2021-11-02 02:11:37 +00:00
|
|
|
impl<'d, T: Instance> PwmSeq<'d, T> {
|
2021-11-01 16:37:34 +00:00
|
|
|
/// Creates the interface to a PWM instance.
|
|
|
|
///
|
|
|
|
/// Defaults the freq to 1Mhz, max_duty 32767, duty 0, and channels low.
|
2021-05-12 02:56:11 +00:00
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
///
|
2021-11-01 16:37:34 +00:00
|
|
|
/// The returned API is safe unless you use `mem::forget` (or similar safe
|
|
|
|
/// mechanisms) on stack allocated buffers which which have been passed to
|
|
|
|
/// [`send()`](Pwm::send) or [`receive`](Pwm::receive).
|
2021-05-12 02:56:11 +00:00
|
|
|
#[allow(unused_unsafe)]
|
|
|
|
pub fn new(
|
2021-05-17 10:23:04 +00:00
|
|
|
_pwm: impl Unborrow<Target = T> + 'd,
|
2021-05-12 02:56:11 +00:00
|
|
|
ch0: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch1: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch2: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch3: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
2021-11-02 02:11:37 +00:00
|
|
|
config: SequenceConfig,
|
|
|
|
) -> Result<Self, Error> {
|
|
|
|
slice_in_ram_or(config.sequence, Error::DMABufferNotInDataMemory)?;
|
|
|
|
|
|
|
|
if config.sequence.len() > 32767 {
|
|
|
|
return Err(Error::SequenceTooLong);
|
|
|
|
}
|
|
|
|
if let SequenceMode::Times(0) = config.times {
|
|
|
|
return Err(Error::SequenceTooShort);
|
|
|
|
}
|
|
|
|
|
2021-05-17 10:23:04 +00:00
|
|
|
unborrow!(ch0, ch1, ch2, ch3);
|
2021-05-12 02:56:11 +00:00
|
|
|
|
|
|
|
let r = T::regs();
|
|
|
|
|
|
|
|
if let Some(pin) = ch0.pin_mut() {
|
2021-05-14 22:02:35 +00:00
|
|
|
pin.set_low();
|
2021-05-12 02:56:11 +00:00
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch1.pin_mut() {
|
2021-05-14 22:02:35 +00:00
|
|
|
pin.set_low();
|
2021-05-12 02:56:11 +00:00
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch2.pin_mut() {
|
2021-05-14 22:02:35 +00:00
|
|
|
pin.set_low();
|
2021-05-12 02:56:11 +00:00
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch3.pin_mut() {
|
2021-05-14 22:02:35 +00:00
|
|
|
pin.set_low();
|
2021-05-12 02:56:11 +00:00
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
2021-11-01 15:54:07 +00:00
|
|
|
|
|
|
|
// if NoPin provided writes disconnected (top bit 1) 0x80000000 else
|
|
|
|
// writes pin number ex 13 (0x0D) which is connected (top bit 0)
|
2021-05-12 02:56:11 +00:00
|
|
|
r.psel.out[0].write(|w| unsafe { w.bits(ch0.psel_bits()) });
|
|
|
|
r.psel.out[1].write(|w| unsafe { w.bits(ch1.psel_bits()) });
|
|
|
|
r.psel.out[2].write(|w| unsafe { w.bits(ch2.psel_bits()) });
|
|
|
|
r.psel.out[3].write(|w| unsafe { w.bits(ch3.psel_bits()) });
|
|
|
|
|
|
|
|
// Disable all interrupts
|
2021-11-01 16:37:34 +00:00
|
|
|
r.intenset.reset();
|
2021-05-12 02:56:11 +00:00
|
|
|
r.intenclr.write(|w| unsafe { w.bits(0xFFFF_FFFF) });
|
2021-11-01 16:37:34 +00:00
|
|
|
r.shorts.reset();
|
2021-05-12 02:56:11 +00:00
|
|
|
|
|
|
|
// Enable
|
|
|
|
r.enable.write(|w| w.enable().enabled());
|
2021-10-26 07:37:52 +00:00
|
|
|
r.mode
|
|
|
|
.write(|w| unsafe { w.bits(config.counter_mode as u32) });
|
|
|
|
r.prescaler
|
|
|
|
.write(|w| w.prescaler().bits(config.prescaler as u8));
|
|
|
|
r.countertop
|
|
|
|
.write(|w| unsafe { w.countertop().bits(config.top) });
|
|
|
|
|
|
|
|
r.decoder.write(|w| {
|
|
|
|
w.load().bits(config.sequence_load as u8);
|
|
|
|
w.mode().refresh_count()
|
|
|
|
});
|
|
|
|
|
|
|
|
r.seq0
|
|
|
|
.ptr
|
|
|
|
.write(|w| unsafe { w.bits(config.sequence.as_ptr() as u32) });
|
|
|
|
r.seq0
|
|
|
|
.cnt
|
|
|
|
.write(|w| unsafe { w.bits(config.sequence.len() as u32) });
|
2021-10-30 23:16:10 +00:00
|
|
|
r.seq0.refresh.write(|w| unsafe { w.bits(config.refresh) });
|
2021-10-26 07:37:52 +00:00
|
|
|
r.seq0
|
|
|
|
.enddelay
|
2021-11-01 08:20:01 +00:00
|
|
|
.write(|w| unsafe { w.bits(config.end_delay) });
|
2021-10-26 07:37:52 +00:00
|
|
|
|
|
|
|
r.seq1
|
|
|
|
.ptr
|
|
|
|
.write(|w| unsafe { w.bits(config.sequence.as_ptr() as u32) });
|
|
|
|
r.seq1
|
|
|
|
.cnt
|
|
|
|
.write(|w| unsafe { w.bits(config.sequence.len() as u32) });
|
2021-10-30 23:16:10 +00:00
|
|
|
r.seq1.refresh.write(|w| unsafe { w.bits(config.refresh) });
|
2021-10-26 07:37:52 +00:00
|
|
|
r.seq1
|
|
|
|
.enddelay
|
2021-11-01 08:20:01 +00:00
|
|
|
.write(|w| unsafe { w.bits(config.end_delay) });
|
2021-10-26 07:37:52 +00:00
|
|
|
|
2021-11-01 15:45:07 +00:00
|
|
|
match config.times {
|
2021-11-01 07:21:17 +00:00
|
|
|
// just the one time, no loop count
|
2021-11-01 15:54:07 +00:00
|
|
|
SequenceMode::Times(1) => {
|
2021-10-30 23:16:10 +00:00
|
|
|
r.loop_.write(|w| w.cnt().disabled());
|
2021-11-01 07:21:17 +00:00
|
|
|
// tasks_seqstart doesnt exist in all svds so write its bit instead
|
2021-10-30 23:16:10 +00:00
|
|
|
r.tasks_seqstart[0].write(|w| unsafe { w.bits(0x01) });
|
|
|
|
}
|
2021-11-01 07:21:17 +00:00
|
|
|
// loop count is how many times to play BOTH sequences
|
2021-11-01 15:45:07 +00:00
|
|
|
// 2 total (1 x 2)
|
|
|
|
// 3 total, (2 x 2) - 1
|
2021-11-01 15:54:07 +00:00
|
|
|
SequenceMode::Times(n) => {
|
2021-11-01 15:45:07 +00:00
|
|
|
let odd = n & 1 == 1;
|
|
|
|
let times = if odd { (n / 2) + 1 } else { n / 2 };
|
|
|
|
|
2021-10-30 23:16:10 +00:00
|
|
|
r.loop_.write(|w| unsafe { w.cnt().bits(times) });
|
|
|
|
|
2021-11-01 20:16:24 +00:00
|
|
|
// we can subtract 1 by starting at seq1 instead of seq0
|
2021-11-01 15:45:07 +00:00
|
|
|
if odd {
|
2021-10-30 23:23:45 +00:00
|
|
|
// tasks_seqstart doesnt exist in all svds so write its bit instead
|
2021-11-01 15:45:07 +00:00
|
|
|
r.tasks_seqstart[1].write(|w| unsafe { w.bits(0x01) });
|
2021-10-30 23:16:10 +00:00
|
|
|
} else {
|
2021-10-30 23:23:45 +00:00
|
|
|
// tasks_seqstart doesnt exist in all svds so write its bit instead
|
2021-11-01 15:45:07 +00:00
|
|
|
r.tasks_seqstart[0].write(|w| unsafe { w.bits(0x01) });
|
2021-10-30 23:16:10 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-01 07:30:42 +00:00
|
|
|
// to play infinitely, repeat the sequence one time, then have loops done self trigger seq0 again
|
2021-11-01 15:54:07 +00:00
|
|
|
SequenceMode::Infinite => {
|
2021-10-30 23:16:10 +00:00
|
|
|
r.loop_.write(|w| unsafe { w.cnt().bits(0x1) });
|
2021-11-01 07:30:42 +00:00
|
|
|
r.shorts.write(|w| w.loopsdone_seqstart0().enabled());
|
2021-11-01 07:21:17 +00:00
|
|
|
// tasks_seqstart doesnt exist in all svds so write its bit instead
|
2021-11-01 07:30:42 +00:00
|
|
|
r.tasks_seqstart[0].write(|w| unsafe { w.bits(0x01) });
|
2021-10-30 23:16:10 +00:00
|
|
|
}
|
|
|
|
}
|
2021-11-02 02:11:37 +00:00
|
|
|
|
|
|
|
Ok(Self {
|
|
|
|
phantom: PhantomData,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Stop playback
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn stop(&self) {
|
|
|
|
let r = T::regs();
|
|
|
|
|
|
|
|
r.shorts.reset();
|
|
|
|
|
|
|
|
// tasks_stop doesnt exist in all svds so write its bit instead
|
|
|
|
r.tasks_stop.write(|w| unsafe { w.bits(0x01) });
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Disables the PWM generator.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn disable(&self) {
|
|
|
|
let r = T::regs();
|
|
|
|
r.enable.write(|w| w.enable().disabled());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T: Instance> Drop for PwmSeq<'a, T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.stop();
|
|
|
|
self.disable();
|
|
|
|
|
|
|
|
info!("pwm drop: done");
|
|
|
|
|
|
|
|
// TODO: disable pins
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
|
pub enum SequenceMode {
|
|
|
|
/// Run sequence n Times total
|
|
|
|
Times(u16),
|
|
|
|
/// Repeat until `stop` is called.
|
|
|
|
Infinite,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Configure an infinite looping sequence for `simple_playback`
|
|
|
|
pub struct SequenceConfig<'a> {
|
|
|
|
/// Selects up mode or up-and-down mode for the counter
|
|
|
|
pub counter_mode: CounterMode,
|
|
|
|
/// Top value to be compared against buffer values
|
|
|
|
pub top: u16,
|
|
|
|
/// Configuration for PWM_CLK
|
|
|
|
pub prescaler: Prescaler,
|
|
|
|
/// In ram buffer to be played back
|
|
|
|
pub sequence: &'a [u16],
|
|
|
|
/// How a sequence is read from RAM and is spread to the compare register
|
|
|
|
pub sequence_load: SequenceLoad,
|
|
|
|
/// Number of Times PWM periods to delay between each sequence sample
|
|
|
|
pub refresh: u32,
|
|
|
|
/// Number of Times PWM periods after the sequence ends before starting the next sequence
|
|
|
|
pub end_delay: u32,
|
|
|
|
/// How many times to play the sequence
|
|
|
|
pub times: SequenceMode,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
|
#[non_exhaustive]
|
|
|
|
pub enum Error {
|
|
|
|
/// Max Sequence size is 32767
|
|
|
|
SequenceTooLong,
|
|
|
|
/// Min Sequence size is 1
|
|
|
|
SequenceTooShort,
|
|
|
|
/// EasyDMA can only read from data memory, read only buffers in flash will fail.
|
|
|
|
DMABufferNotInDataMemory,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'d, T: Instance> Pwm<'d, T> {
|
|
|
|
/// Creates the interface to a PWM instance.
|
|
|
|
///
|
|
|
|
/// Defaults the freq to 1Mhz, max_duty 32767, duty 0, and channels low.
|
|
|
|
/// Must be started by calling `set_duty`
|
|
|
|
///
|
|
|
|
/// # Safety
|
|
|
|
///
|
|
|
|
/// The returned API is safe unless you use `mem::forget` (or similar safe
|
|
|
|
/// mechanisms) on stack allocated buffers which which have been passed to
|
|
|
|
/// [`send()`](Pwm::send) or [`receive`](Pwm::receive).
|
|
|
|
#[allow(unused_unsafe)]
|
|
|
|
pub fn new(
|
|
|
|
_pwm: impl Unborrow<Target = T> + 'd,
|
|
|
|
ch0: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch1: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch2: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
ch3: impl Unborrow<Target = impl GpioOptionalPin> + 'd,
|
|
|
|
) -> Self {
|
|
|
|
unborrow!(ch0, ch1, ch2, ch3);
|
|
|
|
|
|
|
|
let r = T::regs();
|
|
|
|
let s = T::state();
|
|
|
|
|
|
|
|
if let Some(pin) = ch0.pin_mut() {
|
|
|
|
pin.set_low();
|
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch1.pin_mut() {
|
|
|
|
pin.set_low();
|
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch2.pin_mut() {
|
|
|
|
pin.set_low();
|
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
if let Some(pin) = ch3.pin_mut() {
|
|
|
|
pin.set_low();
|
|
|
|
pin.conf().write(|w| w.dir().output());
|
|
|
|
}
|
|
|
|
|
|
|
|
// if NoPin provided writes disconnected (top bit 1) 0x80000000 else
|
|
|
|
// writes pin number ex 13 (0x0D) which is connected (top bit 0)
|
|
|
|
r.psel.out[0].write(|w| unsafe { w.bits(ch0.psel_bits()) });
|
|
|
|
r.psel.out[1].write(|w| unsafe { w.bits(ch1.psel_bits()) });
|
|
|
|
r.psel.out[2].write(|w| unsafe { w.bits(ch2.psel_bits()) });
|
|
|
|
r.psel.out[3].write(|w| unsafe { w.bits(ch3.psel_bits()) });
|
|
|
|
|
|
|
|
// Disable all interrupts
|
|
|
|
r.intenset.reset();
|
|
|
|
r.intenclr.write(|w| unsafe { w.bits(0xFFFF_FFFF) });
|
|
|
|
r.shorts.reset();
|
|
|
|
|
|
|
|
// Enable
|
|
|
|
r.enable.write(|w| w.enable().enabled());
|
|
|
|
|
|
|
|
r.seq0
|
|
|
|
.ptr
|
|
|
|
.write(|w| unsafe { w.bits(&s.duty as *const _ as u32) });
|
|
|
|
r.seq0.cnt.write(|w| unsafe { w.bits(4) });
|
|
|
|
r.seq0.refresh.write(|w| unsafe { w.bits(0) });
|
|
|
|
r.seq0.enddelay.write(|w| unsafe { w.bits(0) });
|
|
|
|
|
|
|
|
r.decoder.write(|w| {
|
|
|
|
w.load().individual();
|
|
|
|
w.mode().refresh_count()
|
|
|
|
});
|
|
|
|
r.mode.write(|w| w.updown().up());
|
|
|
|
r.prescaler.write(|w| w.prescaler().div_16());
|
|
|
|
r.countertop.write(|w| unsafe { w.countertop().bits(1000) });
|
|
|
|
r.loop_.write(|w| w.cnt().disabled());
|
|
|
|
|
|
|
|
Self {
|
|
|
|
phantom: PhantomData,
|
|
|
|
}
|
2021-10-26 07:37:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Stop playback
|
|
|
|
#[inline(always)]
|
2021-11-02 02:11:37 +00:00
|
|
|
pub fn stop(&self) {
|
2021-10-26 07:37:52 +00:00
|
|
|
let r = T::regs();
|
|
|
|
|
2021-11-01 16:37:34 +00:00
|
|
|
r.shorts.reset();
|
2021-10-26 07:37:52 +00:00
|
|
|
|
|
|
|
// tasks_stop doesnt exist in all svds so write its bit instead
|
|
|
|
r.tasks_stop.write(|w| unsafe { w.bits(0x01) });
|
|
|
|
}
|
|
|
|
|
2021-11-02 02:11:37 +00:00
|
|
|
// todo should this do.. something useful
|
2021-05-14 22:02:50 +00:00
|
|
|
/// Enables the PWM generator.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn enable(&self) {
|
|
|
|
let r = T::regs();
|
|
|
|
r.enable.write(|w| w.enable().enabled());
|
|
|
|
}
|
|
|
|
|
2021-11-02 02:11:37 +00:00
|
|
|
// todo should this stop the task? or should you just use set_duty to 0?
|
2021-05-14 22:02:50 +00:00
|
|
|
/// Disables the PWM generator.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn disable(&self) {
|
|
|
|
let r = T::regs();
|
|
|
|
r.enable.write(|w| w.enable().disabled());
|
|
|
|
}
|
|
|
|
|
2021-05-12 02:56:11 +00:00
|
|
|
/// Sets duty cycle (15 bit) for a PWM channel.
|
|
|
|
pub fn set_duty(&self, channel: usize, duty: u16) {
|
|
|
|
let s = T::state();
|
|
|
|
unsafe { (*s.duty.get())[channel] = duty & 0x7FFF };
|
|
|
|
|
2021-11-01 20:00:50 +00:00
|
|
|
// todo justify? should i fence elsehwere we task start? or
|
2021-05-12 02:56:11 +00:00
|
|
|
compiler_fence(Ordering::SeqCst);
|
2021-11-01 20:00:50 +00:00
|
|
|
|
|
|
|
// play duty cycle infinitely
|
|
|
|
let r = T::regs();
|
|
|
|
r.loop_.write(|w| unsafe { w.cnt().bits(0x1) });
|
|
|
|
r.shorts.write(|w| w.loopsdone_seqstart0().enabled());
|
|
|
|
// tasks_seqstart doesnt exist in all svds so write its bit instead
|
|
|
|
r.tasks_seqstart[0].write(|w| unsafe { w.bits(1) });
|
2021-05-12 02:56:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the PWM clock prescaler.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn set_prescaler(&self, div: Prescaler) {
|
|
|
|
T::regs().prescaler.write(|w| w.prescaler().bits(div as u8));
|
|
|
|
}
|
|
|
|
|
2021-10-26 07:37:52 +00:00
|
|
|
/// Gets the PWM clock prescaler.
|
2021-05-12 02:56:11 +00:00
|
|
|
#[inline(always)]
|
|
|
|
pub fn prescaler(&self) -> Prescaler {
|
|
|
|
match T::regs().prescaler.read().prescaler().bits() {
|
|
|
|
0 => Prescaler::Div1,
|
|
|
|
1 => Prescaler::Div2,
|
|
|
|
2 => Prescaler::Div4,
|
|
|
|
3 => Prescaler::Div8,
|
|
|
|
4 => Prescaler::Div16,
|
|
|
|
5 => Prescaler::Div32,
|
|
|
|
6 => Prescaler::Div64,
|
|
|
|
7 => Prescaler::Div128,
|
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the maximum duty cycle value.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn set_max_duty(&self, duty: u16) {
|
|
|
|
T::regs()
|
|
|
|
.countertop
|
|
|
|
.write(|w| unsafe { w.countertop().bits(duty.min(32767u16)) });
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the maximum duty cycle value.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn max_duty(&self) -> u16 {
|
|
|
|
T::regs().countertop.read().countertop().bits()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Sets the PWM output frequency.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn set_period(&self, freq: u32) {
|
|
|
|
let clk = 16_000_000u32 >> (self.prescaler() as u8);
|
|
|
|
let duty = clk / freq;
|
|
|
|
self.set_max_duty(duty.min(32767) as u16);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the PWM output frequency.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn period(&self) -> u32 {
|
|
|
|
let clk = 16_000_000u32 >> (self.prescaler() as u8);
|
|
|
|
let max_duty = self.max_duty() as u32;
|
|
|
|
clk / max_duty
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T: Instance> Drop for Pwm<'a, T> {
|
|
|
|
fn drop(&mut self) {
|
2021-11-02 02:11:37 +00:00
|
|
|
self.stop();
|
2021-10-30 23:16:10 +00:00
|
|
|
self.disable();
|
2021-05-12 02:56:11 +00:00
|
|
|
|
|
|
|
info!("pwm drop: done");
|
|
|
|
|
|
|
|
// TODO: disable pins
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) mod sealed {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
pub struct State {
|
|
|
|
pub duty: UnsafeCell<[u16; 4]>,
|
|
|
|
}
|
|
|
|
unsafe impl Sync for State {}
|
|
|
|
|
|
|
|
impl State {
|
|
|
|
pub const fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
duty: UnsafeCell::new([0; 4]),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait Instance {
|
|
|
|
fn regs() -> &'static pac::pwm0::RegisterBlock;
|
|
|
|
fn state() -> &'static State;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-14 22:05:32 +00:00
|
|
|
pub trait Instance: Unborrow<Target = Self> + sealed::Instance + 'static {
|
2021-05-12 02:56:11 +00:00
|
|
|
type Interrupt: Interrupt;
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! impl_pwm {
|
|
|
|
($type:ident, $pac_type:ident, $irq:ident) => {
|
|
|
|
impl crate::pwm::sealed::Instance for peripherals::$type {
|
|
|
|
fn regs() -> &'static pac::pwm0::RegisterBlock {
|
|
|
|
unsafe { &*pac::$pac_type::ptr() }
|
|
|
|
}
|
|
|
|
fn state() -> &'static crate::pwm::sealed::State {
|
|
|
|
static STATE: crate::pwm::sealed::State = crate::pwm::sealed::State::new();
|
|
|
|
&STATE
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl crate::pwm::Instance for peripherals::$type {
|
|
|
|
type Interrupt = crate::interrupt::$irq;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|