- lock-validator-v3.patch removed from -mm tree

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



The patch titled

     lock validator: -V3

has been removed from the -mm tree.  Its filename is

     lock-validator-v3.patch

This patch was dropped because lockdep is being redone

------------------------------------------------------
Subject: lock validator: -V3
From: Ingo Molnar <mingo@xxxxxxx>


This patch contains items that had to be done in one go, so i guess i'll
call this lock validator -V3, for lack of better name.  (This goes to the
tail of the current lock validator queue in -mm)

Changes:

Added 'detect if freed' (or if reinitialized/destroyed when held) feature
for all the lock types covered by the validator: spinlocks, rwlocks,
mutexes and rwsems - using the same central lock debugging code.  This
feature was available for mutexes before, now it's in the generic 'lock
debugging' code and hence extended to all lock types.

Accordingly, new lock debugging options are now available:

 CONFIG_DEBUG_SPINLOCK_ALLOC=y
 CONFIG_DEBUG_RWLOCK_ALLOC=y
 CONFIG_DEBUG_MUTEX_ALLOC=y
 CONFIG_DEBUG_RWSEM_ALLOC=y

Added 12 new testcases to the locking selftest, to make sure the new
features are working.

I unified all the lock debugging options, they are now all structured
as: 'basic', 'free/exit checking' and 'full validation', and are
dependent on each other. The lock types can be debugged independenty of
each other.

I also resurrected SysRq-D (print all locks) support.

All in one, the structure of the lock debugging code is alot more
consistent now, and all the lock debugging features are available across
the spectrum.

I have tested this on x86 and x86_64, using various combinations of the
new (and old) lock debugging options. Works fine here.

Signed-off-by: Ingo Molnar <mingo@xxxxxxx>
Cc: Arjan van de Ven <arjan@xxxxxxxxxxxxxxx>
Signed-off-by: Andrew Morton <akpm@xxxxxxxx>
---

 drivers/char/sysrq.c              |    1 
 include/asm-i386/rwsem.h          |    4 
 include/linux/debug_locks.h       |   19 +-
 include/linux/lockdep.h           |   73 +++++++--
 include/linux/mutex.h             |    6 
 include/linux/rwsem-spinlock.h    |    4 
 include/linux/rwsem.h             |    4 
 include/linux/sched.h             |    1 
 include/linux/spinlock.h          |   16 --
 include/linux/spinlock_types.h    |    8 -
 include/linux/spinlock_types_up.h |    8 -
 include/linux/spinlock_up.h       |    5 
 kernel/lockdep.c                  |  211 +++++++++++++++++++++++-----
 kernel/mutex-debug.c              |    6 
 kernel/mutex-debug.h              |    3 
 kernel/mutex-lockdep.h            |   40 -----
 kernel/mutex.c                    |   17 --
 kernel/mutex.h                    |    2 
 kernel/spinlock.c                 |   38 -----
 lib/Kconfig.debug                 |   98 +++++++++----
 lib/locking-selftest-mutex.h      |    6 
 lib/locking-selftest-rlock.h      |    9 +
 lib/locking-selftest-rsem.h       |    9 +
 lib/locking-selftest-spin.h       |    6 
 lib/locking-selftest-wlock.h      |    9 +
 lib/locking-selftest-wsem.h       |    9 +
 lib/locking-selftest.c            |  107 ++++++++++++--
 lib/rwsem-spinlock.c              |   10 -
 lib/rwsem.c                       |   10 -
 lib/spinlock_debug.c              |   36 ++++
 30 files changed, 545 insertions(+), 230 deletions(-)

diff -puN drivers/char/sysrq.c~lock-validator-v3 drivers/char/sysrq.c
--- a/drivers/char/sysrq.c~lock-validator-v3
+++ a/drivers/char/sysrq.c
@@ -153,7 +153,6 @@ static void sysrq_handle_showlocks(int k
 				struct tty_struct *tty)
 {
 	debug_show_all_locks();
-	print_lock_types();
 }
 
 static struct sysrq_key_op sysrq_showlocks_op = {
diff -puN include/asm-i386/rwsem.h~lock-validator-v3 include/asm-i386/rwsem.h
--- a/include/asm-i386/rwsem.h~lock-validator-v3
+++ a/include/asm-i386/rwsem.h
@@ -65,7 +65,7 @@ struct rw_semaphore {
 #if RWSEM_DEBUG
 	int			debug;
 #endif
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 	struct lockdep_map dep_map;
 #endif
 };
@@ -79,7 +79,7 @@ struct rw_semaphore {
 #define __RWSEM_DEBUG_INIT	/* */
 #endif
 
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 # define __RWSEM_DEP_MAP_INIT(lockname) , .dep_map = { .name = #lockname }
 #else
 # define __RWSEM_DEP_MAP_INIT(lockname)
diff -puN include/linux/debug_locks.h~lock-validator-v3 include/linux/debug_locks.h
--- a/include/linux/debug_locks.h~lock-validator-v3
+++ a/include/linux/debug_locks.h
@@ -41,22 +41,29 @@ extern int debug_locks_off(void);
 # define locking_selftest()	do { } while (0)
 #endif
 
-static inline void
-debug_check_no_locks_freed(const void *from, unsigned long len)
+#ifdef CONFIG_LOCKDEP
+extern void debug_show_all_locks(void);
+extern void debug_show_held_locks(struct task_struct *task);
+extern void debug_check_no_locks_freed(const void *from, unsigned long len);
+extern void debug_check_no_locks_held(struct task_struct *task);
+#else
+static inline void debug_show_all_locks(void)
 {
 }
 
-static inline void
-debug_check_no_locks_held(struct task_struct *task)
+static inline void debug_show_held_locks(struct task_struct *task)
 {
 }
 
-static inline void debug_show_all_locks(void)
+static inline void
+debug_check_no_locks_freed(const void *from, unsigned long len)
 {
 }
 
-static inline void debug_show_held_locks(struct task_struct *task)
+static inline void
+debug_check_no_locks_held(struct task_struct *task)
 {
 }
+#endif
 
 #endif
diff -puN include/linux/lockdep.h~lock-validator-v3 include/linux/lockdep.h
--- a/include/linux/lockdep.h~lock-validator-v3
+++ a/include/linux/lockdep.h
@@ -178,6 +178,7 @@ struct held_lock {
 	int				irq_context;
 	int				trylock;
 	int				read;
+	int				check;
 	int				hardirqs_off;
 };
 
@@ -214,11 +215,26 @@ extern void lockdep_init_map(struct lock
 #define lockdep_reinit_key(lock, key) \
 		lockdep_init_map(&(lock)->dep_map, #key, key)
 
-extern void lockdep_acquire(struct lockdep_map *lock, unsigned int subtype,
-			    int trylock, int read, unsigned long ip);
+/*
+ * Acquire a lock.
+ *
+ * Values for "read":
+ *
+ *   0: exclusive (write) acquire
+ *   1: read-acquire (no recursion allowed)
+ *   2: read-acquire with same-instance recursion allowed
+ *
+ * Values for check:
+ *
+ *   0: disabled
+ *   1: simple checks (freeing, held-at-exit-time, etc.)
+ *   2: full validation
+ */
+extern void lock_acquire(struct lockdep_map *lock, unsigned int subtype,
+			 int trylock, int read, int check, unsigned long ip);
 
-extern void lockdep_release(struct lockdep_map *lock, int nested,
-			    unsigned long ip);
+extern void lock_release(struct lockdep_map *lock, int nested,
+			 unsigned long ip);
 
 # define INIT_LOCKDEP				.lockdep_recursion = 0,
 
@@ -232,6 +248,8 @@ static inline void lockdep_on(void)
 {
 }
 
+# define lock_acquire(l, s, t, r, c, i)		do { } while (0)
+# define lock_release(l, n, i)			do { } while (0)
 # define lockdep_init()				do { } while (0)
 # define lockdep_info()				do { } while (0)
 # define print_lock_types()			do { } while (0)
@@ -268,36 +286,55 @@ extern void early_boot_irqs_on(void);
  * Map the dependency ops to NOP or to real lockdep ops, depending
  * on the per lock-type debug mode:
  */
-#ifdef CONFIG_PROVE_SPIN_LOCKING
-# define spin_acquire(l, s, t, i)		lockdep_acquire(l, s, t, 0, i)
-# define spin_release(l, n, i)			lockdep_release(l, n, i)
+
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
+# ifdef CONFIG_PROVE_SPIN_LOCKING
+#  define spin_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 2, i)
+# else
+#  define spin_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 1, i)
+# endif
+# define spin_release(l, n, i)			lock_release(l, n, i)
 #else
 # define spin_acquire(l, s, t, i)		do { } while (0)
 # define spin_release(l, n, i)			do { } while (0)
 #endif
 
-#ifdef CONFIG_PROVE_RW_LOCKING
-# define rwlock_acquire(l, s, t, i)		lockdep_acquire(l, s, t, 0, i)
-# define rwlock_acquire_read(l, s, t, i)	lockdep_acquire(l, s, t, 1, i)
-# define rwlock_release(l, n, i)		lockdep_release(l, n, i)
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
+# ifdef CONFIG_PROVE_RW_LOCKING
+#  define rwlock_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 2, i)
+#  define rwlock_acquire_read(l, s, t, i)	lock_acquire(l, s, t, 2, 2, i)
+# else
+#  define rwlock_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 1, i)
+#  define rwlock_acquire_read(l, s, t, i)	lock_acquire(l, s, t, 2, 1, i)
+# endif
+# define rwlock_release(l, n, i)		lock_release(l, n, i)
 #else
 # define rwlock_acquire(l, s, t, i)		do { } while (0)
 # define rwlock_acquire_read(l, s, t, i)	do { } while (0)
 # define rwlock_release(l, n, i)		do { } while (0)
 #endif
 
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
-# define mutex_acquire(l, s, t, i)		lockdep_acquire(l, s, t, 0, i)
-# define mutex_release(l, n, i)			lockdep_release(l, n, i)
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
+# ifdef CONFIG_PROVE_MUTEX_LOCKING
+#  define mutex_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 2, i)
+# else
+#  define mutex_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 1, i)
+# endif
+# define mutex_release(l, n, i)			lock_release(l, n, i)
 #else
 # define mutex_acquire(l, s, t, i)		do { } while (0)
 # define mutex_release(l, n, i)			do { } while (0)
 #endif
 
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
-# define rwsem_acquire(l, s, t, i)		lockdep_acquire(l, s, t, 0, i)
-# define rwsem_acquire_read(l, s, t, i)		lockdep_acquire(l, s, t, -1, i)
-# define rwsem_release(l, n, i)			lockdep_release(l, n, i)
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
+# ifdef CONFIG_PROVE_RWSEM_LOCKING
+#  define rwsem_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 2, i)
+#  define rwsem_acquire_read(l, s, t, i)	lock_acquire(l, s, t, 1, 2, i)
+# else
+#  define rwsem_acquire(l, s, t, i)		lock_acquire(l, s, t, 0, 1, i)
+#  define rwsem_acquire_read(l, s, t, i)	lock_acquire(l, s, t, 1, 1, i)
+# endif
+# define rwsem_release(l, n, i)			lock_release(l, n, i)
 #else
 # define rwsem_acquire(l, s, t, i)		do { } while (0)
 # define rwsem_acquire_read(l, s, t, i)		do { } while (0)
diff -puN include/linux/mutex.h~lock-validator-v3 include/linux/mutex.h
--- a/include/linux/mutex.h~lock-validator-v3
+++ a/include/linux/mutex.h
@@ -54,7 +54,7 @@ struct mutex {
 	const char 		*name;
 	void			*magic;
 #endif
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 	struct lockdep_map	dep_map;
 #endif
 };
@@ -85,7 +85,7 @@ do {							\
 # define mutex_destroy(mutex)				do { } while (0)
 #endif
 
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 # define __DEP_MAP_MUTEX_INITIALIZER(lockname) \
 		, .dep_map = { .name = #lockname }
 #else
@@ -125,7 +125,7 @@ static inline int fastcall mutex_is_lock
 extern void fastcall mutex_lock(struct mutex *lock);
 extern int fastcall mutex_lock_interruptible(struct mutex *lock);
 
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 extern void mutex_lock_nested(struct mutex *lock, unsigned int subtype);
 #else
 # define mutex_lock_nested(lock, subtype) mutex_lock(lock)
diff -puN include/linux/rwsem.h~lock-validator-v3 include/linux/rwsem.h
--- a/include/linux/rwsem.h~lock-validator-v3
+++ a/include/linux/rwsem.h
@@ -30,7 +30,7 @@ struct rw_semaphore;
  * Lockdep: type splitting can also be done for dynamic locks, if for
  * example there are per-CPU dynamically allocated locks:
  */
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 #define init_rwsem_key(sem, key)				\
 	__init_rwsem((sem), #sem, key)
 #else
@@ -100,7 +100,7 @@ static inline void down_write(struct rw_
 /*
  * lock for writing
  */
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 static inline void down_write_nested(struct rw_semaphore *sem, int subtype)
 {
 	might_sleep();
diff -puN include/linux/rwsem-spinlock.h~lock-validator-v3 include/linux/rwsem-spinlock.h
--- a/include/linux/rwsem-spinlock.h~lock-validator-v3
+++ a/include/linux/rwsem-spinlock.h
@@ -35,7 +35,7 @@ struct rw_semaphore {
 #if RWSEM_DEBUG
 	int			debug;
 #endif
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 	struct lockdep_map dep_map;
 #endif
 };
@@ -49,7 +49,7 @@ struct rw_semaphore {
 #define __RWSEM_DEBUG_INIT	/* */
 #endif
 
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 # define __RWSEM_DEP_MAP_INIT(lockname) , .dep_map = { .name = #lockname }
 #else
 # define __RWSEM_DEP_MAP_INIT(lockname)
diff -puN include/linux/sched.h~lock-validator-v3 include/linux/sched.h
--- a/include/linux/sched.h~lock-validator-v3
+++ a/include/linux/sched.h
@@ -947,6 +947,7 @@ struct task_struct {
 	struct held_lock held_locks[MAX_LOCK_DEPTH];
 #endif
 	unsigned int lockdep_recursion;
+	int lockdep_verbose;
 
 /* journalling filesystem info */
 	void *journal_info;
diff -puN include/linux/spinlock.h~lock-validator-v3 include/linux/spinlock.h
--- a/include/linux/spinlock.h~lock-validator-v3
+++ a/include/linux/spinlock.h
@@ -88,7 +88,7 @@ extern int __lockfunc generic__raw_read_
 # include <linux/spinlock_up.h>
 #endif
 
-#if defined(CONFIG_DEBUG_SPINLOCK) || defined(CONFIG_PROVE_SPIN_LOCKING)
+#ifdef CONFIG_DEBUG_SPINLOCK
   extern void __spin_lock_init(spinlock_t *lock, const char *name,
 			       struct lockdep_type_key *key);
 # define spin_lock_init(lock)					\
@@ -123,7 +123,7 @@ do {								\
 	do { spin_lock_init(lock); (void)(key); } while (0)
 #endif
 
-#if defined(CONFIG_DEBUG_SPINLOCK) || defined(CONFIG_PROVE_RW_LOCKING)
+#ifdef CONFIG_DEBUG_SPINLOCK
   extern void __rwlock_init(rwlock_t *lock, const char *name,
 			    struct lockdep_type_key *key);
 # define rwlock_init(lock)					\
@@ -152,9 +152,7 @@ do {								\
 /*
  * Pull the _spin_*()/_read_*()/_write_*() functions/declarations:
  */
-#if defined(CONFIG_SMP) || defined(CONFIG_DEBUG_SPINLOCK) || \
-	defined(CONFIG_PROVE_SPIN_LOCKING) || \
-	defined(CONFIG_PROVE_RW_LOCKING)
+#if defined(CONFIG_SMP) || defined(CONFIG_DEBUG_SPINLOCK)
 # include <linux/spinlock_api_smp.h>
 #else
 # include <linux/spinlock_api_up.h>
@@ -203,9 +201,7 @@ do {								\
 #define write_lock(lock)		_write_lock(lock)
 #define read_lock(lock)			_read_lock(lock)
 
-#if defined(CONFIG_SMP) || defined(CONFIG_DEBUG_SPINLOCK) || \
-	defined(CONFIG_PROVE_SPIN_LOCKING) || \
-	defined(CONFIG_PROVE_RW_LOCKING)
+#if defined(CONFIG_SMP) || defined(CONFIG_DEBUG_SPINLOCK)
 #define spin_lock_irqsave(lock, flags)	flags = _spin_lock_irqsave(lock)
 #define read_lock_irqsave(lock, flags)	flags = _read_lock_irqsave(lock)
 #define write_lock_irqsave(lock, flags)	flags = _write_lock_irqsave(lock)
@@ -228,9 +224,7 @@ do {								\
  * We inline the unlock functions in the nondebug case:
  */
 #if defined(CONFIG_DEBUG_SPINLOCK) || defined(CONFIG_PREEMPT) || \
-	!defined(CONFIG_SMP) || \
-	defined(CONFIG_PROVE_SPIN_LOCKING) || \
-	defined(CONFIG_PROVE_RW_LOCKING)
+	!defined(CONFIG_SMP)
 # define spin_unlock(lock)		_spin_unlock(lock)
 # define spin_unlock_non_nested(lock)	_spin_unlock_non_nested(lock)
 # define read_unlock(lock)		_read_unlock(lock)
diff -puN include/linux/spinlock_types.h~lock-validator-v3 include/linux/spinlock_types.h
--- a/include/linux/spinlock_types.h~lock-validator-v3
+++ a/include/linux/spinlock_types.h
@@ -26,7 +26,7 @@ typedef struct {
 	unsigned int magic, owner_cpu;
 	void *owner;
 #endif
-#ifdef CONFIG_PROVE_SPIN_LOCKING
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
 	struct lockdep_map dep_map;
 #endif
 } spinlock_t;
@@ -42,7 +42,7 @@ typedef struct {
 	unsigned int magic, owner_cpu;
 	void *owner;
 #endif
-#ifdef CONFIG_PROVE_RW_LOCKING
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
 	struct lockdep_map dep_map;
 #endif
 } rwlock_t;
@@ -51,13 +51,13 @@ typedef struct {
 
 #define SPINLOCK_OWNER_INIT	((void *)-1L)
 
-#ifdef CONFIG_PROVE_SPIN_LOCKING
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
 # define SPIN_DEP_MAP_INIT(lockname)	.dep_map = { .name = #lockname }
 #else
 # define SPIN_DEP_MAP_INIT(lockname)
 #endif
 
-#ifdef CONFIG_PROVE_RW_LOCKING
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
 # define RW_DEP_MAP_INIT(lockname)	.dep_map = { .name = #lockname }
 #else
 # define RW_DEP_MAP_INIT(lockname)
diff -puN include/linux/spinlock_types_up.h~lock-validator-v3 include/linux/spinlock_types_up.h
--- a/include/linux/spinlock_types_up.h~lock-validator-v3
+++ a/include/linux/spinlock_types_up.h
@@ -13,12 +13,12 @@
  */
 
 #if defined(CONFIG_DEBUG_SPINLOCK) || \
-	defined(CONFIG_PROVE_SPIN_LOCKING) || \
-	defined(CONFIG_PROVE_RW_LOCKING)
+	defined(CONFIG_DEBUG_SPINLOCK_ALLOC) || \
+	defined(CONFIG_DEBUG_RWLOCK_ALLOC)
 
 typedef struct {
 	volatile unsigned int slock;
-#ifdef CONFIG_PROVE_SPIN_LOCKING
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
 	struct lockdep_map dep_map;
 #endif
 } raw_spinlock_t;
@@ -35,7 +35,7 @@ typedef struct { } raw_spinlock_t;
 
 typedef struct {
 	/* no debug version on UP */
-#ifdef CONFIG_PROVE_RW_LOCKING
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
 	struct lockdep_map dep_map;
 #endif
 } raw_rwlock_t;
diff -puN include/linux/spinlock_up.h~lock-validator-v3 include/linux/spinlock_up.h
--- a/include/linux/spinlock_up.h~lock-validator-v3
+++ a/include/linux/spinlock_up.h
@@ -17,10 +17,7 @@
  * No atomicity anywhere, we are on UP.
  */
 
-#if defined(CONFIG_DEBUG_SPINLOCK) || \
-	defined(CONFIG_PROVE_SPIN_LOCKING) || \
-	defined(CONFIG_PROVE_RW_LOCKING)
-
+#ifdef CONFIG_DEBUG_SPINLOCK
 #define __raw_spin_is_locked(x)		((x)->slock == 0)
 
 static inline void __raw_spin_lock(raw_spinlock_t *lock)
diff -puN kernel/lockdep.c~lock-validator-v3 kernel/lockdep.c
--- a/kernel/lockdep.c~lock-validator-v3
+++ a/kernel/lockdep.c
@@ -189,6 +189,8 @@ static int verbose(struct lock_type *typ
 	return 0;
 }
 
+#ifdef CONFIG_TRACE_IRQFLAGS
+
 static int hardirq_verbose(struct lock_type *type)
 {
 #if HARDIRQ_VERBOSE
@@ -205,6 +207,8 @@ static int softirq_verbose(struct lock_t
 	return 0;
 }
 
+#endif
+
 /*
  * Stack-trace: tightly packed array of stack backtrace
  * addresses. Protected by the hash_lock.
@@ -415,16 +419,16 @@ static void print_lock(struct held_lock 
 
 void lockdep_print_held_locks(struct task_struct *curr)
 {
-	int i;
+	int i, depth = curr->lockdep_depth;
 
-	if (!curr->lockdep_depth) {
+	if (!depth) {
 		printk("no locks held by %s/%d.\n", curr->comm, curr->pid);
 		return;
 	}
-	printk("%d locks held by %s/%d:\n",
-		curr->lockdep_depth, curr->comm, curr->pid);
+	printk("%d lock%s held by %s/%d:\n",
+		depth, depth > 1 ? "s" : "", curr->comm, curr->pid);
 
-	for (i = 0; i < curr->lockdep_depth; i++) {
+	for (i = 0; i < depth; i++) {
 		printk(" #%d: ", i);
 		print_lock(curr->held_locks + i);
 	}
@@ -919,7 +923,7 @@ check_deadlock(struct task_struct *curr,
 		 * Allow read-after-read recursion of the same
 		 * lock instance (i.e. read_lock(lock)+read_lock(lock)):
 		 */
-		if ((read > 0) && prev->read &&
+		if ((read == 2) && prev->read &&
 				(prev->instance == next_instance))
 			return 2;
 		return print_deadlock_bug(curr, prev, next);
@@ -1018,7 +1022,7 @@ check_prev_add(struct task_struct *curr,
 	 * write-lock never takes any other locks, then the reads are
 	 * equivalent to a NOP.
 	 */
-	if (next->read == 1 || prev->read == 1)
+	if (next->read == 2 || prev->read == 2)
 		return 1;
 	/*
 	 * Is the <prev> -> <next> dependency already present?
@@ -2026,9 +2030,9 @@ EXPORT_SYMBOL_GPL(lockdep_init_map);
  * This gets called for every mutex_lock*()/spin_lock*() operation.
  * We maintain the dependency maps and validate the locking attempt:
  */
-static int __lockdep_acquire(struct lockdep_map *lock, unsigned int subtype,
-			     int trylock, int read, int hardirqs_off,
-			     unsigned long ip)
+static int __lock_acquire(struct lockdep_map *lock, unsigned int subtype,
+			  int trylock, int read, int check, int hardirqs_off,
+			  unsigned long ip)
 {
 	struct task_struct *curr = current;
 	struct held_lock *hlock;
@@ -2082,8 +2086,11 @@ static int __lockdep_acquire(struct lock
 	hlock->instance = lock;
 	hlock->trylock = trylock;
 	hlock->read = read;
+	hlock->check = check;
 	hlock->hardirqs_off = hardirqs_off;
 
+	if (check != 2)
+		goto out_calc_hash;
 #ifdef CONFIG_TRACE_IRQFLAGS
 	/*
 	 * If non-trylock use in a hardirq or softirq context, then
@@ -2131,6 +2138,7 @@ static int __lockdep_acquire(struct lock
 	/* mark it as used: */
 	if (!mark_lock(curr, hlock, LOCK_USED, ip))
 		return 0;
+out_calc_hash:
 	/*
 	 * Calculate the chain hash: it's the combined has of all the
 	 * lock keys along the dependency chain. We save the hash value
@@ -2188,7 +2196,7 @@ static int __lockdep_acquire(struct lock
 	 * (If lookup_chain_cache() returns with 1 it acquires
 	 * hash_lock for us)
 	 */
-	if (!trylock && lookup_chain_cache(chain_key)) {
+	if (!trylock && (check == 2) && lookup_chain_cache(chain_key)) {
 		/*
 		 * Check whether last held lock:
 		 *
@@ -2241,7 +2249,8 @@ static int
 print_unlock_order_bug(struct task_struct *curr, struct lockdep_map *lock,
 		       struct held_lock *hlock, unsigned long ip)
 {
-	debug_locks_off();
+	if (!debug_locks_off())
+		return 0;
 	if (debug_locks_silent)
 		return 0;
 
@@ -2271,7 +2280,8 @@ static int
 print_unlock_inbalance_bug(struct task_struct *curr, struct lockdep_map *lock,
 			   unsigned long ip)
 {
-	debug_locks_off();
+	if (!debug_locks_off())
+		return 0;
 	if (debug_locks_silent)
 		return 0;
 
@@ -2314,11 +2324,11 @@ static int check_unlock(struct task_stru
  * Remove the lock to the list of currently held locks in a
  * potentially non-nested (out of order) manner. This is a
  * relatively rare operation, as all the unlock APIs default
- * to nested mode (which uses lockdep_release()):
+ * to nested mode (which uses lock_release()):
  */
 static int
-lockdep_release_non_nested(struct task_struct *curr,
-			   struct lockdep_map *lock, unsigned long ip)
+lock_release_non_nested(struct task_struct *curr,
+			struct lockdep_map *lock, unsigned long ip)
 {
 	struct held_lock *hlock, *prev_hlock;
 	unsigned int depth;
@@ -2357,9 +2367,9 @@ found_it:
 
 	for (i++; i < depth; i++) {
 		hlock = curr->held_locks + i;
-		if (!__lockdep_acquire(hlock->instance,
+		if (!__lock_acquire(hlock->instance,
 			hlock->type->subtype, hlock->trylock,
-				hlock->read, hlock->hardirqs_off,
+				hlock->read, hlock->check, hlock->hardirqs_off,
 				hlock->acquire_ip))
 			return 0;
 	}
@@ -2375,8 +2385,8 @@ found_it:
  * mutex_lock_interruptible()). This is done for unlocks that nest
  * perfectly. (i.e. the current top of the lock-stack is unlocked)
  */
-static int lockdep_release_nested(struct task_struct *curr,
-				  struct lockdep_map *lock, unsigned long ip)
+static int lock_release_nested(struct task_struct *curr,
+			       struct lockdep_map *lock, unsigned long ip)
 {
 	struct held_lock *hlock;
 	unsigned int depth;
@@ -2394,7 +2404,7 @@ static int lockdep_release_nested(struct
 #ifdef CONFIG_DEBUG_NON_NESTED_UNLOCKS
 		return print_unlock_order_bug(curr, lock, hlock, ip);
 #else
-		return lockdep_release_non_nested(curr, lock, ip);
+		return lock_release_non_nested(curr, lock, ip);
 #endif
 	}
 	curr->lockdep_depth--;
@@ -2419,8 +2429,8 @@ static int lockdep_release_nested(struct
  * mutex_lock_interruptible()). This is done for unlocks that nest
  * perfectly. (i.e. the current top of the lock-stack is unlocked)
  */
-static void __lockdep_release(struct lockdep_map *lock, int nested,
-			      unsigned long ip)
+static void
+__lock_release(struct lockdep_map *lock, int nested, unsigned long ip)
 {
 	struct task_struct *curr = current;
 
@@ -2428,10 +2438,10 @@ static void __lockdep_release(struct loc
 		return;
 
 	if (nested) {
-		if (!lockdep_release_nested(curr, lock, ip))
+		if (!lock_release_nested(curr, lock, ip))
 			return;
 	} else {
-		if (!lockdep_release_non_nested(curr, lock, ip))
+		if (!lock_release_non_nested(curr, lock, ip))
 			return;
 	}
 
@@ -2473,8 +2483,8 @@ static void check_flags(unsigned long fl
  * We are not always called with irqs disabled - do that here,
  * and also avoid lockdep recursion:
  */
-void lockdep_acquire(struct lockdep_map *lock, unsigned int subtype,
-		     int trylock, int read, unsigned long ip)
+void lock_acquire(struct lockdep_map *lock, unsigned int subtype,
+		  int trylock, int read, int check, unsigned long ip)
 {
 	unsigned long flags;
 
@@ -2484,15 +2494,16 @@ void lockdep_acquire(struct lockdep_map 
 	if (unlikely(current->lockdep_recursion))
 		goto out;
 	current->lockdep_recursion = 1;
-	__lockdep_acquire(lock, subtype, trylock, read, irqs_disabled_flags(flags), ip);
+	__lock_acquire(lock, subtype, trylock, read, check,
+		       irqs_disabled_flags(flags), ip);
 	current->lockdep_recursion = 0;
 out:
 	raw_local_irq_restore(flags);
 }
 
-EXPORT_SYMBOL_GPL(lockdep_acquire);
+EXPORT_SYMBOL_GPL(lock_acquire);
 
-void lockdep_release(struct lockdep_map *lock, int nested, unsigned long ip)
+void lock_release(struct lockdep_map *lock, int nested, unsigned long ip)
 {
 	unsigned long flags;
 
@@ -2501,13 +2512,13 @@ void lockdep_release(struct lockdep_map 
 	if (unlikely(current->lockdep_recursion))
 		goto out;
 	current->lockdep_recursion = 1;
-	__lockdep_release(lock, nested, ip);
+	__lock_release(lock, nested, ip);
 	current->lockdep_recursion = 0;
 out:
 	raw_local_irq_restore(flags);
 }
 
-EXPORT_SYMBOL_GPL(lockdep_release);
+EXPORT_SYMBOL_GPL(lock_release);
 
 /*
  * Used by the testsuite, sanitize the validator state
@@ -2678,3 +2689,139 @@ void __init lockdep_info(void)
 #endif
 }
 
+static inline int in_range(const void *start, const void *addr, const void *end)
+{
+	return addr >= start && addr <= end;
+}
+
+static void
+print_freed_lock_bug(struct task_struct *curr, const void *mem_from,
+		     const void *mem_to)
+{
+	if (!debug_locks_off())
+		return;
+	if (debug_locks_silent)
+		return;
+
+	printk("\n=========================\n");
+	printk(  "[ BUG: held lock freed! ]\n");
+	printk(  "-------------------------\n");
+	printk("%s/%d is freeing memory %p-%p, with a lock still held there!\n",
+		curr->comm, curr->pid, mem_from, mem_to-1);
+	lockdep_print_held_locks(curr);
+
+	printk("\nstack backtrace:\n");
+	dump_stack();
+}
+
+/*
+ * Called when kernel memory is freed (or unmapped), or if a lock
+ * is destroyed or reinitialized - this code checks whether there is
+ * any held lock in the memory range of <from> to <to>:
+ */
+void debug_check_no_locks_freed(const void *mem_from, unsigned long mem_len)
+{
+	const void *mem_to = mem_from + mem_len, *lock_from, *lock_to;
+	struct task_struct *curr = current;
+	struct held_lock *hlock;
+	unsigned long flags;
+	int i;
+
+	if (unlikely(!debug_locks))
+		return;
+
+	local_irq_save(flags);
+	for (i = 0; i < curr->lockdep_depth; i++) {
+		hlock = curr->held_locks + i;
+
+		lock_from = (void *)hlock->instance;
+		lock_to = (void *)(hlock->instance + 1);
+
+		if (!in_range(mem_from, lock_from, mem_to) &&
+					!in_range(mem_from, lock_to, mem_to))
+			continue;
+
+		print_freed_lock_bug(curr, mem_from, mem_to);
+		break;
+	}
+	local_irq_restore(flags);
+}
+
+static void print_held_locks_bug(struct task_struct *curr)
+{
+	if (!debug_locks_off())
+		return;
+	if (debug_locks_silent)
+		return;
+
+	printk("\n=====================================\n");
+	printk(  "[ BUG: lock held at task exit time! ]\n");
+	printk(  "-------------------------------------\n");
+	printk("%s/%d is exiting with locks still held!\n",
+		curr->comm, curr->pid);
+	lockdep_print_held_locks(curr);
+
+	printk("\nstack backtrace:\n");
+	dump_stack();
+}
+
+void debug_check_no_locks_held(struct task_struct *task)
+{
+	if (unlikely(task->lockdep_depth > 0))
+		print_held_locks_bug(task);
+}
+
+void debug_show_all_locks(void)
+{
+	struct task_struct *g, *p;
+	int count = 10;
+	int unlock = 1;
+
+	printk("\nShowing all locks held in the system:\n");
+
+	/*
+	 * Here we try to get the tasklist_lock as hard as possible,
+	 * if not successful after 2 seconds we ignore it (but keep
+	 * trying). This is to enable a debug printout even if a
+	 * tasklist_lock-holding task deadlocks or crashes.
+	 */
+retry:
+	if (!read_trylock(&tasklist_lock)) {
+		if (count == 10)
+			printk("hm, tasklist_lock locked, retrying... ");
+		if (count) {
+			count--;
+			printk(" #%d", 10-count);
+			mdelay(200);
+			goto retry;
+		}
+		printk(" ignoring it.\n");
+		unlock = 0;
+	}
+	if (count != 10)
+		printk(" locked it.\n");
+
+	do_each_thread(g, p) {
+		if (p->lockdep_depth)
+			lockdep_print_held_locks(p);
+		if (!unlock)
+			if (read_trylock(&tasklist_lock))
+				unlock = 1;
+	} while_each_thread(g, p);
+
+	printk("\n");
+	printk("=============================================\n\n");
+
+	if (unlock)
+		read_unlock(&tasklist_lock);
+}
+
+EXPORT_SYMBOL_GPL(debug_show_all_locks);
+
+void debug_show_held_locks(struct task_struct *task)
+{
+	lockdep_print_held_locks(task);
+}
+
+EXPORT_SYMBOL_GPL(debug_show_held_locks);
+
diff -puN kernel/mutex.c~lock-validator-v3 kernel/mutex.c
--- a/kernel/mutex.c~lock-validator-v3
+++ a/kernel/mutex.c
@@ -27,13 +27,8 @@
 # include "mutex-debug.h"
 # include <asm-generic/mutex-null.h>
 #else
-# ifdef CONFIG_PROVE_MUTEX_LOCKING
-#  include "mutex-lockdep.h"
-#  include <asm-generic/mutex-null.h>
-# else
-#  include "mutex.h"
-#  include <asm/mutex.h>
-# endif
+# include "mutex.h"
+# include <asm/mutex.h>
 #endif
 
 /***
@@ -51,11 +46,7 @@ __mutex_init(struct mutex *lock, const c
 	spin_lock_init(&lock->wait_lock);
 	INIT_LIST_HEAD(&lock->wait_list);
 
-	debug_mutex_init(lock, name);
-
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
-	lockdep_init_map(&lock->dep_map, name, key);
-#endif
+	debug_mutex_init(lock, name, key);
 }
 
 EXPORT_SYMBOL(__mutex_init);
@@ -225,7 +216,7 @@ __mutex_lock_slowpath(atomic_t *lock_cou
 	__mutex_lock_common(lock, TASK_UNINTERRUPTIBLE, 0);
 }
 
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 void __sched
 mutex_lock_nested(struct mutex *lock, unsigned int subtype)
 {
diff -puN kernel/mutex-debug.c~lock-validator-v3 kernel/mutex-debug.c
--- a/kernel/mutex-debug.c~lock-validator-v3
+++ a/kernel/mutex-debug.c
@@ -83,12 +83,16 @@ void debug_mutex_unlock(struct mutex *lo
 	DEBUG_WARN_ON(lock->owner != current_thread_info());
 }
 
-void debug_mutex_init(struct mutex *lock, const char *name)
+void debug_mutex_init(struct mutex *lock, const char *name,
+		      struct lockdep_type_key *key)
 {
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 	/*
 	 * Make sure we are not reinitializing a held lock:
 	 */
 	debug_check_no_locks_freed((void *)lock, sizeof(*lock));
+	lockdep_init_map(&lock->dep_map, name, key);
+#endif
 	lock->owner = NULL;
 	lock->magic = lock;
 }
diff -puN kernel/mutex-debug.h~lock-validator-v3 kernel/mutex-debug.h
--- a/kernel/mutex-debug.h~lock-validator-v3
+++ a/kernel/mutex-debug.h
@@ -32,7 +32,8 @@ extern void debug_mutex_add_waiter(struc
 extern void mutex_remove_waiter(struct mutex *lock, struct mutex_waiter *waiter,
 				struct thread_info *ti);
 extern void debug_mutex_unlock(struct mutex *lock);
-extern void debug_mutex_init(struct mutex *lock, const char *name);
+extern void debug_mutex_init(struct mutex *lock, const char *name,
+			     struct lockdep_type_key *key);
 
 #define spin_lock_mutex(lock, flags)			\
 	do {						\
diff -puN kernel/mutex.h~lock-validator-v3 kernel/mutex.h
--- a/kernel/mutex.h~lock-validator-v3
+++ a/kernel/mutex.h
@@ -24,7 +24,7 @@
 #define debug_mutex_free_waiter(waiter)			do { } while (0)
 #define debug_mutex_add_waiter(lock, waiter, ti)	do { } while (0)
 #define debug_mutex_unlock(lock)			do { } while (0)
-#define debug_mutex_init(lock, name)			do { } while (0)
+#define debug_mutex_init(lock, name, key)		do { } while (0)
 
 static inline void
 debug_mutex_lock_common(struct mutex *lock, struct mutex_waiter *waiter)
diff -puN kernel/mutex-lockdep.h~lock-validator-v3 /dev/null
--- a/kernel/mutex-lockdep.h
+++ /dev/null
@@ -1,40 +0,0 @@
-/*
- * Mutexes: blocking mutual exclusion locks
- *
- * started by Ingo Molnar:
- *
- *  Copyright (C) 2004-2006 Red Hat, Inc., Ingo Molnar <mingo@xxxxxxxxxx>
- *
- * This file contains mutex debugging related internal prototypes, for the
- * !CONFIG_DEBUG_MUTEXES && CONFIG_PROVE_MUTEX_LOCKING case. Most of
- * them are NOPs:
- */
-
-#define spin_lock_mutex(lock, flags)			\
-	do {						\
-		local_irq_save(flags);			\
-		__raw_spin_lock(&(lock)->raw_lock);	\
-	} while (0)
-
-#define spin_unlock_mutex(lock, flags)			\
-	do {						\
-		__raw_spin_unlock(&(lock)->raw_lock);	\
-		local_irq_restore(flags);		\
-	} while (0)
-
-#define mutex_remove_waiter(lock, waiter, ti) \
-		__list_del((waiter)->list.prev, (waiter)->list.next)
-
-#define debug_mutex_set_owner(lock, new_owner)		do { } while (0)
-#define debug_mutex_clear_owner(lock)			do { } while (0)
-#define debug_mutex_wake_waiter(lock, waiter)		do { } while (0)
-#define debug_mutex_free_waiter(waiter)			do { } while (0)
-#define debug_mutex_add_waiter(lock, waiter, ti)	do { } while (0)
-#define debug_mutex_unlock(lock)			do { } while (0)
-#define debug_mutex_init(lock, name)			do { } while (0)
-
-static inline void
-debug_mutex_lock_common(struct mutex *lock,
-			struct mutex_waiter *waiter)
-{
-}
diff -puN kernel/spinlock.c~lock-validator-v3 kernel/spinlock.c
--- a/kernel/spinlock.c~lock-validator-v3
+++ a/kernel/spinlock.c
@@ -17,44 +17,6 @@
 #include <linux/debug_locks.h>
 #include <linux/module.h>
 
-#if defined(CONFIG_DEBUG_SPINLOCK) || defined(CONFIG_PROVE_SPIN_LOCKING)
-void __spin_lock_init(spinlock_t *lock, const char *name,
-		      struct lockdep_type_key *key)
-{
-	lock->raw_lock = (raw_spinlock_t)__RAW_SPIN_LOCK_UNLOCKED;
-#ifdef CONFIG_DEBUG_SPINLOCK
-	lock->magic = SPINLOCK_MAGIC;
-	lock->owner = SPINLOCK_OWNER_INIT;
-	lock->owner_cpu = -1;
-#endif
-#ifdef CONFIG_PROVE_SPIN_LOCKING
-	lockdep_init_map(&lock->dep_map, name, key);
-#endif
-}
-
-EXPORT_SYMBOL(__spin_lock_init);
-
-#endif
-
-#if defined(CONFIG_DEBUG_SPINLOCK) || defined(CONFIG_PROVE_RW_LOCKING)
-
-void __rwlock_init(rwlock_t *lock, const char *name,
-		   struct lockdep_type_key *key)
-{
-	lock->raw_lock = (raw_rwlock_t) __RAW_RW_LOCK_UNLOCKED;
-#ifdef CONFIG_DEBUG_SPINLOCK
-	lock->magic = RWLOCK_MAGIC;
-	lock->owner = SPINLOCK_OWNER_INIT;
-	lock->owner_cpu = -1;
-#endif
-#ifdef CONFIG_PROVE_RW_LOCKING
-	lockdep_init_map(&lock->dep_map, name, key);
-#endif
-}
-
-EXPORT_SYMBOL(__rwlock_init);
-
-#endif
 /*
  * Generic declaration of the raw read_trylock() function,
  * architectures are supposed to optimize this:
diff -puN lib/Kconfig.debug~lock-validator-v3 lib/Kconfig.debug
--- a/lib/Kconfig.debug~lock-validator-v3
+++ a/lib/Kconfig.debug
@@ -116,24 +116,6 @@ config DEBUG_PREEMPT
 	  if kernel code uses it in a preemption-unsafe way. Also, the kernel
 	  will detect preemption count underflows.
 
-config DEBUG_MUTEXES
-	bool "Mutex debugging, basic checks"
-	default y
-	depends on DEBUG_KERNEL
-	help
-	 This feature allows mutex semantics violations to be detected and
-	 reported.
-
-config DEBUG_MUTEX_ALLOC
-	bool "Detect incorrect freeing of live mutexes"
-	default y
-	depends on DEBUG_MUTEXES
-	help
-	 This feature will check whether any held mutex is incorrectly
-	 freed by the kernel, via any of the memory-freeing routines
-	 (kfree(), kmem_cache_free(), free_pages(), vfree(), etc.),
-	 or whether there is any lock held during task exit.
-
 config DEBUG_RT_MUTEXES
 	bool "RT Mutex debugging, deadlock detection"
 	depends on DEBUG_KERNEL && RT_MUTEXES
@@ -153,7 +135,7 @@ config RT_MUTEX_TESTER
 	  This option enables a rt-mutex tester.
 
 config DEBUG_SPINLOCK
-	bool "Spinlock debugging"
+	bool "Spinlock and rw-lock debugging: basic checks"
 	depends on DEBUG_KERNEL
 	help
 	  Say Y here and build SMP to catch missing spinlock initialization
@@ -161,9 +143,20 @@ config DEBUG_SPINLOCK
 	  best used in conjunction with the NMI watchdog so that spinlock
 	  deadlocks are also debuggable.
 
+config DEBUG_SPINLOCK_ALLOC
+	bool "Spinlock debugging: detect incorrect freeing of live spinlocks"
+	depends on DEBUG_SPINLOCK
+	select LOCKDEP
+	help
+	 This feature will check whether any held spinlock is incorrectly
+	 freed by the kernel, via any of the memory-freeing routines
+	 (kfree(), kmem_cache_free(), free_pages(), vfree(), etc.),
+	 whether a live spinlock is being reinitialized via spin_lock_init(),
+	 or whether there is any spinlock held during task exit.
+
 config PROVE_SPIN_LOCKING
-	bool "Prove spin-locking correctness"
-	depends on TRACE_IRQFLAGS_SUPPORT
+	bool "Spinlock debugging: prove spin-locking correctness"
+	depends on TRACE_IRQFLAGS_SUPPORT && DEBUG_SPINLOCK_ALLOC
 	default n
 	help
 	 This feature enables the kernel to prove that all spinlock
@@ -199,9 +192,20 @@ config PROVE_SPIN_LOCKING
 
 	 For more details, see Documentation/locking-correctness.txt.
 
+config DEBUG_RWLOCK_ALLOC
+	bool "rw-lock debugging: detect incorrect freeing of live rwlocks"
+	depends on DEBUG_SPINLOCK
+	select LOCKDEP
+	help
+	 This feature will check whether any held rwlock is incorrectly
+	 freed by the kernel, via any of the memory-freeing routines
+	 (kfree(), kmem_cache_free(), free_pages(), vfree(), etc.),
+	 whether a live rwlock is being reinitialized via rwlock_init(),
+	 or whether there is any rwlock held during task exit.
+
 config PROVE_RW_LOCKING
-	bool "Prove rw-locking correctness"
-	depends on TRACE_IRQFLAGS_SUPPORT
+	bool "rw-lock debugging: prove rw-locking correctness"
+	depends on TRACE_IRQFLAGS_SUPPORT && DEBUG_RWLOCK_ALLOC
 	default n
 	help
 	 This feature enables the kernel to prove that all rwlock
@@ -237,9 +241,28 @@ config PROVE_RW_LOCKING
 
 	 For more details, see Documentation/locking-correctness.txt.
 
+config DEBUG_MUTEXES
+	bool "Mutex debugging: basic checks"
+	default y
+	depends on DEBUG_KERNEL
+	help
+	 This feature allows mutex semantics violations to be detected and
+	 reported.
+
+config DEBUG_MUTEX_ALLOC
+	bool "Mutex debugging: detect incorrect freeing of live mutexes"
+	depends on DEBUG_MUTEXES
+	select LOCKDEP
+	help
+	 This feature will check whether any held mutex is incorrectly
+	 freed by the kernel, via any of the memory-freeing routines
+	 (kfree(), kmem_cache_free(), free_pages(), vfree(), etc.),
+	 whether a live mutex is being reinitialized via mutex_init(),
+	 or whether there is any mutex held during task exit.
+
 config PROVE_MUTEX_LOCKING
-	bool "Prove mutex-locking correctness"
-	depends on TRACE_IRQFLAGS_SUPPORT
+	bool "Mutex debugging: prove mutex-locking correctness"
+	depends on DEBUG_MUTEX_ALLOC
 	default n
 	help
 	 This feature enables the kernel to prove that all mutexlock
@@ -275,9 +298,27 @@ config PROVE_MUTEX_LOCKING
 
 	 For more details, see Documentation/locking-correctness.txt.
 
+config DEBUG_RWSEMS
+	bool "rwsem debugging: basic checks"
+	depends on DEBUG_KERNEL
+	help
+	 This feature allows read-write semaphore semantics violations to
+	 be detected and reported.
+
+config DEBUG_RWSEM_ALLOC
+	bool "rwsem debugging: detect incorrect freeing of live rwsems"
+	depends on DEBUG_RWSEMS
+	select LOCKDEP
+	help
+	 This feature will check whether any held rwsem is incorrectly
+	 freed by the kernel, via any of the memory-freeing routines
+	 (kfree(), kmem_cache_free(), free_pages(), vfree(), etc.),
+	 whether a live rwsem is being reinitialized via init_rwsem(),
+	 or whether there is any rwsem held during task exit.
+
 config PROVE_RWSEM_LOCKING
 	bool "Prove rwsem-locking correctness"
-	depends on TRACE_IRQFLAGS_SUPPORT
+	depends on DEBUG_RWSEM_ALLOC
 	default n
 	help
 	 This feature enables the kernel to prove that all rwsemlock
@@ -315,11 +356,9 @@ config PROVE_RWSEM_LOCKING
 
 config LOCKDEP
 	bool
-	default y
 	select FRAME_POINTER
 	select KALLSYMS
 	select KALLSYMS_ALL
-	depends on PROVE_SPIN_LOCKING || PROVE_RW_LOCKING || PROVE_MUTEX_LOCKING || PROVE_RWSEM_LOCKING
 
 config DEBUG_NON_NESTED_UNLOCKS
 	bool "Detect non-nested unlocks"
@@ -338,7 +377,6 @@ config DEBUG_LOCKDEP
 	bool "Lock dependency engine debugging"
 	depends on LOCKDEP
 	default y
-	depends on TRACE_IRQFLAGS_SUPPORT
 	help
 	  If you say Y here, the lock dependency engine will do
 	  additional runtime checks to debug itself, at the price
@@ -351,7 +389,7 @@ config TRACE_IRQFLAGS
 	depends on PROVE_SPIN_LOCKING || PROVE_RW_LOCKING
 
 config DEBUG_SPINLOCK_SLEEP
-	bool "Sleep-inside-spinlock checking"
+	bool "Spinlock debugging: sleep-inside-spinlock checking"
 	depends on DEBUG_KERNEL
 	help
 	  If you say Y here, various routines which may sleep will become very
diff -puN lib/locking-selftest.c~lock-validator-v3 lib/locking-selftest.c
--- a/lib/locking-selftest.c~lock-validator-v3
+++ a/lib/locking-selftest.c
@@ -173,6 +173,7 @@ static void init_shared_types(void)
 #define U(x)			spin_unlock(&lock_##x)
 #define UNN(x)			spin_unlock_non_nested(&lock_##x)
 #define LU(x)			L(x); U(x)
+#define SI(x)			spin_lock_init(&lock_##x)
 
 #define WL(x)			write_lock(&rwlock_##x)
 #define WU(x)			write_unlock(&rwlock_##x)
@@ -182,10 +183,12 @@ static void init_shared_types(void)
 #define RU(x)			read_unlock(&rwlock_##x)
 #define RUNN(x)			read_unlock_non_nested(&rwlock_##x)
 #define RLU(x)			RL(x); RU(x)
+#define RWI(x)			rwlock_init(&rwlock_##x)
 
 #define ML(x)			mutex_lock(&mutex_##x)
 #define MU(x)			mutex_unlock(&mutex_##x)
 #define MUNN(x)			mutex_unlock_non_nested(&mutex_##x)
+#define MI(x)			mutex_init(&mutex_##x)
 
 #define WSL(x)			down_write(&rwsem_##x)
 #define WSU(x)			up_write(&rwsem_##x)
@@ -193,6 +196,7 @@ static void init_shared_types(void)
 #define RSL(x)			down_read(&rwsem_##x)
 #define RSU(x)			up_read(&rwsem_##x)
 #define RSUNN(x)		up_read_non_nested(&rwsem_##x)
+#define RWSI(x)			init_rwsem(&rwsem_##x)
 
 #define LOCK_UNLOCK_2(x,y)	LOCK(x); LOCK(y); UNLOCK(y); UNLOCK(x)
 
@@ -226,9 +230,7 @@ static void name##_321(void) { E3(); E2(
 #define E()					\
 						\
 	LOCK(X1);				\
-	LOCK(X2); /* this one should fail */	\
-	UNLOCK(X2);				\
-	UNLOCK(X1);
+	LOCK(X2); /* this one should fail */
 
 /*
  * 6 testcases:
@@ -256,16 +258,50 @@ static void rlock_AA1(void)
 {
 	RL(X1);
 	RL(X1); // this one should NOT fail
-	RU(X1);
-	RU(X1);
+}
+
+static void rlock_AA1B(void)
+{
+	RL(X1);
+	RL(X2); // this one should fail
 }
 
 static void rsem_AA1(void)
 {
 	RSL(X1);
 	RSL(X1); // this one should fail
-	RSU(X1);
-	RSU(X1);
+}
+
+static void rsem_AA1B(void)
+{
+	RSL(X1);
+	RSL(X2); // this one should fail
+}
+/*
+ * The mixing of read and write locks is not allowed:
+ */
+static void rlock_AA2(void)
+{
+	RL(X1);
+	WL(X2); // this one should fail
+}
+
+static void rsem_AA2(void)
+{
+	RSL(X1);
+	WSL(X2); // this one should fail
+}
+
+static void rlock_AA3(void)
+{
+	WL(X1);
+	RL(X2); // this one should fail
+}
+
+static void rsem_AA3(void)
+{
+	WSL(X1);
+	RSL(X2); // this one should fail
 }
 
 /*
@@ -491,6 +527,32 @@ GENERATE_TESTCASE(bad_unlock_order_rsem)
 
 #undef E
 
+/*
+ * initializing a held lock:
+ */
+#define E()					\
+						\
+	LOCK(A);				\
+	INIT(A); /* fail */
+
+/*
+ * 6 testcases:
+ */
+#include "locking-selftest-spin.h"
+GENERATE_TESTCASE(init_held_spin)
+#include "locking-selftest-wlock.h"
+GENERATE_TESTCASE(init_held_wlock)
+#include "locking-selftest-rlock.h"
+GENERATE_TESTCASE(init_held_rlock)
+#include "locking-selftest-mutex.h"
+GENERATE_TESTCASE(init_held_mutex)
+#include "locking-selftest-wsem.h"
+GENERATE_TESTCASE(init_held_wsem)
+#include "locking-selftest-rsem.h"
+GENERATE_TESTCASE(init_held_rsem)
+
+#undef E
+
 #ifdef CONFIG_LOCKDEP
 /*
  * bad unlock ordering - but using the _non_nested API,
@@ -871,25 +933,25 @@ GENERATE_PERMUTATIONS_3_EVENTS(irq_read_
 #include "locking-selftest-softirq.h"
 // GENERATE_PERMUTATIONS_3_EVENTS(irq_read_recursion2_soft)
 
-#ifdef CONFIG_PROVE_SPIN_LOCKING
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
 # define I_SPINLOCK(x)	lockdep_reset_lock(&lock_##x.dep_map)
 #else
 # define I_SPINLOCK(x)
 #endif
 
-#ifdef CONFIG_PROVE_RW_LOCKING
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
 # define I_RWLOCK(x)	lockdep_reset_lock(&rwlock_##x.dep_map)
 #else
 # define I_RWLOCK(x)
 #endif
 
-#ifdef CONFIG_PROVE_MUTEX_LOCKING
+#ifdef CONFIG_DEBUG_MUTEX_ALLOC
 # define I_MUTEX(x)	lockdep_reset_lock(&mutex_##x.dep_map)
 #else
 # define I_MUTEX(x)
 #endif
 
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
 # define I_RWSEM(x)	lockdep_reset_lock(&rwsem_##x.dep_map)
 #else
 # define I_RWSEM(x)
@@ -1139,6 +1201,7 @@ void locking_selftest(void)
 	DO_TESTCASE_6R("A-B-C-D-B-D-D-A deadlock", ABCDBDDA);
 	DO_TESTCASE_6R("A-B-C-D-B-C-D-A deadlock", ABCDBCDA);
 	DO_TESTCASE_6("double unlock", double_unlock);
+	DO_TESTCASE_6("initialize held", init_held);
 #ifdef CONFIG_DEBUG_NON_NESTED_UNLOCKS
 	DO_TESTCASE_6("bad unlock order", bad_unlock_order);
 #else
@@ -1153,8 +1216,30 @@ void locking_selftest(void)
 	dotest(rsem_AA1, FAILURE, LOCKTYPE_RWSEM);
 	printk("\n");
 
+	print_testname("recursive read-lock #2");
+	printk("             |");
+	dotest(rlock_AA1B, FAILURE, LOCKTYPE_RWLOCK);
+	printk("             |");
+	dotest(rsem_AA1B, FAILURE, LOCKTYPE_RWSEM);
+	printk("\n");
+
+	print_testname("mixed read-write-lock");
+	printk("             |");
+	dotest(rlock_AA2, FAILURE, LOCKTYPE_RWLOCK);
+	printk("             |");
+	dotest(rsem_AA2, FAILURE, LOCKTYPE_RWSEM);
+	printk("\n");
+
+	print_testname("mixed write-read-lock");
+	printk("             |");
+	dotest(rlock_AA3, FAILURE, LOCKTYPE_RWLOCK);
+	printk("             |");
+	dotest(rsem_AA3, FAILURE, LOCKTYPE_RWSEM);
+	printk("\n");
+
 	printk("  --------------------------------------------------------------------------\n");
 
+
 #ifdef CONFIG_LOCKDEP
 	print_testname("non-nested unlock");
 	dotest(spin_order_nn, SUCCESS, LOCKTYPE_SPIN);
diff -puN lib/locking-selftest-mutex.h~lock-validator-v3 lib/locking-selftest-mutex.h
--- a/lib/locking-selftest-mutex.h~lock-validator-v3
+++ a/lib/locking-selftest-mutex.h
@@ -3,3 +3,9 @@
 
 #undef UNLOCK
 #define UNLOCK		MU
+
+#undef RLOCK
+#undef WLOCK
+
+#undef INIT
+#define INIT		MI
diff -puN lib/locking-selftest-rlock.h~lock-validator-v3 lib/locking-selftest-rlock.h
--- a/lib/locking-selftest-rlock.h~lock-validator-v3
+++ a/lib/locking-selftest-rlock.h
@@ -3,3 +3,12 @@
 
 #undef UNLOCK
 #define UNLOCK		RU
+
+#undef RLOCK
+#define RLOCK		RL
+
+#undef WLOCK
+#define WLOCK		WL
+
+#undef INIT
+#define INIT		RWI
diff -puN lib/locking-selftest-rsem.h~lock-validator-v3 lib/locking-selftest-rsem.h
--- a/lib/locking-selftest-rsem.h~lock-validator-v3
+++ a/lib/locking-selftest-rsem.h
@@ -3,3 +3,12 @@
 
 #undef UNLOCK
 #define UNLOCK		RSU
+
+#undef RLOCK
+#define RLOCK		RSL
+
+#undef WLOCK
+#define WLOCK		WSL
+
+#undef INIT
+#define INIT		RWSI
diff -puN lib/locking-selftest-spin.h~lock-validator-v3 lib/locking-selftest-spin.h
--- a/lib/locking-selftest-spin.h~lock-validator-v3
+++ a/lib/locking-selftest-spin.h
@@ -3,3 +3,9 @@
 
 #undef UNLOCK
 #define UNLOCK		U
+
+#undef RLOCK
+#undef WLOCK
+
+#undef INIT
+#define INIT		SI
diff -puN lib/locking-selftest-wlock.h~lock-validator-v3 lib/locking-selftest-wlock.h
--- a/lib/locking-selftest-wlock.h~lock-validator-v3
+++ a/lib/locking-selftest-wlock.h
@@ -3,3 +3,12 @@
 
 #undef UNLOCK
 #define UNLOCK		WU
+
+#undef RLOCK
+#define RLOCK		RL
+
+#undef WLOCK
+#define WLOCK		WL
+
+#undef INIT
+#define INIT		RWI
diff -puN lib/locking-selftest-wsem.h~lock-validator-v3 lib/locking-selftest-wsem.h
--- a/lib/locking-selftest-wsem.h~lock-validator-v3
+++ a/lib/locking-selftest-wsem.h
@@ -3,3 +3,12 @@
 
 #undef UNLOCK
 #define UNLOCK		WSU
+
+#undef RLOCK
+#define RLOCK		RSL
+
+#undef WLOCK
+#define WLOCK		WSL
+
+#undef INIT
+#define INIT		RWSI
diff -puN lib/rwsem.c~lock-validator-v3 lib/rwsem.c
--- a/lib/rwsem.c~lock-validator-v3
+++ a/lib/rwsem.c
@@ -14,15 +14,19 @@
 void __init_rwsem(struct rw_semaphore *sem, const char *name,
 		  struct lockdep_type_key *key)
 {
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
+	/*
+	 * Make sure we are not reinitializing a held semaphore:
+	 */
+	debug_check_no_locks_freed((void *)sem, sizeof(*sem));
+	lockdep_init_map(&sem->dep_map, name, key);
+#endif
 	sem->count = RWSEM_UNLOCKED_VALUE;
 	spin_lock_init(&sem->wait_lock);
 	INIT_LIST_HEAD(&sem->wait_list);
 #if RWSEM_DEBUG
 	sem->debug = 0;
 #endif
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
-	lockdep_init_map(&sem->dep_map, name, key);
-#endif
 }
 
 EXPORT_SYMBOL(__init_rwsem);
diff -puN lib/rwsem-spinlock.c~lock-validator-v3 lib/rwsem-spinlock.c
--- a/lib/rwsem-spinlock.c~lock-validator-v3
+++ a/lib/rwsem-spinlock.c
@@ -33,15 +33,19 @@ void rwsemtrace(struct rw_semaphore *sem
 void __init_rwsem(struct rw_semaphore *sem, const char *name,
 		  struct lockdep_type_key *key)
 {
+#ifdef CONFIG_DEBUG_RWSEM_ALLOC
+	/*
+	 * Make sure we are not reinitializing a held semaphore:
+	 */
+	debug_check_no_locks_freed((void *)sem, sizeof(*sem));
+	lockdep_init_map(&sem->dep_map, name, key);
+#endif
 	sem->activity = 0;
 	spin_lock_init(&sem->wait_lock);
 	INIT_LIST_HEAD(&sem->wait_list);
 #if RWSEM_DEBUG
 	sem->debug = 0;
 #endif
-#ifdef CONFIG_PROVE_RWSEM_LOCKING
-	lockdep_init_map(&sem->dep_map, name, key);
-#endif
 }
 
 /*
diff -puN lib/spinlock_debug.c~lock-validator-v3 lib/spinlock_debug.c
--- a/lib/spinlock_debug.c~lock-validator-v3
+++ a/lib/spinlock_debug.c
@@ -13,6 +13,42 @@
 #include <linux/delay.h>
 #include <linux/module.h>
 
+void __spin_lock_init(spinlock_t *lock, const char *name,
+		      struct lockdep_type_key *key)
+{
+#ifdef CONFIG_DEBUG_SPINLOCK_ALLOC
+	/*
+	 * Make sure we are not reinitializing a held lock:
+	 */
+	debug_check_no_locks_freed((void *)lock, sizeof(*lock));
+	lockdep_init_map(&lock->dep_map, name, key);
+#endif
+	lock->raw_lock = (raw_spinlock_t)__RAW_SPIN_LOCK_UNLOCKED;
+	lock->magic = SPINLOCK_MAGIC;
+	lock->owner = SPINLOCK_OWNER_INIT;
+	lock->owner_cpu = -1;
+}
+
+EXPORT_SYMBOL(__spin_lock_init);
+
+void __rwlock_init(rwlock_t *lock, const char *name,
+		   struct lockdep_type_key *key)
+{
+#ifdef CONFIG_DEBUG_RWLOCK_ALLOC
+	/*
+	 * Make sure we are not reinitializing a held lock:
+	 */
+	debug_check_no_locks_freed((void *)lock, sizeof(*lock));
+	lockdep_init_map(&lock->dep_map, name, key);
+#endif
+	lock->raw_lock = (raw_rwlock_t) __RAW_RW_LOCK_UNLOCKED;
+	lock->magic = RWLOCK_MAGIC;
+	lock->owner = SPINLOCK_OWNER_INIT;
+	lock->owner_cpu = -1;
+}
+
+EXPORT_SYMBOL(__rwlock_init);
+
 static void spin_bug(spinlock_t *lock, const char *msg)
 {
 	struct task_struct *owner = NULL;
_

Patches currently in -mm which might be from mingo@xxxxxxx are

origin.patch
disable-debugging-version-of-write_lock.patch
fix-drivers-mfd-ucb1x00-corec-irq-probing-bug.patch
cifs-remove-f_ownerlock-use.patch
lock-validator-fix-ns83820c-irq-flags-bug.patch
revert-gregkh-pci-pci-test-that-drivers-properly-call-pci_set_master.patch
x86-re-enable-generic-numa.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma-tidy.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma-arch_vma_name-fix.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma-vs-x86_64-mm-reliable-stack-trace-support-i386.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma-vs-x86_64-mm-reliable-stack-trace-support-i386-2.patch
vdso-randomize-the-i386-vdso-by-moving-it-into-a-vma-vs-x86_64-mm-reliable-stack-trace-support-i386-2-revert-maxmem-change.patch
work-around-ppc64-bootup-bug-by-making-mutex-debugging-save-restore-irqs.patch
kernel-kernel-cpuc-to-mutexes.patch
define-__raw_get_cpu_var-and-use-it.patch
ide-cd-end-of-media-error-fix.patch
spin-rwlock-init-cleanups.patch
lock-validator-introduce-warn_on_oncecond.patch
lock-validator-introduce-warn_on_oncecond-speedup.patch
emu10k1-mark-midi_spinlock-as-used.patch
epoll-use-unlocked-wqueue-operations.patch
reintegrate-irqreturnh-into-hardirqh.patch
time-clocksource-infrastructure.patch
sched-fix-smt-nice-lock-contention-and-optimization.patch
sched-fix-smt-nice-lock-contention-and-optimization-tidy.patch
sched-comment-bitmap-size-accounting.patch
sched-fix-interactive-ceiling-code.patch
sched-cpu-hotplug-race-vs-set_cpus_allowed.patch
sched-implement-smpnice.patch
sched-protect-calculation-of-max_pull-from-integer-wrap.patch
sched-store-weighted-load-on-up.patch
sched-add-discrete-weighted-cpu-load-function.patch
sched-prevent-high-load-weight-tasks-suppressing-balancing.patch
sched-improve-stability-of-smpnice-load-balancing.patch
sched-improve-smpnice-load-balancing-when-load-per-task.patch
smpnice-dont-consider-sched-groups-which-are-lightly-loaded-for-balancing.patch
smpnice-dont-consider-sched-groups-which-are-lightly-loaded-for-balancing-fix.patch
sched-modify-move_tasks-to-improve-load-balancing-outcomes.patch
sched-avoid-unnecessarily-moving-highest-priority-task-move_tasks.patch
sched-avoid-unnecessarily-moving-highest-priority-task-move_tasks-fix-2.patch
sched_domain-handle-kmalloc-failure.patch
sched_domain-handle-kmalloc-failure-fix.patch
sched_domain-dont-use-gfp_atomic.patch
sched_domain-use-kmalloc_node.patch
sched_domain-allocate-sched_group-structures-dynamically.patch
sched-mc-smt-power-savings-sched-policy.patch
sched_exit-fix-parent-time_slice-calculation.patch
sched_exit-move-the-callsite-to-do_exit.patch
sched-uninline-task_rq_lock.patch
bug-if-setscheduler-is-called-from-interrupt-context.patch
sched-add-above-background-load-function.patch
mm-implement-swap-prefetching.patch
pi-futex-futex-code-cleanups.patch
pi-futex-robust-futex-docs-fix.patch
pi-futex-introduce-debug_check_no_locks_freed.patch
pi-futex-introduce-warn_on_smp.patch
pi-futex-add-plist-implementation.patch
pi-futex-scheduler-support-for-pi.patch
pi-futex-rt-mutex-core.patch
pi-futex-rt-mutex-docs.patch
pi-futex-rt-mutex-docs-update.patch
pi-futex-rt-mutex-debug.patch
pi-futex-rt-mutex-tester.patch
pi-futex-rt-mutex-tester-fix.patch
pi-futex-rt-mutex-futex-api.patch
pi-futex-futex_lock_pi-futex_unlock_pi-support.patch
pi-futex-futex_lock_pi-futex_unlock_pi-support-fix.patch
rtmutex-modify-rtmutex-tester-to-test-the-setscheduler.patch
futex_requeue-optimization.patch
fix-ide-deadlock-in-error-reporting-code.patch
genirq-rename-desc-handler-to-desc-chip.patch
genirq-rename-desc-handler-to-desc-chip-power-fix.patch
genirq-rename-desc-handler-to-desc-chip-ia64-fix.patch
genirq-rename-desc-handler-to-desc-chip-ia64-fix-2.patch
genirq-rename-desc-handler-to-desc-chip-terminate_irqs-fix.patch
genirq-sem2mutex-probe_sem-probing_active.patch
genirq-cleanup-merge-irq_affinity-into-irq_desc.patch
genirq-cleanup-remove-irq_descp.patch
genirq-cleanup-remove-irq_descp-fix.patch
genirq-cleanup-remove-fastcall.patch
genirq-cleanup-misc-code-cleanups.patch
genirq-cleanup-reduce-irq_desc_t-use-mark-it-obsolete.patch
genirq-cleanup-include-linux-irqh.patch
genirq-cleanup-merge-irq_dir-smp_affinity_entry-into-irq_desc.patch
genirq-cleanup-merge-pending_irq_cpumask-into-irq_desc.patch
genirq-cleanup-turn-arch_has_irq_per_cpu-into-config_irq_per_cpu.patch
genirq-debug-better-debug-printout-in-enable_irq.patch
genirq-add-retrigger-irq-op-to-consolidate-hw_irq_resend.patch
genirq-doc-comment-include-linux-irqh-structures.patch
genirq-doc-handle_irq_event-and-__do_irq-comments.patch
genirq-cleanup-no_irq_type-cleanups.patch
genirq-doc-add-design-documentation.patch
genirq-add-genirq-sw-irq-retrigger.patch
genirq-add-irq_noprobe-support.patch
genirq-add-irq_norequest-support.patch
genirq-add-irq_noautoen-support.patch
genirq-update-copyrights.patch
genirq-core.patch
genirq-msi-fixes-2.patch
genirq-add-irq-chip-support.patch
genirq-add-irq-chip-support-fix.patch
genirq-add-irq-chip-support-misroute-irq-dont-call-desc-chip-end.patch
genirq-add-handle_bad_irq.patch
genirq-add-irq-wake-power-management-support.patch
genirq-add-sa_trigger-support.patch
genirq-cleanup-no_irq_type-no_irq_chip-rename.patch
genirq-more-verbose-debugging-on-unexpected-irq-vectors.patch
genirq-ia64-build-fix.patch
genirq-add-irq_type_sense_mask.patch
genirq-add-irq-chip-support-fasteoi-handler-handle-interrupt-disabling.patch
genirq-irq-document-what-an-irq-is.patch
genirq-add-chip-eoi-fastack-fasteoi-core.patch
genirq-add-chip-eoi-fastack-fasteoi-fix.patch
acpi-reduce-code-size-clean-up-fix-validator-message.patch
lock-validator-special-locking-kgdb.patch
detect-atomic-counter-underflows.patch
debug-shared-irqs.patch
make-frame_pointer-default=y.patch
mutex-subsystem-synchro-test-module.patch
vdso-print-fatal-signals.patch
vdso-improve-print_fatal_signals-support-by-adding-memory-maps.patch

-
To unsubscribe from this list: send the line "unsubscribe mm-commits" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Kernel Newbies FAQ]     [Kernel Archive]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [Bugtraq]     [Photo]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]

  Powered by Linux