@@ -239,7 +239,7 @@ static inline unsigned long long cycles_2_ns(unsigned long long cyc)
239
239
return ns ;
240
240
}
241
241
242
- static void set_cyc2ns_scale (unsigned long cpu_khz , int cpu )
242
+ static void set_cyc2ns_scale (unsigned long khz , int cpu )
243
243
{
244
244
unsigned long long tsc_now , ns_now ;
245
245
struct cyc2ns_data * data ;
@@ -248,7 +248,7 @@ static void set_cyc2ns_scale(unsigned long cpu_khz, int cpu)
248
248
local_irq_save (flags );
249
249
sched_clock_idle_sleep_event ();
250
250
251
- if (!cpu_khz )
251
+ if (!khz )
252
252
goto done ;
253
253
254
254
data = cyc2ns_write_begin (cpu );
@@ -261,7 +261,7 @@ static void set_cyc2ns_scale(unsigned long cpu_khz, int cpu)
261
261
* time function is continuous; see the comment near struct
262
262
* cyc2ns_data.
263
263
*/
264
- clocks_calc_mult_shift (& data -> cyc2ns_mul , & data -> cyc2ns_shift , cpu_khz ,
264
+ clocks_calc_mult_shift (& data -> cyc2ns_mul , & data -> cyc2ns_shift , khz ,
265
265
NSEC_PER_MSEC , 0 );
266
266
267
267
/*
@@ -665,15 +665,72 @@ static unsigned long quick_pit_calibrate(void)
665
665
}
666
666
667
667
/**
668
- * native_calibrate_tsc - calibrate the tsc on boot
668
+ * native_calibrate_tsc
669
+ * Determine TSC frequency via CPUID, else return 0.
669
670
*/
670
671
unsigned long native_calibrate_tsc (void )
672
+ {
673
+ unsigned int eax_denominator , ebx_numerator , ecx_hz , edx ;
674
+ unsigned int crystal_khz ;
675
+
676
+ if (boot_cpu_data .x86_vendor != X86_VENDOR_INTEL )
677
+ return 0 ;
678
+
679
+ if (boot_cpu_data .cpuid_level < 0x15 )
680
+ return 0 ;
681
+
682
+ eax_denominator = ebx_numerator = ecx_hz = edx = 0 ;
683
+
684
+ /* CPUID 15H TSC/Crystal ratio, plus optionally Crystal Hz */
685
+ cpuid (0x15 , & eax_denominator , & ebx_numerator , & ecx_hz , & edx );
686
+
687
+ if (ebx_numerator == 0 || eax_denominator == 0 )
688
+ return 0 ;
689
+
690
+ crystal_khz = ecx_hz / 1000 ;
691
+
692
+ if (crystal_khz == 0 ) {
693
+ switch (boot_cpu_data .x86_model ) {
694
+ case 0x4E : /* SKL */
695
+ case 0x5E : /* SKL */
696
+ crystal_khz = 24000 ; /* 24 MHz */
697
+ }
698
+ }
699
+
700
+ return crystal_khz * ebx_numerator / eax_denominator ;
701
+ }
702
+
703
+ static unsigned long cpu_khz_from_cpuid (void )
704
+ {
705
+ unsigned int eax_base_mhz , ebx_max_mhz , ecx_bus_mhz , edx ;
706
+
707
+ if (boot_cpu_data .x86_vendor != X86_VENDOR_INTEL )
708
+ return 0 ;
709
+
710
+ if (boot_cpu_data .cpuid_level < 0x16 )
711
+ return 0 ;
712
+
713
+ eax_base_mhz = ebx_max_mhz = ecx_bus_mhz = edx = 0 ;
714
+
715
+ cpuid (0x16 , & eax_base_mhz , & ebx_max_mhz , & ecx_bus_mhz , & edx );
716
+
717
+ return eax_base_mhz * 1000 ;
718
+ }
719
+
720
+ /**
721
+ * native_calibrate_cpu - calibrate the cpu on boot
722
+ */
723
+ unsigned long native_calibrate_cpu (void )
671
724
{
672
725
u64 tsc1 , tsc2 , delta , ref1 , ref2 ;
673
726
unsigned long tsc_pit_min = ULONG_MAX , tsc_ref_min = ULONG_MAX ;
674
727
unsigned long flags , latch , ms , fast_calibrate ;
675
728
int hpet = is_hpet_enabled (), i , loopmin ;
676
729
730
+ fast_calibrate = cpu_khz_from_cpuid ();
731
+ if (fast_calibrate )
732
+ return fast_calibrate ;
733
+
677
734
fast_calibrate = cpu_khz_from_msr ();
678
735
if (fast_calibrate )
679
736
return fast_calibrate ;
@@ -834,8 +891,10 @@ int recalibrate_cpu_khz(void)
834
891
if (!boot_cpu_has (X86_FEATURE_TSC ))
835
892
return - ENODEV ;
836
893
894
+ cpu_khz = x86_platform .calibrate_cpu ();
837
895
tsc_khz = x86_platform .calibrate_tsc ();
838
- cpu_khz = tsc_khz ;
896
+ if (tsc_khz == 0 )
897
+ tsc_khz = cpu_khz ;
839
898
cpu_data (0 ).loops_per_jiffy = cpufreq_scale (cpu_data (0 ).loops_per_jiffy ,
840
899
cpu_khz_old , cpu_khz );
841
900
@@ -1241,8 +1300,10 @@ void __init tsc_init(void)
1241
1300
return ;
1242
1301
}
1243
1302
1303
+ cpu_khz = x86_platform .calibrate_cpu ();
1244
1304
tsc_khz = x86_platform .calibrate_tsc ();
1245
- cpu_khz = tsc_khz ;
1305
+ if (tsc_khz == 0 )
1306
+ tsc_khz = cpu_khz ;
1246
1307
1247
1308
if (!tsc_khz ) {
1248
1309
mark_tsc_unstable ("could not calculate TSC khz" );
@@ -1262,7 +1323,7 @@ void __init tsc_init(void)
1262
1323
*/
1263
1324
for_each_possible_cpu (cpu ) {
1264
1325
cyc2ns_init (cpu );
1265
- set_cyc2ns_scale (cpu_khz , cpu );
1326
+ set_cyc2ns_scale (tsc_khz , cpu );
1266
1327
}
1267
1328
1268
1329
if (tsc_disabled > 0 )
0 commit comments