/* SPDX-License-Identifier: GPL-2.0 */ #ifndef __LINUX_DEBUG_LOCKING_H #define __LINUX_DEBUG_LOCKING_H #include <linux/atomic.h> #include <linux/cache.h> struct task_struct; extern int debug_locks __read_mostly; extern int debug_locks_silent __read_mostly; static __always_inline int __debug_locks_off(void) { return xchg(&debug_locks, 0); } /* * Generic 'turn off all lock debugging' function: */ extern int debug_locks_off(void); #define DEBUG_LOCKS_WARN_ON(c) \ ({ \ int __ret = 0; \ \ if (!oops_in_progress && unlikely(c)) { \ instrumentation_begin(); \ if (debug_locks_off() && !debug_locks_silent) \ WARN(1, "DEBUG_LOCKS_WARN_ON(%s)", #c); \ instrumentation_end(); \ __ret = 1; \ } \ __ret; \ }) #ifdef CONFIG_SMP # define SMP_DEBUG_LOCKS_WARN_ON(c) DEBUG_LOCKS_WARN_ON(c) #else # define SMP_DEBUG_LOCKS_WARN_ON(c) do { } while (0) #endif #ifdef CONFIG_DEBUG_LOCKING_API_SELFTESTS extern void locking_selftest(void); #else # define locking_selftest() do { } while (0) #endif #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(void); #else static inline void debug_show_all_locks(void) { } static inline void debug_show_held_locks(struct task_struct *task) { } static inline void debug_check_no_locks_freed(const void *from, unsigned long len) { } static inline void debug_check_no_locks_held(void) { } #endif #endif