Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys Implement debugging feature for pserialize(9)
details: https://anonhg.NetBSD.org/src/rev/9fe69169c95b
branches: trunk
changeset: 357686:9fe69169c95b
user: ozaki-r <ozaki-r%NetBSD.org@localhost>
date: Tue Nov 21 08:49:14 2017 +0000
description:
Implement debugging feature for pserialize(9)
The debugging feature detects violations of pserialize constraints.
It causes a panic:
- if a context switch happens in a read section, or
- if a sleepable function is called in a read section.
The feature is enabled only if LOCKDEBUG is on.
Discussed on tech-kern@
diffstat:
sys/kern/kern_lock.c | 8 ++-
sys/kern/subr_pserialize.c | 98 ++++++++++++++++++++++++++++++++++++++-
sys/rump/librump/rumpkern/emul.c | 16 +++++-
sys/rump/librump/rumpkern/rump.c | 6 +-
sys/sys/pserialize.h | 4 +-
5 files changed, 121 insertions(+), 11 deletions(-)
diffs (282 lines):
diff -r 8e4b71b1acd5 -r 9fe69169c95b sys/kern/kern_lock.c
--- a/sys/kern/kern_lock.c Tue Nov 21 08:19:54 2017 +0000
+++ b/sys/kern/kern_lock.c Tue Nov 21 08:49:14 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: kern_lock.c,v 1.159 2017/09/16 23:55:33 christos Exp $ */
+/* $NetBSD: kern_lock.c,v 1.160 2017/11/21 08:49:14 ozaki-r Exp $ */
/*-
* Copyright (c) 2002, 2006, 2007, 2008, 2009 The NetBSD Foundation, Inc.
@@ -31,7 +31,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: kern_lock.c,v 1.159 2017/09/16 23:55:33 christos Exp $");
+__KERNEL_RCSID(0, "$NetBSD: kern_lock.c,v 1.160 2017/11/21 08:49:14 ozaki-r Exp $");
#include <sys/param.h>
#include <sys/proc.h>
@@ -43,6 +43,7 @@
#include <sys/syslog.h>
#include <sys/atomic.h>
#include <sys/lwp.h>
+#include <sys/pserialize.h>
#include <machine/lock.h>
@@ -88,6 +89,9 @@
if (cpu_softintr_p()) {
reason = "softint";
}
+ if (!pserialize_not_in_read_section()) {
+ reason = "pserialize";
+ }
if (reason) {
panic("%s: %s caller=%p", __func__, reason,
diff -r 8e4b71b1acd5 -r 9fe69169c95b sys/kern/subr_pserialize.c
--- a/sys/kern/subr_pserialize.c Tue Nov 21 08:19:54 2017 +0000
+++ b/sys/kern/subr_pserialize.c Tue Nov 21 08:49:14 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: subr_pserialize.c,v 1.8 2015/06/12 19:18:30 dholland Exp $ */
+/* $NetBSD: subr_pserialize.c,v 1.9 2017/11/21 08:49:14 ozaki-r Exp $ */
/*-
* Copyright (c) 2010, 2011 The NetBSD Foundation, Inc.
@@ -38,7 +38,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: subr_pserialize.c,v 1.8 2015/06/12 19:18:30 dholland Exp $");
+__KERNEL_RCSID(0, "$NetBSD: subr_pserialize.c,v 1.9 2017/11/21 08:49:14 ozaki-r Exp $");
#include <sys/param.h>
@@ -73,6 +73,12 @@
static TAILQ_HEAD(, pserialize) psz_queue1 __cacheline_aligned;
static TAILQ_HEAD(, pserialize) psz_queue2 __cacheline_aligned;
+#ifdef LOCKDEBUG
+#include <sys/percpu.h>
+
+static percpu_t *psz_debug_nreads __cacheline_aligned;
+#endif
+
/*
* pserialize_init:
*
@@ -89,6 +95,9 @@
mutex_init(&psz_lock, MUTEX_DEFAULT, IPL_SCHED);
evcnt_attach_dynamic(&psz_ev_excl, EVCNT_TYPE_MISC, NULL,
"pserialize", "exclusive access");
+#ifdef LOCKDEBUG
+ psz_debug_nreads = percpu_alloc(sizeof(uint32_t));
+#endif
}
/*
@@ -185,15 +194,37 @@
int
pserialize_read_enter(void)
{
+ int s;
KASSERT(!cpu_intr_p());
- return splsoftserial();
+ s = splsoftserial();
+#ifdef LOCKDEBUG
+ {
+ uint32_t *nreads;
+ nreads = percpu_getref(psz_debug_nreads);
+ (*nreads)++;
+ if (*nreads == 0)
+ panic("nreads overflow");
+ percpu_putref(psz_debug_nreads);
+ }
+#endif
+ return s;
}
void
pserialize_read_exit(int s)
{
+#ifdef LOCKDEBUG
+ {
+ uint32_t *nreads;
+ nreads = percpu_getref(psz_debug_nreads);
+ (*nreads)--;
+ if (*nreads == UINT_MAX)
+ panic("nreads underflow");
+ percpu_putref(psz_debug_nreads);
+ }
+#endif
splx(s);
}
@@ -209,6 +240,9 @@
pserialize_t psz, next;
cpuid_t cid;
+ /* We must to ensure not to come here from inside a read section. */
+ KASSERT(pserialize_not_in_read_section());
+
/*
* If no updates pending, bail out. No need to lock in order to
* test psz_work_todo; the only ill effect of missing an update
@@ -261,3 +295,61 @@
}
mutex_spin_exit(&psz_lock);
}
+
+/*
+ * pserialize_in_read_section:
+ *
+ * True if the caller is in a pserialize read section. To be used only
+ * for diagnostic assertions where we want to guarantee the condition like:
+ *
+ * KASSERT(pserialize_in_read_section());
+ */
+bool
+pserialize_in_read_section(void)
+{
+#ifdef LOCKDEBUG
+ uint32_t *nreads;
+ bool in;
+
+ /* Not initialized yet */
+ if (__predict_false(psz_debug_nreads == NULL))
+ return true;
+
+ nreads = percpu_getref(psz_debug_nreads);
+ in = *nreads != 0;
+ percpu_putref(psz_debug_nreads);
+
+ return in;
+#else
+ return true;
+#endif
+}
+
+/*
+ * pserialize_not_in_read_section:
+ *
+ * True if the caller is not in a pserialize read section. To be used only
+ * for diagnostic assertions where we want to guarantee the condition like:
+ *
+ * KASSERT(pserialize_not_in_read_section());
+ */
+bool
+pserialize_not_in_read_section(void)
+{
+#ifdef LOCKDEBUG
+ uint32_t *nreads;
+ bool notin;
+
+ /* Not initialized yet */
+ if (__predict_false(psz_debug_nreads == NULL))
+ return true;
+
+ nreads = percpu_getref(psz_debug_nreads);
+ notin = *nreads == 0;
+ percpu_putref(psz_debug_nreads);
+
+ return notin;
+#else
+ return true;
+#endif
+}
diff -r 8e4b71b1acd5 -r 9fe69169c95b sys/rump/librump/rumpkern/emul.c
--- a/sys/rump/librump/rumpkern/emul.c Tue Nov 21 08:19:54 2017 +0000
+++ b/sys/rump/librump/rumpkern/emul.c Tue Nov 21 08:49:14 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: emul.c,v 1.183 2017/11/09 12:46:55 christos Exp $ */
+/* $NetBSD: emul.c,v 1.184 2017/11/21 08:49:14 ozaki-r Exp $ */
/*
* Copyright (c) 2007-2011 Antti Kantee. All Rights Reserved.
@@ -26,7 +26,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: emul.c,v 1.183 2017/11/09 12:46:55 christos Exp $");
+__KERNEL_RCSID(0, "$NetBSD: emul.c,v 1.184 2017/11/21 08:49:14 ozaki-r Exp $");
#include <sys/param.h>
#include <sys/cprng.h>
@@ -166,11 +166,23 @@
return t;
}
+#define RETURN_ADDRESS (uintptr_t)__builtin_return_address(0)
+
void
assert_sleepable(void)
{
+ const char *reason = NULL;
/* always sleepable, although we should improve this */
+
+ if (!pserialize_not_in_read_section()) {
+ reason = "pserialize";
+ }
+
+ if (reason) {
+ panic("%s: %s caller=%p", __func__, reason,
+ (void *)RETURN_ADDRESS);
+ }
}
void
diff -r 8e4b71b1acd5 -r 9fe69169c95b sys/rump/librump/rumpkern/rump.c
--- a/sys/rump/librump/rumpkern/rump.c Tue Nov 21 08:19:54 2017 +0000
+++ b/sys/rump/librump/rumpkern/rump.c Tue Nov 21 08:49:14 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: rump.c,v 1.329 2016/03/08 14:30:48 joerg Exp $ */
+/* $NetBSD: rump.c,v 1.330 2017/11/21 08:49:14 ozaki-r Exp $ */
/*
* Copyright (c) 2007-2011 Antti Kantee. All Rights Reserved.
@@ -26,7 +26,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: rump.c,v 1.329 2016/03/08 14:30:48 joerg Exp $");
+__KERNEL_RCSID(0, "$NetBSD: rump.c,v 1.330 2017/11/21 08:49:14 ozaki-r Exp $");
#include <sys/systm.h>
#define ELFSIZE ARCH_ELFSIZE
@@ -301,6 +301,7 @@
callout_startup();
kprintf_init();
+ percpu_init();
pserialize_init();
kauth_init();
@@ -351,7 +352,6 @@
rump_schedule();
bootlwp = curlwp;
- percpu_init();
inittimecounter();
ntp_init();
diff -r 8e4b71b1acd5 -r 9fe69169c95b sys/sys/pserialize.h
--- a/sys/sys/pserialize.h Tue Nov 21 08:19:54 2017 +0000
+++ b/sys/sys/pserialize.h Tue Nov 21 08:49:14 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pserialize.h,v 1.1 2011/07/30 17:01:04 christos Exp $ */
+/* $NetBSD: pserialize.h,v 1.2 2017/11/21 08:49:15 ozaki-r Exp $ */
/*-
* Copyright (c) 2010, 2011 The NetBSD Foundation, Inc.
@@ -44,6 +44,8 @@
int pserialize_read_enter(void);
void pserialize_read_exit(int);
+bool pserialize_in_read_section(void);
+bool pserialize_not_in_read_section(void);
#endif /* _KERNEL */
#endif /* _SYS_PSERIALIZE_H_ */
Home |
Main Index |
Thread Index |
Old Index