⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 locking-selftest.c

📁 Lib files of linux kernel
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * lib/locking-selftest.c * * Testsuite for various locking APIs: spinlocks, rwlocks, * mutexes and rw-semaphores. * * It is checking both false positives and false negatives. * * Started by Ingo Molnar: * *  Copyright (C) 2006 Red Hat, Inc., Ingo Molnar <mingo@redhat.com> */#include <linux/rwsem.h>#include <linux/mutex.h>#include <linux/sched.h>#include <linux/delay.h>#include <linux/module.h>#include <linux/lockdep.h>#include <linux/spinlock.h>#include <linux/kallsyms.h>#include <linux/interrupt.h>#include <linux/debug_locks.h>#include <linux/irqflags.h>/* * Change this to 1 if you want to see the failure printouts: */static unsigned int debug_locks_verbose;static int __init setup_debug_locks_verbose(char *str){	get_option(&str, &debug_locks_verbose);	return 1;}__setup("debug_locks_verbose=", setup_debug_locks_verbose);#define FAILURE		0#define SUCCESS		1#define LOCKTYPE_SPIN	0x1#define LOCKTYPE_RWLOCK	0x2#define LOCKTYPE_MUTEX	0x4#define LOCKTYPE_RWSEM	0x8/* * Normal standalone locks, for the circular and irq-context * dependency tests: */static DEFINE_SPINLOCK(lock_A);static DEFINE_SPINLOCK(lock_B);static DEFINE_SPINLOCK(lock_C);static DEFINE_SPINLOCK(lock_D);static DEFINE_RWLOCK(rwlock_A);static DEFINE_RWLOCK(rwlock_B);static DEFINE_RWLOCK(rwlock_C);static DEFINE_RWLOCK(rwlock_D);static DEFINE_MUTEX(mutex_A);static DEFINE_MUTEX(mutex_B);static DEFINE_MUTEX(mutex_C);static DEFINE_MUTEX(mutex_D);static DECLARE_RWSEM(rwsem_A);static DECLARE_RWSEM(rwsem_B);static DECLARE_RWSEM(rwsem_C);static DECLARE_RWSEM(rwsem_D);/* * Locks that we initialize dynamically as well so that * e.g. X1 and X2 becomes two instances of the same class, * but X* and Y* are different classes. We do this so that * we do not trigger a real lockup: */static DEFINE_SPINLOCK(lock_X1);static DEFINE_SPINLOCK(lock_X2);static DEFINE_SPINLOCK(lock_Y1);static DEFINE_SPINLOCK(lock_Y2);static DEFINE_SPINLOCK(lock_Z1);static DEFINE_SPINLOCK(lock_Z2);static DEFINE_RWLOCK(rwlock_X1);static DEFINE_RWLOCK(rwlock_X2);static DEFINE_RWLOCK(rwlock_Y1);static DEFINE_RWLOCK(rwlock_Y2);static DEFINE_RWLOCK(rwlock_Z1);static DEFINE_RWLOCK(rwlock_Z2);static DEFINE_MUTEX(mutex_X1);static DEFINE_MUTEX(mutex_X2);static DEFINE_MUTEX(mutex_Y1);static DEFINE_MUTEX(mutex_Y2);static DEFINE_MUTEX(mutex_Z1);static DEFINE_MUTEX(mutex_Z2);static DECLARE_RWSEM(rwsem_X1);static DECLARE_RWSEM(rwsem_X2);static DECLARE_RWSEM(rwsem_Y1);static DECLARE_RWSEM(rwsem_Y2);static DECLARE_RWSEM(rwsem_Z1);static DECLARE_RWSEM(rwsem_Z2);/* * non-inlined runtime initializers, to let separate locks share * the same lock-class: */#define INIT_CLASS_FUNC(class) 				\static noinline void					\init_class_##class(spinlock_t *lock, rwlock_t *rwlock, struct mutex *mutex, \		 struct rw_semaphore *rwsem)		\{							\	spin_lock_init(lock);				\	rwlock_init(rwlock);				\	mutex_init(mutex);				\	init_rwsem(rwsem);				\}INIT_CLASS_FUNC(X)INIT_CLASS_FUNC(Y)INIT_CLASS_FUNC(Z)static void init_shared_classes(void){	init_class_X(&lock_X1, &rwlock_X1, &mutex_X1, &rwsem_X1);	init_class_X(&lock_X2, &rwlock_X2, &mutex_X2, &rwsem_X2);	init_class_Y(&lock_Y1, &rwlock_Y1, &mutex_Y1, &rwsem_Y1);	init_class_Y(&lock_Y2, &rwlock_Y2, &mutex_Y2, &rwsem_Y2);	init_class_Z(&lock_Z1, &rwlock_Z1, &mutex_Z1, &rwsem_Z1);	init_class_Z(&lock_Z2, &rwlock_Z2, &mutex_Z2, &rwsem_Z2);}/* * For spinlocks and rwlocks we also do hardirq-safe / softirq-safe tests. * The following functions use a lock from a simulated hardirq/softirq * context, causing the locks to be marked as hardirq-safe/softirq-safe: */#define HARDIRQ_DISABLE		local_irq_disable#define HARDIRQ_ENABLE		local_irq_enable#define HARDIRQ_ENTER()				\	local_irq_disable();			\	irq_enter();				\	WARN_ON(!in_irq());#define HARDIRQ_EXIT()				\	__irq_exit();				\	local_irq_enable();#define SOFTIRQ_DISABLE		local_bh_disable#define SOFTIRQ_ENABLE		local_bh_enable#define SOFTIRQ_ENTER()				\		local_bh_disable();		\		local_irq_disable();		\		trace_softirq_enter();		\		WARN_ON(!in_softirq());#define SOFTIRQ_EXIT()				\		trace_softirq_exit();		\		local_irq_enable();		\		local_bh_enable();/* * Shortcuts for lock/unlock API variants, to keep * the testcases compact: */#define L(x)			spin_lock(&lock_##x)#define U(x)			spin_unlock(&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)#define WLU(x)			WL(x); WU(x)#define RL(x)			read_lock(&rwlock_##x)#define RU(x)			read_unlock(&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 MI(x)			mutex_init(&mutex_##x)#define WSL(x)			down_write(&rwsem_##x)#define WSU(x)			up_write(&rwsem_##x)#define RSL(x)			down_read(&rwsem_##x)#define RSU(x)			up_read(&rwsem_##x)#define RWSI(x)			init_rwsem(&rwsem_##x)#define LOCK_UNLOCK_2(x,y)	LOCK(x); LOCK(y); UNLOCK(y); UNLOCK(x)/* * Generate different permutations of the same testcase, using * the same basic lock-dependency/state events: */#define GENERATE_TESTCASE(name)			\						\static void name(void) { E(); }#define GENERATE_PERMUTATIONS_2_EVENTS(name)	\						\static void name##_12(void) { E1(); E2(); }	\static void name##_21(void) { E2(); E1(); }#define GENERATE_PERMUTATIONS_3_EVENTS(name)		\							\static void name##_123(void) { E1(); E2(); E3(); }	\static void name##_132(void) { E1(); E3(); E2(); }	\static void name##_213(void) { E2(); E1(); E3(); }	\static void name##_231(void) { E2(); E3(); E1(); }	\static void name##_312(void) { E3(); E1(); E2(); }	\static void name##_321(void) { E3(); E2(); E1(); }/* * AA deadlock: */#define E()					\						\	LOCK(X1);				\	LOCK(X2); /* this one should fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(AA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(AA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(AA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(AA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(AA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(AA_rsem)#undef E/* * Special-case for read-locking, they are * allowed to recurse on the same lock class: */static void rlock_AA1(void){	RL(X1);	RL(X1); // this one should NOT fail}static void rlock_AA1B(void){	RL(X1);	RL(X2); // this one should NOT fail}static void rsem_AA1(void){	RSL(X1);	RSL(X1); // this one should fail}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}/* * ABBA deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(B, A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABBA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABBA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABBA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABBA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABBA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABBA_rsem)#undef E/* * AB BC CA deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(B, C);			\	LOCK_UNLOCK_2(C, A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABBCCA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABBCCA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABBCCA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABBCCA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABBCCA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABBCCA_rsem)#undef E/* * AB CA BC deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(C, A);			\	LOCK_UNLOCK_2(B, C); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABCABC_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABCABC_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABCABC_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABCABC_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABCABC_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABCABC_rsem)#undef E/* * AB BC CD DA deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(B, C);			\	LOCK_UNLOCK_2(C, D);			\	LOCK_UNLOCK_2(D, A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABBCCDDA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABBCCDDA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABBCCDDA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABBCCDDA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABBCCDDA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABBCCDDA_rsem)#undef E/* * AB CD BD DA deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(C, D);			\	LOCK_UNLOCK_2(B, D);			\	LOCK_UNLOCK_2(D, A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABCDBDDA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABCDBDDA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABCDBDDA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABCDBDDA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABCDBDDA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABCDBDDA_rsem)#undef E/* * AB CD BC DA deadlock: */#define E()					\						\	LOCK_UNLOCK_2(A, B);			\	LOCK_UNLOCK_2(C, D);			\	LOCK_UNLOCK_2(B, C);			\	LOCK_UNLOCK_2(D, A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(ABCDBCDA_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(ABCDBCDA_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(ABCDBCDA_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(ABCDBCDA_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(ABCDBCDA_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(ABCDBCDA_rsem)#undef E/* * Double unlock: */#define E()					\						\	LOCK(A);				\	UNLOCK(A);				\	UNLOCK(A); /* fail *//* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(double_unlock_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(double_unlock_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(double_unlock_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(double_unlock_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(double_unlock_wsem)#include "locking-selftest-rsem.h"GENERATE_TESTCASE(double_unlock_rsem)#undef E/* * Bad unlock ordering: */#define E()					\						\	LOCK(A);				\	LOCK(B);				\	UNLOCK(A); /* fail */			\	UNLOCK(B);/* * 6 testcases: */#include "locking-selftest-spin.h"GENERATE_TESTCASE(bad_unlock_order_spin)#include "locking-selftest-wlock.h"GENERATE_TESTCASE(bad_unlock_order_wlock)#include "locking-selftest-rlock.h"GENERATE_TESTCASE(bad_unlock_order_rlock)#include "locking-selftest-mutex.h"GENERATE_TESTCASE(bad_unlock_order_mutex)#include "locking-selftest-wsem.h"GENERATE_TESTCASE(bad_unlock_order_wsem)#include "locking-selftest-rsem.h"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/* * locking an irq-safe lock with irqs enabled: */#define E1()				\					\	IRQ_ENTER();			\	LOCK(A);			\	UNLOCK(A);			\	IRQ_EXIT();#define E2()				\					\	LOCK(A);			\	UNLOCK(A);/* * Generate 24 testcases: */#include "locking-selftest-spin-hardirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_hard_spin)#include "locking-selftest-rlock-hardirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_hard_rlock)#include "locking-selftest-wlock-hardirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_hard_wlock)#include "locking-selftest-spin-softirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_soft_spin)#include "locking-selftest-rlock-softirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_soft_rlock)#include "locking-selftest-wlock-softirq.h"GENERATE_PERMUTATIONS_2_EVENTS(irqsafe1_soft_wlock)#undef E1#undef E2/* * Enabling hardirqs with a softirq-safe lock held: */#define E1()				\					\	SOFTIRQ_ENTER();		\	LOCK(A);			\	UNLOCK(A);			\	SOFTIRQ_EXIT();#define E2()				\					\	HARDIRQ_DISABLE();		\	LOCK(A);			\	HARDIRQ_ENABLE();		\	UNLOCK(A);/* * Generate 12 testcases: */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -