| /* |
| * 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); |
| } |