2018-04-16 19:03:33 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Ken Rabold
|
|
|
|
*
|
|
|
|
* 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_fe310
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file rtt.c
|
|
|
|
* @brief Low-level RTT interface implementation for SiFive FE310
|
|
|
|
* The FE320's RTC module is what RIOT calls a Real-Time
|
|
|
|
* Timer (RTT), a simple counter which counts seconds; RIOT Real-
|
|
|
|
* Time Clocks (RTC) counts seconds, minutes, hours etc. There is
|
|
|
|
* an RTT->RTC wrapper layer in a separate file to allow using the
|
|
|
|
* RTT as a system real time clock.
|
|
|
|
*
|
|
|
|
* @author Ken Rabold
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph_cpu.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/rtt.h"
|
2020-08-28 20:35:37 +02:00
|
|
|
#include "plic.h"
|
2018-04-16 19:03:33 +02:00
|
|
|
#include "vendor/encoding.h"
|
|
|
|
#include "vendor/platform.h"
|
|
|
|
#include "vendor/plic_driver.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* Convert RTT freq to pre-scaler value */
|
2020-04-17 10:12:26 +02:00
|
|
|
#ifdef RTT_FREQUENCY
|
|
|
|
#if ((RTT_CLOCK_FREQUENCY % RTT_FREQUENCY) != 0)
|
2018-04-16 19:03:33 +02:00
|
|
|
#error "Invalid RTT_FREQUENCY: Must be power of 2"
|
|
|
|
#endif
|
2020-04-17 10:12:26 +02:00
|
|
|
#endif
|
|
|
|
#define RTT_SCALE __builtin_ctz(RTT_CLOCK_FREQUENCY / RTT_FREQUENCY)
|
2018-04-16 19:03:33 +02:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
uint32_t alarm_val; /**< cached alarm val */
|
|
|
|
rtt_cb_t alarm_cb; /**< callback called from RTC alarm */
|
|
|
|
void *alarm_arg; /**< argument passed to the callback */
|
|
|
|
rtt_cb_t overflow_cb; /**< callback called when RTC overflows */
|
|
|
|
void *overflow_arg; /**< argument passed to the callback */
|
|
|
|
}rtt_state_t;
|
|
|
|
|
|
|
|
static rtt_state_t rtt_callback;
|
|
|
|
|
|
|
|
void rtt_isr(int num)
|
|
|
|
{
|
|
|
|
(void) num;
|
|
|
|
|
|
|
|
/* Clear intr */
|
|
|
|
AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
|
|
|
|
|
|
|
|
/* Invoke callback function for alarm or overflow */
|
|
|
|
if (rtt_callback.alarm_cb) {
|
|
|
|
rtt_callback.alarm_cb(rtt_callback.alarm_arg);
|
|
|
|
}
|
|
|
|
else if (rtt_callback.overflow_cb) {
|
|
|
|
rtt_callback.overflow_cb(rtt_callback.overflow_arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_init(void)
|
|
|
|
{
|
|
|
|
/* Disable ext interrupts when setting up */
|
|
|
|
clear_csr(mie, MIP_MEIP);
|
|
|
|
|
|
|
|
/* Configure RTC ISR with PLIC */
|
2020-08-28 20:35:37 +02:00
|
|
|
plic_set_isr_cb(INT_RTCCMP, rtt_isr);
|
|
|
|
plic_enable_interrupt(INT_RTCCMP);
|
|
|
|
plic_set_priority(INT_RTCCMP, RTT_INTR_PRIORITY);
|
2018-04-16 19:03:33 +02:00
|
|
|
|
|
|
|
/* Configure RTC scaler, etc... */
|
|
|
|
AON_REG(AON_RTCCFG) = RTT_SCALE;
|
|
|
|
AON_REG(AON_RTCLO) = 0;
|
|
|
|
AON_REG(AON_RTCHI) = 0;
|
|
|
|
AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
|
|
|
|
|
|
|
|
/* Configure state struct */
|
|
|
|
rtt_callback.alarm_val = RTT_MAX_VALUE;
|
|
|
|
rtt_callback.alarm_cb = NULL;
|
|
|
|
rtt_callback.alarm_arg = NULL;
|
|
|
|
rtt_callback.overflow_cb = NULL;
|
|
|
|
rtt_callback.overflow_arg = NULL;
|
|
|
|
|
|
|
|
/* Enable the RTC counter */
|
|
|
|
rtt_poweron();
|
|
|
|
|
|
|
|
/* Re-eanble ext interrupts */
|
|
|
|
set_csr(mie, MIP_MEIP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_set_overflow_cb(rtt_cb_t cb, void *arg)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* No separate overflow intr
|
|
|
|
* If no alarm cb is set, this cb will fire on RTC intr
|
|
|
|
*/
|
|
|
|
rtt_callback.overflow_cb = cb;
|
|
|
|
rtt_callback.overflow_arg = arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_clear_overflow_cb(void)
|
|
|
|
{
|
|
|
|
/* No separate overflow intr */
|
|
|
|
rtt_callback.overflow_cb = NULL;
|
|
|
|
rtt_callback.overflow_arg = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t rtt_get_counter(void)
|
|
|
|
{
|
|
|
|
/* Read scaled counter reg value */
|
|
|
|
uint32_t t = AON_REG(AON_RTCS);
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_set_counter(uint32_t counter)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Can't write to scaled RTC reg
|
|
|
|
* Must program HI/LO regs
|
|
|
|
* Write scaled counter reg value
|
|
|
|
*/
|
2020-04-17 10:12:26 +02:00
|
|
|
/* Use ifdef to avoid out of bound shift when RTT_SCALE == 0 */
|
|
|
|
#if RTT_CLOCK_FREQUENCY == RTT_FREQUENCY
|
|
|
|
AON_REG(AON_RTCLO) = counter;
|
|
|
|
AON_REG(AON_RTCHI) = 0;
|
|
|
|
#else
|
|
|
|
AON_REG(AON_RTCLO) = counter << RTT_SCALE;
|
|
|
|
AON_REG(AON_RTCHI) = counter >> (32 - RTT_SCALE);
|
|
|
|
#endif
|
2018-04-16 19:03:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_set_alarm(uint32_t alarm, rtt_cb_t cb, void *arg)
|
|
|
|
{
|
|
|
|
/* Set cmp reg to given value */
|
|
|
|
AON_REG(AON_RTCCMP) = alarm;
|
|
|
|
rtt_callback.alarm_val = alarm;
|
|
|
|
rtt_callback.alarm_cb = cb;
|
|
|
|
rtt_callback.alarm_arg = arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t rtt_get_alarm(void)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Read back cached alarm value
|
|
|
|
* When alarm fires, cmp reg is set to highest val
|
|
|
|
* to clear intr, so used cached value for this function
|
|
|
|
*/
|
|
|
|
return rtt_callback.alarm_val;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_clear_alarm(void)
|
|
|
|
{
|
|
|
|
/* Set cmp reg to highest value */
|
|
|
|
AON_REG(AON_RTCCMP) = RTT_MAX_VALUE;
|
|
|
|
rtt_callback.alarm_val = RTT_MAX_VALUE;
|
|
|
|
rtt_callback.alarm_cb = NULL;
|
|
|
|
rtt_callback.alarm_arg = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_poweron(void)
|
|
|
|
{
|
|
|
|
/* Enable the RTC counter */
|
|
|
|
AON_REG(AON_RTCCFG) |= AON_RTCCFG_ENALWAYS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rtt_poweroff(void)
|
|
|
|
{
|
|
|
|
/* Disable the RTC counter */
|
|
|
|
AON_REG(AON_RTCCFG) &= ~AON_RTCCFG_ENALWAYS;
|
|
|
|
}
|