xref: /linux/include/linux/debug_locks.h (revision 36110669ddf832e6c9ceba4dd203749d5be31d31)
1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef __LINUX_DEBUG_LOCKING_H
3 #define __LINUX_DEBUG_LOCKING_H
4 
5 #include <linux/atomic.h>
6 #include <linux/cache.h>
7 
8 struct task_struct;
9 
10 extern int debug_locks __read_mostly;
11 extern int debug_locks_silent __read_mostly;
12 
13 
14 static __always_inline int __debug_locks_off(void)
15 {
16 	return xchg(&debug_locks, 0);
17 }
18 
19 /*
20  * Generic 'turn off all lock debugging' function:
21  */
22 extern int debug_locks_off(void);
23 
24 #define DEBUG_LOCKS_WARN_ON(c)						\
25 ({									\
26 	int __ret = 0;							\
27 									\
28 	if (!oops_in_progress && unlikely(c)) {				\
29 		instrumentation_begin();				\
30 		if (debug_locks_off() && !debug_locks_silent)		\
31 			WARN(1, "DEBUG_LOCKS_WARN_ON(%s)", #c);		\
32 		instrumentation_end();					\
33 		__ret = 1;						\
34 	}								\
35 	__ret;								\
36 })
37 
38 #ifdef CONFIG_SMP
39 # define SMP_DEBUG_LOCKS_WARN_ON(c)			DEBUG_LOCKS_WARN_ON(c)
40 #else
41 # define SMP_DEBUG_LOCKS_WARN_ON(c)			do { } while (0)
42 #endif
43 
44 #ifdef CONFIG_DEBUG_LOCKING_API_SELFTESTS
45   extern void locking_selftest(void);
46 #else
47 # define locking_selftest()	do { } while (0)
48 #endif
49 
50 #ifdef CONFIG_LOCKDEP
51 extern void debug_show_all_locks(void);
52 extern void debug_show_held_locks(struct task_struct *task);
53 extern void debug_check_no_locks_freed(const void *from, unsigned long len);
54 extern void debug_check_no_locks_held(void);
55 #else
56 static inline void debug_show_all_locks(void)
57 {
58 }
59 
60 static inline void debug_show_held_locks(struct task_struct *task)
61 {
62 }
63 
64 static inline void
65 debug_check_no_locks_freed(const void *from, unsigned long len)
66 {
67 }
68 
69 static inline void
70 debug_check_no_locks_held(void)
71 {
72 }
73 #endif
74 
75 #endif
76