Skip site navigation (1)Skip section navigation (2)
Date:      Fri, 25 Jan 2008 01:34:30 GMT
From:      John Baldwin <jhb@FreeBSD.org>
To:        Perforce Change Reviews <perforce@freebsd.org>
Subject:   PERFORCE change 134047 for review
Message-ID:  <200801250134.m0P1YUoX048085@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=134047

Change 134047 by jhb@jhb_mutex on 2008/01/25 01:33:43

	- Compile (catch up to kdb_enter() API change).
	- Add a test for the tsleep/pause race.

Affected files ...

.. //depot/projects/smpng/sys/modules/crash/crash.c#42 edit

Differences ...

==== //depot/projects/smpng/sys/modules/crash/crash.c#42 (text+ko) ====

@@ -52,11 +52,14 @@
 #include <sys/refcount.h>
 #include <sys/rwlock.h>
 #include <sys/sched.h>
+#include <sys/sleepqueue.h>
 #include <sys/sysctl.h>
 #include <sys/systm.h>
 #include <sys/sx.h>
 #include <sys/unistd.h>
 
+#define	KDB_WHY_CRASH		"crash"
+
 struct crash_event {
 	const char *ev_name;
 	void (*ev_handler)(void);
@@ -86,6 +89,48 @@
 
 /* Events. */
 
+static int race_wchan;
+
+static void
+tsleep_race_fallback(void *dummy)
+{
+
+	wakeup(&race_wchan);
+}
+
+static void
+exercise_tsleep_race(void)
+{
+	struct callout c;
+	struct bintime start, finish;
+	struct timeval elapsed;
+	int rval;
+
+	/*
+	 * Inline copy of tsleep() with a long DELAY() to open up the
+	 * race window where the thread lock is dropped in between
+	 * sleepq_add() and sleepq_*wait*().
+	 */
+	callout_init(&c, CALLOUT_MPSAFE);
+	callout_reset(&c, 5 * hz, tsleep_race_fallback, NULL);
+	getbinuptime(&start);
+	sleepq_lock(&race_wchan);
+	DROP_GIANT();
+	sleepq_add(&race_wchan, NULL, "race", SLEEPQ_SLEEP, 0);
+	sleepq_set_timeout(&race_wchan, 1);
+	DELAY(2 * 1000 * 1000);
+	rval = sleepq_timedwait(&race_wchan);
+	getbinuptime(&finish);
+	PICKUP_GIANT();
+	printf("crash: sleepq_timedwait() returned %d\n", rval);
+	bintime_sub(&finish, &start);
+	bintime2timeval(&finish, &elapsed);
+	printf("crash: slept for %d.%04ld sec\n", elapsed.tv_sec,
+	    elapsed.tv_usec / 1000);
+	callout_drain(&c);
+}
+CRASH_EVENT("exercise tsleep() race", exercise_tsleep_race);
+
 static void
 sx_recurse(void)
 {
@@ -93,7 +138,7 @@
 	sx_init_flags(&qux, "qux", SX_RECURSE);
 	sx_xlock(&qux);
 	sx_xlock(&qux);
-	kdb_enter("qux should be recursed");
+	kdb_enter(KDB_WHY_CRASH, "qux should be recursed");
 	sx_xunlock(&qux);
 	if (sx_try_xlock(&qux) == 0)
 		panic("try_xlock failed!");
@@ -200,9 +245,9 @@
 
 	sx_init(&qux, "qux");
 	sx_destroy(&qux);
-	kdb_enter("examine qux");
+	kdb_enter(KDB_WHY_CRASH, "examine qux");
 	sx_xlock(&qux);
-	kdb_enter("examine again");
+	kdb_enter(KDB_WHY_CRASH, "examine again");
 }
 CRASH_EVENT("lock destroyed sx", lock_destroyed_sx);
 
@@ -212,9 +257,9 @@
 
 	rw_init(&baz, "baz");
 	rw_destroy(&baz);
-	kdb_enter("examine baz");
+	kdb_enter(KDB_WHY_CRASH, "examine baz");
 	rw_wlock(&baz);
-	kdb_enter("examine again");
+	kdb_enter(KDB_WHY_CRASH, "examine again");
 }
 CRASH_EVENT("lock destroyed rwlock", lock_destroyed_rw);
 
@@ -226,9 +271,9 @@
 	mtx_init(&test1_mtx, "test1", NULL, MTX_DEF | MTX_RECURSE);
 	mtx_lock(&test1_mtx);
 	mtx_destroy(&test1_mtx);
-	kdb_enter("examine test1");
+	kdb_enter(KDB_WHY_CRASH, "examine test1");
 	mtx_lock(&test1_mtx);
-	kdb_enter("examine again");
+	kdb_enter(KDB_WHY_CRASH, "examine again");
 }
 CRASH_EVENT("lock destroyed mutex", lock_destroyed_mtx);
 
@@ -408,11 +453,11 @@
 {
 
 	rw_init(&baz, "baz");
-	kdb_enter("baz exists");
+	kdb_enter(KDB_WHY_CRASH, "baz exists");
 	rw_wlock(&baz);
-	kdb_enter("baz has a writer");
+	kdb_enter(KDB_WHY_CRASH, "baz has a writer");
 	rw_wunlock(&baz);
-	kdb_enter("baz is unlocked");
+	kdb_enter(KDB_WHY_CRASH, "baz is unlocked");
 	rw_destroy(&baz);
 }
 CRASH_EVENT("rw_wlock", rwlock_wlock);
@@ -422,11 +467,11 @@
 {
 
 	rw_init(&baz, "baz");
-	kdb_enter("baz exists");
+	kdb_enter(KDB_WHY_CRASH, "baz exists");
 	rw_rlock(&baz);
-	kdb_enter("baz has a reader");
+	kdb_enter(KDB_WHY_CRASH, "baz has a reader");
 	rw_runlock(&baz);
-	kdb_enter("baz is unlocked");
+	kdb_enter(KDB_WHY_CRASH, "baz is unlocked");
 	rw_destroy(&baz);
 }
 CRASH_EVENT("rw_rlock", rwlock_rlock);
@@ -448,11 +493,11 @@
 	mtx_lock_spin(&test1_mtx);
 	error = msleep_spin(&test1_mtx, &test1_mtx, "spincr", 5);
 	printf("msleep_spin returned %d\n", error);
-	kdb_enter("timed out");
+	kdb_enter(KDB_WHY_CRASH, "timed out");
 	timeout(msleep_spin_callout, NULL, 5);
 	error = msleep_spin(&test1_mtx, &test1_mtx, "spincr2", 0);
 	printf("msleep_spin returned %d\n", error);
-	kdb_enter("awoke from spin");
+	kdb_enter(KDB_WHY_CRASH, "awoke from spin");
 	mtx_unlock_spin(&test1_mtx);
 	mtx_destroy(&test1_mtx);
 }
@@ -463,18 +508,18 @@
 {
 
 	mtx_lock(&Giant);
-	kdb_enter("Giant locked");
+	kdb_enter(KDB_WHY_CRASH, "Giant locked");
 	mtx_lock(&Giant);
-	kdb_enter("Giant recursed");
+	kdb_enter(KDB_WHY_CRASH, "Giant recursed");
 	mtx_unlock(&Giant);
 	mtx_unlock(&Giant);
 	sx_slock(&foo);
-	kdb_enter("foo slocked");
+	kdb_enter(KDB_WHY_CRASH, "foo slocked");
 	sx_sunlock(&foo);
 	sx_xlock(&foo);
-	kdb_enter("foo xlocked");
+	kdb_enter(KDB_WHY_CRASH, "foo xlocked");
 	sx_xunlock(&foo);
-	kdb_enter("both unlocked");
+	kdb_enter(KDB_WHY_CRASH, "both unlocked");
 }
 CRASH_EVENT("ddb lock commands", ddblock_test);
 
@@ -620,9 +665,9 @@
 static void
 double_mtx_init(void)
 {
-	kdb_enter("about to init again");
+	kdb_enter(KDB_WHY_CRASH, "about to init again");
 	mtx_init(&test_mtx, "test", NULL, MTX_DEF);
-	kdb_enter("if we haven't panic'd by now, ouch. :(");
+	kdb_enter(KDB_WHY_CRASH, "if we haven't panic'd by now, ouch. :(");
 	mtx_destroy(&test_mtx);
 }
 CRASH_EVENT("re-init of test_mtx", double_mtx_init);
@@ -664,18 +709,19 @@
 	bzero(&test2_mtx, sizeof(test2_mtx));
 	mtx_init(&test1_mtx, "test1", NULL, MTX_DEF);
 	mtx_init(&test2_mtx, "test2", NULL, MTX_DEF);
-	kdb_enter("no order yet");
+	kdb_enter(KDB_WHY_CRASH, "no order yet");
 	mtx_lock(&Giant);
 	mtx_lock(&test1_mtx);
 	mtx_lock(&test2_mtx);
 	mtx_unlock(&test2_mtx);
 	mtx_unlock(&test1_mtx);
 	mtx_unlock(&Giant);
-	kdb_enter("test1 and test2 should be ordered");
+	kdb_enter(KDB_WHY_CRASH, "test1 and test2 should be ordered");
 	mtx_destroy(&test1_mtx);
-	kdb_enter("test1 should be gone, test2 should be after Giant");
+	kdb_enter(KDB_WHY_CRASH,
+	    "test1 should be gone, test2 should be after Giant");
 	mtx_destroy(&test2_mtx);
-	kdb_enter("test1 and test2 should be gone");
+	kdb_enter(KDB_WHY_CRASH, "test1 and test2 should be gone");
 }
 CRASH_EVENT("use test1 and test2 mutexes to test witness removal",
     test_witness_removal);
@@ -721,10 +767,10 @@
 	bzero(&test2_mtx, sizeof(test2_mtx));
 	mtx_init(&test1_mtx, "test1", NULL, MTX_DEF);
 	mtx_init(&test2_mtx, "test2", NULL, MTX_DEF);
-	kdb_enter("no order yet");
+	kdb_enter(KDB_WHY_CRASH, "no order yet");
 	status = WITNESS_DEFINEORDER(&test1_mtx, &test2_mtx);
 	printf("Status of test1 -> test2 set order should be 0: %d\n", status);
-	kdb_enter("order should be test1 then test2");
+	kdb_enter(KDB_WHY_CRASH, "order should be test1 then test2");
 	printf("Check order of test1 -> test2 should succeed.\n");
 	mtx_lock(&test1_mtx);
 	witness_check(&test2_mtx);
@@ -786,7 +832,7 @@
 
 	if (depth == 20) {
 		kdb_backtrace();
-		kdb_enter(__func__);
+		kdb_enter(KDB_WHY_CRASH, __func__);
 	} else
 		recurse_deep(depth + 1);
 }



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200801250134.m0P1YUoX048085>