diff options
author | Yong Zhang <yong.zhang0@gmail.com> | 2011-11-09 03:04:51 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2011-12-06 02:16:51 -0500 |
commit | d3d03d4fc5b1bec3a579112de170a9676e9d97cb (patch) | |
tree | f8ab00599868ffe3094edea1e940750681587835 /kernel/lockdep.c | |
parent | fbdc4b9a6c29befbcca65e5366e5aaf2abb7a013 (diff) |
lockdep, kmemcheck: Annotate ->lock in lockdep_init_map()
Since commit f59de89 ("lockdep: Clear whole lockdep_map on initialization"),
lockdep_init_map() will clear all the struct. But it will break
lock_set_class()/lock_set_subclass(). A typical race condition
is like below:
CPU A CPU B
lock_set_subclass(lockA);
lock_set_class(lockA);
lockdep_init_map(lockA);
/* lockA->name is cleared */
memset(lockA);
__lock_acquire(lockA);
/* lockA->class_cache[] is cleared */
register_lock_class(lockA);
look_up_lock_class(lockA);
WARN_ON_ONCE(class->name !=
lock->name);
lock->name = name;
So restore to what we have done before commit f59de89 but annotate
->lock with kmemcheck_mark_initialized() to suppress the kmemcheck
warning reported in commit f59de89.
Reported-by: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Reported-by: Borislav Petkov <bp@alien8.de>
Suggested-by: Vegard Nossum <vegard.nossum@gmail.com>
Signed-off-by: Yong Zhang <yong.zhang0@gmail.com>
Cc: Tejun Heo <tj@kernel.org>
Cc: David Rientjes <rientjes@google.com>
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/20111109080451.GB8124@zhy
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'kernel/lockdep.c')
-rw-r--r-- | kernel/lockdep.c | 8 |
1 files changed, 7 insertions, 1 deletions
diff --git a/kernel/lockdep.c b/kernel/lockdep.c index a2ab30c12af9..54cc0dc7b303 100644 --- a/kernel/lockdep.c +++ b/kernel/lockdep.c | |||
@@ -44,6 +44,7 @@ | |||
44 | #include <linux/stringify.h> | 44 | #include <linux/stringify.h> |
45 | #include <linux/bitops.h> | 45 | #include <linux/bitops.h> |
46 | #include <linux/gfp.h> | 46 | #include <linux/gfp.h> |
47 | #include <linux/kmemcheck.h> | ||
47 | 48 | ||
48 | #include <asm/sections.h> | 49 | #include <asm/sections.h> |
49 | 50 | ||
@@ -2950,7 +2951,12 @@ static int mark_lock(struct task_struct *curr, struct held_lock *this, | |||
2950 | void lockdep_init_map(struct lockdep_map *lock, const char *name, | 2951 | void lockdep_init_map(struct lockdep_map *lock, const char *name, |
2951 | struct lock_class_key *key, int subclass) | 2952 | struct lock_class_key *key, int subclass) |
2952 | { | 2953 | { |
2953 | memset(lock, 0, sizeof(*lock)); | 2954 | int i; |
2955 | |||
2956 | kmemcheck_mark_initialized(lock, sizeof(*lock)); | ||
2957 | |||
2958 | for (i = 0; i < NR_LOCKDEP_CACHING_CLASSES; i++) | ||
2959 | lock->class_cache[i] = NULL; | ||
2954 | 2960 | ||
2955 | #ifdef CONFIG_LOCK_STAT | 2961 | #ifdef CONFIG_LOCK_STAT |
2956 | lock->cpu = raw_smp_processor_id(); | 2962 | lock->cpu = raw_smp_processor_id(); |