ra-hal 0.3.0

Hardware Abstraction Layer (HAL) for the Renesas RA family of MCUs.
//! Realtime Clock (`RTC`).

use core::marker::PhantomData;

use chrono::{Datelike, NaiveDate, NaiveDateTime, NaiveTime, Timelike};
use cortex_m::asm;
use embassy_hal_internal::{Peri, PeripheralType};
use embassy_time::Timer;

use crate::{
    pac::{
        self,
        rtc::vals::{Rcksel, RwkcntDayw},
    },
    peripherals,
    write_protect::ProtectedPeripheral as _,
};

/// Realtime clock driver.
///
/// # Notes
///
/// On the `RA4M1` the clock source can either be an internal or external oscillator.
/// The internal oscillator used is the low-speed on-chip oscillator (`LOCO`) which is accurate to within 15% (§ 48.3.2).
/// This can lead to significant drift which can be mitigated somewhat with trimming.
/// Trimming has not yet been implemented.
pub struct Rtc<'d, I: Instance> {
    _phantom: PhantomData<&'d I>,
}

/// Realtime clock instance.
#[allow(private_bounds)]
pub trait Instance: SealedInstance + PeripheralType + 'static + Send {}

trait SealedInstance {
    fn regs() -> pac::rtc::Rtc;
}

impl Instance for peripherals::RTC {}

impl SealedInstance for peripherals::RTC {
    fn regs() -> pac::rtc::Rtc {
        pac::RTC
    }
}

impl<'d, I: Instance> Rtc<'d, I> {
    /// # Arguments
    /// * `_rtc` A peripheral that implements [`Instance`].
    ///   The `RA4M1` only has one realtime clock available as [`RTC`](peripherals::RTC).
    ///
    /// # Returns
    ///
    /// An RTC driver.
    pub async fn new(_rtc: Peri<'d, I>) -> Self {
        let system = pac::SYSTEM;
        if system.lococr().read().lcstp() {
            warn!("RTC: LOCO not running. Attempting start.");

            let vbatt_enabled = system.vbtcr1().read().bpwswstp();

            if vbatt_enabled {
                system.vbtcr1().write(|w| w.set_bpwswstp(true));
            }

            system.lococr().modify(|w| w.set_lcstp(false));

            // § 48.3.2
            Timer::after_micros(100).await;

            if vbatt_enabled {
                system.vbtcr1().write(|w| w.set_bpwswstp(false));

                while !system.vbtsr().read().vbtrvld() {
                    asm::nop();
                }
            }
        }

        let rtc = I::regs();

        let mut instance = Self {
            _phantom: PhantomData,
        };

        // §24.2.20
        rtc.rfrl().write_value(0xFF - 16);

        let system = pac::SYSTEM;
        system.protected_write(|| {
            system.locoutcr().write(|w| w.set_locoutrm((-128_i8) as u8));
        });

        info!(
            "Trim: act={:08b}, expect={:08b}",
            system.locoutcr().read().locoutrm(),
            0b1111_1110
        );

        rtc.rcr4().modify(|w| w.set_rcksel(Rcksel::Loco));

        for _ in 0..6 {
            rtc.rcr4().read();
        }

        instance.stop();

        rtc.rcr2().modify(|w| {
            w.set_cntmd(false);
            // w.set_aadje(true);
            // w.set_aadjp(true);
        });
        while rtc.rcr2().read().cntmd() {
            asm::nop();
        }

        rtc.rcr2().modify(|w| w.set_reset(true));
        while rtc.rcr2().read().reset() {
            asm::nop();
        }

        rtc.rcr2().modify(|w| w.set_start(false));
        while rtc.rcr2().read().start() {
            asm::nop();
        }

        rtc.rseccnt().write(|w| {
            w.set_sec1(0);
            w.set_sec10(0);
        });

        rtc.rmincnt().write(|w| {
            w.set_min1(0);
            w.set_min10(0);
        });

        rtc.rhrcnt().write(|w| {
            w.set_hr1(0);
            w.set_hr10(0);
            w.set_pm(false);
        });

        rtc.rwkcnt().write(|w| w.set_dayw(RwkcntDayw::Friday));

        rtc.rmoncnt().write(|w| {
            w.set_mon10(false);
            w.set_mon1(1);
        });

        rtc.ryrcnt().write(|w| {
            w.set_yr10(7);
            w.set_yr1(0);
        });

        instance.start();

        instance
    }

    /// Starts the `RTC` peripheral.
    pub fn start(&mut self) {
        let rtc = I::regs();

        rtc.rcr2().modify(|w| w.set_start(true));

        while !rtc.rcr2().read().start() {
            asm::nop();
        }
    }

    /// Stops the `RTC` peripheral.
    pub fn stop(&mut self) {
        let rtc = I::regs();

        rtc.rcr2().modify(|w| w.set_start(false));

        while rtc.rcr2().read().start() {
            asm::nop();
        }
    }

    /// Sets the current datetime from values contained in a [`NaiveDateTime`].
    pub fn set_time(&mut self, date_time: NaiveDateTime) {
        let year = date_time.year_ce().1;
        let month = date_time.month();
        let day = date_time.day();
        let hour = date_time.hour();
        let minute = date_time.minute();
        let second = date_time.second();

        #[cfg(feature = "strict-assert")]
        {
            assert!(year < u16::MAX as u32);
            assert!(month < u8::MAX as u32);
            assert!(day < u8::MAX as u32);
            assert!(hour < u8::MAX as u32);
            assert!(minute < u8::MAX as u32);
            assert!(second < u8::MAX as u32);
        }

        self.set_time_ymd_hms(
            year as u16,
            month as u8,
            day as u8,
            hour as u8,
            minute as u8,
            second as u8,
        );
    }

    /// Sets the current datetime based on individual components.
    ///
    /// # Arguments
    /// * `year`  The `RA4M1` clock stores the year as two digits internally. The century component is discarded by this function.
    /// * `month` The month, `1..=12`
    /// * `day` The day, `1..=31` as appropriate.
    /// * `hour` The hour of day, `0..=23`.
    /// * `minute`  The minute, `0..=59`.
    /// * `second` The second, `0..=59`.
    pub fn set_time_ymd_hms(
        &mut self,
        year: u16,
        month: u8,
        day: u8,
        hour: u8,
        minute: u8,
        second: u8,
    ) {
        let rtc = I::regs();

        self.stop();

        rtc.ryrcnt().write(|w| {
            w.set_yr10(((year / 10) % 10) as _);
            w.set_yr1((year % 10) as _);
        });

        rtc.rmoncnt().write(|w| {
            w.set_mon10(((month / 10) % 10) == 1);
            w.set_mon1((month % 10) as _);
        });

        rtc.rdaycnt().write(|w| {
            w.set_date10(((day / 10) % 10) as _);
            w.set_date1((day % 10) as _);
        });

        rtc.rhrcnt().write(|w| {
            w.set_hr10(((hour / 10) % 10) as _);
            w.set_hr1((hour % 10) as _);
        });

        rtc.rmincnt().write(|w| {
            w.set_min10(((minute / 10) % 10) as _);
            w.set_min1((minute % 10) as _);
        });
        rtc.rseccnt().write(|w| {
            w.set_sec10(((second / 10) % 10) as _);
            w.set_sec1((second % 10) as _);
        });

        self.start();
    }

    /// # Returns
    ///
    /// The current datetime
    pub fn now(&self) -> NaiveDateTime {
        let rtc = I::regs();

        let year = rtc.ryrcnt().read();
        let mut year = i32::from(year.yr1() + (10 * year.yr10()));
        if year < 69 {
            year += 2000;
        } else {
            year += 1900;
        }

        let month = rtc.rmoncnt().read();
        let month = u32::from(month.mon1() + (10 * month.mon10() as u8));

        let day = rtc.rdaycnt().read();
        let day = u32::from(day.date1() + (10 * day.date10()));

        let date = NaiveDate::from_ymd_opt(year, month, day).unwrap();

        let hour = rtc.rhrcnt().read();
        let minute = rtc.rmincnt().read();
        let seconds = rtc.rseccnt().read();

        let hour = u32::from(hour.hr1() + (10 * hour.hr10()));
        let minute = u32::from(minute.min1() + (10 * minute.min10()));
        let seconds = u32::from(seconds.sec1() + (10 * seconds.sec10()));

        let time = NaiveTime::from_hms_opt(hour, minute, seconds).unwrap();

        NaiveDateTime::new(date, time)
    }
}