rtmutex-tester.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. /*
  2. * RT-Mutex-tester: scriptable tester for rt mutexes
  3. *
  4. * started by Thomas Gleixner:
  5. *
  6. * Copyright (C) 2006, Timesys Corp., Thomas Gleixner <tglx@timesys.com>
  7. *
  8. */
  9. #include <linux/kthread.h>
  10. #include <linux/module.h>
  11. #include <linux/sched.h>
  12. #include <linux/smp_lock.h>
  13. #include <linux/spinlock.h>
  14. #include <linux/sysdev.h>
  15. #include <linux/timer.h>
  16. #include <linux/freezer.h>
  17. #include "rtmutex.h"
  18. #define MAX_RT_TEST_THREADS 8
  19. #define MAX_RT_TEST_MUTEXES 8
  20. static spinlock_t rttest_lock;
  21. static atomic_t rttest_event;
  22. struct test_thread_data {
  23. int opcode;
  24. int opdata;
  25. int mutexes[MAX_RT_TEST_MUTEXES];
  26. int bkl;
  27. int event;
  28. struct sys_device sysdev;
  29. };
  30. static struct test_thread_data thread_data[MAX_RT_TEST_THREADS];
  31. static struct task_struct *threads[MAX_RT_TEST_THREADS];
  32. static struct rt_mutex mutexes[MAX_RT_TEST_MUTEXES];
  33. enum test_opcodes {
  34. RTTEST_NOP = 0,
  35. RTTEST_SCHEDOT, /* 1 Sched other, data = nice */
  36. RTTEST_SCHEDRT, /* 2 Sched fifo, data = prio */
  37. RTTEST_LOCK, /* 3 Lock uninterruptible, data = lockindex */
  38. RTTEST_LOCKNOWAIT, /* 4 Lock uninterruptible no wait in wakeup, data = lockindex */
  39. RTTEST_LOCKINT, /* 5 Lock interruptible, data = lockindex */
  40. RTTEST_LOCKINTNOWAIT, /* 6 Lock interruptible no wait in wakeup, data = lockindex */
  41. RTTEST_LOCKCONT, /* 7 Continue locking after the wakeup delay */
  42. RTTEST_UNLOCK, /* 8 Unlock, data = lockindex */
  43. RTTEST_LOCKBKL, /* 9 Lock BKL */
  44. RTTEST_UNLOCKBKL, /* 10 Unlock BKL */
  45. RTTEST_SIGNAL, /* 11 Signal other test thread, data = thread id */
  46. RTTEST_RESETEVENT = 98, /* 98 Reset event counter */
  47. RTTEST_RESET = 99, /* 99 Reset all pending operations */
  48. };
  49. static int handle_op(struct test_thread_data *td, int lockwakeup)
  50. {
  51. int i, id, ret = -EINVAL;
  52. switch(td->opcode) {
  53. case RTTEST_NOP:
  54. return 0;
  55. case RTTEST_LOCKCONT:
  56. td->mutexes[td->opdata] = 1;
  57. td->event = atomic_add_return(1, &rttest_event);
  58. return 0;
  59. case RTTEST_RESET:
  60. for (i = 0; i < MAX_RT_TEST_MUTEXES; i++) {
  61. if (td->mutexes[i] == 4) {
  62. rt_mutex_unlock(&mutexes[i]);
  63. td->mutexes[i] = 0;
  64. }
  65. }
  66. if (!lockwakeup && td->bkl == 4) {
  67. #ifdef CONFIG_LOCK_KERNEL
  68. unlock_kernel();
  69. #endif
  70. td->bkl = 0;
  71. }
  72. return 0;
  73. case RTTEST_RESETEVENT:
  74. atomic_set(&rttest_event, 0);
  75. return 0;
  76. default:
  77. if (lockwakeup)
  78. return ret;
  79. }
  80. switch(td->opcode) {
  81. case RTTEST_LOCK:
  82. case RTTEST_LOCKNOWAIT:
  83. id = td->opdata;
  84. if (id < 0 || id >= MAX_RT_TEST_MUTEXES)
  85. return ret;
  86. td->mutexes[id] = 1;
  87. td->event = atomic_add_return(1, &rttest_event);
  88. rt_mutex_lock(&mutexes[id]);
  89. td->event = atomic_add_return(1, &rttest_event);
  90. td->mutexes[id] = 4;
  91. return 0;
  92. case RTTEST_LOCKINT:
  93. case RTTEST_LOCKINTNOWAIT:
  94. id = td->opdata;
  95. if (id < 0 || id >= MAX_RT_TEST_MUTEXES)
  96. return ret;
  97. td->mutexes[id] = 1;
  98. td->event = atomic_add_return(1, &rttest_event);
  99. ret = rt_mutex_lock_interruptible(&mutexes[id], 0);
  100. td->event = atomic_add_return(1, &rttest_event);
  101. td->mutexes[id] = ret ? 0 : 4;
  102. return ret ? -EINTR : 0;
  103. case RTTEST_UNLOCK:
  104. id = td->opdata;
  105. if (id < 0 || id >= MAX_RT_TEST_MUTEXES || td->mutexes[id] != 4)
  106. return ret;
  107. td->event = atomic_add_return(1, &rttest_event);
  108. rt_mutex_unlock(&mutexes[id]);
  109. td->event = atomic_add_return(1, &rttest_event);
  110. td->mutexes[id] = 0;
  111. return 0;
  112. case RTTEST_LOCKBKL:
  113. if (td->bkl)
  114. return 0;
  115. td->bkl = 1;
  116. #ifdef CONFIG_LOCK_KERNEL
  117. lock_kernel();
  118. #endif
  119. td->bkl = 4;
  120. return 0;
  121. case RTTEST_UNLOCKBKL:
  122. if (td->bkl != 4)
  123. break;
  124. #ifdef CONFIG_LOCK_KERNEL
  125. unlock_kernel();
  126. #endif
  127. td->bkl = 0;
  128. return 0;
  129. default:
  130. break;
  131. }
  132. return ret;
  133. }
  134. /*
  135. * Schedule replacement for rtsem_down(). Only called for threads with
  136. * PF_MUTEX_TESTER set.
  137. *
  138. * This allows us to have finegrained control over the event flow.
  139. *
  140. */
  141. void schedule_rt_mutex_test(struct rt_mutex *mutex)
  142. {
  143. int tid, op, dat;
  144. struct test_thread_data *td;
  145. /* We have to lookup the task */
  146. for (tid = 0; tid < MAX_RT_TEST_THREADS; tid++) {
  147. if (threads[tid] == current)
  148. break;
  149. }
  150. BUG_ON(tid == MAX_RT_TEST_THREADS);
  151. td = &thread_data[tid];
  152. op = td->opcode;
  153. dat = td->opdata;
  154. switch (op) {
  155. case RTTEST_LOCK:
  156. case RTTEST_LOCKINT:
  157. case RTTEST_LOCKNOWAIT:
  158. case RTTEST_LOCKINTNOWAIT:
  159. if (mutex != &mutexes[dat])
  160. break;
  161. if (td->mutexes[dat] != 1)
  162. break;
  163. td->mutexes[dat] = 2;
  164. td->event = atomic_add_return(1, &rttest_event);
  165. break;
  166. case RTTEST_LOCKBKL:
  167. default:
  168. break;
  169. }
  170. schedule();
  171. switch (op) {
  172. case RTTEST_LOCK:
  173. case RTTEST_LOCKINT:
  174. if (mutex != &mutexes[dat])
  175. return;
  176. if (td->mutexes[dat] != 2)
  177. return;
  178. td->mutexes[dat] = 3;
  179. td->event = atomic_add_return(1, &rttest_event);
  180. break;
  181. case RTTEST_LOCKNOWAIT:
  182. case RTTEST_LOCKINTNOWAIT:
  183. if (mutex != &mutexes[dat])
  184. return;
  185. if (td->mutexes[dat] != 2)
  186. return;
  187. td->mutexes[dat] = 1;
  188. td->event = atomic_add_return(1, &rttest_event);
  189. return;
  190. case RTTEST_LOCKBKL:
  191. return;
  192. default:
  193. return;
  194. }
  195. td->opcode = 0;
  196. for (;;) {
  197. set_current_state(TASK_INTERRUPTIBLE);
  198. if (td->opcode > 0) {
  199. int ret;
  200. set_current_state(TASK_RUNNING);
  201. ret = handle_op(td, 1);
  202. set_current_state(TASK_INTERRUPTIBLE);
  203. if (td->opcode == RTTEST_LOCKCONT)
  204. break;
  205. td->opcode = ret;
  206. }
  207. /* Wait for the next command to be executed */
  208. schedule();
  209. }
  210. /* Restore previous command and data */
  211. td->opcode = op;
  212. td->opdata = dat;
  213. }
  214. static int test_func(void *data)
  215. {
  216. struct test_thread_data *td = data;
  217. int ret;
  218. current->flags |= PF_MUTEX_TESTER;
  219. set_freezable();
  220. allow_signal(SIGHUP);
  221. for(;;) {
  222. set_current_state(TASK_INTERRUPTIBLE);
  223. if (td->opcode > 0) {
  224. set_current_state(TASK_RUNNING);
  225. ret = handle_op(td, 0);
  226. set_current_state(TASK_INTERRUPTIBLE);
  227. td->opcode = ret;
  228. }
  229. /* Wait for the next command to be executed */
  230. schedule();
  231. try_to_freeze();
  232. if (signal_pending(current))
  233. flush_signals(current);
  234. if(kthread_should_stop())
  235. break;
  236. }
  237. return 0;
  238. }
  239. /**
  240. * sysfs_test_command - interface for test commands
  241. * @dev: thread reference
  242. * @buf: command for actual step
  243. * @count: length of buffer
  244. *
  245. * command syntax:
  246. *
  247. * opcode:data
  248. */
  249. static ssize_t sysfs_test_command(struct sys_device *dev, struct sysdev_attribute *attr,
  250. const char *buf, size_t count)
  251. {
  252. struct sched_param schedpar;
  253. struct test_thread_data *td;
  254. char cmdbuf[32];
  255. int op, dat, tid, ret;
  256. td = container_of(dev, struct test_thread_data, sysdev);
  257. tid = td->sysdev.id;
  258. /* strings from sysfs write are not 0 terminated! */
  259. if (count >= sizeof(cmdbuf))
  260. return -EINVAL;
  261. /* strip of \n: */
  262. if (buf[count-1] == '\n')
  263. count--;
  264. if (count < 1)
  265. return -EINVAL;
  266. memcpy(cmdbuf, buf, count);
  267. cmdbuf[count] = 0;
  268. if (sscanf(cmdbuf, "%d:%d", &op, &dat) != 2)
  269. return -EINVAL;
  270. switch (op) {
  271. case RTTEST_SCHEDOT:
  272. schedpar.sched_priority = 0;
  273. ret = sched_setscheduler(threads[tid], SCHED_NORMAL, &schedpar);
  274. if (ret)
  275. return ret;
  276. set_user_nice(current, 0);
  277. break;
  278. case RTTEST_SCHEDRT:
  279. schedpar.sched_priority = dat;
  280. ret = sched_setscheduler(threads[tid], SCHED_FIFO, &schedpar);
  281. if (ret)
  282. return ret;
  283. break;
  284. case RTTEST_SIGNAL:
  285. send_sig(SIGHUP, threads[tid], 0);
  286. break;
  287. default:
  288. if (td->opcode > 0)
  289. return -EBUSY;
  290. td->opdata = dat;
  291. td->opcode = op;
  292. wake_up_process(threads[tid]);
  293. }
  294. return count;
  295. }
  296. /**
  297. * sysfs_test_status - sysfs interface for rt tester
  298. * @dev: thread to query
  299. * @buf: char buffer to be filled with thread status info
  300. */
  301. static ssize_t sysfs_test_status(struct sys_device *dev, struct sysdev_attribute *attr,
  302. char *buf)
  303. {
  304. struct test_thread_data *td;
  305. struct task_struct *tsk;
  306. char *curr = buf;
  307. int i;
  308. td = container_of(dev, struct test_thread_data, sysdev);
  309. tsk = threads[td->sysdev.id];
  310. spin_lock(&rttest_lock);
  311. curr += sprintf(curr,
  312. "O: %4d, E:%8d, S: 0x%08lx, P: %4d, N: %4d, B: %p, K: %d, M:",
  313. td->opcode, td->event, tsk->state,
  314. (MAX_RT_PRIO - 1) - tsk->prio,
  315. (MAX_RT_PRIO - 1) - tsk->normal_prio,
  316. tsk->pi_blocked_on, td->bkl);
  317. for (i = MAX_RT_TEST_MUTEXES - 1; i >=0 ; i--)
  318. curr += sprintf(curr, "%d", td->mutexes[i]);
  319. spin_unlock(&rttest_lock);
  320. curr += sprintf(curr, ", T: %p, R: %p\n", tsk,
  321. mutexes[td->sysdev.id].owner);
  322. return curr - buf;
  323. }
  324. static SYSDEV_ATTR(status, 0600, sysfs_test_status, NULL);
  325. static SYSDEV_ATTR(command, 0600, NULL, sysfs_test_command);
  326. static struct sysdev_class rttest_sysclass = {
  327. .name = "rttest",
  328. };
  329. static int init_test_thread(int id)
  330. {
  331. thread_data[id].sysdev.cls = &rttest_sysclass;
  332. thread_data[id].sysdev.id = id;
  333. threads[id] = kthread_run(test_func, &thread_data[id], "rt-test-%d", id);
  334. if (IS_ERR(threads[id]))
  335. return PTR_ERR(threads[id]);
  336. return sysdev_register(&thread_data[id].sysdev);
  337. }
  338. static int init_rttest(void)
  339. {
  340. int ret, i;
  341. spin_lock_init(&rttest_lock);
  342. for (i = 0; i < MAX_RT_TEST_MUTEXES; i++)
  343. rt_mutex_init(&mutexes[i]);
  344. ret = sysdev_class_register(&rttest_sysclass);
  345. if (ret)
  346. return ret;
  347. for (i = 0; i < MAX_RT_TEST_THREADS; i++) {
  348. ret = init_test_thread(i);
  349. if (ret)
  350. break;
  351. ret = sysdev_create_file(&thread_data[i].sysdev, &attr_status);
  352. if (ret)
  353. break;
  354. ret = sysdev_create_file(&thread_data[i].sysdev, &attr_command);
  355. if (ret)
  356. break;
  357. }
  358. printk("Initializing RT-Tester: %s\n", ret ? "Failed" : "OK" );
  359. return ret;
  360. }
  361. device_initcall(init_rttest);