mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
207 lines
5.2 KiB
C
207 lines
5.2 KiB
C
|
/*
|
||
|
* 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"
|
||
|
#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 */
|
||
|
#if (RTT_FREQUENCY == 32768)
|
||
|
#define RTT_SCALE (0)
|
||
|
#elif (RTT_FREQUENCY == 16384)
|
||
|
#define RTT_SCALE (1)
|
||
|
#elif (RTT_FREQUENCY == 8192)
|
||
|
#define RTT_SCALE (2)
|
||
|
#elif (RTT_FREQUENCY == 4096)
|
||
|
#define RTT_SCALE (3)
|
||
|
#elif (RTT_FREQUENCY == 2048)
|
||
|
#define RTT_SCALE (4)
|
||
|
#elif (RTT_FREQUENCY == 1024)
|
||
|
#define RTT_SCALE (5)
|
||
|
#elif (RTT_FREQUENCY == 512)
|
||
|
#define RTT_SCALE (6)
|
||
|
#elif (RTT_FREQUENCY == 256)
|
||
|
#define RTT_SCALE (7)
|
||
|
#elif (RTT_FREQUENCY == 128)
|
||
|
#define RTT_SCALE (8)
|
||
|
#elif (RTT_FREQUENCY == 64)
|
||
|
#define RTT_SCALE (9)
|
||
|
#elif (RTT_FREQUENCY == 32)
|
||
|
#define RTT_SCALE (10)
|
||
|
#elif (RTT_FREQUENCY == 16)
|
||
|
#define RTT_SCALE (11)
|
||
|
#elif (RTT_FREQUENCY == 8)
|
||
|
#define RTT_SCALE (12)
|
||
|
#elif (RTT_FREQUENCY == 4)
|
||
|
#define RTT_SCALE (13)
|
||
|
#elif (RTT_FREQUENCY == 2)
|
||
|
#define RTT_SCALE (14)
|
||
|
#elif (RTT_FREQUENCY == 1)
|
||
|
#define RTT_SCALE (15)
|
||
|
#else
|
||
|
#error "Invalid RTT_FREQUENCY: Must be power of 2"
|
||
|
#endif
|
||
|
|
||
|
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 */
|
||
|
set_external_isr_cb(INT_RTCCMP, rtt_isr);
|
||
|
PLIC_enable_interrupt(INT_RTCCMP);
|
||
|
PLIC_set_priority(INT_RTCCMP, RTT_INTR_PRIORITY);
|
||
|
|
||
|
/* 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
|
||
|
*/
|
||
|
AON_REG(AON_RTCLO) = counter << RTT_SCALE;
|
||
|
AON_REG(AON_RTCHI) = counter >> (32 - RTT_SCALE);
|
||
|
}
|
||
|
|
||
|
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;
|
||
|
}
|