2022-02-11 22:00:31 +01:00
|
|
|
//! This example showcases channels and timing utilities of embassy.
|
|
|
|
//!
|
|
|
|
//! This example works best on STM32F3DISCOVERY board. It flashes a single LED, then if the USER
|
|
|
|
//! button is pressed the next LED is flashed (in clockwise fashion). If the USER button is double
|
|
|
|
//! clicked, then the direction changes. If the USER button is pressed for 1 second, then all the
|
|
|
|
//! LEDS flash 3 times.
|
|
|
|
//!
|
|
|
|
|
|
|
|
#![no_std]
|
|
|
|
#![no_main]
|
|
|
|
#![feature(type_alias_impl_trait)]
|
|
|
|
|
2022-04-02 04:35:06 +02:00
|
|
|
use defmt::*;
|
2022-02-11 23:25:30 +01:00
|
|
|
use embassy::blocking_mutex::raw::NoopRawMutex;
|
2022-02-11 22:00:31 +01:00
|
|
|
use embassy::channel::mpsc::{self, Channel, Receiver, Sender};
|
|
|
|
use embassy::executor::Spawner;
|
|
|
|
use embassy::time::{with_timeout, Duration, Timer};
|
|
|
|
use embassy::util::Forever;
|
|
|
|
use embassy_stm32::exti::ExtiInput;
|
|
|
|
use embassy_stm32::gpio::{AnyPin, Input, Level, Output, Pin, Pull, Speed};
|
|
|
|
use embassy_stm32::peripherals::PA0;
|
|
|
|
use embassy_stm32::Peripherals;
|
2022-04-02 04:35:06 +02:00
|
|
|
|
|
|
|
use defmt_rtt as _; // global logger
|
|
|
|
use panic_probe as _;
|
2022-02-11 22:00:31 +01:00
|
|
|
|
|
|
|
struct Leds<'a> {
|
|
|
|
leds: [Output<'a, AnyPin>; 8],
|
|
|
|
direction: i8,
|
|
|
|
current_led: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Leds<'a> {
|
|
|
|
fn new(pins: [Output<'a, AnyPin>; 8]) -> Self {
|
|
|
|
Self {
|
|
|
|
leds: pins,
|
|
|
|
direction: 1,
|
|
|
|
current_led: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn change_direction(&mut self) {
|
|
|
|
self.direction *= -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn move_next(&mut self) {
|
|
|
|
if self.direction > 0 {
|
|
|
|
self.current_led = (self.current_led + 1) & 7;
|
|
|
|
} else {
|
|
|
|
self.current_led = (8 + self.current_led - 1) & 7;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-16 21:22:35 +01:00
|
|
|
async fn show(&mut self, queue: &mut Receiver<'static, NoopRawMutex, ButtonEvent, 4>) {
|
2022-02-11 22:00:31 +01:00
|
|
|
self.leds[self.current_led].set_high();
|
2022-02-16 21:22:35 +01:00
|
|
|
if let Ok(new_message) = with_timeout(Duration::from_millis(500), queue.recv()).await {
|
|
|
|
self.leds[self.current_led].set_low();
|
|
|
|
self.process_event(new_message).await;
|
|
|
|
} else {
|
|
|
|
self.leds[self.current_led].set_low();
|
|
|
|
if let Ok(new_message) = with_timeout(Duration::from_millis(200), queue.recv()).await {
|
|
|
|
self.process_event(new_message).await;
|
|
|
|
}
|
|
|
|
}
|
2022-02-11 22:00:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn flash(&mut self) {
|
|
|
|
for _ in 0..3 {
|
|
|
|
for led in &mut self.leds {
|
|
|
|
led.set_high();
|
|
|
|
}
|
|
|
|
Timer::after(Duration::from_millis(500)).await;
|
|
|
|
for led in &mut self.leds {
|
|
|
|
led.set_low();
|
|
|
|
}
|
|
|
|
Timer::after(Duration::from_millis(200)).await;
|
|
|
|
}
|
|
|
|
}
|
2022-02-16 21:22:35 +01:00
|
|
|
|
|
|
|
async fn process_event(&mut self, event: Option<ButtonEvent>) {
|
|
|
|
match event {
|
|
|
|
Some(ButtonEvent::SingleClick) => self.move_next(),
|
|
|
|
Some(ButtonEvent::DoubleClick) => {
|
|
|
|
self.change_direction();
|
|
|
|
self.move_next()
|
|
|
|
}
|
|
|
|
Some(ButtonEvent::Hold) => self.flash().await,
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
2022-02-11 22:00:31 +01:00
|
|
|
}
|
|
|
|
|
2022-02-16 21:22:35 +01:00
|
|
|
#[derive(Format)]
|
2022-02-11 22:00:31 +01:00
|
|
|
enum ButtonEvent {
|
|
|
|
SingleClick,
|
|
|
|
DoubleClick,
|
|
|
|
Hold,
|
|
|
|
}
|
|
|
|
|
2022-02-11 23:25:30 +01:00
|
|
|
static BUTTON_EVENTS_QUEUE: Forever<Channel<NoopRawMutex, ButtonEvent, 4>> = Forever::new();
|
2022-02-11 22:00:31 +01:00
|
|
|
|
|
|
|
#[embassy::main]
|
|
|
|
async fn main(spawner: Spawner, p: Peripherals) {
|
|
|
|
let button = Input::new(p.PA0, Pull::Down);
|
|
|
|
let button = ExtiInput::new(button, p.EXTI0);
|
|
|
|
info!("Press the USER button...");
|
|
|
|
let leds = [
|
|
|
|
Output::new(p.PE9.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE10.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE11.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE12.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE13.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE14.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE15.degrade(), Level::Low, Speed::Low),
|
|
|
|
Output::new(p.PE8.degrade(), Level::Low, Speed::Low),
|
|
|
|
];
|
|
|
|
let leds = Leds::new(leds);
|
|
|
|
|
|
|
|
let buttons_queue = BUTTON_EVENTS_QUEUE.put(Channel::new());
|
|
|
|
let (sender, receiver) = mpsc::split(buttons_queue);
|
|
|
|
spawner.spawn(button_waiter(button, sender)).unwrap();
|
|
|
|
spawner.spawn(led_blinker(leds, receiver)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[embassy::task]
|
2022-02-11 23:25:30 +01:00
|
|
|
async fn led_blinker(
|
|
|
|
mut leds: Leds<'static>,
|
2022-02-16 21:22:35 +01:00
|
|
|
mut queue: Receiver<'static, NoopRawMutex, ButtonEvent, 4>,
|
2022-02-11 23:25:30 +01:00
|
|
|
) {
|
2022-02-11 22:00:31 +01:00
|
|
|
loop {
|
2022-02-16 21:22:35 +01:00
|
|
|
leds.show(&mut queue).await;
|
2022-02-11 22:00:31 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[embassy::task]
|
|
|
|
async fn button_waiter(
|
|
|
|
mut button: ExtiInput<'static, PA0>,
|
2022-02-11 23:25:30 +01:00
|
|
|
queue: Sender<'static, NoopRawMutex, ButtonEvent, 4>,
|
2022-02-11 22:00:31 +01:00
|
|
|
) {
|
|
|
|
const DOUBLE_CLICK_DELAY: u64 = 250;
|
|
|
|
const HOLD_DELAY: u64 = 1000;
|
|
|
|
|
|
|
|
button.wait_for_rising_edge().await;
|
|
|
|
loop {
|
|
|
|
if with_timeout(
|
|
|
|
Duration::from_millis(HOLD_DELAY),
|
|
|
|
button.wait_for_falling_edge(),
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.is_err()
|
|
|
|
{
|
|
|
|
info!("Hold");
|
|
|
|
if queue.send(ButtonEvent::Hold).await.is_err() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
button.wait_for_falling_edge().await;
|
|
|
|
} else if with_timeout(
|
|
|
|
Duration::from_millis(DOUBLE_CLICK_DELAY),
|
|
|
|
button.wait_for_rising_edge(),
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.is_err()
|
|
|
|
{
|
|
|
|
if queue.send(ButtonEvent::SingleClick).await.is_err() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
info!("Single click");
|
|
|
|
} else {
|
|
|
|
info!("Double click");
|
|
|
|
if queue.send(ButtonEvent::DoubleClick).await.is_err() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
button.wait_for_falling_edge().await;
|
|
|
|
}
|
|
|
|
button.wait_for_rising_edge().await;
|
|
|
|
}
|
|
|
|
}
|