git: 9ba8ea3f0522 - stable/13 - x86: cpufunc: Add rdtsc_ordered()

From: Konstantin Belousov <kib_at_FreeBSD.org>
Date: Sun, 10 Oct 2021 09:23:57 UTC
The branch stable/13 has been updated by kib:

URL: https://cgit.FreeBSD.org/src/commit/?id=9ba8ea3f05225d0cea3d5d3f656217287c90d3c7

commit 9ba8ea3f05225d0cea3d5d3f656217287c90d3c7
Author:     Adam Fenn <adam@fenn.io>
AuthorDate: 2021-07-28 17:12:00 +0000
Commit:     Konstantin Belousov <kib@FreeBSD.org>
CommitDate: 2021-10-10 09:21:16 +0000

    x86: cpufunc: Add rdtsc_ordered()
    
    (cherry picked from commit 652ae7b11485d9f991ea66a56aa79d4dd9cd1103)
---
 sys/amd64/include/cpufunc.h | 14 ++++++++++++++
 sys/i386/include/cpufunc.h  | 14 ++++++++++++++
 sys/x86/include/x86_var.h   |  1 +
 sys/x86/x86/cpu_machdep.c   | 16 ++++++++++++++++
 4 files changed, 45 insertions(+)

diff --git a/sys/amd64/include/cpufunc.h b/sys/amd64/include/cpufunc.h
index 49975e826fb9..bca74d8ead67 100644
--- a/sys/amd64/include/cpufunc.h
+++ b/sys/amd64/include/cpufunc.h
@@ -355,6 +355,20 @@ rdtsc(void)
 	return (low | ((uint64_t)high << 32));
 }
 
+static __inline uint64_t
+rdtsc_ordered_lfence(void)
+{
+	lfence();
+	return (rdtsc());
+}
+
+static __inline uint64_t
+rdtsc_ordered_mfence(void)
+{
+	mfence();
+	return (rdtsc());
+}
+
 static __inline uint64_t
 rdtscp(void)
 {
diff --git a/sys/i386/include/cpufunc.h b/sys/i386/include/cpufunc.h
index 7b99978facc3..79cdd3004b77 100644
--- a/sys/i386/include/cpufunc.h
+++ b/sys/i386/include/cpufunc.h
@@ -394,6 +394,20 @@ rdtsc(void)
 	return (rv);
 }
 
+static __inline uint64_t
+rdtsc_ordered_lfence(void)
+{
+	lfence();
+	return (rdtsc());
+}
+
+static __inline uint64_t
+rdtsc_ordered_mfence(void)
+{
+	mfence();
+	return (rdtsc());
+}
+
 static __inline uint64_t
 rdtscp(void)
 {
diff --git a/sys/x86/include/x86_var.h b/sys/x86/include/x86_var.h
index 71f7a7eca564..55227cf9e65a 100644
--- a/sys/x86/include/x86_var.h
+++ b/sys/x86/include/x86_var.h
@@ -152,6 +152,7 @@ int	pti_get_default(void);
 int	user_dbreg_trap(register_t dr6);
 int	minidumpsys(struct dumperinfo *);
 struct pcb *get_pcb_td(struct thread *td);
+uint64_t rdtsc_ordered(void);
 
 /*
  * MSR ops for x86_msr_op()
diff --git a/sys/x86/x86/cpu_machdep.c b/sys/x86/x86/cpu_machdep.c
index 8ad718372928..518152a59619 100644
--- a/sys/x86/x86/cpu_machdep.c
+++ b/sys/x86/x86/cpu_machdep.c
@@ -76,6 +76,7 @@ __FBSDID("$FreeBSD$");
 
 #include <machine/clock.h>
 #include <machine/cpu.h>
+#include <machine/cpufunc.h>
 #include <machine/cputypes.h>
 #include <machine/specialreg.h>
 #include <machine/md_var.h>
@@ -88,6 +89,7 @@ __FBSDID("$FreeBSD$");
 #include <machine/elan_mmcr.h>
 #endif
 #include <x86/acpica_machdep.h>
+#include <x86/ifunc.h>
 
 #include <vm/vm.h>
 #include <vm/vm_extern.h>
@@ -1504,3 +1506,17 @@ acpi_get_fadt_bootflags(uint16_t *flagsp)
 	return (false);
 #endif
 }
+
+DEFINE_IFUNC(, uint64_t, rdtsc_ordered, (void))
+{
+	bool cpu_is_amd = cpu_vendor_id == CPU_VENDOR_AMD ||
+	    cpu_vendor_id == CPU_VENDOR_HYGON;
+
+	if ((amd_feature & AMDID_RDTSCP) != 0)
+		return (rdtscp);
+	else if ((cpu_feature & CPUID_SSE2) != 0)
+		return (cpu_is_amd ? rdtsc_ordered_mfence :
+		    rdtsc_ordered_lfence);
+	else
+		return (rdtsc);
+}