Blame view

kernel/mutex-debug.c 3.04 KB
408894ee4   Ingo Molnar   [PATCH] mutex sub...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
  /*
   * kernel/mutex-debug.c
   *
   * Debugging code for mutexes
   *
   * Started by Ingo Molnar:
   *
   *  Copyright (C) 2004, 2005, 2006 Red Hat, Inc., Ingo Molnar <mingo@redhat.com>
   *
   * lock debugging, locking tree, deadlock detection started by:
   *
   *  Copyright (C) 2004, LynuxWorks, Inc., Igor Manyilov, Bill Huey
   *  Released under the General Public License (GPL).
   */
  #include <linux/mutex.h>
408894ee4   Ingo Molnar   [PATCH] mutex sub...
16
17
  #include <linux/delay.h>
  #include <linux/module.h>
a7807a32b   Randy Dunlap   [PATCH] poison: a...
18
  #include <linux/poison.h>
408894ee4   Ingo Molnar   [PATCH] mutex sub...
19
20
21
  #include <linux/spinlock.h>
  #include <linux/kallsyms.h>
  #include <linux/interrupt.h>
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
22
  #include <linux/debug_locks.h>
408894ee4   Ingo Molnar   [PATCH] mutex sub...
23

408894ee4   Ingo Molnar   [PATCH] mutex sub...
24
25
26
  #include "mutex-debug.h"
  
  /*
408894ee4   Ingo Molnar   [PATCH] mutex sub...
27
28
   * Must be called with lock->wait_lock held.
   */
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
29
  void debug_mutex_set_owner(struct mutex *lock, struct thread_info *new_owner)
408894ee4   Ingo Molnar   [PATCH] mutex sub...
30
31
  {
  	lock->owner = new_owner;
408894ee4   Ingo Molnar   [PATCH] mutex sub...
32
  }
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
33
  void debug_mutex_lock_common(struct mutex *lock, struct mutex_waiter *waiter)
408894ee4   Ingo Molnar   [PATCH] mutex sub...
34
  {
a7807a32b   Randy Dunlap   [PATCH] poison: a...
35
  	memset(waiter, MUTEX_DEBUG_INIT, sizeof(*waiter));
408894ee4   Ingo Molnar   [PATCH] mutex sub...
36
37
38
39
40
41
  	waiter->magic = waiter;
  	INIT_LIST_HEAD(&waiter->list);
  }
  
  void debug_mutex_wake_waiter(struct mutex *lock, struct mutex_waiter *waiter)
  {
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
42
43
44
45
  	SMP_DEBUG_LOCKS_WARN_ON(!spin_is_locked(&lock->wait_lock));
  	DEBUG_LOCKS_WARN_ON(list_empty(&lock->wait_list));
  	DEBUG_LOCKS_WARN_ON(waiter->magic != waiter);
  	DEBUG_LOCKS_WARN_ON(list_empty(&waiter->list));
408894ee4   Ingo Molnar   [PATCH] mutex sub...
46
47
48
49
  }
  
  void debug_mutex_free_waiter(struct mutex_waiter *waiter)
  {
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
50
  	DEBUG_LOCKS_WARN_ON(!list_empty(&waiter->list));
a7807a32b   Randy Dunlap   [PATCH] poison: a...
51
  	memset(waiter, MUTEX_DEBUG_FREE, sizeof(*waiter));
408894ee4   Ingo Molnar   [PATCH] mutex sub...
52
53
54
  }
  
  void debug_mutex_add_waiter(struct mutex *lock, struct mutex_waiter *waiter,
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
55
  			    struct thread_info *ti)
408894ee4   Ingo Molnar   [PATCH] mutex sub...
56
  {
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
57
  	SMP_DEBUG_LOCKS_WARN_ON(!spin_is_locked(&lock->wait_lock));
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
58

408894ee4   Ingo Molnar   [PATCH] mutex sub...
59
60
61
62
63
64
65
66
  	/* Mark the current thread as blocked on the lock: */
  	ti->task->blocked_on = waiter;
  	waiter->lock = lock;
  }
  
  void mutex_remove_waiter(struct mutex *lock, struct mutex_waiter *waiter,
  			 struct thread_info *ti)
  {
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
67
68
69
  	DEBUG_LOCKS_WARN_ON(list_empty(&waiter->list));
  	DEBUG_LOCKS_WARN_ON(waiter->task != ti->task);
  	DEBUG_LOCKS_WARN_ON(ti->task->blocked_on != waiter);
408894ee4   Ingo Molnar   [PATCH] mutex sub...
70
71
72
73
74
75
76
77
  	ti->task->blocked_on = NULL;
  
  	list_del_init(&waiter->list);
  	waiter->task = NULL;
  }
  
  void debug_mutex_unlock(struct mutex *lock)
  {
2ee91f197   Ingo Molnar   [PATCH] lockdep: ...
78
79
  	if (unlikely(!debug_locks))
  		return;
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
80
  	DEBUG_LOCKS_WARN_ON(lock->magic != lock);
493d35863   Jeremy Kerr   mutex-debug: chec...
81
  	DEBUG_LOCKS_WARN_ON(lock->owner != current_thread_info());
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
82
83
  	DEBUG_LOCKS_WARN_ON(!lock->wait_list.prev && !lock->wait_list.next);
  	DEBUG_LOCKS_WARN_ON(lock->owner != current_thread_info());
408894ee4   Ingo Molnar   [PATCH] mutex sub...
84
  }
ef5d4707b   Ingo Molnar   [PATCH] lockdep: ...
85
86
  void debug_mutex_init(struct mutex *lock, const char *name,
  		      struct lock_class_key *key)
408894ee4   Ingo Molnar   [PATCH] mutex sub...
87
  {
ef5d4707b   Ingo Molnar   [PATCH] lockdep: ...
88
  #ifdef CONFIG_DEBUG_LOCK_ALLOC
408894ee4   Ingo Molnar   [PATCH] mutex sub...
89
90
91
  	/*
  	 * Make sure we are not reinitializing a held lock:
  	 */
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
92
  	debug_check_no_locks_freed((void *)lock, sizeof(*lock));
4dfbb9d8c   Peter Zijlstra   Lockdep: add lock...
93
  	lockdep_init_map(&lock->dep_map, name, key, 0);
ef5d4707b   Ingo Molnar   [PATCH] lockdep: ...
94
  #endif
408894ee4   Ingo Molnar   [PATCH] mutex sub...
95
  	lock->owner = NULL;
408894ee4   Ingo Molnar   [PATCH] mutex sub...
96
97
98
99
100
101
102
103
104
105
106
  	lock->magic = lock;
  }
  
  /***
   * mutex_destroy - mark a mutex unusable
   * @lock: the mutex to be destroyed
   *
   * This function marks the mutex uninitialized, and any subsequent
   * use of the mutex is forbidden. The mutex must not be locked when
   * this function is called.
   */
7ad5b3a50   Harvey Harrison   kernel: remove fa...
107
  void mutex_destroy(struct mutex *lock)
408894ee4   Ingo Molnar   [PATCH] mutex sub...
108
  {
9e7f4d451   Ingo Molnar   [PATCH] lockdep: ...
109
  	DEBUG_LOCKS_WARN_ON(mutex_is_locked(lock));
408894ee4   Ingo Molnar   [PATCH] mutex sub...
110
111
112
113
  	lock->magic = NULL;
  }
  
  EXPORT_SYMBOL_GPL(mutex_destroy);