/* SPDX-License-Identifier: GPL-2.0-only */ #include #include #include #include #include void init_timer(void) { (void)tsc_freq_mhz(); } void udelay(unsigned int us) { unsigned long long start; unsigned long long current; unsigned long long clocks; if (!thread_yield_microseconds(us)) return; start = rdtscll(); clocks = us; clocks *= tsc_freq_mhz(); current = rdtscll(); while ((current - start) < clocks) { cpu_relax(); current = rdtscll(); } } #if CONFIG(TSC_MONOTONIC_TIMER) #include static struct monotonic_counter { int initialized; struct mono_time time; uint64_t last_value; } mono_counter; void timer_monotonic_get(struct mono_time *mt) { uint64_t current_tick; uint64_t ticks_elapsed; unsigned long ticks_per_usec; if (!mono_counter.initialized) { init_timer(); mono_counter.last_value = rdtscll(); mono_counter.initialized = 1; } current_tick = rdtscll(); ticks_elapsed = current_tick - mono_counter.last_value; ticks_per_usec = tsc_freq_mhz(); /* Update current time and tick values only if a full tick occurred. */ if (ticks_elapsed >= ticks_per_usec) { uint64_t usecs_elapsed; usecs_elapsed = ticks_elapsed / ticks_per_usec; mono_time_add_usecs(&mono_counter.time, (long)usecs_elapsed); mono_counter.last_value = current_tick; } /* Save result. */ *mt = mono_counter.time; } #endif