/* * SPDX-License-Identifier: BSD-2-Clause * * Copyright (c) 2019 Western Digital Corporation or its affiliates. * * Authors: * Anup Patel */ #include #include #include #include #include #include #include #include #include #include #include #include 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); }