Compare commits
19 Commits
hil-test
...
disable-st
Author | SHA1 | Date | |
---|---|---|---|
57edf289ea | |||
b6fc682117 | |||
0beb84768e | |||
b98a279367 | |||
e8a3cfaed6 | |||
0cc3e18db6 | |||
6b19c0abd1 | |||
f956d19e6e | |||
ceb0d0bf08 | |||
b3879ec223 | |||
bda99e59ec | |||
25c2a9baaa | |||
1e362c750b | |||
1a51a84313 | |||
7f72dbdaf2 | |||
e8c162ac03 | |||
1aaa19748a | |||
188ee59ba6 | |||
591612db7e |
3
ci.sh
3
ci.sh
@ -218,6 +218,9 @@ cargo batch \
|
|||||||
rm out/tests/stm32wb55rg/wpan_mac
|
rm out/tests/stm32wb55rg/wpan_mac
|
||||||
rm out/tests/stm32wb55rg/wpan_ble
|
rm out/tests/stm32wb55rg/wpan_ble
|
||||||
|
|
||||||
|
# unstable
|
||||||
|
rm out/tests/stm32f429zi/stop
|
||||||
|
|
||||||
# unstable, I think it's running out of RAM?
|
# unstable, I think it's running out of RAM?
|
||||||
rm out/tests/stm32f207zg/eth
|
rm out/tests/stm32f207zg/eth
|
||||||
|
|
||||||
|
@ -3,7 +3,7 @@ use core::task::{RawWaker, RawWakerVTable, Waker};
|
|||||||
|
|
||||||
use super::{wake_task, TaskHeader, TaskRef};
|
use super::{wake_task, TaskHeader, TaskRef};
|
||||||
|
|
||||||
const VTABLE: RawWakerVTable = RawWakerVTable::new(clone, wake, wake, drop);
|
static VTABLE: RawWakerVTable = RawWakerVTable::new(clone, wake, wake, drop);
|
||||||
|
|
||||||
unsafe fn clone(p: *const ()) -> RawWaker {
|
unsafe fn clone(p: *const ()) -> RawWaker {
|
||||||
RawWaker::new(p, &VTABLE)
|
RawWaker::new(p, &VTABLE)
|
||||||
|
@ -564,7 +564,7 @@ fn main() {
|
|||||||
fn enable_and_reset_with_cs(_cs: critical_section::CriticalSection) {
|
fn enable_and_reset_with_cs(_cs: critical_section::CriticalSection) {
|
||||||
#before_enable
|
#before_enable
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
crate::rcc::clock_refcount_add(_cs);
|
unsafe { crate::rcc::REFCOUNT_STOP2 += 1 };
|
||||||
crate::pac::RCC.#en_reg().modify(|w| w.#set_en_field(true));
|
crate::pac::RCC.#en_reg().modify(|w| w.#set_en_field(true));
|
||||||
#after_enable
|
#after_enable
|
||||||
#rst
|
#rst
|
||||||
@ -573,7 +573,7 @@ fn main() {
|
|||||||
#before_disable
|
#before_disable
|
||||||
crate::pac::RCC.#en_reg().modify(|w| w.#set_en_field(false));
|
crate::pac::RCC.#en_reg().modify(|w| w.#set_en_field(false));
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
crate::rcc::clock_refcount_sub(_cs);
|
unsafe { crate::rcc::REFCOUNT_STOP2 -= 1 };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -465,7 +465,7 @@ pub(crate) fn assert_not_corrupted_read(end_address: u32) {
|
|||||||
feature = "stm32f439vg",
|
feature = "stm32f439vg",
|
||||||
feature = "stm32f439zg",
|
feature = "stm32f439zg",
|
||||||
))]
|
))]
|
||||||
if second_bank_read && unsafe { pac::DBGMCU.idcode().read().rev_id() < REVISION_3 && !pa12_is_output_pull_low() } {
|
if second_bank_read && pac::DBGMCU.idcode().read().rev_id() < REVISION_3 && !pa12_is_output_pull_low() {
|
||||||
panic!("Read corruption for stm32f42xxG and stm32f43xxG in dual bank mode when PA12 is in use for chips below revision 3, see errata 2.2.11");
|
panic!("Read corruption for stm32f42xxG and stm32f43xxG in dual bank mode when PA12 is in use for chips below revision 3, see errata 2.2.11");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -226,9 +226,9 @@ pub fn init(config: Config) -> Peripherals {
|
|||||||
time_driver::init(cs);
|
time_driver::init(cs);
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
while !crate::rcc::low_power_ready() {
|
{
|
||||||
crate::rcc::clock_refcount_sub(cs);
|
crate::rcc::REFCOUNT_STOP2 = 0
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
p
|
p
|
||||||
|
@ -6,7 +6,6 @@ use cortex_m::peripheral::SCB;
|
|||||||
use embassy_executor::*;
|
use embassy_executor::*;
|
||||||
|
|
||||||
use crate::interrupt;
|
use crate::interrupt;
|
||||||
use crate::rcc::low_power_ready;
|
|
||||||
use crate::time_driver::{get_driver, RtcDriver};
|
use crate::time_driver::{get_driver, RtcDriver};
|
||||||
|
|
||||||
const THREAD_PENDER: usize = usize::MAX;
|
const THREAD_PENDER: usize = usize::MAX;
|
||||||
@ -33,6 +32,15 @@ pub fn stop_with_rtc(rtc: &'static Rtc) {
|
|||||||
unsafe { EXECUTOR.as_mut().unwrap() }.stop_with_rtc(rtc)
|
unsafe { EXECUTOR.as_mut().unwrap() }.stop_with_rtc(rtc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn stop_ready(stop_mode: StopMode) -> bool {
|
||||||
|
unsafe { EXECUTOR.as_mut().unwrap() }.stop_ready(stop_mode)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[non_exhaustive]
|
||||||
|
pub enum StopMode {
|
||||||
|
Stop2,
|
||||||
|
}
|
||||||
|
|
||||||
/// Thread mode executor, using WFE/SEV.
|
/// Thread mode executor, using WFE/SEV.
|
||||||
///
|
///
|
||||||
/// This is the simplest and most common kind of executor. It runs on
|
/// This is the simplest and most common kind of executor. It runs on
|
||||||
@ -80,12 +88,18 @@ impl Executor {
|
|||||||
trace!("low power: stop with rtc configured");
|
trace!("low power: stop with rtc configured");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn stop_ready(&self, stop_mode: StopMode) -> bool {
|
||||||
|
match stop_mode {
|
||||||
|
StopMode::Stop2 => unsafe { crate::rcc::REFCOUNT_STOP2 == 0 },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn configure_pwr(&mut self) {
|
fn configure_pwr(&mut self) {
|
||||||
self.scb.clear_sleepdeep();
|
self.scb.clear_sleepdeep();
|
||||||
|
|
||||||
compiler_fence(Ordering::SeqCst);
|
compiler_fence(Ordering::SeqCst);
|
||||||
|
|
||||||
if !low_power_ready() {
|
if !self.stop_ready(StopMode::Stop2) {
|
||||||
trace!("low power: not ready to stop");
|
trace!("low power: not ready to stop");
|
||||||
} else if self.time_driver.pause_time().is_err() {
|
} else if self.time_driver.pause_time().is_err() {
|
||||||
trace!("low power: failed to pause time");
|
trace!("low power: failed to pause time");
|
||||||
|
@ -23,8 +23,6 @@ pub use mco::*;
|
|||||||
#[cfg_attr(rcc_u5, path = "u5.rs")]
|
#[cfg_attr(rcc_u5, path = "u5.rs")]
|
||||||
#[cfg_attr(rcc_wba, path = "wba.rs")]
|
#[cfg_attr(rcc_wba, path = "wba.rs")]
|
||||||
mod _version;
|
mod _version;
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
use core::sync::atomic::{AtomicU32, Ordering};
|
|
||||||
|
|
||||||
pub use _version::*;
|
pub use _version::*;
|
||||||
|
|
||||||
@ -183,27 +181,7 @@ pub struct Clocks {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
static CLOCK_REFCOUNT: AtomicU32 = AtomicU32::new(0);
|
pub(crate) static mut REFCOUNT_STOP2: u32 = 0;
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
pub fn low_power_ready() -> bool {
|
|
||||||
// trace!("clock refcount: {}", CLOCK_REFCOUNT.load(Ordering::SeqCst));
|
|
||||||
CLOCK_REFCOUNT.load(Ordering::SeqCst) == 0
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
pub(crate) fn clock_refcount_add(_cs: critical_section::CriticalSection) {
|
|
||||||
// We don't check for overflow because constructing more than u32 peripherals is unlikely
|
|
||||||
let n = CLOCK_REFCOUNT.load(Ordering::Relaxed);
|
|
||||||
CLOCK_REFCOUNT.store(n + 1, Ordering::Relaxed);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
pub(crate) fn clock_refcount_sub(_cs: critical_section::CriticalSection) {
|
|
||||||
let n = CLOCK_REFCOUNT.load(Ordering::Relaxed);
|
|
||||||
assert!(n != 0);
|
|
||||||
CLOCK_REFCOUNT.store(n - 1, Ordering::Relaxed);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Frozen clock frequencies
|
/// Frozen clock frequencies
|
||||||
///
|
///
|
||||||
|
@ -4,8 +4,60 @@ use core::convert::From;
|
|||||||
#[cfg(feature = "chrono")]
|
#[cfg(feature = "chrono")]
|
||||||
use chrono::{self, Datelike, NaiveDate, Timelike, Weekday};
|
use chrono::{self, Datelike, NaiveDate, Timelike, Weekday};
|
||||||
|
|
||||||
use super::byte_to_bcd2;
|
#[cfg(any(feature = "defmt", feature = "time"))]
|
||||||
use crate::pac::rtc::Rtc;
|
use crate::peripherals::RTC;
|
||||||
|
#[cfg(any(feature = "defmt", feature = "time"))]
|
||||||
|
use crate::rtc::sealed::Instance;
|
||||||
|
|
||||||
|
/// Represents an instant in time that can be substracted to compute a duration
|
||||||
|
pub struct RtcInstant {
|
||||||
|
/// 0..59
|
||||||
|
pub second: u8,
|
||||||
|
/// 0..256
|
||||||
|
pub subsecond: u16,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RtcInstant {
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub(super) fn from(second: u8, subsecond: u16) -> Result<Self, super::RtcError> {
|
||||||
|
Ok(Self { second, subsecond })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "defmt")]
|
||||||
|
impl defmt::Format for RtcInstant {
|
||||||
|
fn format(&self, fmt: defmt::Formatter) {
|
||||||
|
defmt::write!(
|
||||||
|
fmt,
|
||||||
|
"{}:{}",
|
||||||
|
self.second,
|
||||||
|
RTC::regs().prer().read().prediv_s() - self.subsecond,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "time")]
|
||||||
|
impl core::ops::Sub for RtcInstant {
|
||||||
|
type Output = embassy_time::Duration;
|
||||||
|
|
||||||
|
fn sub(self, rhs: Self) -> Self::Output {
|
||||||
|
use embassy_time::{Duration, TICK_HZ};
|
||||||
|
|
||||||
|
let second = if self.second < rhs.second {
|
||||||
|
self.second + 60
|
||||||
|
} else {
|
||||||
|
self.second
|
||||||
|
};
|
||||||
|
|
||||||
|
let psc = RTC::regs().prer().read().prediv_s() as u32;
|
||||||
|
|
||||||
|
let self_ticks = second as u32 * (psc + 1) + (psc - self.subsecond as u32);
|
||||||
|
let other_ticks = rhs.second as u32 * (psc + 1) + (psc - rhs.subsecond as u32);
|
||||||
|
let rtc_ticks = self_ticks - other_ticks;
|
||||||
|
|
||||||
|
Duration::from_ticks(((rtc_ticks * TICK_HZ as u32) / (psc + 1)) as u64)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Errors regarding the [`DateTime`] struct.
|
/// Errors regarding the [`DateTime`] struct.
|
||||||
#[derive(Clone, Debug, PartialEq, Eq)]
|
#[derive(Clone, Debug, PartialEq, Eq)]
|
||||||
@ -32,19 +84,85 @@ pub enum Error {
|
|||||||
/// Structure containing date and time information
|
/// Structure containing date and time information
|
||||||
pub struct DateTime {
|
pub struct DateTime {
|
||||||
/// 0..4095
|
/// 0..4095
|
||||||
pub year: u16,
|
year: u16,
|
||||||
/// 1..12, 1 is January
|
/// 1..12, 1 is January
|
||||||
pub month: u8,
|
month: u8,
|
||||||
/// 1..28,29,30,31 depending on month
|
/// 1..28,29,30,31 depending on month
|
||||||
pub day: u8,
|
day: u8,
|
||||||
///
|
///
|
||||||
pub day_of_week: DayOfWeek,
|
day_of_week: DayOfWeek,
|
||||||
/// 0..23
|
/// 0..23
|
||||||
pub hour: u8,
|
hour: u8,
|
||||||
/// 0..59
|
/// 0..59
|
||||||
pub minute: u8,
|
minute: u8,
|
||||||
/// 0..59
|
/// 0..59
|
||||||
pub second: u8,
|
second: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DateTime {
|
||||||
|
pub const fn year(&self) -> u16 {
|
||||||
|
self.year
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn month(&self) -> u8 {
|
||||||
|
self.month
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn day(&self) -> u8 {
|
||||||
|
self.day
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn day_of_week(&self) -> DayOfWeek {
|
||||||
|
self.day_of_week
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn hour(&self) -> u8 {
|
||||||
|
self.hour
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn minute(&self) -> u8 {
|
||||||
|
self.minute
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn second(&self) -> u8 {
|
||||||
|
self.second
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from(
|
||||||
|
year: u16,
|
||||||
|
month: u8,
|
||||||
|
day: u8,
|
||||||
|
day_of_week: u8,
|
||||||
|
hour: u8,
|
||||||
|
minute: u8,
|
||||||
|
second: u8,
|
||||||
|
) -> Result<Self, Error> {
|
||||||
|
let day_of_week = day_of_week_from_u8(day_of_week)?;
|
||||||
|
|
||||||
|
if year > 4095 {
|
||||||
|
Err(Error::InvalidYear)
|
||||||
|
} else if month < 1 || month > 12 {
|
||||||
|
Err(Error::InvalidMonth)
|
||||||
|
} else if day < 1 || day > 31 {
|
||||||
|
Err(Error::InvalidDay)
|
||||||
|
} else if hour > 23 {
|
||||||
|
Err(Error::InvalidHour)
|
||||||
|
} else if minute > 59 {
|
||||||
|
Err(Error::InvalidMinute)
|
||||||
|
} else if second > 59 {
|
||||||
|
Err(Error::InvalidSecond)
|
||||||
|
} else {
|
||||||
|
Ok(Self {
|
||||||
|
year,
|
||||||
|
month,
|
||||||
|
day,
|
||||||
|
day_of_week,
|
||||||
|
hour,
|
||||||
|
minute,
|
||||||
|
second,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "chrono")]
|
#[cfg(feature = "chrono")]
|
||||||
@ -142,58 +260,3 @@ pub(super) fn validate_datetime(dt: &DateTime) -> Result<(), Error> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn write_date_time(rtc: &Rtc, t: DateTime) {
|
|
||||||
let (ht, hu) = byte_to_bcd2(t.hour as u8);
|
|
||||||
let (mnt, mnu) = byte_to_bcd2(t.minute as u8);
|
|
||||||
let (st, su) = byte_to_bcd2(t.second as u8);
|
|
||||||
|
|
||||||
let (dt, du) = byte_to_bcd2(t.day as u8);
|
|
||||||
let (mt, mu) = byte_to_bcd2(t.month as u8);
|
|
||||||
let yr = t.year as u16;
|
|
||||||
let yr_offset = (yr - 1970_u16) as u8;
|
|
||||||
let (yt, yu) = byte_to_bcd2(yr_offset);
|
|
||||||
|
|
||||||
use crate::pac::rtc::vals::Ampm;
|
|
||||||
|
|
||||||
rtc.tr().write(|w| {
|
|
||||||
w.set_ht(ht);
|
|
||||||
w.set_hu(hu);
|
|
||||||
w.set_mnt(mnt);
|
|
||||||
w.set_mnu(mnu);
|
|
||||||
w.set_st(st);
|
|
||||||
w.set_su(su);
|
|
||||||
w.set_pm(Ampm::AM);
|
|
||||||
});
|
|
||||||
|
|
||||||
rtc.dr().write(|w| {
|
|
||||||
w.set_dt(dt);
|
|
||||||
w.set_du(du);
|
|
||||||
w.set_mt(mt > 0);
|
|
||||||
w.set_mu(mu);
|
|
||||||
w.set_yt(yt);
|
|
||||||
w.set_yu(yu);
|
|
||||||
w.set_wdu(day_of_week_to_u8(t.day_of_week));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(super) fn datetime(
|
|
||||||
year: u16,
|
|
||||||
month: u8,
|
|
||||||
day: u8,
|
|
||||||
day_of_week: u8,
|
|
||||||
hour: u8,
|
|
||||||
minute: u8,
|
|
||||||
second: u8,
|
|
||||||
) -> Result<DateTime, Error> {
|
|
||||||
let day_of_week = day_of_week_from_u8(day_of_week)?;
|
|
||||||
Ok(DateTime {
|
|
||||||
year,
|
|
||||||
month,
|
|
||||||
day,
|
|
||||||
day_of_week,
|
|
||||||
hour,
|
|
||||||
minute,
|
|
||||||
second,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
@ -9,7 +9,8 @@ use embassy_sync::blocking_mutex::raw::CriticalSectionRawMutex;
|
|||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
use embassy_sync::blocking_mutex::Mutex;
|
use embassy_sync::blocking_mutex::Mutex;
|
||||||
|
|
||||||
pub use self::datetime::{DateTime, DayOfWeek, Error as DateTimeError};
|
pub use self::datetime::{DateTime, DayOfWeek, Error as DateTimeError, RtcInstant};
|
||||||
|
use crate::rtc::datetime::day_of_week_to_u8;
|
||||||
use crate::time::Hertz;
|
use crate::time::Hertz;
|
||||||
|
|
||||||
/// refer to AN4759 to compare features of RTC2 and RTC3
|
/// refer to AN4759 to compare features of RTC2 and RTC3
|
||||||
@ -39,48 +40,6 @@ pub enum RtcError {
|
|||||||
NotRunning,
|
NotRunning,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
/// Represents an instant in time that can be substracted to compute a duration
|
|
||||||
struct RtcInstant {
|
|
||||||
second: u8,
|
|
||||||
subsecond: u16,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(all(feature = "low-power", feature = "defmt"))]
|
|
||||||
impl defmt::Format for RtcInstant {
|
|
||||||
fn format(&self, fmt: defmt::Formatter) {
|
|
||||||
defmt::write!(
|
|
||||||
fmt,
|
|
||||||
"{}:{}",
|
|
||||||
self.second,
|
|
||||||
RTC::regs().prer().read().prediv_s() - self.subsecond,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
|
||||||
impl core::ops::Sub for RtcInstant {
|
|
||||||
type Output = embassy_time::Duration;
|
|
||||||
|
|
||||||
fn sub(self, rhs: Self) -> Self::Output {
|
|
||||||
use embassy_time::{Duration, TICK_HZ};
|
|
||||||
|
|
||||||
let second = if self.second < rhs.second {
|
|
||||||
self.second + 60
|
|
||||||
} else {
|
|
||||||
self.second
|
|
||||||
};
|
|
||||||
|
|
||||||
let psc = RTC::regs().prer().read().prediv_s() as u32;
|
|
||||||
|
|
||||||
let self_ticks = second as u32 * (psc + 1) + (psc - self.subsecond as u32);
|
|
||||||
let other_ticks = rhs.second as u32 * (psc + 1) + (psc - rhs.subsecond as u32);
|
|
||||||
let rtc_ticks = self_ticks - other_ticks;
|
|
||||||
|
|
||||||
Duration::from_ticks(((rtc_ticks * TICK_HZ as u32) / (psc + 1)) as u64)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct RtcTimeProvider {
|
pub struct RtcTimeProvider {
|
||||||
_private: (),
|
_private: (),
|
||||||
}
|
}
|
||||||
@ -113,7 +72,7 @@ impl RtcTimeProvider {
|
|||||||
let month = bcd2_to_byte((dr.mt() as u8, dr.mu()));
|
let month = bcd2_to_byte((dr.mt() as u8, dr.mu()));
|
||||||
let year = bcd2_to_byte((dr.yt(), dr.yu())) as u16 + 1970_u16;
|
let year = bcd2_to_byte((dr.yt(), dr.yu())) as u16 + 1970_u16;
|
||||||
|
|
||||||
return self::datetime::datetime(year, month, day, weekday, hour, minute, second)
|
return DateTime::from(year, month, day, weekday, hour, minute, second)
|
||||||
.map_err(RtcError::InvalidDateTime);
|
.map_err(RtcError::InvalidDateTime);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -134,7 +93,7 @@ impl RtcTimeProvider {
|
|||||||
let month = bcd2_to_byte((dr.mt() as u8, dr.mu()));
|
let month = bcd2_to_byte((dr.mt() as u8, dr.mu()));
|
||||||
let year = bcd2_to_byte((dr.yt(), dr.yu())) as u16 + 1970_u16;
|
let year = bcd2_to_byte((dr.yt(), dr.yu())) as u16 + 1970_u16;
|
||||||
|
|
||||||
self::datetime::datetime(year, month, day, weekday, hour, minute, second).map_err(RtcError::InvalidDateTime)
|
DateTime::from(year, month, day, weekday, hour, minute, second).map_err(RtcError::InvalidDateTime)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -187,7 +146,9 @@ impl Rtc {
|
|||||||
critical_section::with(|cs| {
|
critical_section::with(|cs| {
|
||||||
<RTC as crate::rcc::sealed::RccPeripheral>::enable_and_reset_with_cs(cs);
|
<RTC as crate::rcc::sealed::RccPeripheral>::enable_and_reset_with_cs(cs);
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
crate::rcc::clock_refcount_sub(cs);
|
unsafe {
|
||||||
|
crate::rcc::REFCOUNT_STOP2 -= 1
|
||||||
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
let mut this = Self {
|
let mut this = Self {
|
||||||
@ -223,14 +184,46 @@ impl Rtc {
|
|||||||
/// Will return `RtcError::InvalidDateTime` if the datetime is not a valid range.
|
/// Will return `RtcError::InvalidDateTime` if the datetime is not a valid range.
|
||||||
pub fn set_datetime(&mut self, t: DateTime) -> Result<(), RtcError> {
|
pub fn set_datetime(&mut self, t: DateTime) -> Result<(), RtcError> {
|
||||||
self::datetime::validate_datetime(&t).map_err(RtcError::InvalidDateTime)?;
|
self::datetime::validate_datetime(&t).map_err(RtcError::InvalidDateTime)?;
|
||||||
self.write(true, |rtc| self::datetime::write_date_time(rtc, t));
|
self.write(true, |rtc| {
|
||||||
|
let (ht, hu) = byte_to_bcd2(t.hour() as u8);
|
||||||
|
let (mnt, mnu) = byte_to_bcd2(t.minute() as u8);
|
||||||
|
let (st, su) = byte_to_bcd2(t.second() as u8);
|
||||||
|
|
||||||
|
let (dt, du) = byte_to_bcd2(t.day() as u8);
|
||||||
|
let (mt, mu) = byte_to_bcd2(t.month() as u8);
|
||||||
|
let yr = t.year() as u16;
|
||||||
|
let yr_offset = (yr - 1970_u16) as u8;
|
||||||
|
let (yt, yu) = byte_to_bcd2(yr_offset);
|
||||||
|
|
||||||
|
use crate::pac::rtc::vals::Ampm;
|
||||||
|
|
||||||
|
rtc.tr().write(|w| {
|
||||||
|
w.set_ht(ht);
|
||||||
|
w.set_hu(hu);
|
||||||
|
w.set_mnt(mnt);
|
||||||
|
w.set_mnu(mnu);
|
||||||
|
w.set_st(st);
|
||||||
|
w.set_su(su);
|
||||||
|
w.set_pm(Ampm::AM);
|
||||||
|
});
|
||||||
|
|
||||||
|
rtc.dr().write(|w| {
|
||||||
|
w.set_dt(dt);
|
||||||
|
w.set_du(du);
|
||||||
|
w.set_mt(mt > 0);
|
||||||
|
w.set_mu(mu);
|
||||||
|
w.set_yt(yt);
|
||||||
|
w.set_yu(yu);
|
||||||
|
w.set_wdu(day_of_week_to_u8(t.day_of_week()));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(not(rtc_v2f2))]
|
||||||
/// Return the current instant.
|
/// Return the current instant.
|
||||||
fn instant(&self) -> RtcInstant {
|
pub fn instant(&self) -> Result<RtcInstant, RtcError> {
|
||||||
let r = RTC::regs();
|
let r = RTC::regs();
|
||||||
let tr = r.tr().read();
|
let tr = r.tr().read();
|
||||||
let subsecond = r.ssr().read().ss();
|
let subsecond = r.ssr().read().ss();
|
||||||
@ -239,7 +232,7 @@ impl Rtc {
|
|||||||
// Unlock the registers
|
// Unlock the registers
|
||||||
r.dr().read();
|
r.dr().read();
|
||||||
|
|
||||||
RtcInstant { second, subsecond }
|
RtcInstant::from(second, subsecond.try_into().unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return the current datetime.
|
/// Return the current datetime.
|
||||||
|
@ -95,15 +95,16 @@ impl super::Rtc {
|
|||||||
regs.cr().modify(|w| w.set_wutie(true));
|
regs.cr().modify(|w| w.set_wutie(true));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let instant = self.instant().unwrap();
|
||||||
trace!(
|
trace!(
|
||||||
"rtc: start wakeup alarm for {} ms (psc: {}, ticks: {}) at {}",
|
"rtc: start wakeup alarm for {} ms (psc: {}, ticks: {}) at {}",
|
||||||
Duration::from_ticks(rtc_ticks as u64 * TICK_HZ * prescaler as u64 / rtc_hz).as_millis(),
|
Duration::from_ticks(rtc_ticks as u64 * TICK_HZ * prescaler as u64 / rtc_hz).as_millis(),
|
||||||
prescaler as u32,
|
prescaler as u32,
|
||||||
rtc_ticks,
|
rtc_ticks,
|
||||||
self.instant(),
|
instant,
|
||||||
);
|
);
|
||||||
|
|
||||||
assert!(self.stop_time.borrow(cs).replace(Some(self.instant())).is_none())
|
assert!(self.stop_time.borrow(cs).replace(Some(instant)).is_none())
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
@ -112,8 +113,9 @@ impl super::Rtc {
|
|||||||
pub(crate) fn stop_wakeup_alarm(&self, cs: critical_section::CriticalSection) -> Option<embassy_time::Duration> {
|
pub(crate) fn stop_wakeup_alarm(&self, cs: critical_section::CriticalSection) -> Option<embassy_time::Duration> {
|
||||||
use crate::interrupt::typelevel::Interrupt;
|
use crate::interrupt::typelevel::Interrupt;
|
||||||
|
|
||||||
|
let instant = self.instant().unwrap();
|
||||||
if RTC::regs().cr().read().wute() {
|
if RTC::regs().cr().read().wute() {
|
||||||
trace!("rtc: stop wakeup alarm at {}", self.instant());
|
trace!("rtc: stop wakeup alarm at {}", instant);
|
||||||
|
|
||||||
self.write(false, |regs| {
|
self.write(false, |regs| {
|
||||||
regs.cr().modify(|w| w.set_wutie(false));
|
regs.cr().modify(|w| w.set_wutie(false));
|
||||||
@ -128,10 +130,7 @@ impl super::Rtc {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
self.stop_time
|
self.stop_time.borrow(cs).take().map(|stop_time| instant - stop_time)
|
||||||
.borrow(cs)
|
|
||||||
.take()
|
|
||||||
.map(|stop_time| self.instant() - stop_time)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "low-power")]
|
#[cfg(feature = "low-power")]
|
||||||
|
@ -116,28 +116,28 @@ pub struct BufferedUartRx<'d, T: BasicInstance> {
|
|||||||
|
|
||||||
impl<'d, T: BasicInstance> SetConfig for BufferedUart<'d, T> {
|
impl<'d, T: BasicInstance> SetConfig for BufferedUart<'d, T> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'d, T: BasicInstance> SetConfig for BufferedUartRx<'d, T> {
|
impl<'d, T: BasicInstance> SetConfig for BufferedUartRx<'d, T> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'d, T: BasicInstance> SetConfig for BufferedUartTx<'d, T> {
|
impl<'d, T: BasicInstance> SetConfig for BufferedUartTx<'d, T> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -233,9 +233,6 @@ impl<'d, T: BasicInstance> BufferedUart<'d, T> {
|
|||||||
configure(r, &config, T::frequency(), T::KIND, true, true)?;
|
configure(r, &config, T::frequency(), T::KIND, true, true)?;
|
||||||
|
|
||||||
r.cr1().modify(|w| {
|
r.cr1().modify(|w| {
|
||||||
#[cfg(lpuart_v2)]
|
|
||||||
w.set_fifoen(true);
|
|
||||||
|
|
||||||
w.set_rxneie(true);
|
w.set_rxneie(true);
|
||||||
w.set_idleie(true);
|
w.set_idleie(true);
|
||||||
});
|
});
|
||||||
@ -254,7 +251,14 @@ impl<'d, T: BasicInstance> BufferedUart<'d, T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
||||||
reconfigure::<T>(config)
|
reconfigure::<T>(config)?;
|
||||||
|
|
||||||
|
T::regs().cr1().modify(|w| {
|
||||||
|
w.set_rxneie(true);
|
||||||
|
w.set_idleie(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -334,7 +338,14 @@ impl<'d, T: BasicInstance> BufferedUartRx<'d, T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
||||||
reconfigure::<T>(config)
|
reconfigure::<T>(config)?;
|
||||||
|
|
||||||
|
T::regs().cr1().modify(|w| {
|
||||||
|
w.set_rxneie(true);
|
||||||
|
w.set_idleie(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -408,7 +419,14 @@ impl<'d, T: BasicInstance> BufferedUartTx<'d, T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
pub fn set_config(&mut self, config: &Config) -> Result<(), ConfigError> {
|
||||||
reconfigure::<T>(config)
|
reconfigure::<T>(config)?;
|
||||||
|
|
||||||
|
T::regs().cr1().modify(|w| {
|
||||||
|
w.set_rxneie(true);
|
||||||
|
w.set_idleie(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -108,6 +108,7 @@ pub enum StopBits {
|
|||||||
pub enum ConfigError {
|
pub enum ConfigError {
|
||||||
BaudrateTooLow,
|
BaudrateTooLow,
|
||||||
BaudrateTooHigh,
|
BaudrateTooHigh,
|
||||||
|
RxOrTxNotEnabled,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[non_exhaustive]
|
#[non_exhaustive]
|
||||||
@ -181,11 +182,11 @@ pub struct Uart<'d, T: BasicInstance, TxDma = NoDma, RxDma = NoDma> {
|
|||||||
|
|
||||||
impl<'d, T: BasicInstance, TxDma, RxDma> SetConfig for Uart<'d, T, TxDma, RxDma> {
|
impl<'d, T: BasicInstance, TxDma, RxDma> SetConfig for Uart<'d, T, TxDma, RxDma> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.tx.set_config(config).map_err(|_| ())?;
|
self.tx.set_config(config)?;
|
||||||
self.rx.set_config(config).map_err(|_| ())
|
self.rx.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -196,10 +197,10 @@ pub struct UartTx<'d, T: BasicInstance, TxDma = NoDma> {
|
|||||||
|
|
||||||
impl<'d, T: BasicInstance, TxDma> SetConfig for UartTx<'d, T, TxDma> {
|
impl<'d, T: BasicInstance, TxDma> SetConfig for UartTx<'d, T, TxDma> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -213,10 +214,10 @@ pub struct UartRx<'d, T: BasicInstance, RxDma = NoDma> {
|
|||||||
|
|
||||||
impl<'d, T: BasicInstance, RxDma> SetConfig for UartRx<'d, T, RxDma> {
|
impl<'d, T: BasicInstance, RxDma> SetConfig for UartRx<'d, T, RxDma> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -866,7 +867,7 @@ fn configure(
|
|||||||
enable_tx: bool,
|
enable_tx: bool,
|
||||||
) -> Result<(), ConfigError> {
|
) -> Result<(), ConfigError> {
|
||||||
if !enable_rx && !enable_tx {
|
if !enable_rx && !enable_tx {
|
||||||
panic!("USART: At least one of RX or TX should be enabled");
|
return Err(ConfigError::RxOrTxNotEnabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(usart_v4))]
|
#[cfg(not(usart_v4))]
|
||||||
@ -909,6 +910,11 @@ fn configure(
|
|||||||
brr + rounding
|
brr + rounding
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UART must be disabled during configuration.
|
||||||
|
r.cr1().modify(|w| {
|
||||||
|
w.set_ue(false);
|
||||||
|
});
|
||||||
|
|
||||||
#[cfg(not(usart_v1))]
|
#[cfg(not(usart_v1))]
|
||||||
let mut over8 = false;
|
let mut over8 = false;
|
||||||
let mut found_brr = None;
|
let mut found_brr = None;
|
||||||
@ -968,6 +974,12 @@ fn configure(
|
|||||||
#[cfg(any(usart_v3, usart_v4))]
|
#[cfg(any(usart_v3, usart_v4))]
|
||||||
w.set_swap(config.swap_rx_tx);
|
w.set_swap(config.swap_rx_tx);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
#[cfg(not(usart_v1))]
|
||||||
|
r.cr3().modify(|w| {
|
||||||
|
w.set_onebit(config.assume_noise_free);
|
||||||
|
});
|
||||||
|
|
||||||
r.cr1().write(|w| {
|
r.cr1().write(|w| {
|
||||||
// enable uart
|
// enable uart
|
||||||
w.set_ue(true);
|
w.set_ue(true);
|
||||||
@ -976,6 +988,7 @@ fn configure(
|
|||||||
// enable receiver
|
// enable receiver
|
||||||
w.set_re(enable_rx);
|
w.set_re(enable_rx);
|
||||||
// configure word size
|
// configure word size
|
||||||
|
// if using odd or even parity it must be configured to 9bits
|
||||||
w.set_m0(if config.parity != Parity::ParityNone {
|
w.set_m0(if config.parity != Parity::ParityNone {
|
||||||
vals::M0::BIT9
|
vals::M0::BIT9
|
||||||
} else {
|
} else {
|
||||||
@ -994,11 +1007,6 @@ fn configure(
|
|||||||
w.set_fifoen(true);
|
w.set_fifoen(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
#[cfg(not(usart_v1))]
|
|
||||||
r.cr3().modify(|w| {
|
|
||||||
w.set_onebit(config.assume_noise_free);
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,10 +18,10 @@ pub struct RingBufferedUartRx<'d, T: BasicInstance, RxDma: super::RxDma<T>> {
|
|||||||
|
|
||||||
impl<'d, T: BasicInstance, RxDma: super::RxDma<T>> SetConfig for RingBufferedUartRx<'d, T, RxDma> {
|
impl<'d, T: BasicInstance, RxDma: super::RxDma<T>> SetConfig for RingBufferedUartRx<'d, T, RxDma> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
type ConfigError = ();
|
type ConfigError = ConfigError;
|
||||||
|
|
||||||
fn set_config(&mut self, config: &Self::Config) -> Result<(), ()> {
|
fn set_config(&mut self, config: &Self::Config) -> Result<(), Self::ConfigError> {
|
||||||
self.set_config(config).map_err(|_| ())
|
self.set_config(config)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,6 +5,10 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## 0.1.6 - ???
|
||||||
|
|
||||||
|
- Added tick rates in multiples of 10 kHz
|
||||||
|
|
||||||
## 0.1.5 - 2023-10-16
|
## 0.1.5 - 2023-10-16
|
||||||
|
|
||||||
- Added `links` key to Cargo.toml, to prevent multiple copies of this crate in the same binary.
|
- Added `links` key to Cargo.toml, to prevent multiple copies of this crate in the same binary.
|
||||||
|
@ -126,6 +126,25 @@ tick-hz-65_536_000 = []
|
|||||||
tick-hz-131_072_000 = []
|
tick-hz-131_072_000 = []
|
||||||
tick-hz-262_144_000 = []
|
tick-hz-262_144_000 = []
|
||||||
tick-hz-524_288_000 = []
|
tick-hz-524_288_000 = []
|
||||||
|
tick-hz-20_000 = []
|
||||||
|
tick-hz-40_000 = []
|
||||||
|
tick-hz-80_000 = []
|
||||||
|
tick-hz-160_000 = []
|
||||||
|
tick-hz-320_000 = []
|
||||||
|
tick-hz-640_000 = []
|
||||||
|
tick-hz-1_280_000 = []
|
||||||
|
tick-hz-2_560_000 = []
|
||||||
|
tick-hz-5_120_000 = []
|
||||||
|
tick-hz-10_240_000 = []
|
||||||
|
tick-hz-20_480_000 = []
|
||||||
|
tick-hz-40_960_000 = []
|
||||||
|
tick-hz-81_920_000 = []
|
||||||
|
tick-hz-163_840_000 = []
|
||||||
|
tick-hz-327_680_000 = []
|
||||||
|
tick-hz-655_360_000 = []
|
||||||
|
tick-hz-1_310_720_000 = []
|
||||||
|
tick-hz-2_621_440_000 = []
|
||||||
|
tick-hz-5_242_880_000 = []
|
||||||
tick-hz-2_000_000 = []
|
tick-hz-2_000_000 = []
|
||||||
tick-hz-3_000_000 = []
|
tick-hz-3_000_000 = []
|
||||||
tick-hz-4_000_000 = []
|
tick-hz-4_000_000 = []
|
||||||
|
@ -13,6 +13,8 @@ for i in range(1, 25):
|
|||||||
ticks.append(2**i)
|
ticks.append(2**i)
|
||||||
for i in range(1, 20):
|
for i in range(1, 20):
|
||||||
ticks.append(2**i * 1000)
|
ticks.append(2**i * 1000)
|
||||||
|
for i in range(1, 20):
|
||||||
|
ticks.append(2**i * 10000)
|
||||||
for i in range(1, 10):
|
for i in range(1, 10):
|
||||||
ticks.append(2**i * 1000000)
|
ticks.append(2**i * 1000000)
|
||||||
ticks.append(2**i * 9 // 8 * 1000000)
|
ticks.append(2**i * 9 // 8 * 1000000)
|
||||||
|
@ -106,6 +106,44 @@ pub const TICK_HZ: u64 = 131_072_000;
|
|||||||
pub const TICK_HZ: u64 = 262_144_000;
|
pub const TICK_HZ: u64 = 262_144_000;
|
||||||
#[cfg(feature = "tick-hz-524_288_000")]
|
#[cfg(feature = "tick-hz-524_288_000")]
|
||||||
pub const TICK_HZ: u64 = 524_288_000;
|
pub const TICK_HZ: u64 = 524_288_000;
|
||||||
|
#[cfg(feature = "tick-hz-20_000")]
|
||||||
|
pub const TICK_HZ: u64 = 20_000;
|
||||||
|
#[cfg(feature = "tick-hz-40_000")]
|
||||||
|
pub const TICK_HZ: u64 = 40_000;
|
||||||
|
#[cfg(feature = "tick-hz-80_000")]
|
||||||
|
pub const TICK_HZ: u64 = 80_000;
|
||||||
|
#[cfg(feature = "tick-hz-160_000")]
|
||||||
|
pub const TICK_HZ: u64 = 160_000;
|
||||||
|
#[cfg(feature = "tick-hz-320_000")]
|
||||||
|
pub const TICK_HZ: u64 = 320_000;
|
||||||
|
#[cfg(feature = "tick-hz-640_000")]
|
||||||
|
pub const TICK_HZ: u64 = 640_000;
|
||||||
|
#[cfg(feature = "tick-hz-1_280_000")]
|
||||||
|
pub const TICK_HZ: u64 = 1_280_000;
|
||||||
|
#[cfg(feature = "tick-hz-2_560_000")]
|
||||||
|
pub const TICK_HZ: u64 = 2_560_000;
|
||||||
|
#[cfg(feature = "tick-hz-5_120_000")]
|
||||||
|
pub const TICK_HZ: u64 = 5_120_000;
|
||||||
|
#[cfg(feature = "tick-hz-10_240_000")]
|
||||||
|
pub const TICK_HZ: u64 = 10_240_000;
|
||||||
|
#[cfg(feature = "tick-hz-20_480_000")]
|
||||||
|
pub const TICK_HZ: u64 = 20_480_000;
|
||||||
|
#[cfg(feature = "tick-hz-40_960_000")]
|
||||||
|
pub const TICK_HZ: u64 = 40_960_000;
|
||||||
|
#[cfg(feature = "tick-hz-81_920_000")]
|
||||||
|
pub const TICK_HZ: u64 = 81_920_000;
|
||||||
|
#[cfg(feature = "tick-hz-163_840_000")]
|
||||||
|
pub const TICK_HZ: u64 = 163_840_000;
|
||||||
|
#[cfg(feature = "tick-hz-327_680_000")]
|
||||||
|
pub const TICK_HZ: u64 = 327_680_000;
|
||||||
|
#[cfg(feature = "tick-hz-655_360_000")]
|
||||||
|
pub const TICK_HZ: u64 = 655_360_000;
|
||||||
|
#[cfg(feature = "tick-hz-1_310_720_000")]
|
||||||
|
pub const TICK_HZ: u64 = 1_310_720_000;
|
||||||
|
#[cfg(feature = "tick-hz-2_621_440_000")]
|
||||||
|
pub const TICK_HZ: u64 = 2_621_440_000;
|
||||||
|
#[cfg(feature = "tick-hz-5_242_880_000")]
|
||||||
|
pub const TICK_HZ: u64 = 5_242_880_000;
|
||||||
#[cfg(feature = "tick-hz-2_000_000")]
|
#[cfg(feature = "tick-hz-2_000_000")]
|
||||||
pub const TICK_HZ: u64 = 2_000_000;
|
pub const TICK_HZ: u64 = 2_000_000;
|
||||||
#[cfg(feature = "tick-hz-3_000_000")]
|
#[cfg(feature = "tick-hz-3_000_000")]
|
||||||
@ -334,6 +372,25 @@ pub const TICK_HZ: u64 = 980_000_000;
|
|||||||
feature = "tick-hz-131_072_000",
|
feature = "tick-hz-131_072_000",
|
||||||
feature = "tick-hz-262_144_000",
|
feature = "tick-hz-262_144_000",
|
||||||
feature = "tick-hz-524_288_000",
|
feature = "tick-hz-524_288_000",
|
||||||
|
feature = "tick-hz-20_000",
|
||||||
|
feature = "tick-hz-40_000",
|
||||||
|
feature = "tick-hz-80_000",
|
||||||
|
feature = "tick-hz-160_000",
|
||||||
|
feature = "tick-hz-320_000",
|
||||||
|
feature = "tick-hz-640_000",
|
||||||
|
feature = "tick-hz-1_280_000",
|
||||||
|
feature = "tick-hz-2_560_000",
|
||||||
|
feature = "tick-hz-5_120_000",
|
||||||
|
feature = "tick-hz-10_240_000",
|
||||||
|
feature = "tick-hz-20_480_000",
|
||||||
|
feature = "tick-hz-40_960_000",
|
||||||
|
feature = "tick-hz-81_920_000",
|
||||||
|
feature = "tick-hz-163_840_000",
|
||||||
|
feature = "tick-hz-327_680_000",
|
||||||
|
feature = "tick-hz-655_360_000",
|
||||||
|
feature = "tick-hz-1_310_720_000",
|
||||||
|
feature = "tick-hz-2_621_440_000",
|
||||||
|
feature = "tick-hz-5_242_880_000",
|
||||||
feature = "tick-hz-2_000_000",
|
feature = "tick-hz-2_000_000",
|
||||||
feature = "tick-hz-3_000_000",
|
feature = "tick-hz-3_000_000",
|
||||||
feature = "tick-hz-4_000_000",
|
feature = "tick-hz-4_000_000",
|
||||||
|
@ -10,8 +10,8 @@ use chrono::NaiveDate;
|
|||||||
use common::*;
|
use common::*;
|
||||||
use cortex_m_rt::entry;
|
use cortex_m_rt::entry;
|
||||||
use embassy_executor::Spawner;
|
use embassy_executor::Spawner;
|
||||||
use embassy_stm32::low_power::{stop_with_rtc, Executor};
|
use embassy_stm32::low_power::{stop_ready, stop_with_rtc, Executor, StopMode};
|
||||||
use embassy_stm32::rcc::{low_power_ready, LsConfig};
|
use embassy_stm32::rcc::LsConfig;
|
||||||
use embassy_stm32::rtc::{Rtc, RtcConfig};
|
use embassy_stm32::rtc::{Rtc, RtcConfig};
|
||||||
use embassy_stm32::Config;
|
use embassy_stm32::Config;
|
||||||
use embassy_time::Timer;
|
use embassy_time::Timer;
|
||||||
@ -28,7 +28,7 @@ fn main() -> ! {
|
|||||||
async fn task_1() {
|
async fn task_1() {
|
||||||
for _ in 0..9 {
|
for _ in 0..9 {
|
||||||
info!("task 1: waiting for 500ms...");
|
info!("task 1: waiting for 500ms...");
|
||||||
defmt::assert!(low_power_ready());
|
defmt::assert!(stop_ready(StopMode::Stop2));
|
||||||
Timer::after_millis(500).await;
|
Timer::after_millis(500).await;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -37,7 +37,7 @@ async fn task_1() {
|
|||||||
async fn task_2() {
|
async fn task_2() {
|
||||||
for _ in 0..5 {
|
for _ in 0..5 {
|
||||||
info!("task 2: waiting for 1000ms...");
|
info!("task 2: waiting for 1000ms...");
|
||||||
defmt::assert!(low_power_ready());
|
defmt::assert!(stop_ready(StopMode::Stop2));
|
||||||
Timer::after_millis(1000).await;
|
Timer::after_millis(1000).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user