Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch Print a rate limited warning if the TSC timecounter...
details: https://anonhg.NetBSD.org/src/rev/2f42756f3d5d
branches: trunk
changeset: 1010970:2f42756f3d5d
user: ad <ad%NetBSD.org@localhost>
date: Sat Jun 13 23:58:51 2020 +0000
description:
Print a rate limited warning if the TSC timecounter goes backwards from the
viewpoint of any single LWP.
diffstat:
sys/arch/amd64/amd64/cpufunc.S | 9 +++++----
sys/arch/amd64/include/proc.h | 3 ++-
sys/arch/i386/i386/cpufunc.S | 11 +++++------
sys/arch/i386/include/proc.h | 3 ++-
sys/arch/x86/x86/tsc.c | 40 +++++++++++++++++++++++++++++++++++++---
5 files changed, 51 insertions(+), 15 deletions(-)
diffs (176 lines):
diff -r 714a774173d4 -r 2f42756f3d5d sys/arch/amd64/amd64/cpufunc.S
--- a/sys/arch/amd64/amd64/cpufunc.S Sat Jun 13 22:35:23 2020 +0000
+++ b/sys/arch/amd64/amd64/cpufunc.S Sat Jun 13 23:58:51 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: cpufunc.S,v 1.59 2020/06/01 22:58:06 ad Exp $ */
+/* $NetBSD: cpufunc.S,v 1.60 2020/06/13 23:58:51 ad Exp $ */
/*
* Copyright (c) 1998, 2007, 2008, 2020 The NetBSD Foundation, Inc.
@@ -206,8 +206,10 @@
END(x86_hotpatch)
#endif /* !XENPV */
-/* Could be exact same as cpu_counter, but KMSAN needs to have the correct
- * size of the return value. */
+/*
+ * Could be exact same as cpu_counter, but KMSAN needs to have the correct
+ * size of the return value.
+ */
ENTRY(cpu_counter32)
movq CPUVAR(CURLWP), %rcx
1:
@@ -221,7 +223,6 @@
2:
jmp 1b
END(cpu_counter32)
-STRONG_ALIAS(tsc_get_timecount, cpu_counter32)
ENTRY(cpu_counter)
movq CPUVAR(CURLWP), %rcx
diff -r 714a774173d4 -r 2f42756f3d5d sys/arch/amd64/include/proc.h
--- a/sys/arch/amd64/include/proc.h Sat Jun 13 22:35:23 2020 +0000
+++ b/sys/arch/amd64/include/proc.h Sat Jun 13 23:58:51 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: proc.h,v 1.24 2020/01/13 00:26:52 ad Exp $ */
+/* $NetBSD: proc.h,v 1.25 2020/06/13 23:58:51 ad Exp $ */
/*
* Copyright (c) 1991 Regents of the University of California.
@@ -46,6 +46,7 @@
struct vm_page;
struct mdlwp {
+ volatile uint64_t md_tsc; /* last TSC reading */
struct trapframe *md_regs; /* registers on current frame */
int md_flags; /* machine-dependent flags */
volatile int md_astpending;
diff -r 714a774173d4 -r 2f42756f3d5d sys/arch/i386/i386/cpufunc.S
--- a/sys/arch/i386/i386/cpufunc.S Sat Jun 13 22:35:23 2020 +0000
+++ b/sys/arch/i386/i386/cpufunc.S Sat Jun 13 23:58:51 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: cpufunc.S,v 1.45 2020/05/28 20:03:19 ad Exp $ */
+/* $NetBSD: cpufunc.S,v 1.46 2020/06/13 23:58:52 ad Exp $ */
/*-
* Copyright (c) 1998, 2007, 2020 The NetBSD Foundation, Inc.
@@ -38,7 +38,7 @@
#include <sys/errno.h>
#include <machine/asm.h>
-__KERNEL_RCSID(0, "$NetBSD: cpufunc.S,v 1.45 2020/05/28 20:03:19 ad Exp $");
+__KERNEL_RCSID(0, "$NetBSD: cpufunc.S,v 1.46 2020/06/13 23:58:52 ad Exp $");
#include "opt_xen.h"
@@ -163,7 +163,7 @@
ret
END(msr_onfault)
-ENTRY(tsc_get_timecount)
+ENTRY(cpu_counter)
pushl %ebx
movl CPUVAR(CURLWP), %ecx
1:
@@ -177,10 +177,9 @@
ret
2:
jmp 1b
-END(tsc_get_timecount)
+END(cpu_counter)
-STRONG_ALIAS(cpu_counter, tsc_get_timecount)
-STRONG_ALIAS(cpu_counter32, tsc_get_timecount)
+STRONG_ALIAS(cpu_counter32, cpu_counter)
ENTRY(breakpoint)
pushl %ebp
diff -r 714a774173d4 -r 2f42756f3d5d sys/arch/i386/include/proc.h
--- a/sys/arch/i386/include/proc.h Sat Jun 13 22:35:23 2020 +0000
+++ b/sys/arch/i386/include/proc.h Sat Jun 13 23:58:51 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: proc.h,v 1.47 2020/01/13 00:26:52 ad Exp $ */
+/* $NetBSD: proc.h,v 1.48 2020/06/13 23:58:52 ad Exp $ */
/*
* Copyright (c) 1991 Regents of the University of California.
@@ -46,6 +46,7 @@
#define MDL_FPU_IN_CPU 0x0020 /* the FPU state is in the CPU */
struct mdlwp {
+ volatile uint64_t md_tsc; /* last TSC reading */
struct trapframe *md_regs; /* registers on current frame */
int md_flags; /* machine-dependent flags */
volatile int md_astpending; /* AST pending for this process */
diff -r 714a774173d4 -r 2f42756f3d5d sys/arch/x86/x86/tsc.c
--- a/sys/arch/x86/x86/tsc.c Sat Jun 13 22:35:23 2020 +0000
+++ b/sys/arch/x86/x86/tsc.c Sat Jun 13 23:58:51 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: tsc.c,v 1.48 2020/05/27 18:46:15 ad Exp $ */
+/* $NetBSD: tsc.c,v 1.49 2020/06/13 23:58:52 ad Exp $ */
/*-
* Copyright (c) 2008, 2020 The NetBSD Foundation, Inc.
@@ -27,7 +27,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: tsc.c,v 1.48 2020/05/27 18:46:15 ad Exp $");
+__KERNEL_RCSID(0, "$NetBSD: tsc.c,v 1.49 2020/06/13 23:58:52 ad Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -38,6 +38,7 @@
#include <sys/kernel.h>
#include <sys/cpu.h>
#include <sys/xcall.h>
+#include <sys/lock.h>
#include <machine/cpu_counter.h>
#include <machine/cpuvar.h>
@@ -50,7 +51,7 @@
#define TSC_SYNC_ROUNDS 1000
#define ABS(a) ((a) >= 0 ? (a) : -(a))
-u_int tsc_get_timecount(struct timecounter *);
+static u_int tsc_get_timecount(struct timecounter *);
static void tsc_delay(unsigned int);
@@ -352,3 +353,36 @@
x86_pause();
}
}
+
+static u_int
+tsc_get_timecount(struct timecounter *tc)
+{
+ static __cpu_simple_lock_t lock = __SIMPLELOCK_UNLOCKED;
+ static int lastwarn;
+ uint64_t cur, prev;
+ lwp_t *l = curlwp;
+ int ticks;
+
+ /*
+ * Previous value must be read before the counter and stored to
+ * after, because this routine can be called from interrupt context
+ * and may run over the top of an existing invocation. Ordering is
+ * guaranteed by "volatile" on md_tsc.
+ */
+ prev = l->l_md.md_tsc;
+ cur = cpu_counter();
+ if (__predict_false(cur < prev)) {
+ if ((cur >> 63) == (prev >> 63) &&
+ __cpu_simple_lock_try(&lock)) {
+ ticks = getticks();
+ if (ticks - lastwarn >= hz) {
+ printf("WARNING: TSC time went backwards "
+ " by %u\n", (unsigned)(prev - cur));
+ lastwarn = ticks;
+ }
+ __cpu_simple_unlock(&lock);
+ }
+ }
+ l->l_md.md_tsc = cur;
+ return (uint32_t)cur;
+}
Home |
Main Index |
Thread Index |
Old Index