mirror of
https://github.com/riscv-software-src/opensbi.git
synced 2026-05-23 14:21:32 +01:00
1932ee3f0a
The users of timer event have to compute next_event (aka timer value in the future) based on desired units and unit frequency. Introduce sbi_timer_compute_delta() and friends to simplify computing next_event for timer event users. Signed-off-by: Anup Patel <anup.patel@oss.qualcomm.com> Reviewed-by: Nicholas Piggin <npiggin@gmail.com> Link: https://lore.kernel.org/r/20260425104048.2335262-5-anup.patel@oss.qualcomm.com Signed-off-by: Anup Patel <anup@brainfault.org>
397 lines
9.5 KiB
C
397 lines
9.5 KiB
C
/*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*
|
|
* Copyright (c) 2019 Western Digital Corporation or its affiliates.
|
|
*
|
|
* Authors:
|
|
* Anup Patel <anup.patel@wdc.com>
|
|
*/
|
|
|
|
#include <sbi/riscv_asm.h>
|
|
#include <sbi/riscv_barrier.h>
|
|
#include <sbi/riscv_encoding.h>
|
|
#include <sbi/riscv_locks.h>
|
|
#include <sbi/sbi_console.h>
|
|
#include <sbi/sbi_error.h>
|
|
#include <sbi/sbi_hart.h>
|
|
#include <sbi/sbi_hartmask.h>
|
|
#include <sbi/sbi_platform.h>
|
|
#include <sbi/sbi_pmu.h>
|
|
#include <sbi/sbi_scratch.h>
|
|
#include <sbi/sbi_timer.h>
|
|
|
|
struct timer_state {
|
|
u64 time_delta;
|
|
spinlock_t event_list_lock;
|
|
struct sbi_dlist event_list;
|
|
struct sbi_timer_event smode_ev;
|
|
};
|
|
|
|
static unsigned long timer_state_off;
|
|
static u64 (*get_time_val)(void);
|
|
static const struct sbi_timer_device *timer_dev = NULL;
|
|
|
|
#if __riscv_xlen == 32
|
|
static u64 get_ticks(void)
|
|
{
|
|
u32 lo, hi, tmp;
|
|
__asm__ __volatile__("1:\n"
|
|
"rdtimeh %0\n"
|
|
"rdtime %1\n"
|
|
"rdtimeh %2\n"
|
|
"bne %0, %2, 1b"
|
|
: "=&r"(hi), "=&r"(lo), "=&r"(tmp));
|
|
return ((u64)hi << 32) | lo;
|
|
}
|
|
#else
|
|
static u64 get_ticks(void)
|
|
{
|
|
unsigned long n;
|
|
|
|
__asm__ __volatile__("rdtime %0" : "=r"(n));
|
|
return n;
|
|
}
|
|
#endif
|
|
|
|
static void nop_delay_fn(void *opaque)
|
|
{
|
|
cpu_relax();
|
|
}
|
|
|
|
u64 sbi_timer_compute_delta(ulong units, u64 unit_freq)
|
|
{
|
|
u64 delta;
|
|
|
|
delta = ((u64)timer_dev->timer_freq * (u64)units);
|
|
delta = delta / unit_freq;
|
|
return delta;
|
|
}
|
|
|
|
void sbi_timer_delay_loop(ulong units, u64 unit_freq,
|
|
void (*delay_fn)(void *), void *opaque)
|
|
{
|
|
u64 start_val, delta;
|
|
|
|
/* Do nothing if we don't have timer device */
|
|
if (!timer_dev || !get_time_val) {
|
|
sbi_printf("%s: called without timer device\n", __func__);
|
|
return;
|
|
}
|
|
|
|
/* Save starting timer value */
|
|
start_val = get_time_val();
|
|
|
|
/* Use NOP delay function if delay function not available */
|
|
if (!delay_fn)
|
|
delay_fn = nop_delay_fn;
|
|
|
|
/* Busy loop until desired timer value delta reached */
|
|
delta = sbi_timer_compute_delta(units, unit_freq);
|
|
while ((get_time_val() - start_val) < delta)
|
|
delay_fn(opaque);
|
|
}
|
|
|
|
bool sbi_timer_waitms_until(bool (*predicate)(void *), void *arg,
|
|
uint64_t timeout_ms)
|
|
{
|
|
uint64_t start_time = sbi_timer_value();
|
|
uint64_t ticks =
|
|
(sbi_timer_get_device()->timer_freq / 1000) *
|
|
timeout_ms;
|
|
while(!predicate(arg))
|
|
if (sbi_timer_value() - start_time >= ticks)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
u64 sbi_timer_value(void)
|
|
{
|
|
if (get_time_val)
|
|
return get_time_val();
|
|
return 0;
|
|
}
|
|
|
|
u64 sbi_timer_virt_value(void)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
|
|
return sbi_timer_value() + tstate->time_delta;
|
|
}
|
|
|
|
u64 sbi_timer_get_delta(void)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
|
|
return tstate->time_delta;
|
|
}
|
|
|
|
void sbi_timer_set_delta(ulong delta)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
|
|
#if __riscv_xlen == 32
|
|
tstate->time_delta &= ~0xffffffffUL;
|
|
tstate->time_delta |= (u32)delta;
|
|
#else
|
|
tstate->time_delta = delta;
|
|
#endif
|
|
}
|
|
|
|
#if __riscv_xlen == 32
|
|
void sbi_timer_set_delta_upper(ulong delta_upper)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
|
|
tstate->time_delta &= 0xffffffffUL;
|
|
tstate->time_delta |= (u64)delta_upper << 32;
|
|
}
|
|
#endif
|
|
|
|
static void __sbi_timer_update_device(struct timer_state *tstate)
|
|
{
|
|
struct sbi_timer_event *ev;
|
|
|
|
if (!timer_dev)
|
|
return;
|
|
|
|
if (sbi_list_empty(&tstate->event_list)) {
|
|
if (timer_dev->timer_event_stop)
|
|
timer_dev->timer_event_stop();
|
|
csr_clear(CSR_MIE, MIP_MTIP);
|
|
} else {
|
|
ev = sbi_list_first_entry(&tstate->event_list, struct sbi_timer_event, head);
|
|
if (timer_dev->timer_event_start)
|
|
timer_dev->timer_event_start(ev->time_stamp);
|
|
csr_set(CSR_MIE, MIP_MTIP);
|
|
}
|
|
}
|
|
|
|
static void __sbi_timer_event_stop(struct sbi_timer_event *ev)
|
|
{
|
|
if (ev->hart_index > -1) {
|
|
sbi_list_del(&ev->head);
|
|
ev->hart_index = -1;
|
|
}
|
|
}
|
|
|
|
static void __sbi_timer_event_start(struct timer_state *tstate,
|
|
struct sbi_timer_event *ev, u64 next_event)
|
|
{
|
|
struct sbi_timer_event *tev, *next_ev = NULL;
|
|
|
|
/* Find where to insert the event in per-HART event list */
|
|
sbi_list_for_each_entry(tev, &tstate->event_list, head) {
|
|
if (next_event < tev->time_stamp) {
|
|
next_ev = tev;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Insert the event in per-HART event list */
|
|
ev->hart_index = current_hartindex();
|
|
ev->time_stamp = next_event;
|
|
if (next_ev)
|
|
sbi_list_add(&ev->head, &next_ev->head);
|
|
else
|
|
sbi_list_add_tail(&ev->head, &tstate->event_list);
|
|
}
|
|
|
|
void sbi_timer_event_start(struct sbi_timer_event *ev, u64 next_event)
|
|
{
|
|
struct timer_state *tstate;
|
|
|
|
if (!ev)
|
|
return;
|
|
|
|
/* Ensure that event is not on the per-HART event list */
|
|
if (ev->hart_index > -1) {
|
|
tstate = sbi_scratch_offset_ptr(sbi_hartindex_to_scratch(ev->hart_index),
|
|
timer_state_off);
|
|
spin_lock(&tstate->event_list_lock);
|
|
__sbi_timer_event_stop(ev);
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|
|
|
|
tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
spin_lock(&tstate->event_list_lock);
|
|
|
|
__sbi_timer_event_start(tstate, ev, next_event);
|
|
__sbi_timer_update_device(tstate);
|
|
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|
|
|
|
void sbi_timer_event_stop(struct sbi_timer_event *ev)
|
|
{
|
|
struct timer_state *tstate;
|
|
int ev_hart_index;
|
|
|
|
if (!ev)
|
|
return;
|
|
|
|
/* Ensure that event is not on the per-HART event list */
|
|
ev_hart_index = ev->hart_index;
|
|
if (ev->hart_index > -1) {
|
|
tstate = sbi_scratch_offset_ptr(sbi_hartindex_to_scratch(ev->hart_index),
|
|
timer_state_off);
|
|
spin_lock(&tstate->event_list_lock);
|
|
__sbi_timer_event_stop(ev);
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|
|
|
|
/* Re-program timer device on the current HART */
|
|
if (ev_hart_index == current_hartindex()) {
|
|
tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
spin_lock(&tstate->event_list_lock);
|
|
__sbi_timer_update_device(tstate);
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|
|
}
|
|
|
|
static void sbi_timer_smode_event_callback(struct sbi_timer_event *ev,
|
|
struct sbi_timer_event_restart *restart)
|
|
{
|
|
/*
|
|
* If sstc extension is available, supervisor can receive the timer
|
|
* directly without M-mode come in between. This function should
|
|
* only invoked if M-mode programs the timer for its own purpose.
|
|
*/
|
|
if (!sbi_hart_has_extension(sbi_scratch_thishart_ptr(), SBI_HART_EXT_SSTC))
|
|
csr_set(CSR_MIP, MIP_STIP);
|
|
}
|
|
|
|
static void sbi_timer_smode_event_cleanup(struct sbi_timer_event *ev)
|
|
{
|
|
if (!sbi_hart_has_extension(sbi_scratch_thishart_ptr(), SBI_HART_EXT_SSTC))
|
|
csr_clear(CSR_MIP, MIP_STIP);
|
|
}
|
|
|
|
void sbi_timer_smode_event_start(u64 next_event)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_offset_ptr(sbi_scratch_thishart_ptr(),
|
|
timer_state_off);
|
|
|
|
sbi_pmu_ctr_incr_fw(SBI_PMU_FW_SET_TIMER);
|
|
|
|
/**
|
|
* Update the stimecmp directly if available. This allows
|
|
* the older software to leverage sstc extension on newer hardware.
|
|
*/
|
|
if (sbi_hart_has_extension(sbi_scratch_thishart_ptr(), SBI_HART_EXT_SSTC)) {
|
|
csr_write64(CSR_STIMECMP, next_event);
|
|
} else {
|
|
csr_clear(CSR_MIP, MIP_STIP);
|
|
sbi_timer_event_start(&tstate->smode_ev, next_event);
|
|
}
|
|
}
|
|
|
|
void sbi_timer_process(void)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
struct sbi_timer_event_restart restart;
|
|
SBI_LIST_HEAD(restart_list);
|
|
struct sbi_timer_event *ev;
|
|
|
|
spin_lock(&tstate->event_list_lock);
|
|
|
|
while (!sbi_list_empty(&tstate->event_list)) {
|
|
ev = sbi_list_first_entry(&tstate->event_list, struct sbi_timer_event, head);
|
|
if (ev->time_stamp > sbi_timer_value())
|
|
break;
|
|
|
|
__sbi_timer_event_stop(ev);
|
|
if (ev->callback) {
|
|
restart.required = false;
|
|
restart.next_event = 0;
|
|
ev->callback(ev, &restart);
|
|
if (restart.required) {
|
|
ev->time_stamp = restart.next_event;
|
|
sbi_list_add_tail(&ev->head, &restart_list);
|
|
}
|
|
}
|
|
}
|
|
|
|
while (!sbi_list_empty(&restart_list)) {
|
|
ev = sbi_list_first_entry(&tstate->event_list, struct sbi_timer_event, head);
|
|
sbi_list_del(&ev->head);
|
|
__sbi_timer_event_start(tstate, ev, ev->time_stamp);
|
|
}
|
|
|
|
__sbi_timer_update_device(tstate);
|
|
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|
|
|
|
const struct sbi_timer_device *sbi_timer_get_device(void)
|
|
{
|
|
return timer_dev;
|
|
}
|
|
|
|
void sbi_timer_set_device(const struct sbi_timer_device *dev)
|
|
{
|
|
if (!dev || timer_dev)
|
|
return;
|
|
|
|
timer_dev = dev;
|
|
if (!get_time_val && timer_dev->timer_value)
|
|
get_time_val = timer_dev->timer_value;
|
|
}
|
|
|
|
int sbi_timer_init(struct sbi_scratch *scratch, bool cold_boot)
|
|
{
|
|
const struct sbi_platform *plat = sbi_platform_ptr(scratch);
|
|
struct timer_state *tstate;
|
|
int ret;
|
|
|
|
if (cold_boot) {
|
|
timer_state_off = sbi_scratch_alloc_offset(sizeof(*tstate));
|
|
if (!timer_state_off)
|
|
return SBI_ENOMEM;
|
|
|
|
if (sbi_hart_has_csr(scratch, SBI_HART_CSR_TIME))
|
|
get_time_val = get_ticks;
|
|
|
|
ret = sbi_platform_timer_init(plat);
|
|
if (ret)
|
|
return ret;
|
|
} else {
|
|
if (!timer_state_off)
|
|
return SBI_ENOMEM;
|
|
}
|
|
|
|
tstate = sbi_scratch_offset_ptr(scratch, timer_state_off);
|
|
tstate->time_delta = 0;
|
|
SPIN_LOCK_INIT(tstate->event_list_lock);
|
|
SBI_INIT_LIST_HEAD(&tstate->event_list);
|
|
SBI_INIT_TIMER_EVENT(&tstate->smode_ev,
|
|
sbi_timer_smode_event_callback,
|
|
sbi_timer_smode_event_cleanup, NULL);
|
|
|
|
if (timer_dev && timer_dev->warm_init) {
|
|
ret = timer_dev->warm_init();
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void sbi_timer_exit(struct sbi_scratch *scratch)
|
|
{
|
|
struct timer_state *tstate = sbi_scratch_thishart_offset_ptr(timer_state_off);
|
|
struct sbi_timer_event *ev;
|
|
|
|
spin_lock(&tstate->event_list_lock);
|
|
|
|
while (!sbi_list_empty(&tstate->event_list)) {
|
|
ev = sbi_list_first_entry(&tstate->event_list, struct sbi_timer_event, head);
|
|
__sbi_timer_event_stop(ev);
|
|
if (ev->cleanup)
|
|
ev->cleanup(ev);
|
|
}
|
|
|
|
__sbi_timer_update_device(tstate);
|
|
|
|
spin_unlock(&tstate->event_list_lock);
|
|
}
|