mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
077056b949
The currently supported SAM0 MCUs (samd21, saml21, saml1x) share the same RTC peripheral, yet each of them carries it's own copy of the RTT driver. Unify the drivers and move them to sam0_common.
225 lines
5.2 KiB
C
225 lines
5.2 KiB
C
/*
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
* 2015 FreshTemp, LLC.
|
|
*
|
|
* 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_sam0_common
|
|
* @ingroup drivers_periph_rtt
|
|
* @{
|
|
*
|
|
* @file rtt.c
|
|
* @brief Low-level RTT driver implementation
|
|
*
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include "periph/rtt.h"
|
|
#include "board.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
static rtt_cb_t _overflow_cb;
|
|
static void* _overflow_arg;
|
|
|
|
static rtt_cb_t _cmp0_cb;
|
|
static void* _cmp0_arg;
|
|
|
|
static void _wait_syncbusy(void)
|
|
{
|
|
#ifdef REG_RTC_MODE0_SYNCBUSY
|
|
while (RTC->MODE0.SYNCBUSY.reg) {}
|
|
#else
|
|
while(RTC->MODE0.STATUS.bit.SYNCBUSY) {}
|
|
#endif
|
|
}
|
|
|
|
static inline void _rtt_reset(void)
|
|
{
|
|
#ifdef RTC_MODE0_CTRL_SWRST
|
|
RTC->MODE0.CTRL.bit.SWRST = 1;
|
|
while (RTC->MODE0.CTRL.bit.SWRST) {}
|
|
#else
|
|
RTC->MODE0.CTRLA.bit.SWRST = 1;
|
|
while (RTC->MODE0.CTRLA.bit.SWRST) {}
|
|
#endif
|
|
}
|
|
|
|
#ifdef CPU_SAMD21
|
|
static void _rtt_clock_setup(void)
|
|
{
|
|
/* RTC uses External 32,768KHz Oscillator because OSC32K isn't accurate
|
|
* enough (p1075/1138). Also keep running in standby. */
|
|
SYSCTRL->XOSC32K.reg = SYSCTRL_XOSC32K_ONDEMAND |
|
|
SYSCTRL_XOSC32K_EN32K |
|
|
SYSCTRL_XOSC32K_XTALEN |
|
|
SYSCTRL_XOSC32K_STARTUP(6) |
|
|
#if RTT_RUNSTDBY
|
|
SYSCTRL_XOSC32K_RUNSTDBY |
|
|
#endif
|
|
SYSCTRL_XOSC32K_ENABLE;
|
|
|
|
/* Setup clock GCLK2 with divider 1 */
|
|
GCLK->GENDIV.reg = GCLK_GENDIV_ID(2) | GCLK_GENDIV_DIV(1);
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
|
|
|
/* Enable GCLK2 with XOSC32K as source. Use divider without modification
|
|
* and keep running in standby. */
|
|
GCLK->GENCTRL.reg = GCLK_GENCTRL_ID(2) |
|
|
GCLK_GENCTRL_GENEN |
|
|
#if RTT_RUNSTDBY
|
|
GCLK_GENCTRL_RUNSTDBY |
|
|
#endif
|
|
GCLK_GENCTRL_SRC_XOSC32K;
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
|
|
|
/* Connect GCLK2 to RTC */
|
|
GCLK->CLKCTRL.reg = GCLK_CLKCTRL_GEN_GCLK2 |
|
|
GCLK_CLKCTRL_CLKEN |
|
|
GCLK_CLKCTRL_ID(RTC_GCLK_ID);
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
|
}
|
|
/* !CPU_SAMD21 */
|
|
#else
|
|
static void _rtt_clock_setup(void)
|
|
{
|
|
/* Turn on power manager for RTC */
|
|
MCLK->APBAMASK.reg |= MCLK_APBAMASK_OSC32KCTRL;
|
|
|
|
/* set clock source */
|
|
OSC32KCTRL->RTCCTRL.reg = OSC32KCTRL_RTCCTRL_RTCSEL_ULP32K;
|
|
}
|
|
#endif
|
|
|
|
void rtt_init(void)
|
|
{
|
|
_rtt_clock_setup();
|
|
rtt_poweron();
|
|
|
|
_rtt_reset();
|
|
|
|
/* set 32bit counting mode & enable the RTC */
|
|
#ifdef REG_RTC_MODE0_CTRLA
|
|
RTC->MODE0.CTRLA.reg = RTC_MODE0_CTRLA_MODE(0) | RTC_MODE0_CTRLA_ENABLE;
|
|
#else
|
|
RTC->MODE0.CTRL.reg = RTC_MODE0_CTRL_MODE(0) | RTC_MODE0_CTRL_ENABLE;
|
|
#endif
|
|
_wait_syncbusy();
|
|
|
|
/* initially clear flag */
|
|
RTC->MODE0.INTFLAG.reg |= RTC_MODE0_INTFLAG_CMP0
|
|
| RTC_MODE0_INTFLAG_OVF;
|
|
|
|
NVIC_EnableIRQ(RTC_IRQn);
|
|
|
|
DEBUG("%s:%d %u\n", __func__, __LINE__, (unsigned)rtt_get_counter());
|
|
}
|
|
|
|
void rtt_set_overflow_cb(rtt_cb_t cb, void *arg)
|
|
{
|
|
/* clear overflow cb to avoid race while assigning */
|
|
rtt_clear_overflow_cb();
|
|
|
|
/* set callback variables */
|
|
_overflow_cb = cb;
|
|
_overflow_arg = arg;
|
|
|
|
/* enable overflow interrupt */
|
|
RTC->MODE0.INTENSET.bit.OVF = 1;
|
|
}
|
|
void rtt_clear_overflow_cb(void)
|
|
{
|
|
/* disable overflow interrupt */
|
|
RTC->MODE0.INTENCLR.bit.OVF = 1;
|
|
}
|
|
|
|
uint32_t rtt_get_counter(void)
|
|
{
|
|
_wait_syncbusy();
|
|
return RTC->MODE0.COUNT.reg;
|
|
}
|
|
|
|
void rtt_set_counter(uint32_t count)
|
|
{
|
|
RTC->MODE0.COUNT.reg = count;
|
|
_wait_syncbusy();
|
|
}
|
|
|
|
uint32_t rtt_get_alarm(void)
|
|
{
|
|
_wait_syncbusy();
|
|
return RTC->MODE0.COMP[0].reg;
|
|
}
|
|
|
|
void rtt_set_alarm(uint32_t alarm, rtt_cb_t cb, void *arg)
|
|
{
|
|
DEBUG("%s:%d alarm=%u\n", __func__, __LINE__, (unsigned)alarm);
|
|
|
|
/* disable interrupt to avoid race */
|
|
rtt_clear_alarm();
|
|
|
|
/* setup callback */
|
|
_cmp0_cb = cb;
|
|
_cmp0_arg = arg;
|
|
|
|
/* set COM register */
|
|
RTC->MODE0.COMP[0].reg = alarm;
|
|
_wait_syncbusy();
|
|
|
|
/* enable compare interrupt and clear flag */
|
|
RTC->MODE0.INTFLAG.reg |= RTC_MODE0_INTFLAG_CMP0;
|
|
RTC->MODE0.INTENSET.reg |= RTC_MODE0_INTENSET_CMP0;
|
|
}
|
|
|
|
void rtt_clear_alarm(void)
|
|
{
|
|
/* clear compare interrupt */
|
|
RTC->MODE0.INTENCLR.bit.CMP0 = 1;
|
|
}
|
|
|
|
void rtt_poweron(void)
|
|
{
|
|
#ifdef MCLK
|
|
MCLK->APBAMASK.reg |= MCLK_APBAMASK_RTC;
|
|
#else
|
|
PM->APBAMASK.reg |= PM_APBAMASK_RTC;
|
|
#endif
|
|
}
|
|
|
|
void rtt_poweroff(void)
|
|
{
|
|
#ifdef MCLK
|
|
MCLK->APBAMASK.reg &= ~MCLK_APBAMASK_RTC;
|
|
#else
|
|
PM->APBAMASK.reg &= ~PM_APBAMASK_RTC;
|
|
#endif
|
|
}
|
|
|
|
void isr_rtc(void)
|
|
{
|
|
if (RTC->MODE0.INTFLAG.bit.OVF) {
|
|
RTC->MODE0.INTFLAG.reg |= RTC_MODE0_INTFLAG_OVF;
|
|
if (_overflow_cb) {
|
|
_overflow_cb(_overflow_arg);
|
|
}
|
|
}
|
|
if (RTC->MODE0.INTFLAG.bit.CMP0) {
|
|
/* clear flag */
|
|
RTC->MODE0.INTFLAG.reg |= RTC_MODE0_INTFLAG_CMP0;
|
|
/* disable interrupt */
|
|
RTC->MODE0.INTENCLR.bit.CMP0 = 1;
|
|
if (_cmp0_cb) {
|
|
_cmp0_cb(_cmp0_arg);
|
|
}
|
|
}
|
|
cortexm_isr_end();
|
|
}
|