fix timing, turn TIM UDE on only necessary, clean DMA FEIF after each Transfer

This commit is contained in:
eZio Pan 2023-12-22 01:24:31 +08:00
parent d7e517e2d3
commit 53fc344e4d

View file

@ -64,9 +64,6 @@ async fn main(_spawner: Spawner) {
CountingMode::EdgeAlignedUp, CountingMode::EdgeAlignedUp,
); );
// PAC level hacking, enable timer-update-event trigger DMA
pac::TIM3.dier().modify(|v| v.set_ude(true));
// construct ws2812 non-return-to-zero (NRZ) code bit by bit // construct ws2812 non-return-to-zero (NRZ) code bit by bit
// ws2812 only need 24 bits for each LED, but we add one bit more to keep PWM output low // ws2812 only need 24 bits for each LED, but we add one bit more to keep PWM output low
@ -104,13 +101,16 @@ async fn main(_spawner: Spawner) {
dma_transfer_option.mburst = Burst::Incr8; dma_transfer_option.mburst = Burst::Incr8;
// flip color at 2 Hz // flip color at 2 Hz
let mut ticker = Ticker::every(Duration::from_micros(500)); let mut ticker = Ticker::every(Duration::from_millis(500));
loop { loop {
for &color in color_list { for &color in color_list {
// start PWM output // start PWM output
ws2812_pwm.enable(pwm_channel); ws2812_pwm.enable(pwm_channel);
// PAC level hacking, enable timer-update-event trigger DMA
pac::TIM3.dier().modify(|v| v.set_ude(true));
unsafe { unsafe {
Transfer::new_write( Transfer::new_write(
// with &mut, we can easily reuse same DMA channel multiple times // with &mut, we can easily reuse same DMA channel multiple times
@ -121,6 +121,14 @@ async fn main(_spawner: Spawner) {
dma_transfer_option, dma_transfer_option,
) )
.await; .await;
// Turn off timer-update-event trigger DMA as soon as possible.
// Then clean the FIFO Error Flag if set.
pac::TIM3.dier().modify(|v| v.set_ude(false));
if pac::DMA1.isr(0).read().feif(2) {
pac::DMA1.ifcr(0).write(|v| v.set_feif(2, true));
}
// ws2812 need at least 50 us low level input to confirm the input data and change it's state // ws2812 need at least 50 us low level input to confirm the input data and change it's state
Timer::after_micros(50).await; Timer::after_micros(50).await;
} }