/* * Copyright (c) 2012 The Chromium OS Authors. * * SPDX-License-Identifier: GPL-2.0+ */ #include #include #include #include #include #include #include DECLARE_GLOBAL_DATA_PTR; void timer_set_base(u64 base) { gd->arch.tsc_base = base; } /* * Get the number of CPU time counter ticks since it was read first time after * restart. This yields a free running counter guaranteed to take almost 6 * years to wrap around even at 100GHz clock rate. */ u64 __attribute__((no_instrument_function)) get_ticks(void) { u64 now_tick = rdtsc(); /* We assume that 0 means the base hasn't been set yet */ if (!gd->arch.tsc_base) panic("No tick base available"); return now_tick - gd->arch.tsc_base; } #define PLATFORM_INFO_MSR 0xce /* Get the speed of the TSC timer in MHz */ unsigned __attribute__((no_instrument_function)) long get_tbclk_mhz(void) { u32 ratio; u64 platform_info = native_read_msr(PLATFORM_INFO_MSR); /* 100MHz times Max Non Turbo ratio */ ratio = (platform_info >> 8) & 0xff; return 100 * ratio; } unsigned long get_tbclk(void) { return get_tbclk_mhz() * 1000 * 1000; } static ulong get_ms_timer(void) { return (get_ticks() * 1000) / get_tbclk(); } ulong get_timer(ulong base) { return get_ms_timer() - base; } ulong __attribute__((no_instrument_function)) timer_get_us(void) { return get_ticks() / get_tbclk_mhz(); } ulong timer_get_boot_us(void) { return timer_get_us(); } void __udelay(unsigned long usec) { u64 now = get_ticks(); u64 stop; stop = now + usec * get_tbclk_mhz(); while ((int64_t)(stop - get_ticks()) > 0) ; } int timer_init(void) { #ifdef CONFIG_SYS_PCAT_TIMER /* Set up the PCAT timer if required */ pcat_timer_init(); #endif return 0; }