diff options
Diffstat (limited to 'sys/kern/subr_witness.c')
-rw-r--r-- | sys/kern/subr_witness.c | 37 |
1 files changed, 36 insertions, 1 deletions
diff --git a/sys/kern/subr_witness.c b/sys/kern/subr_witness.c index e47ea4a5b13..31a97cb4856 100644 --- a/sys/kern/subr_witness.c +++ b/sys/kern/subr_witness.c @@ -1,4 +1,4 @@ -/* $OpenBSD: subr_witness.c,v 1.14 2018/05/16 14:53:43 visa Exp $ */ +/* $OpenBSD: subr_witness.c,v 1.15 2018/05/16 14:55:44 visa Exp $ */ /*- * Copyright (c) 2008 Isilon Systems, Inc. @@ -1039,6 +1039,41 @@ witness_checkorder(struct lock_object *lock, int flags, const char *file, lock->lo_name, w->w_type->lt_name, fixup_filename(file), line); } + if (witness_watch > 1) { + struct witness_lock_order_data *wlod1, *wlod2; + + mtx_enter(&w_mtx); + wlod1 = witness_lock_order_get(w, w1); + wlod2 = witness_lock_order_get(w1, w); + mtx_leave(&w_mtx); + + /* + * It is safe to access saved stack traces, + * w_type, and w_class without the lock. + * Once written, they never change. + */ + + if (wlod1 != NULL) { + printf("lock order \"%s\"(%s) -> " + "\"%s\"(%s) first seen at:\n", + w->w_type->lt_name, + w->w_class->lc_name, + w1->w_type->lt_name, + w1->w_class->lc_name); + db_print_stack_trace( + &wlod1->wlod_stack, printf); + } + if (wlod2 != NULL) { + printf("lock order \"%s\"(%s) -> " + "\"%s\"(%s) first seen at:\n", + w1->w_type->lt_name, + w1->w_class->lc_name, + w->w_type->lt_name, + w->w_class->lc_name); + db_print_stack_trace( + &wlod2->wlod_stack, printf); + } + } witness_debugger(0); goto out_splx; } |