esp-idf-hal/examples/timer_notify.rs
ivmarkov ebeef0fcc8
All executors (#315)
* Support for all async executors

* Make all Notifiucation primitives a public API

* Address a small typo

* Address a small typo

* Fix CI

* Tighten the notify APIs by not taking 0s

* Fix CI

* Fix changelog

* Fix changelog

* Fix the examples

* edge-executor support moved to edge-executor

* wake-from-isr feature for edge-executor

* Fix CI

* Bugfixing

* Clippy
2023-10-06 22:01:20 +03:00

52 lines
2.1 KiB
Rust

use std::num::NonZeroU32;
use esp_idf_hal::sys::{EspError, TaskHandle_t}; // If using the `binstart` feature of `esp-idf-sys`, always keep this module imported
fn main() -> Result<(), EspError> {
// It is necessary to call this function once. Otherwise some patches to the runtime
// implemented by esp-idf-sys might not link properly. See https://github.com/esp-rs/esp-idf-template/issues/71
esp_idf_hal::sys::link_patches();
let per = esp_idf_hal::peripherals::Peripherals::take().unwrap();
// This handle will be used as the address for the event in the callback.
// Make sure that the handle / thread lives always longer as the callback it is used in
let main_task_handle: TaskHandle_t = esp_idf_hal::task::current().unwrap();
// BaseClock for the Timer is the APB_CLK that is running on 80MHz at default
// The default clock-divider is -> 80
// default APB clk is available with the APB_CLK_FREQ constant
let timer_conf = esp_idf_hal::timer::config::Config::new().auto_reload(true);
let mut timer = esp_idf_hal::timer::TimerDriver::new(per.timer00, &timer_conf)?;
// Calculate value needed for alarm in seconds
// (APB_CLK_FREQ / DEVIDER ) * seconds = count
// example every 200 us
// ( 80*10^6 / 80 ) * 200 *10^(-6) = 200
timer.set_alarm(200)?;
// Saftey: make sure the task handle stays valid for longer than the subscribtion
// is active
unsafe {
timer.subscribe(move || {
let bitset = 0b10001010101;
esp_idf_hal::task::notify_and_yield(main_task_handle, NonZeroU32::new(bitset).unwrap());
})?;
}
timer.enable_alarm(true)?;
timer.enable(true)?;
loop {
// Notify approach
// The benefit with this approach over checking a global static variable is
// that the scheduler can hold the task, and resume when signaled
// so no spinlock is needed
let bitset = esp_idf_hal::task::wait_notification(esp_idf_hal::delay::BLOCK);
if let Some(bitset) = bitset {
println!("got event with bits {bitset:#b} from ISR");
}
}
}