2017-11-13 16:08:08 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016-2017 Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
*
|
|
|
|
* 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-11-29 19:50:50 +01:00
|
|
|
* @ingroup cpu_efm32
|
2017-11-13 16:08:08 +01:00
|
|
|
* @ingroup drivers_periph_rtc
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief RTC peripheral driver implementation for EFM32 Series 1 MCUs
|
|
|
|
*
|
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/rtc.h"
|
|
|
|
|
|
|
|
#include "em_cmu.h"
|
|
|
|
#include "em_rtcc.h"
|
|
|
|
#include "em_rtcc_utils.h"
|
|
|
|
|
|
|
|
#define RTC_YEAR_OFFSET (100) /**< RTCC has only two-digit notation */
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
rtc_alarm_cb_t alarm_cb; /**< callback called from RTC interrupt */
|
|
|
|
void *alarm_arg; /**< argument passed to the callback */
|
|
|
|
uint32_t alarm_year; /**< alarm year */
|
|
|
|
} rtc_state_t;
|
|
|
|
|
|
|
|
static rtc_state_t rtc_state;
|
|
|
|
|
|
|
|
void rtc_init(void)
|
|
|
|
{
|
|
|
|
/* enable clocks */
|
|
|
|
CMU_ClockEnable(cmuClock_CORELE, true);
|
|
|
|
CMU_ClockEnable(cmuClock_RTCC, true);
|
|
|
|
|
|
|
|
/* reset and initialize peripheral */
|
|
|
|
RTCC_Init_TypeDef init = RTCC_INIT_DEFAULT;
|
|
|
|
|
|
|
|
init.enable = false;
|
|
|
|
init.presc = rtccCntPresc_32768;
|
|
|
|
init.cntMode = rtccCntModeCalendar;
|
|
|
|
|
|
|
|
RTCC_Reset();
|
|
|
|
RTCC_Init(&init);
|
|
|
|
|
|
|
|
/* initialize alarm channel */
|
|
|
|
RTCC_CCChConf_TypeDef init_channel = RTCC_CH_INIT_COMPARE_DEFAULT;
|
|
|
|
|
|
|
|
RTCC_ChannelInit(0, &init_channel);
|
|
|
|
|
|
|
|
/* enable interrupt */
|
|
|
|
NVIC_ClearPendingIRQ(RTCC_IRQn);
|
|
|
|
NVIC_EnableIRQ(RTCC_IRQn);
|
|
|
|
|
|
|
|
/* enable peripheral */
|
|
|
|
RTCC_Enable(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_set_time(struct tm *time)
|
|
|
|
{
|
|
|
|
RTCC_DateSet(
|
|
|
|
RTCC_Year2BCD(time->tm_year - RTC_YEAR_OFFSET) |
|
|
|
|
RTCC_Month2BCD(time->tm_mon) |
|
|
|
|
RTCC_DayOfMonth2BCD(time->tm_mday) |
|
|
|
|
RTCC_DayOfWeek2BCD(time->tm_wday));
|
|
|
|
RTCC_TimeSet(
|
|
|
|
RTCC_Hour2BCD(time->tm_hour) |
|
|
|
|
RTCC_Minute2BCD(time->tm_min) |
|
|
|
|
RTCC_Second2BCD(time->tm_sec));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_get_time(struct tm *time)
|
|
|
|
{
|
|
|
|
uint32_t datestamp = RTCC_DateGet();
|
|
|
|
uint32_t timestamp = RTCC_TimeGet();
|
|
|
|
|
|
|
|
time->tm_year = RTCC_BCD2Year(datestamp) + RTC_YEAR_OFFSET;
|
|
|
|
time->tm_mon = RTCC_BCD2Month(datestamp);
|
|
|
|
time->tm_mday = RTCC_BCD2DayOfMonth(datestamp);
|
|
|
|
time->tm_wday = RTCC_BCD2DayOfWeek(datestamp);
|
|
|
|
time->tm_hour = RTCC_BCD2Hour(timestamp);
|
|
|
|
time->tm_min = RTCC_BCD2Minute(timestamp);
|
|
|
|
time->tm_sec = RTCC_BCD2Second(timestamp);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
|
|
|
|
{
|
|
|
|
rtc_state.alarm_cb = cb;
|
|
|
|
rtc_state.alarm_arg = arg;
|
|
|
|
rtc_state.alarm_year = time->tm_year;
|
|
|
|
|
|
|
|
/* disable interrupt so it doesn't accidentally trigger */
|
|
|
|
RTCC_IntDisable(RTCC_IEN_CC0);
|
|
|
|
|
|
|
|
/* set compare registers */
|
|
|
|
RTCC_ChannelDateSet(0,
|
|
|
|
RTCC_Channel_Month2BCD(time->tm_mon) |
|
|
|
|
RTCC_Channel_Day2BCD(time->tm_mday));
|
|
|
|
RTCC_ChannelTimeSet(0,
|
|
|
|
RTCC_Channel_Hour2BCD(time->tm_hour) |
|
|
|
|
RTCC_Channel_Minute2BCD(time->tm_min) |
|
|
|
|
RTCC_Channel_Second2BCD(time->tm_sec));
|
|
|
|
|
|
|
|
/* enable the interrupt */
|
|
|
|
RTCC_IntClear(RTCC_IFC_CC0);
|
|
|
|
RTCC_IntEnable(RTCC_IEN_CC0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int rtc_get_alarm(struct tm *time)
|
|
|
|
{
|
|
|
|
uint32_t datestamp = RTCC_ChannelDateGet(0);
|
|
|
|
uint32_t timestamp = RTCC_ChannelTimeGet(0);
|
|
|
|
|
|
|
|
time->tm_year = rtc_state.alarm_year;
|
|
|
|
time->tm_mon = RTCC_Channel_BCD2Month(datestamp);
|
|
|
|
time->tm_mday = RTCC_Channel_BCD2Day(datestamp);
|
|
|
|
time->tm_hour = RTCC_Channel_BCD2Hour(timestamp);
|
|
|
|
time->tm_min = RTCC_Channel_BCD2Minute(timestamp);
|
|
|
|
time->tm_sec = RTCC_Channel_BCD2Second(timestamp);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_clear_alarm(void)
|
|
|
|
{
|
|
|
|
rtc_state.alarm_cb = NULL;
|
|
|
|
rtc_state.alarm_arg = NULL;
|
|
|
|
rtc_state.alarm_year = 0;
|
|
|
|
|
|
|
|
RTCC_IntDisable(RTCC_IEN_CC0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_poweron(void)
|
|
|
|
{
|
|
|
|
CMU_ClockEnable(cmuClock_RTCC, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtc_poweroff(void)
|
|
|
|
{
|
|
|
|
CMU_ClockEnable(cmuClock_RTCC, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void isr_rtcc(void)
|
|
|
|
{
|
|
|
|
if (RTCC_IntGet() & RTCC_IF_CC0) {
|
|
|
|
if (rtc_state.alarm_cb != NULL) {
|
|
|
|
/* check if year matches, otherwise alarm would go off each year */
|
|
|
|
if (RTCC_BCD2Year(RTCC_DateGet()) + RTC_YEAR_OFFSET == rtc_state.alarm_year) {
|
|
|
|
rtc_state.alarm_cb(rtc_state.alarm_arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RTCC_IntClear(RTCC_IFC_CC0);
|
|
|
|
}
|
|
|
|
cortexm_isr_end();
|
|
|
|
}
|