mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
359 lines
9.8 KiB
C
359 lines
9.8 KiB
C
/*
|
|
* Copyright (C) 2015 Lari Lehtomäki
|
|
* 2016 Laksh Bhatia
|
|
* 2016-2017 OTA keys S.A.
|
|
* 2017 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_stm32_common
|
|
* @{
|
|
* @file
|
|
* @brief Low-level RTC driver implementation
|
|
*
|
|
* @author Lari Lehtomäki <lari@lehtomaki.fi>
|
|
* @author Laksh Bhatia <bhatialaksh3@gmail.com>
|
|
* @author Vincent Dupont <vincent@otakeys.com>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
* @}
|
|
*/
|
|
|
|
#include <time.h>
|
|
#include "cpu.h"
|
|
#include "stmclk.h"
|
|
#include "periph/rtc.h"
|
|
|
|
/* this implementation does not work for the stm32f1 */
|
|
#if !defined(CPU_FAM_STM32F1)
|
|
|
|
/* map some CPU specific register names */
|
|
#if defined (CPU_FAM_STM32L0) || defined(CPU_FAM_STM32L1)
|
|
#define EN_REG (RCC->CSR)
|
|
#define EN_BIT (RCC_CSR_RTCEN)
|
|
#define CLKSEL_MASK (RCC_CSR_RTCSEL)
|
|
#define CLKSEL_LSE (RCC_CSR_RTCSEL_LSE)
|
|
#define CLKSEL_LSI (RCC_CSR_RTCSEL_LSI)
|
|
#else
|
|
#define EN_REG (RCC->BDCR)
|
|
#define EN_BIT (RCC_BDCR_RTCEN)
|
|
#define CLKSEL_MASK (RCC_BDCR_RTCSEL_0 | RCC_BDCR_RTCSEL_1)
|
|
#define CLKSEL_LSE (RCC_BDCR_RTCSEL_0)
|
|
#define CLKSEL_LSI (RCC_BDCR_RTCSEL_1)
|
|
#endif
|
|
|
|
/* interrupt line name mapping */
|
|
#if defined(CPU_FAM_STM32F0) || defined(CPU_FAM_STM32L0)
|
|
#define IRQN (RTC_IRQn)
|
|
#define ISR_NAME isr_rtc
|
|
#else
|
|
#define IRQN (RTC_Alarm_IRQn)
|
|
#define ISR_NAME isr_rtc_alarm
|
|
#endif
|
|
|
|
/* EXTI bitfield mapping */
|
|
#if defined(CPU_FAM_STM32L4)
|
|
#define EXTI_IMR_BIT (EXTI_IMR1_IM18)
|
|
#define EXTI_FTSR_BIT (EXTI_FTSR1_FT18)
|
|
#define EXTI_RTSR_BIT (EXTI_RTSR1_RT18)
|
|
#define EXTI_PR_BIT (EXTI_PR1_PIF18)
|
|
#elif defined(CPU_FAM_STM32WB)
|
|
#define EXTI_IMR_BIT (EXTI_IMR1_IM17)
|
|
#define EXTI_FTSR_BIT (EXTI_FTSR1_FT17)
|
|
#define EXTI_RTSR_BIT (EXTI_RTSR1_RT17)
|
|
#define EXTI_PR_BIT (EXTI_PR1_PIF17)
|
|
#else
|
|
#if defined(CPU_FAM_STM32L0)
|
|
#define EXTI_IMR_BIT (EXTI_IMR_IM17)
|
|
#else
|
|
#define EXTI_IMR_BIT (EXTI_IMR_MR17)
|
|
#endif
|
|
#define EXTI_FTSR_BIT (EXTI_FTSR_TR17)
|
|
#define EXTI_RTSR_BIT (EXTI_RTSR_TR17)
|
|
#define EXTI_PR_BIT (EXTI_PR_PR17)
|
|
#endif
|
|
|
|
/* write protection values */
|
|
#define WPK1 (0xCA)
|
|
#define WPK2 (0x53)
|
|
|
|
/* define TR, DR, and ALRMAR position and masks */
|
|
#define TR_H_MASK (RTC_TR_HU | RTC_TR_HT)
|
|
#define TR_M_MASK (RTC_TR_MNU | RTC_TR_MNT)
|
|
#define TR_S_MASK (RTC_TR_SU | RTC_TR_ST)
|
|
#define DR_Y_MASK (RTC_DR_YU | RTC_DR_YT)
|
|
#define DR_M_MASK (RTC_DR_MU | RTC_DR_MT)
|
|
#define DR_D_MASK (RTC_DR_DU | RTC_DR_DT)
|
|
#define ALRM_D_MASK (RTC_ALRMAR_DU | RTC_ALRMAR_DT)
|
|
#define ALRM_H_MASK (RTC_ALRMAR_HU | RTC_ALRMAR_HT)
|
|
#define ALRM_M_MASK (RTC_ALRMAR_MNU | RTC_ALRMAR_MNT)
|
|
#define ALRM_S_MASK (RTC_ALRMAR_SU | RTC_ALRMAR_ST)
|
|
#ifndef RTC_DR_YU_Pos
|
|
#define RTC_DR_YU_Pos (16U)
|
|
#endif
|
|
#ifndef RTC_DR_MU_Pos
|
|
#define RTC_DR_MU_Pos (8U)
|
|
#endif
|
|
#ifndef RTC_DR_DU_Pos
|
|
#define RTC_DR_DU_Pos (0U)
|
|
#endif
|
|
#ifndef RTC_TR_HU_Pos
|
|
#define RTC_TR_HU_Pos (16U)
|
|
#endif
|
|
#ifndef RTC_TR_MNU_Pos
|
|
#define RTC_TR_MNU_Pos (8U)
|
|
#endif
|
|
#ifndef RTC_TR_SU_Pos
|
|
#define RTC_TR_SU_Pos (0U)
|
|
#endif
|
|
#ifndef RTC_ALRMAR_DU_Pos
|
|
#define RTC_ALRMAR_DU_Pos (24U)
|
|
#endif
|
|
#ifndef RTC_ALRMAR_HU_Pos
|
|
#define RTC_ALRMAR_HU_Pos (16U)
|
|
#endif
|
|
#ifndef RTC_ALRMAR_MNU_Pos
|
|
#define RTC_ALRMAR_MNU_Pos (8U)
|
|
#endif
|
|
#ifndef RTC_ALRMAR_SU_Pos
|
|
#define RTC_ALRMAR_SU_Pos (0U)
|
|
#endif
|
|
|
|
/* figure out sync and async prescaler */
|
|
#if CLOCK_LSE
|
|
#define PRE_SYNC (255)
|
|
#define PRE_ASYNC (127)
|
|
#elif (CLOCK_LSI == 40000)
|
|
#define PRE_SYNC (319)
|
|
#define PRE_ASYNC (124)
|
|
#elif (CLOCK_LSI == 37000)
|
|
#define PRE_SYNC (295)
|
|
#define PRE_ASYNC (124)
|
|
#elif (CLOCK_LSI == 32000)
|
|
#define PRE_SYNC (249)
|
|
#define PRE_ASYNC (127)
|
|
#else
|
|
#error "rtc: unable to determine RTC SYNC and ASYNC prescalers from LSI value"
|
|
#endif
|
|
|
|
/* struct tm counts years since 1900 but RTC has only two-digit year hence the
|
|
* offset of 100 years. */
|
|
#define YEAR_OFFSET (100)
|
|
|
|
/* Use a magic number to determine the initial RTC source. This will be used
|
|
to know if a reset of the RTC is required at initialization. */
|
|
#if CLOCK_LSE
|
|
#define MAGIC_CLCK_NUMBER (0x1970)
|
|
#else
|
|
#define MAGIC_CLCK_NUMBER (0x1971)
|
|
#endif
|
|
|
|
static struct {
|
|
rtc_alarm_cb_t cb; /**< callback called from RTC interrupt */
|
|
void *arg; /**< argument passed to the callback */
|
|
} isr_ctx;
|
|
|
|
static uint32_t val2bcd(int val, int shift, uint32_t mask)
|
|
{
|
|
uint32_t bcdhigh = 0;
|
|
|
|
while (val >= 10) {
|
|
bcdhigh++;
|
|
val -= 10;
|
|
}
|
|
|
|
return ((((bcdhigh << 4) | val) << shift) & mask);
|
|
}
|
|
|
|
static int bcd2val(uint32_t val, int shift, uint32_t mask)
|
|
{
|
|
int tmp = (int)((val & mask) >> shift);
|
|
return (((tmp >> 4) * 10) + (tmp & 0x0f));
|
|
}
|
|
|
|
static inline void rtc_unlock(void)
|
|
{
|
|
/* enable backup clock domain */
|
|
stmclk_dbp_unlock();
|
|
/* unlock RTC */
|
|
RTC->WPR = WPK1;
|
|
RTC->WPR = WPK2;
|
|
/* enter RTC init mode */
|
|
RTC->ISR |= RTC_ISR_INIT;
|
|
while (!(RTC->ISR & RTC_ISR_INITF)) {}
|
|
}
|
|
|
|
static inline void rtc_lock(void)
|
|
{
|
|
/* exit RTC init mode */
|
|
RTC->ISR &= ~RTC_ISR_INIT;
|
|
while (RTC->ISR & RTC_ISR_INITF) {}
|
|
/* lock RTC device */
|
|
RTC->WPR = 0xff;
|
|
/* disable backup clock domain */
|
|
stmclk_dbp_lock();
|
|
}
|
|
|
|
void rtc_init(void)
|
|
{
|
|
stmclk_dbp_unlock();
|
|
#if defined(CPU_FAM_STM32L0) || defined(CPU_FAM_STM32L1)
|
|
/* Compare the stored magic number with the current one. If it's different
|
|
it means the clock source has changed and thus a RTC reset is
|
|
required. */
|
|
if (RTC->BKP0R != MAGIC_CLCK_NUMBER) {
|
|
RCC->CSR |= RCC_CSR_RTCRST;
|
|
RCC->CSR &= ~RCC_CSR_RTCRST;
|
|
RTC->BKP0R = MAGIC_CLCK_NUMBER; /* Store the new magic number */
|
|
}
|
|
#endif
|
|
stmclk_dbp_lock();
|
|
|
|
/* enable low frequency clock */
|
|
stmclk_enable_lfclk();
|
|
|
|
/* select input clock and enable the RTC */
|
|
stmclk_dbp_unlock();
|
|
EN_REG &= ~(CLKSEL_MASK);
|
|
#if CLOCK_LSE
|
|
EN_REG |= (CLKSEL_LSE | EN_BIT);
|
|
#else
|
|
EN_REG |= (CLKSEL_LSI | EN_BIT);
|
|
#endif
|
|
|
|
rtc_unlock();
|
|
/* reset configuration */
|
|
RTC->CR = 0;
|
|
RTC->ISR = RTC_ISR_INIT;
|
|
/* configure prescaler (RTC PRER) */
|
|
RTC->PRER = (PRE_SYNC | (PRE_ASYNC << 16));
|
|
rtc_lock();
|
|
|
|
/* configure the EXTI channel, as RTC interrupts are routed through it.
|
|
* Needs to be configured to trigger on rising edges. */
|
|
EXTI->FTSR &= ~(EXTI_FTSR_BIT);
|
|
EXTI->RTSR |= EXTI_RTSR_BIT;
|
|
EXTI->IMR |= EXTI_IMR_BIT;
|
|
EXTI->PR |= EXTI_PR_BIT;
|
|
/* enable global RTC interrupt */
|
|
NVIC_EnableIRQ(IRQN);
|
|
}
|
|
|
|
int rtc_set_time(struct tm *time)
|
|
{
|
|
/* normalize input */
|
|
rtc_tm_normalize(time);
|
|
|
|
rtc_unlock();
|
|
|
|
RTC->DR = (val2bcd((time->tm_year % 100), RTC_DR_YU_Pos, DR_Y_MASK) |
|
|
val2bcd(time->tm_mon + 1, RTC_DR_MU_Pos, DR_M_MASK) |
|
|
val2bcd(time->tm_mday, RTC_DR_DU_Pos, DR_D_MASK));
|
|
RTC->TR = (val2bcd(time->tm_hour, RTC_TR_HU_Pos, TR_H_MASK) |
|
|
val2bcd(time->tm_min, RTC_TR_MNU_Pos, TR_M_MASK) |
|
|
val2bcd(time->tm_sec, RTC_TR_SU_Pos, TR_S_MASK));
|
|
rtc_lock();
|
|
while (!(RTC->ISR & RTC_ISR_RSF)) {}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rtc_get_time(struct tm *time)
|
|
{
|
|
/* save current time */
|
|
uint32_t tr = RTC->TR;
|
|
uint32_t dr = RTC->DR;
|
|
time->tm_year = bcd2val(dr, RTC_DR_YU_Pos, DR_Y_MASK) + YEAR_OFFSET;
|
|
time->tm_mon = bcd2val(dr, RTC_DR_MU_Pos, DR_M_MASK) - 1;
|
|
time->tm_mday = bcd2val(dr, RTC_DR_DU_Pos, DR_D_MASK);
|
|
time->tm_hour = bcd2val(tr, RTC_TR_HU_Pos, TR_H_MASK);
|
|
time->tm_min = bcd2val(tr, RTC_TR_MNU_Pos, TR_M_MASK);
|
|
time->tm_sec = bcd2val(tr, RTC_TR_SU_Pos, TR_S_MASK);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
|
|
{
|
|
/* normalize input */
|
|
rtc_tm_normalize(time);
|
|
|
|
rtc_unlock();
|
|
|
|
/* disable existing alarm (if enabled) */
|
|
rtc_clear_alarm();
|
|
|
|
/* save callback and argument */
|
|
isr_ctx.cb = cb;
|
|
isr_ctx.arg = arg;
|
|
|
|
/* set wakeup time */
|
|
RTC->ALRMAR = (val2bcd(time->tm_mday, RTC_ALRMAR_DU_Pos, ALRM_D_MASK) |
|
|
val2bcd(time->tm_hour, RTC_ALRMAR_HU_Pos, ALRM_H_MASK) |
|
|
val2bcd(time->tm_min, RTC_ALRMAR_MNU_Pos, ALRM_M_MASK) |
|
|
val2bcd(time->tm_sec, RTC_ALRMAR_SU_Pos, ALRM_S_MASK));
|
|
|
|
/* Enable Alarm A */
|
|
RTC->ISR &= ~(RTC_ISR_ALRAF);
|
|
RTC->CR |= (RTC_CR_ALRAE | RTC_CR_ALRAIE);
|
|
|
|
rtc_lock();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rtc_get_alarm(struct tm *time)
|
|
{
|
|
uint32_t dr = RTC->DR;
|
|
uint32_t alrm = RTC->ALRMAR;
|
|
|
|
time->tm_year = bcd2val(dr, RTC_DR_YU_Pos, DR_Y_MASK) + YEAR_OFFSET;
|
|
time->tm_mon = bcd2val(dr, RTC_DR_MU_Pos, DR_M_MASK) - 1;
|
|
time->tm_mday = bcd2val(alrm, RTC_ALRMAR_DU_Pos, ALRM_D_MASK);
|
|
time->tm_hour = bcd2val(alrm, RTC_ALRMAR_HU_Pos, ALRM_H_MASK);
|
|
time->tm_min = bcd2val(alrm, RTC_ALRMAR_MNU_Pos, ALRM_M_MASK);
|
|
time->tm_sec = bcd2val(alrm, RTC_ALRMAR_SU_Pos, ALRM_S_MASK);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void rtc_clear_alarm(void)
|
|
{
|
|
RTC->CR &= ~(RTC_CR_ALRAE | RTC_CR_ALRAIE);
|
|
while (!(RTC->ISR & RTC_ISR_ALRAWF)) {}
|
|
|
|
isr_ctx.cb = NULL;
|
|
isr_ctx.arg = NULL;
|
|
}
|
|
|
|
void rtc_poweron(void)
|
|
{
|
|
stmclk_dbp_unlock();
|
|
EN_REG |= EN_BIT;
|
|
stmclk_dbp_lock();
|
|
}
|
|
|
|
void rtc_poweroff(void)
|
|
{
|
|
stmclk_dbp_unlock();
|
|
EN_REG &= ~EN_BIT;
|
|
stmclk_dbp_lock();
|
|
}
|
|
|
|
void ISR_NAME(void)
|
|
{
|
|
if (RTC->ISR & RTC_ISR_ALRAF) {
|
|
if (isr_ctx.cb != NULL) {
|
|
isr_ctx.cb(isr_ctx.arg);
|
|
}
|
|
RTC->ISR &= ~RTC_ISR_ALRAF;
|
|
}
|
|
EXTI->PR |= EXTI_PR_BIT;
|
|
cortexm_isr_end();
|
|
}
|
|
|
|
#endif /* !CPU_FAM_STM32F1 */
|