From: Andi Kleen Date: Wed, 30 Jan 2008 12:32:38 +0000 (+0100) Subject: x86: introduce rdtsc_barrier() X-Git-Tag: firefly_0821_release~23636^2~423 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=fde1b3fa947c2512e3715962ebb1d3a6a9b9bb7d;p=firefly-linux-kernel-4.4.55.git x86: introduce rdtsc_barrier() rdtsc_barrier() is a new barrier primitive that stops RDTSC speculation to avoid races with timer interrupts on other CPUs. It expands either to LFENCE (for Intel CPUs) or MFENCE (for AMD CPUs) which stops RDTSC on all currently known microarchitectures that implement SSE. On CPUs without SSE there is generally no RDTSC speculation. [ mingo@elte.hu: renamed it to rdtsc_barrier() and made it x86-only ] Signed-off-by: Andi Kleen Signed-off-by: Ingo Molnar Signed-off-by: Thomas Gleixner --- diff --git a/include/asm-x86/system.h b/include/asm-x86/system.h index 6c7d1fda4995..39474f2957a3 100644 --- a/include/asm-x86/system.h +++ b/include/asm-x86/system.h @@ -5,6 +5,7 @@ #include #include #include +#include #include #include @@ -395,5 +396,17 @@ void default_idle(void); #define set_mb(var, value) do { var = value; barrier(); } while (0) #endif +/* + * Stop RDTSC speculation. This is needed when you need to use RDTSC + * (or get_cycles or vread that possibly accesses the TSC) in a defined + * code region. + * + * (Could use an alternative three way for this if there was one.) + */ +static inline void rdtsc_barrier(void) +{ + alternative(ASM_NOP3, "mfence", X86_FEATURE_MFENCE_RDTSC); + alternative(ASM_NOP3, "lfence", X86_FEATURE_LFENCE_RDTSC); +} #endif