162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci#include <linux/types.h> 362306a36Sopenharmony_ci#include <math.h> 462306a36Sopenharmony_ci#include <string.h> 562306a36Sopenharmony_ci#include <stdlib.h> 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include "../../../util/debug.h" 862306a36Sopenharmony_ci#include "../../../util/tsc.h" 962306a36Sopenharmony_ci#include "cpuid.h" 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ciu64 rdtsc(void) 1262306a36Sopenharmony_ci{ 1362306a36Sopenharmony_ci unsigned int low, high; 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci asm volatile("rdtsc" : "=a" (low), "=d" (high)); 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci return low | ((u64)high) << 32; 1862306a36Sopenharmony_ci} 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci/* 2162306a36Sopenharmony_ci * Derive the TSC frequency in Hz from the /proc/cpuinfo, for example: 2262306a36Sopenharmony_ci * ... 2362306a36Sopenharmony_ci * model name : Intel(R) Xeon(R) Gold 6154 CPU @ 3.00GHz 2462306a36Sopenharmony_ci * ... 2562306a36Sopenharmony_ci * will return 3000000000. 2662306a36Sopenharmony_ci */ 2762306a36Sopenharmony_cistatic double cpuinfo_tsc_freq(void) 2862306a36Sopenharmony_ci{ 2962306a36Sopenharmony_ci double result = 0; 3062306a36Sopenharmony_ci FILE *cpuinfo; 3162306a36Sopenharmony_ci char *line = NULL; 3262306a36Sopenharmony_ci size_t len = 0; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci cpuinfo = fopen("/proc/cpuinfo", "r"); 3562306a36Sopenharmony_ci if (!cpuinfo) { 3662306a36Sopenharmony_ci pr_err("Failed to read /proc/cpuinfo for TSC frequency"); 3762306a36Sopenharmony_ci return NAN; 3862306a36Sopenharmony_ci } 3962306a36Sopenharmony_ci while (getline(&line, &len, cpuinfo) > 0) { 4062306a36Sopenharmony_ci if (!strncmp(line, "model name", 10)) { 4162306a36Sopenharmony_ci char *pos = strstr(line + 11, " @ "); 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci if (pos && sscanf(pos, " @ %lfGHz", &result) == 1) { 4462306a36Sopenharmony_ci result *= 1000000000; 4562306a36Sopenharmony_ci goto out; 4662306a36Sopenharmony_ci } 4762306a36Sopenharmony_ci } 4862306a36Sopenharmony_ci } 4962306a36Sopenharmony_ciout: 5062306a36Sopenharmony_ci if (fpclassify(result) == FP_ZERO) 5162306a36Sopenharmony_ci pr_err("Failed to find TSC frequency in /proc/cpuinfo"); 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci free(line); 5462306a36Sopenharmony_ci fclose(cpuinfo); 5562306a36Sopenharmony_ci return result; 5662306a36Sopenharmony_ci} 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_cidouble arch_get_tsc_freq(void) 5962306a36Sopenharmony_ci{ 6062306a36Sopenharmony_ci unsigned int a, b, c, d, lvl; 6162306a36Sopenharmony_ci static bool cached; 6262306a36Sopenharmony_ci static double tsc; 6362306a36Sopenharmony_ci char vendor[16]; 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci if (cached) 6662306a36Sopenharmony_ci return tsc; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci cached = true; 6962306a36Sopenharmony_ci get_cpuid_0(vendor, &lvl); 7062306a36Sopenharmony_ci if (!strstr(vendor, "Intel")) 7162306a36Sopenharmony_ci return 0; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci /* 7462306a36Sopenharmony_ci * Don't support Time Stamp Counter and 7562306a36Sopenharmony_ci * Nominal Core Crystal Clock Information Leaf. 7662306a36Sopenharmony_ci */ 7762306a36Sopenharmony_ci if (lvl < 0x15) { 7862306a36Sopenharmony_ci tsc = cpuinfo_tsc_freq(); 7962306a36Sopenharmony_ci return tsc; 8062306a36Sopenharmony_ci } 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_ci cpuid(0x15, 0, &a, &b, &c, &d); 8362306a36Sopenharmony_ci /* TSC frequency is not enumerated */ 8462306a36Sopenharmony_ci if (!a || !b || !c) { 8562306a36Sopenharmony_ci tsc = cpuinfo_tsc_freq(); 8662306a36Sopenharmony_ci return tsc; 8762306a36Sopenharmony_ci } 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci tsc = (double)c * (double)b / (double)a; 9062306a36Sopenharmony_ci return tsc; 9162306a36Sopenharmony_ci} 92