mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
25d04c4742
- Unecessary definitions are removed. - Since the 48-bit RTC hardware timer uses a RC oscillator as clock, it is pretty inaccurate and leads to a RTC time deviation of up to 3 seconds per minute. Therefore, a calibration during the boot time determines a correction factor for the 48-bit RTC hardware timer. Function _rtc_time_to_us uses now this correction factor and converts a raw 48-bit RTC time to a corrected time in microseconds. Thus, the 48-bit RTC timer becomes much more accurate, but it can't still reach the accuracy of the PLL driven 64-bit system timer. The Advantage of using RTC over 64-bit sydtem timer is that it also continues in deep sleep mode and after software reset. - If the 64-bit system timer is used to emulate the RTC timer, it uses the RTC hardware timer to continue its operation after software .
343 lines
10 KiB
C
343 lines
10 KiB
C
/*
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
*
|
|
* 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_esp32
|
|
* @ingroup drivers_periph_rtc
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level RTC driver implementation
|
|
*
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
/*
|
|
* If module esp_rtc_timer is enabled, the 48-bit RTC hardware timer is used
|
|
* directly. Otherwise the PLL driven 64-bit microsecond system timer is used
|
|
* to emulate a RTC timer (default). This emulated RTC timer results into much
|
|
* better accuracy. The Advantage of using RTC hardware timer over system timer
|
|
* is that it would also continue in deep sleep mode and after software reset.
|
|
*/
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#include "debug.h"
|
|
#include "esp_common.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include "cpu.h"
|
|
#include "esp_attr.h"
|
|
#include "log.h"
|
|
#include "irq_arch.h"
|
|
#include "periph/rtc.h"
|
|
#include "rom/ets_sys.h"
|
|
#include "soc/dport_reg.h"
|
|
#include "soc/rtc_cntl_struct.h"
|
|
#include "soc/timer_group_struct.h"
|
|
#include "syscalls.h"
|
|
#include "timex.h"
|
|
#include "xtensa/xtensa_api.h"
|
|
|
|
/* TODO move to TIMER_SYSTEM definition in periph_cpu.h */
|
|
#define TIMER_SYSTEM_GROUP TIMERG0
|
|
#define TIMER_SYSTEM_INT_MASK BIT(0)
|
|
#define TIMER_SYSTEM_INT_SRC ETS_TG0_T0_LEVEL_INTR_SOURCE
|
|
|
|
#define RTC_CLK_CAL_FRACT 19 /* fractional bits of calibration value */
|
|
|
|
/* we can't include soc/rtc.h because of rtc_init declaration conflicts */
|
|
extern uint32_t rtc_clk_slow_freq_get_hz(void);
|
|
extern uint32_t esp_clk_slowclk_cal_get(void);
|
|
|
|
/* static variables */
|
|
static rtc_alarm_cb_t _rtc_alarm_cb = NULL;
|
|
static void* _rtc_alarm_arg = NULL;
|
|
static time_t _sys_alarm_time = 0;
|
|
|
|
#define RTC_BSS_ATTR __attribute__((section(".rtc.bss")))
|
|
|
|
/* save several time stamps */
|
|
static uint64_t RTC_BSS_ATTR _rtc_time_init_us; /* RTC time on init in us */
|
|
static uint64_t RTC_BSS_ATTR _rtc_time_init; /* RTC time on init in cycles */
|
|
static uint64_t RTC_BSS_ATTR _rtc_time_set_us; /* RTC time on set in us */
|
|
static uint64_t RTC_BSS_ATTR _rtc_time_set; /* RTC time on set in cycles */
|
|
static uint64_t RTC_BSS_ATTR _sys_time_set_us; /* system time on set in us */
|
|
static time_t RTC_BSS_ATTR _sys_time_set; /* system time on set in sec */
|
|
static uint64_t RTC_BSS_ATTR _sys_time_off_us; /* system time offset in us */
|
|
|
|
/* forward declarations */
|
|
static time_t _sys_get_time (void); /* system time in seconds */
|
|
static uint64_t _rtc_get_time_raw(void); /* RTC time in cycles */
|
|
static uint64_t _rtc_time_to_us(uint64_t raw); /* convert RTC cycles to us */
|
|
static void IRAM_ATTR _rtc_timer_handler(void* arg);
|
|
|
|
void rtc_init(void)
|
|
{
|
|
uint64_t _rtc_time_us = _rtc_time_to_us(_rtc_get_time_raw());
|
|
|
|
if (_rtc_time_init == 0 && _rtc_time_init_us == 0) {
|
|
/* only set it new, if it was not set before */
|
|
_rtc_time_init = _rtc_get_time_raw();
|
|
_rtc_time_init_us = _rtc_time_us;
|
|
_sys_time_off_us = 0;
|
|
|
|
DEBUG("%s saved rtc_init=%lld rtc_init_us=%lld\n",
|
|
__func__, _rtc_time_init, _rtc_time_init_us);
|
|
|
|
}
|
|
else {
|
|
_sys_time_off_us = _rtc_time_us - _rtc_time_set_us;
|
|
}
|
|
_sys_time_set_us = 0;
|
|
}
|
|
|
|
void rtc_poweron(void)
|
|
{
|
|
/* RTC is always on, also in deep sleep mode */
|
|
return;
|
|
}
|
|
|
|
void rtc_poweroff(void)
|
|
{
|
|
/* RTC is always on, also in deep sleep mode */
|
|
return;
|
|
}
|
|
|
|
int rtc_set_time(struct tm *ttime)
|
|
{
|
|
_rtc_time_set = _rtc_get_time_raw();
|
|
_rtc_time_set_us = _rtc_time_to_us(_rtc_time_set);
|
|
|
|
_sys_time_set = mktime(ttime);
|
|
_sys_time_set_us = system_get_time_64();
|
|
_sys_time_off_us = 0;
|
|
|
|
DEBUG("%s sys_time=%ld rtc_time=%lld rtc_time_us=%lld\n",
|
|
__func__, _sys_time_set, _rtc_time_set, _rtc_time_set_us);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int rtc_get_time(struct tm *ttime)
|
|
{
|
|
time_t _sys_time = _sys_get_time();
|
|
|
|
DEBUG("%s sys_time=%ld rtc_time=%lld rtc_time_us=%lld\n", __func__,
|
|
_sys_time, _rtc_get_time_raw(), _rtc_time_to_us(_rtc_get_time_raw()));
|
|
|
|
struct tm* _time = localtime(&_sys_time);
|
|
if (_time) {
|
|
memcpy(ttime, _time, sizeof(struct tm));
|
|
return 0;
|
|
}
|
|
else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int rtc_get_alarm(struct tm *time)
|
|
{
|
|
struct tm* _time = localtime(&_sys_alarm_time);
|
|
if (_time) {
|
|
memcpy(time, _time, sizeof(struct tm));
|
|
return 0;
|
|
}
|
|
else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int rtc_set_alarm(struct tm *time, rtc_alarm_cb_t cb, void *arg)
|
|
{
|
|
_rtc_alarm_cb = cb;
|
|
_rtc_alarm_arg = arg;
|
|
|
|
/* determine the offset of alarm time to the set time in seconds */
|
|
_sys_alarm_time = mktime(time);
|
|
time_t _sys_time_offset = _sys_alarm_time - _sys_time_set;
|
|
|
|
/*
|
|
* RTC doesn't provide alarm functionality in active mode. At least
|
|
* the RTC main timer seems not to work. Therefore we always use the
|
|
* system timer for alarms. The Advantage of using RTC over system timer
|
|
* is that it also continues in deep sleep and after software reset.
|
|
*/
|
|
#if 0 /* TODO should be MODULE_ESP_RTC_TIMER */
|
|
|
|
/* determine the offset of alarm time to current time in RTC time */
|
|
uint64_t _rtc_time_alarm;
|
|
_rtc_time_alarm = _rtc_time_set + _sys_time_offset * rtc_clk_slow_freq_get_hz();
|
|
|
|
DEBUG("%s sys=%d sys_alarm=%d rtc=%lld rtc_alarm=%lld\n", __func__,
|
|
_sys_get_time(), _sys_time_offset, _rtc_get_time_raw(), _rtc_time_alarm);
|
|
|
|
/* set the timer value */
|
|
RTCCNTL.slp_timer0 = _rtc_time_alarm & 0xffffffff;
|
|
RTCCNTL.slp_timer1.slp_val_hi = _rtc_time_alarm >> 32;
|
|
|
|
DEBUG("%s %08x%08x \n", __func__, RTCCNTL.slp_timer1.slp_val_hi, RTCCNTL.slp_timer0);
|
|
|
|
/* enable RTC timer alarm */
|
|
RTCCNTL.slp_timer1.main_timer_alarm_en = 1;
|
|
|
|
/* clear and enable RTC timer interrupt */
|
|
RTCCNTL.int_clr.rtc_main_timer = 1;
|
|
RTCCNTL.int_ena.rtc_main_timer = 1;
|
|
|
|
/* route all RTC interrupt sources to the same level type interrupt */
|
|
intr_matrix_set(PRO_CPU_NUM, DPORT_PRO_RTC_CORE_INTR_MAP_REG, CPU_INUM_RTC);
|
|
|
|
/* set interrupt handler and enable the CPU interrupt */
|
|
xt_set_interrupt_handler(CPU_INUM_RTC, _rtc_timer_handler, NULL);
|
|
xt_ints_on(BIT(CPU_INUM_RTC));
|
|
|
|
#else
|
|
|
|
/* determine the offset of alarm time to the RTC set time in us */
|
|
uint64_t _sys_alarm_us;
|
|
_sys_alarm_us = _sys_time_set_us - _sys_time_off_us + _sys_time_offset * US_PER_SEC;
|
|
|
|
DEBUG("%s sys_time=%ld sys_time_offset=%ld "
|
|
"sys_time_us=%lld sys_time_alarm_us=%lld\n", __func__,
|
|
_sys_get_time(), _sys_time_offset, system_get_time_64(), _sys_alarm_us);
|
|
|
|
/* set the timer value */
|
|
TIMER_SYSTEM.alarm_high = (uint32_t)(_sys_alarm_us >> 32);
|
|
TIMER_SYSTEM.alarm_low = (uint32_t)(_sys_alarm_us & 0xffffffff);
|
|
|
|
/* clear the bit in status and set the bit in interrupt enable */
|
|
TIMER_SYSTEM_GROUP.int_clr_timers.val |= TIMER_SYSTEM_INT_MASK;
|
|
TIMER_SYSTEM_GROUP.int_ena.val |= TIMER_SYSTEM_INT_MASK;
|
|
|
|
/* route all timer interrupt sources to the same level type interrupt */
|
|
intr_matrix_set(PRO_CPU_NUM, TIMER_SYSTEM_INT_SRC, CPU_INUM_RTC);
|
|
|
|
/* set interrupt handler and enable the CPU interrupt */
|
|
xt_set_interrupt_handler(CPU_INUM_RTC, _rtc_timer_handler, NULL);
|
|
xt_ints_on(BIT(CPU_INUM_RTC));
|
|
|
|
/* enable the timer alarm */
|
|
TIMER_SYSTEM.config.level_int_en = 1;
|
|
TIMER_SYSTEM.config.alarm_en = 1;
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
void rtc_clear_alarm(void)
|
|
{
|
|
_rtc_alarm_cb = NULL;
|
|
_rtc_alarm_arg = NULL;
|
|
|
|
#if 0 /* TODO should be MODULE_ESP_RTC_TIMER, see rtc_set_alarm */
|
|
|
|
/* disable RTC timer alarm and disable the RTC timer interrupt */
|
|
RTCCNTL.slp_timer1.main_timer_alarm_en = 0;
|
|
RTCCNTL.int_ena.rtc_main_timer = 0;
|
|
|
|
/* route all RTC interrupt sources to the same level type interrupt */
|
|
intr_matrix_set(PRO_CPU_NUM, DPORT_PRO_RTC_CORE_INTR_MAP_REG, CPU_INUM_RTC);
|
|
|
|
/* disable the the CPU interrupt */
|
|
|
|
#else
|
|
|
|
/* reset the bit in interrupt enable */
|
|
TIMER_SYSTEM_GROUP.int_ena.val |= TIMER_SYSTEM_INT_MASK;
|
|
|
|
/* disable the CPU interrupt */
|
|
xt_ints_on(BIT(CPU_INUM_RTC));
|
|
|
|
/* disable the timer alarm */
|
|
TIMER_SYSTEM.config.level_int_en = 0;
|
|
TIMER_SYSTEM.config.alarm_en = 0;
|
|
|
|
xt_ints_on(BIT(CPU_INUM_RTC));
|
|
|
|
#endif
|
|
}
|
|
|
|
static time_t _sys_get_time (void)
|
|
{
|
|
#if MODULE_ESP_RTC_TIMER
|
|
return _sys_time_set +
|
|
(_rtc_time_to_us(_rtc_get_time_raw() - _rtc_time_set) / US_PER_SEC);
|
|
#else
|
|
return _sys_time_set +
|
|
((_sys_time_off_us + system_get_time_64() - _sys_time_set_us) / US_PER_SEC);
|
|
#endif
|
|
}
|
|
|
|
static uint64_t _rtc_get_time_raw(void)
|
|
{
|
|
/* trigger timer register update */
|
|
RTCCNTL.time_update.update = 1;
|
|
/* wait until values in registers are valid */
|
|
while (!RTCCNTL.time_update.valid) { }
|
|
/* read the time and return */
|
|
uint64_t rtc_time;
|
|
rtc_time = RTCCNTL.time0;
|
|
rtc_time += ((uint64_t)RTCCNTL.time1.val) << 32;
|
|
return rtc_time;
|
|
}
|
|
|
|
static uint64_t _rtc_time_to_us(uint64_t raw)
|
|
{
|
|
const uint32_t cal = esp_clk_slowclk_cal_get();
|
|
return ((((raw >> 32) * cal) << (32 - RTC_CLK_CAL_FRACT)) + /* high part */
|
|
(((raw & 0xffffffff) * cal) >> RTC_CLK_CAL_FRACT)); /* low part */
|
|
}
|
|
|
|
static void IRAM_ATTR _rtc_timer_handler(void* arg)
|
|
{
|
|
irq_isr_enter();
|
|
|
|
#if 0 /* TODO should be MODULE_ESP_RTC_TIMER */
|
|
|
|
/* check for RTC timer interrupt */
|
|
if (RTCCNTL.int_st.rtc_main_timer) {
|
|
/* clear the interrupt */
|
|
RTCCNTL.int_clr.rtc_main_timer = 1;
|
|
/* call back registered function */
|
|
if (_rtc_alarm_cb) {
|
|
_rtc_alarm_cb(_rtc_alarm_arg);
|
|
}
|
|
}
|
|
/* clear all interrupts */
|
|
RTCCNTL.int_clr.val = 0x1ff;
|
|
|
|
#else
|
|
|
|
/* check for RTC timer interrupt */
|
|
if (TIMER_SYSTEM_GROUP.int_st_timers.val & TIMER_SYSTEM_INT_MASK) {
|
|
|
|
DEBUG("%s\n", __func__);
|
|
|
|
/* disable alarms */
|
|
TIMER_SYSTEM.config.level_int_en = 0;
|
|
TIMER_SYSTEM.config.alarm_en = 0;
|
|
|
|
/* clear the bit in interrupt enable and status register */
|
|
TIMER_SYSTEM_GROUP.int_ena.val &= ~TIMER_SYSTEM_INT_MASK;
|
|
TIMER_SYSTEM_GROUP.int_clr_timers.val |= TIMER_SYSTEM_INT_MASK;
|
|
|
|
/* call back registered function */
|
|
if (_rtc_alarm_cb) {
|
|
_rtc_alarm_cb(_rtc_alarm_arg);
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
irq_isr_exit();
|
|
}
|