mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
264 lines
7.9 KiB
C
264 lines
7.9 KiB
C
|
/*
|
||
|
* Copyright (C) 2015 Lari Lehtomäki
|
||
|
*
|
||
|
* 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_stm32f0
|
||
|
* @{
|
||
|
* @file
|
||
|
* @brief Low-level RTC driver implementation
|
||
|
* @author Lari Lehtomäki <lari@lehtomaki.fi>
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
|
||
|
#include <time.h>
|
||
|
#include "cpu.h"
|
||
|
#include "periph/rtc.h"
|
||
|
#include "periph_conf.h"
|
||
|
#include "sched.h"
|
||
|
#include "thread.h"
|
||
|
|
||
|
/* guard file in case no RTC device was specified */
|
||
|
#if RTC_NUMOF
|
||
|
|
||
|
#define RTC_WRITE_PROTECTION_KEY1 (0xCA)
|
||
|
#define RTC_WRITE_PROTECTION_KEY2 (0x53)
|
||
|
#define RTC_SYNC_PRESCALER (0xff) /**< prescaler for 32.768 kHz oscillator */
|
||
|
#define RTC_ASYNC_PRESCALER (0x7f) /**< prescaler for 32.768 kHz oscillator */
|
||
|
|
||
|
#define MCU_YEAR_OFFSET (100) /**< struct tm counts years since 1900
|
||
|
but RTC has only two-digit year
|
||
|
hence the offset of 100 years. */
|
||
|
|
||
|
typedef struct {
|
||
|
rtc_alarm_cb_t cb; /**< callback called from RTC interrupt */
|
||
|
void *arg; /**< argument passed to the callback */
|
||
|
} rtc_state_t;
|
||
|
|
||
|
static rtc_state_t rtc_callback;
|
||
|
|
||
|
static uint8_t byte2bcd(uint8_t value);
|
||
|
|
||
|
/**
|
||
|
* @brief Initializes the RTC to use LSE (external 32.768 kHz oscillator) as a
|
||
|
* clocl source. Verify that your board has this oscillator. If other clock
|
||
|
* source is used, then also the prescaler constants should be changed.
|
||
|
*/
|
||
|
void rtc_init(void)
|
||
|
{
|
||
|
|
||
|
/* Enable write access to RTC registers */
|
||
|
RCC->APB1ENR |= RCC_APB1ENR_PWREN;
|
||
|
PWR->CR |= PWR_CR_DBP;
|
||
|
|
||
|
/* Reset RTC domain */
|
||
|
RCC->BDCR |= RCC_BDCR_BDRST;
|
||
|
RCC->BDCR &= ~(RCC_BDCR_BDRST);
|
||
|
|
||
|
/* Enable the LSE clock (external 32.768 kHz oscillator) */
|
||
|
RCC->BDCR &= ~(RCC_BDCR_LSEON);
|
||
|
RCC->BDCR &= ~(RCC_BDCR_LSEBYP);
|
||
|
RCC->BDCR |= RCC_BDCR_LSEON;
|
||
|
while ( (RCC->BDCR & RCC_BDCR_LSERDY) == 0 );
|
||
|
|
||
|
/* Switch RTC to LSE clock source */
|
||
|
RCC->BDCR &= ~(RCC_BDCR_RTCSEL);
|
||
|
RCC->BDCR |= RCC_BDCR_RTCSEL_LSE;
|
||
|
|
||
|
/* Enable the RTC */
|
||
|
RCC->BDCR |= RCC_BDCR_RTCEN;
|
||
|
|
||
|
/* Unlock RTC write protection */
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY1;
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY2;
|
||
|
|
||
|
/* Enter RTC Init mode */
|
||
|
RTC->ISR = 0;
|
||
|
RTC->ISR |= RTC_ISR_INIT;
|
||
|
while ( (RTC->ISR & RTC_ISR_INITF) == 0 );
|
||
|
|
||
|
/* Set 24-h clock */
|
||
|
RTC->CR |= RTC_CR_FMT;
|
||
|
/* Timestamps enabled */
|
||
|
RTC->CR |= RTC_CR_TSE;
|
||
|
|
||
|
/* Configure the RTC PRER */
|
||
|
RTC->PRER = RTC_SYNC_PRESCALER;
|
||
|
RTC->PRER |= (RTC_ASYNC_PRESCALER << 16);
|
||
|
|
||
|
/* Exit RTC init mode */
|
||
|
RTC->ISR &= (uint32_t)~RTC_ISR_INIT;
|
||
|
|
||
|
/* Enable RTC write protection */
|
||
|
RTC->WPR = 0xff;
|
||
|
|
||
|
}
|
||
|
|
||
|
int rtc_set_time(struct tm *time)
|
||
|
{
|
||
|
/* Enable write access to RTC registers */
|
||
|
RCC->APB1ENR |= RCC_APB1ENR_PWREN;
|
||
|
PWR->CR |= PWR_CR_DBP;
|
||
|
|
||
|
/* Unlock RTC write protection */
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY1;
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY2;
|
||
|
|
||
|
/* Enter RTC Init mode */
|
||
|
RTC->ISR |= RTC_ISR_INIT;
|
||
|
while ( (RTC->ISR & RTC_ISR_INITF) == 0 );
|
||
|
|
||
|
|
||
|
RTC->DR = ( (((uint32_t)byte2bcd(time->tm_year - MCU_YEAR_OFFSET) << 16) & (RTC_DR_YT | RTC_DR_YU) ) |
|
||
|
(((uint32_t)byte2bcd(time->tm_mon+1)<< 8) & (RTC_DR_MT | RTC_DR_MU) ) |
|
||
|
(((uint32_t)byte2bcd(time->tm_mday) << 0) & (RTC_DR_DT | RTC_DR_DU) ) );
|
||
|
|
||
|
RTC->TR = ( (((uint32_t)byte2bcd(time->tm_hour) << 16) & (RTC_TR_HT | RTC_TR_HU) ) |
|
||
|
(((uint32_t)byte2bcd(time->tm_min) << 8) & (RTC_TR_MNT| RTC_TR_MNU)) |
|
||
|
(((uint32_t)byte2bcd(time->tm_sec) << 0) & (RTC_TR_ST | RTC_TR_SU) ) );
|
||
|
|
||
|
/* Exit RTC init mode */
|
||
|
RTC->ISR &= (uint32_t)~RTC_ISR_INIT;
|
||
|
/* Enable RTC write protection */
|
||
|
RTC->WPR = 0xFF;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int rtc_get_time(struct tm *time)
|
||
|
{
|
||
|
time->tm_year = MCU_YEAR_OFFSET;
|
||
|
time->tm_year +=(((RTC->DR & RTC_DR_YT) >> 20) * 10) + ((RTC->DR & RTC_DR_YU) >> 16);
|
||
|
time->tm_mon = (((RTC->DR & RTC_DR_MT) >> 12) * 10) + ((RTC->DR & RTC_DR_MU) >> 8) - 1;
|
||
|
time->tm_mday = (((RTC->DR & RTC_DR_DT) >> 4) * 10) + ((RTC->DR & RTC_DR_DU) >> 0);
|
||
|
time->tm_hour = (((RTC->TR & RTC_TR_HT) >> 20) * 10) + ((RTC->TR & RTC_TR_HU) >> 16);
|
||
|
if ( RTC->TR & RTC_TR_PM )
|
||
|
time->tm_hour += 12;
|
||
|
time->tm_min = (((RTC->TR & RTC_TR_MNT) >> 12) * 10) + ((RTC->TR & RTC_TR_MNU) >> 8);
|
||
|
time->tm_sec = (((RTC->TR & RTC_TR_ST) >> 4) * 10) + ((RTC->TR & RTC_TR_SU) >> 0);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
|
||
|
{
|
||
|
/* Enable write access to RTC registers */
|
||
|
RCC->APB1ENR |= RCC_APB1ENR_PWREN;
|
||
|
PWR->CR |= PWR_CR_DBP;
|
||
|
|
||
|
/* Unlock RTC write protection */
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY1;
|
||
|
RTC->WPR = RTC_WRITE_PROTECTION_KEY2;
|
||
|
|
||
|
/* Enter RTC Init mode */
|
||
|
RTC->ISR |= RTC_ISR_INIT;
|
||
|
while ( (RTC->ISR & RTC_ISR_INITF) == 0 );
|
||
|
|
||
|
RTC->CR &= ~(RTC_CR_ALRAE);
|
||
|
while ( (RTC->ISR & RTC_ISR_ALRAWF) == 0 );
|
||
|
RTC->ALRMAR &= ~(RTC_ALRMAR_MSK1 | RTC_ALRMAR_MSK2 | RTC_ALRMAR_MSK3 | RTC_ALRMAR_MSK4);
|
||
|
RTC->ALRMAR = ( (((uint32_t)byte2bcd(time->tm_mday) << 24) & (RTC_ALRMAR_DT | RTC_ALRMAR_DU) ) |
|
||
|
(((uint32_t)byte2bcd(time->tm_hour) << 16) & (RTC_ALRMAR_HT | RTC_ALRMAR_HU) ) |
|
||
|
(((uint32_t)byte2bcd(time->tm_min) << 8) & (RTC_ALRMAR_MNT| RTC_ALRMAR_MNU)) |
|
||
|
(((uint32_t)byte2bcd(time->tm_sec) << 0) & (RTC_ALRMAR_ST | RTC_ALRMAR_SU) ) );
|
||
|
/* Enable Alarm A */
|
||
|
RTC->CR |= RTC_CR_ALRAE;
|
||
|
RTC->CR |= RTC_CR_ALRAIE;
|
||
|
RTC->ISR &= ~(RTC_ISR_ALRAF);
|
||
|
|
||
|
/* Exit RTC init mode */
|
||
|
RTC->ISR &= (uint32_t)~RTC_ISR_INIT;
|
||
|
/* Enable RTC write protection */
|
||
|
RTC->WPR = 0xFF;
|
||
|
|
||
|
EXTI->IMR |= EXTI_IMR_MR17;
|
||
|
EXTI->RTSR |= EXTI_RTSR_TR17;
|
||
|
NVIC_SetPriority(RTC_IRQn, 10);
|
||
|
NVIC_EnableIRQ(RTC_IRQn);
|
||
|
|
||
|
rtc_callback.cb = cb;
|
||
|
rtc_callback.arg = arg;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int rtc_get_alarm(struct tm *time)
|
||
|
{
|
||
|
time->tm_year = MCU_YEAR_OFFSET;
|
||
|
time->tm_year +=(((RTC->DR & RTC_DR_YT) >> 20) * 10) + ((RTC->DR & RTC_DR_YU) >> 16);
|
||
|
time->tm_mon = (((RTC->DR & RTC_DR_MT) >> 12) * 10) + ((RTC->DR & RTC_DR_MU) >> 8) - 1;
|
||
|
time->tm_mday = (((RTC->ALRMAR & RTC_ALRMAR_DT) >> 28) * 10) + ((RTC->ALRMAR & RTC_ALRMAR_DU) >> 24);
|
||
|
time->tm_hour = (((RTC->ALRMAR & RTC_ALRMAR_HT) >> 20) * 10) + ((RTC->ALRMAR & RTC_ALRMAR_HU) >> 16);
|
||
|
if ( (RTC->ALRMAR & RTC_ALRMAR_PM) && (RTC->CR & RTC_CR_FMT) )
|
||
|
time->tm_hour += 12;
|
||
|
time->tm_min = (((RTC->ALRMAR & RTC_ALRMAR_MNT) >> 12) * 10) + ((RTC->ALRMAR & RTC_ALRMAR_MNU) >> 8);
|
||
|
time->tm_sec = (((RTC->ALRMAR & RTC_ALRMAR_ST) >> 4) * 10) + ((RTC->ALRMAR & RTC_ALRMAR_SU) >> 0);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void rtc_clear_alarm(void)
|
||
|
{
|
||
|
/* Disable Alarm A */
|
||
|
RTC->CR &= RTC_CR_ALRAE;
|
||
|
RTC->CR &= RTC_CR_ALRAIE;
|
||
|
|
||
|
rtc_callback.cb = NULL;
|
||
|
rtc_callback.arg = NULL;
|
||
|
}
|
||
|
|
||
|
void rtc_poweron(void)
|
||
|
{
|
||
|
/* RTC on STM32F0 is online even on sleep modes. No need to power on. */
|
||
|
}
|
||
|
|
||
|
void rtc_poweroff(void)
|
||
|
{
|
||
|
/* Enable write access to RTC registers */
|
||
|
RCC->APB1ENR |= RCC_APB1ENR_PWREN;
|
||
|
PWR->CR |= PWR_CR_DBP;
|
||
|
|
||
|
/* Reset RTC domain */
|
||
|
RCC->BDCR |= RCC_BDCR_BDRST;
|
||
|
RCC->BDCR &= ~(RCC_BDCR_BDRST);
|
||
|
/* Disable the RTC */
|
||
|
RCC->BDCR &= ~RCC_BDCR_RTCEN;
|
||
|
/* Disable LSE clock */
|
||
|
RCC->BDCR &= ~(RCC_BDCR_LSEON);
|
||
|
}
|
||
|
|
||
|
void isr_rtc(void)
|
||
|
{
|
||
|
|
||
|
if ((RTC->ISR & RTC_ISR_ALRAF) && (rtc_callback.cb != NULL)) {
|
||
|
rtc_callback.cb(rtc_callback.arg);
|
||
|
RTC->ISR &= ~RTC_ISR_ALRAF;
|
||
|
}
|
||
|
if (sched_context_switch_request) {
|
||
|
thread_yield();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Convert a number from unsigned to BCD
|
||
|
*
|
||
|
* @param[in] value to be converted
|
||
|
* @return BCD representation of the value
|
||
|
*/
|
||
|
static uint8_t byte2bcd(uint8_t value)
|
||
|
{
|
||
|
uint8_t bcdhigh = 0;
|
||
|
|
||
|
while (value >= 10)
|
||
|
{
|
||
|
bcdhigh++;
|
||
|
value -= 10;
|
||
|
}
|
||
|
|
||
|
return ((uint8_t)(bcdhigh << 4) | value);
|
||
|
}
|
||
|
|
||
|
#endif /* RTC_NUMOF */
|