//! This shows how to use the TIMG peripheral interrupts. //! There is TIMG0 and TIMG1 each of them containing two general purpose timers //! and a watchdog timer. #![no_std] #![no_main] use core::{cell::RefCell, fmt::Write}; use esp32s2_hal::{ clock::ClockControl, interrupt, interrupt::Priority, pac::{self, Peripherals, TIMG0, TIMG1, UART0}, prelude::*, timer::{Timer, Timer0, Timer1, TimerGroup}, Rtc, Serial, }; use panic_halt as _; use xtensa_lx::mutex::{CriticalSectionMutex, Mutex}; use xtensa_lx_rt::entry; static mut SERIAL: CriticalSectionMutex>>> = CriticalSectionMutex::new(RefCell::new(None)); static mut TIMER00: CriticalSectionMutex>>>> = CriticalSectionMutex::new(RefCell::new(None)); static mut TIMER01: CriticalSectionMutex>>>> = CriticalSectionMutex::new(RefCell::new(None)); static mut TIMER10: CriticalSectionMutex>>>> = CriticalSectionMutex::new(RefCell::new(None)); static mut TIMER11: CriticalSectionMutex>>>> = CriticalSectionMutex::new(RefCell::new(None)); #[entry] fn main() -> ! { let peripherals = Peripherals::take().unwrap(); let system = peripherals.SYSTEM.split(); let clocks = ClockControl::boot_defaults(system.clock_control).freeze(); // Disable the TIMG watchdog timer. let timer_group0 = TimerGroup::new(peripherals.TIMG0, &clocks); let mut timer00 = timer_group0.timer0; let mut timer01 = timer_group0.timer1; let mut wdt0 = timer_group0.wdt; let timer_group1 = TimerGroup::new(peripherals.TIMG1, &clocks); let mut timer10 = timer_group1.timer0; let mut timer11 = timer_group1.timer1; let mut wdt1 = timer_group1.wdt; let serial0 = Serial::new(peripherals.UART0); let mut rtc = Rtc::new(peripherals.RTC_CNTL); // Disable MWDT and RWDT (Watchdog) flash boot protection wdt0.disable(); wdt1.disable(); rtc.rwdt.disable(); interrupt::enable(pac::Interrupt::TG0_T0_LEVEL, Priority::Priority2).unwrap(); interrupt::enable(pac::Interrupt::TG0_T1_LEVEL, Priority::Priority2).unwrap(); interrupt::enable(pac::Interrupt::TG1_T0_LEVEL, Priority::Priority3).unwrap(); interrupt::enable(pac::Interrupt::TG1_T1_LEVEL, Priority::Priority3).unwrap(); timer00.start(500u64.millis()); timer00.listen(); timer01.start(2500u64.millis()); timer01.listen(); timer10.start(1u64.secs()); timer10.listen(); timer11.start(3u64.secs()); timer11.listen(); unsafe { (&SERIAL).lock(|data| (*data).replace(Some(serial0))); (&TIMER00).lock(|data| (*data).replace(Some(timer00))); (&TIMER01).lock(|data| (*data).replace(Some(timer01))); (&TIMER10).lock(|data| (*data).replace(Some(timer10))); (&TIMER11).lock(|data| (*data).replace(Some(timer11))); } loop {} } #[interrupt] fn TG0_T0_LEVEL() { unsafe { (&TIMER00).lock(|data| { let mut timer = data.borrow_mut(); let timer = timer.as_mut().unwrap(); if timer.is_interrupt_set() { timer.clear_interrupt(); timer.start(500u64.millis()); (&SERIAL).lock(|data| { let mut serial = data.borrow_mut(); let serial = serial.as_mut().unwrap(); writeln!(serial, "Interrupt Level 2 - Timer0").ok(); }); } }); } } #[interrupt] fn TG0_T1_LEVEL() { unsafe { (&TIMER01).lock(|data| { let mut timer = data.borrow_mut(); let timer = timer.as_mut().unwrap(); if timer.is_interrupt_set() { timer.clear_interrupt(); timer.start(2500u64.millis()); (&SERIAL).lock(|data| { let mut serial = data.borrow_mut(); let serial = serial.as_mut().unwrap(); writeln!(serial, "Interrupt Level 2 - Timer1").ok(); }); } }); } } #[interrupt] fn TG1_T0_LEVEL() { unsafe { (&TIMER10).lock(|data| { let mut timer = data.borrow_mut(); let timer = timer.as_mut().unwrap(); if timer.is_interrupt_set() { timer.clear_interrupt(); timer.start(1u64.secs()); (&SERIAL).lock(|data| { let mut serial = data.borrow_mut(); let serial = serial.as_mut().unwrap(); writeln!(serial, "Interrupt Level 3 - Timer0").ok(); }); } }); } } #[interrupt] fn TG1_T1_LEVEL() { unsafe { (&TIMER11).lock(|data| { let mut timer = data.borrow_mut(); let timer = timer.as_mut().unwrap(); if timer.is_interrupt_set() { timer.clear_interrupt(); timer.start(3u64.secs()); (&SERIAL).lock(|data| { let mut serial = data.borrow_mut(); let serial = serial.as_mut().unwrap(); writeln!(serial, "Interrupt Level 3 - Timer1").ok(); }); } }); } }