2015-01-06 00:19:24 +01:00
|
|
|
/*
|
2015-02-08 09:18:23 +01:00
|
|
|
* Copyright (C) 2015 Eistec AB
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-10-27 01:26:26 +02:00
|
|
|
* @ingroup cpu_kinetis
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_rtc
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2015-02-08 09:18:23 +01:00
|
|
|
* @brief RTC interface wrapper for use with RTT modules.
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-01-06 00:19:24 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
#include "bit.h"
|
2015-01-06 00:19:24 +01:00
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph/rtc.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
|
2020-10-22 11:34:00 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2015-01-06 00:19:24 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
static rtc_alarm_cb_t rtc_callback = NULL;
|
|
|
|
static rtc_alarm_cb_t rtc_arg;
|
2015-02-08 09:18:23 +01:00
|
|
|
|
2015-01-06 00:19:24 +01:00
|
|
|
void rtc_init(void)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
rtc_callback = NULL;
|
|
|
|
|
|
|
|
/* Enable module clock gate */
|
|
|
|
RTC_CLKEN();
|
|
|
|
|
|
|
|
/* At this point, the CPU core may be clocked by a clock derived from the
|
|
|
|
* RTC oscillator, avoid touching the oscillator enable bit (OSCE) in RTC_CR */
|
|
|
|
|
|
|
|
/* Enable user mode access */
|
|
|
|
bit_set32(&RTC->CR, RTC_CR_SUP_SHIFT);
|
|
|
|
|
|
|
|
/* Disable all RTC interrupts. */
|
|
|
|
RTC->IER = 0;
|
|
|
|
|
|
|
|
/* The RTC module is only reset on VBAT power on reset, we try to preserve
|
|
|
|
* the seconds counter between reboots */
|
|
|
|
if (RTC->SR & RTC_SR_TIF_MASK) {
|
|
|
|
/* Time Invalid Flag is set, clear TIF by writing TSR */
|
|
|
|
|
|
|
|
/* Stop counter to make TSR writable */
|
|
|
|
bit_clear32(&RTC->SR, RTC_SR_TCE_SHIFT);
|
|
|
|
|
|
|
|
RTC->TSR = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear the alarm flag TAF by writing a new alarm target to TAR */
|
|
|
|
RTC->TAR = 0xffffffff;
|
|
|
|
|
|
|
|
/* Enable RTC interrupts */
|
|
|
|
NVIC_EnableIRQ(RTC_IRQn);
|
|
|
|
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_set_time(struct tm *time)
|
|
|
|
{
|
2020-08-27 23:11:19 +02:00
|
|
|
uint32_t t = rtc_mktime(time);
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
/* Disable time counter before writing to the timestamp register */
|
|
|
|
bit_clear32(&RTC->SR, RTC_SR_TCE_SHIFT);
|
|
|
|
RTC->TPR = 0;
|
|
|
|
/* write TSR after TPR, as clearing TPR bit 14 will increment TSR */
|
|
|
|
RTC->TSR = t;
|
|
|
|
/* Enable when done */
|
|
|
|
bit_set32(&RTC->SR, RTC_SR_TCE_SHIFT);
|
2015-02-08 09:18:23 +01:00
|
|
|
|
2015-01-06 00:19:24 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_get_time(struct tm *time)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
uint32_t t;
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
/* Read twice to make sure we get a stable reading */
|
|
|
|
t = RTC->TSR;
|
|
|
|
|
|
|
|
if (t == RTC->TSR) {
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Fallback just return unstable reading */
|
2020-08-27 23:11:19 +02:00
|
|
|
rtc_localtime(t, time);
|
2015-02-08 09:18:23 +01:00
|
|
|
return 0;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
|
|
|
|
{
|
2020-08-27 23:11:19 +02:00
|
|
|
uint32_t t = rtc_mktime(time);
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
/* The alarm is triggered when TSR matches TAR, and TSR increments. This
|
|
|
|
* seem counterintuitive as most users expect the alarm to trigger
|
|
|
|
* immediately when the counter becomes equal to the alarm time.
|
|
|
|
*
|
|
|
|
* Workaround: Set TAF to alarm - 1
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Disable Timer Alarm Interrupt */
|
|
|
|
bit_clear32(&RTC->IER, RTC_IER_TAIE_SHIFT);
|
|
|
|
|
|
|
|
RTC->TAR = t - 1;
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
rtc_callback = cb;
|
|
|
|
rtc_arg = arg;
|
|
|
|
|
|
|
|
/* Enable Timer Alarm Interrupt */
|
|
|
|
bit_set32(&RTC->IER, RTC_IER_TAIE_SHIFT);
|
2015-01-06 00:19:24 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_get_alarm(struct tm *time)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
uint32_t t = RTC->TAR + 1;
|
2015-01-06 00:19:24 +01:00
|
|
|
|
2020-08-27 23:11:19 +02:00
|
|
|
rtc_localtime(t, time);
|
2015-01-06 00:19:24 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_clear_alarm(void)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
/* Disable Timer Alarm Interrupt */
|
|
|
|
bit_clear32(&RTC->IER, RTC_IER_TAIE_SHIFT);
|
|
|
|
rtc_callback = NULL;
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_poweron(void)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
/* Enable Time Counter */
|
|
|
|
bit_set32(&RTC->SR, RTC_SR_TCE_SHIFT);
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_poweroff(void)
|
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
/* Disable Time Counter */
|
|
|
|
bit_clear32(&RTC->SR, RTC_SR_TCE_SHIFT);
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
|
|
|
|
2021-06-11 17:21:01 +02:00
|
|
|
void isr_rtc(void)
|
2015-01-06 00:19:24 +01:00
|
|
|
{
|
2021-06-11 17:21:01 +02:00
|
|
|
if (RTC->SR & RTC_SR_TAF_MASK) {
|
|
|
|
if (rtc_callback != NULL) {
|
|
|
|
/* Disable Timer Alarm Interrupt */
|
|
|
|
bit_clear32(&RTC->IER, RTC_IER_TAIE_SHIFT);
|
|
|
|
rtc_callback(rtc_arg);
|
|
|
|
}
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|
2021-06-11 17:21:01 +02:00
|
|
|
|
|
|
|
cortexm_isr_end();
|
2015-01-06 00:19:24 +01:00
|
|
|
}
|