diff options
Diffstat (limited to 'lib/locking-selftest.c')
-rw-r--r-- | lib/locking-selftest.c | 720 |
1 files changed, 702 insertions, 18 deletions
diff --git a/lib/locking-selftest.c b/lib/locking-selftest.c index c3eb261a7df3..aad024dde3c4 100644 --- a/lib/locking-selftest.c +++ b/lib/locking-selftest.c | |||
@@ -26,6 +26,8 @@ | |||
26 | */ | 26 | */ |
27 | static unsigned int debug_locks_verbose; | 27 | static unsigned int debug_locks_verbose; |
28 | 28 | ||
29 | static DEFINE_WW_CLASS(ww_lockdep); | ||
30 | |||
29 | static int __init setup_debug_locks_verbose(char *str) | 31 | static int __init setup_debug_locks_verbose(char *str) |
30 | { | 32 | { |
31 | get_option(&str, &debug_locks_verbose); | 33 | get_option(&str, &debug_locks_verbose); |
@@ -42,6 +44,10 @@ __setup("debug_locks_verbose=", setup_debug_locks_verbose); | |||
42 | #define LOCKTYPE_RWLOCK 0x2 | 44 | #define LOCKTYPE_RWLOCK 0x2 |
43 | #define LOCKTYPE_MUTEX 0x4 | 45 | #define LOCKTYPE_MUTEX 0x4 |
44 | #define LOCKTYPE_RWSEM 0x8 | 46 | #define LOCKTYPE_RWSEM 0x8 |
47 | #define LOCKTYPE_WW 0x10 | ||
48 | |||
49 | static struct ww_acquire_ctx t, t2; | ||
50 | static struct ww_mutex o, o2, o3; | ||
45 | 51 | ||
46 | /* | 52 | /* |
47 | * Normal standalone locks, for the circular and irq-context | 53 | * Normal standalone locks, for the circular and irq-context |
@@ -193,6 +199,20 @@ static void init_shared_classes(void) | |||
193 | #define RSU(x) up_read(&rwsem_##x) | 199 | #define RSU(x) up_read(&rwsem_##x) |
194 | #define RWSI(x) init_rwsem(&rwsem_##x) | 200 | #define RWSI(x) init_rwsem(&rwsem_##x) |
195 | 201 | ||
202 | #ifndef CONFIG_DEBUG_WW_MUTEX_SLOWPATH | ||
203 | #define WWAI(x) ww_acquire_init(x, &ww_lockdep) | ||
204 | #else | ||
205 | #define WWAI(x) do { ww_acquire_init(x, &ww_lockdep); (x)->deadlock_inject_countdown = ~0U; } while (0) | ||
206 | #endif | ||
207 | #define WWAD(x) ww_acquire_done(x) | ||
208 | #define WWAF(x) ww_acquire_fini(x) | ||
209 | |||
210 | #define WWL(x, c) ww_mutex_lock(x, c) | ||
211 | #define WWT(x) ww_mutex_trylock(x) | ||
212 | #define WWL1(x) ww_mutex_lock(x, NULL) | ||
213 | #define WWU(x) ww_mutex_unlock(x) | ||
214 | |||
215 | |||
196 | #define LOCK_UNLOCK_2(x,y) LOCK(x); LOCK(y); UNLOCK(y); UNLOCK(x) | 216 | #define LOCK_UNLOCK_2(x,y) LOCK(x); LOCK(y); UNLOCK(y); UNLOCK(x) |
197 | 217 | ||
198 | /* | 218 | /* |
@@ -894,11 +914,13 @@ GENERATE_PERMUTATIONS_3_EVENTS(irq_read_recursion_soft) | |||
894 | # define I_RWLOCK(x) lockdep_reset_lock(&rwlock_##x.dep_map) | 914 | # define I_RWLOCK(x) lockdep_reset_lock(&rwlock_##x.dep_map) |
895 | # define I_MUTEX(x) lockdep_reset_lock(&mutex_##x.dep_map) | 915 | # define I_MUTEX(x) lockdep_reset_lock(&mutex_##x.dep_map) |
896 | # define I_RWSEM(x) lockdep_reset_lock(&rwsem_##x.dep_map) | 916 | # define I_RWSEM(x) lockdep_reset_lock(&rwsem_##x.dep_map) |
917 | # define I_WW(x) lockdep_reset_lock(&x.dep_map) | ||
897 | #else | 918 | #else |
898 | # define I_SPINLOCK(x) | 919 | # define I_SPINLOCK(x) |
899 | # define I_RWLOCK(x) | 920 | # define I_RWLOCK(x) |
900 | # define I_MUTEX(x) | 921 | # define I_MUTEX(x) |
901 | # define I_RWSEM(x) | 922 | # define I_RWSEM(x) |
923 | # define I_WW(x) | ||
902 | #endif | 924 | #endif |
903 | 925 | ||
904 | #define I1(x) \ | 926 | #define I1(x) \ |
@@ -920,11 +942,20 @@ GENERATE_PERMUTATIONS_3_EVENTS(irq_read_recursion_soft) | |||
920 | static void reset_locks(void) | 942 | static void reset_locks(void) |
921 | { | 943 | { |
922 | local_irq_disable(); | 944 | local_irq_disable(); |
945 | lockdep_free_key_range(&ww_lockdep.acquire_key, 1); | ||
946 | lockdep_free_key_range(&ww_lockdep.mutex_key, 1); | ||
947 | |||
923 | I1(A); I1(B); I1(C); I1(D); | 948 | I1(A); I1(B); I1(C); I1(D); |
924 | I1(X1); I1(X2); I1(Y1); I1(Y2); I1(Z1); I1(Z2); | 949 | I1(X1); I1(X2); I1(Y1); I1(Y2); I1(Z1); I1(Z2); |
950 | I_WW(t); I_WW(t2); I_WW(o.base); I_WW(o2.base); I_WW(o3.base); | ||
925 | lockdep_reset(); | 951 | lockdep_reset(); |
926 | I2(A); I2(B); I2(C); I2(D); | 952 | I2(A); I2(B); I2(C); I2(D); |
927 | init_shared_classes(); | 953 | init_shared_classes(); |
954 | |||
955 | ww_mutex_init(&o, &ww_lockdep); ww_mutex_init(&o2, &ww_lockdep); ww_mutex_init(&o3, &ww_lockdep); | ||
956 | memset(&t, 0, sizeof(t)); memset(&t2, 0, sizeof(t2)); | ||
957 | memset(&ww_lockdep.acquire_key, 0, sizeof(ww_lockdep.acquire_key)); | ||
958 | memset(&ww_lockdep.mutex_key, 0, sizeof(ww_lockdep.mutex_key)); | ||
928 | local_irq_enable(); | 959 | local_irq_enable(); |
929 | } | 960 | } |
930 | 961 | ||
@@ -938,7 +969,6 @@ static int unexpected_testcase_failures; | |||
938 | static void dotest(void (*testcase_fn)(void), int expected, int lockclass_mask) | 969 | static void dotest(void (*testcase_fn)(void), int expected, int lockclass_mask) |
939 | { | 970 | { |
940 | unsigned long saved_preempt_count = preempt_count(); | 971 | unsigned long saved_preempt_count = preempt_count(); |
941 | int expected_failure = 0; | ||
942 | 972 | ||
943 | WARN_ON(irqs_disabled()); | 973 | WARN_ON(irqs_disabled()); |
944 | 974 | ||
@@ -947,25 +977,17 @@ static void dotest(void (*testcase_fn)(void), int expected, int lockclass_mask) | |||
947 | * Filter out expected failures: | 977 | * Filter out expected failures: |
948 | */ | 978 | */ |
949 | #ifndef CONFIG_PROVE_LOCKING | 979 | #ifndef CONFIG_PROVE_LOCKING |
950 | if ((lockclass_mask & LOCKTYPE_SPIN) && debug_locks != expected) | 980 | if (expected == FAILURE && debug_locks) { |
951 | expected_failure = 1; | 981 | expected_testcase_failures++; |
952 | if ((lockclass_mask & LOCKTYPE_RWLOCK) && debug_locks != expected) | 982 | printk("failed|"); |
953 | expected_failure = 1; | 983 | } |
954 | if ((lockclass_mask & LOCKTYPE_MUTEX) && debug_locks != expected) | 984 | else |
955 | expected_failure = 1; | ||
956 | if ((lockclass_mask & LOCKTYPE_RWSEM) && debug_locks != expected) | ||
957 | expected_failure = 1; | ||
958 | #endif | 985 | #endif |
959 | if (debug_locks != expected) { | 986 | if (debug_locks != expected) { |
960 | if (expected_failure) { | 987 | unexpected_testcase_failures++; |
961 | expected_testcase_failures++; | 988 | printk("FAILED|"); |
962 | printk("failed|"); | 989 | |
963 | } else { | 990 | dump_stack(); |
964 | unexpected_testcase_failures++; | ||
965 | |||
966 | printk("FAILED|"); | ||
967 | dump_stack(); | ||
968 | } | ||
969 | } else { | 991 | } else { |
970 | testcase_successes++; | 992 | testcase_successes++; |
971 | printk(" ok |"); | 993 | printk(" ok |"); |
@@ -1108,6 +1130,666 @@ static inline void print_testname(const char *testname) | |||
1108 | DO_TESTCASE_6IRW(desc, name, 312); \ | 1130 | DO_TESTCASE_6IRW(desc, name, 312); \ |
1109 | DO_TESTCASE_6IRW(desc, name, 321); | 1131 | DO_TESTCASE_6IRW(desc, name, 321); |
1110 | 1132 | ||
1133 | static void ww_test_fail_acquire(void) | ||
1134 | { | ||
1135 | int ret; | ||
1136 | |||
1137 | WWAI(&t); | ||
1138 | t.stamp++; | ||
1139 | |||
1140 | ret = WWL(&o, &t); | ||
1141 | |||
1142 | if (WARN_ON(!o.ctx) || | ||
1143 | WARN_ON(ret)) | ||
1144 | return; | ||
1145 | |||
1146 | /* No lockdep test, pure API */ | ||
1147 | ret = WWL(&o, &t); | ||
1148 | WARN_ON(ret != -EALREADY); | ||
1149 | |||
1150 | ret = WWT(&o); | ||
1151 | WARN_ON(ret); | ||
1152 | |||
1153 | t2 = t; | ||
1154 | t2.stamp++; | ||
1155 | ret = WWL(&o, &t2); | ||
1156 | WARN_ON(ret != -EDEADLK); | ||
1157 | WWU(&o); | ||
1158 | |||
1159 | if (WWT(&o)) | ||
1160 | WWU(&o); | ||
1161 | #ifdef CONFIG_DEBUG_LOCK_ALLOC | ||
1162 | else | ||
1163 | DEBUG_LOCKS_WARN_ON(1); | ||
1164 | #endif | ||
1165 | } | ||
1166 | |||
1167 | static void ww_test_normal(void) | ||
1168 | { | ||
1169 | int ret; | ||
1170 | |||
1171 | WWAI(&t); | ||
1172 | |||
1173 | /* | ||
1174 | * None of the ww_mutex codepaths should be taken in the 'normal' | ||
1175 | * mutex calls. The easiest way to verify this is by using the | ||
1176 | * normal mutex calls, and making sure o.ctx is unmodified. | ||
1177 | */ | ||
1178 | |||
1179 | /* mutex_lock (and indirectly, mutex_lock_nested) */ | ||
1180 | o.ctx = (void *)~0UL; | ||
1181 | mutex_lock(&o.base); | ||
1182 | mutex_unlock(&o.base); | ||
1183 | WARN_ON(o.ctx != (void *)~0UL); | ||
1184 | |||
1185 | /* mutex_lock_interruptible (and *_nested) */ | ||
1186 | o.ctx = (void *)~0UL; | ||
1187 | ret = mutex_lock_interruptible(&o.base); | ||
1188 | if (!ret) | ||
1189 | mutex_unlock(&o.base); | ||
1190 | else | ||
1191 | WARN_ON(1); | ||
1192 | WARN_ON(o.ctx != (void *)~0UL); | ||
1193 | |||
1194 | /* mutex_lock_killable (and *_nested) */ | ||
1195 | o.ctx = (void *)~0UL; | ||
1196 | ret = mutex_lock_killable(&o.base); | ||
1197 | if (!ret) | ||
1198 | mutex_unlock(&o.base); | ||
1199 | else | ||
1200 | WARN_ON(1); | ||
1201 | WARN_ON(o.ctx != (void *)~0UL); | ||
1202 | |||
1203 | /* trylock, succeeding */ | ||
1204 | o.ctx = (void *)~0UL; | ||
1205 | ret = mutex_trylock(&o.base); | ||
1206 | WARN_ON(!ret); | ||
1207 | if (ret) | ||
1208 | mutex_unlock(&o.base); | ||
1209 | else | ||
1210 | WARN_ON(1); | ||
1211 | WARN_ON(o.ctx != (void *)~0UL); | ||
1212 | |||
1213 | /* trylock, failing */ | ||
1214 | o.ctx = (void *)~0UL; | ||
1215 | mutex_lock(&o.base); | ||
1216 | ret = mutex_trylock(&o.base); | ||
1217 | WARN_ON(ret); | ||
1218 | mutex_unlock(&o.base); | ||
1219 | WARN_ON(o.ctx != (void *)~0UL); | ||
1220 | |||
1221 | /* nest_lock */ | ||
1222 | o.ctx = (void *)~0UL; | ||
1223 | mutex_lock_nest_lock(&o.base, &t); | ||
1224 | mutex_unlock(&o.base); | ||
1225 | WARN_ON(o.ctx != (void *)~0UL); | ||
1226 | } | ||
1227 | |||
1228 | static void ww_test_two_contexts(void) | ||
1229 | { | ||
1230 | WWAI(&t); | ||
1231 | WWAI(&t2); | ||
1232 | } | ||
1233 | |||
1234 | static void ww_test_diff_class(void) | ||
1235 | { | ||
1236 | WWAI(&t); | ||
1237 | #ifdef CONFIG_DEBUG_MUTEXES | ||
1238 | t.ww_class = NULL; | ||
1239 | #endif | ||
1240 | WWL(&o, &t); | ||
1241 | } | ||
1242 | |||
1243 | static void ww_test_context_done_twice(void) | ||
1244 | { | ||
1245 | WWAI(&t); | ||
1246 | WWAD(&t); | ||
1247 | WWAD(&t); | ||
1248 | WWAF(&t); | ||
1249 | } | ||
1250 | |||
1251 | static void ww_test_context_unlock_twice(void) | ||
1252 | { | ||
1253 | WWAI(&t); | ||
1254 | WWAD(&t); | ||
1255 | WWAF(&t); | ||
1256 | WWAF(&t); | ||
1257 | } | ||
1258 | |||
1259 | static void ww_test_context_fini_early(void) | ||
1260 | { | ||
1261 | WWAI(&t); | ||
1262 | WWL(&o, &t); | ||
1263 | WWAD(&t); | ||
1264 | WWAF(&t); | ||
1265 | } | ||
1266 | |||
1267 | static void ww_test_context_lock_after_done(void) | ||
1268 | { | ||
1269 | WWAI(&t); | ||
1270 | WWAD(&t); | ||
1271 | WWL(&o, &t); | ||
1272 | } | ||
1273 | |||
1274 | static void ww_test_object_unlock_twice(void) | ||
1275 | { | ||
1276 | WWL1(&o); | ||
1277 | WWU(&o); | ||
1278 | WWU(&o); | ||
1279 | } | ||
1280 | |||
1281 | static void ww_test_object_lock_unbalanced(void) | ||
1282 | { | ||
1283 | WWAI(&t); | ||
1284 | WWL(&o, &t); | ||
1285 | t.acquired = 0; | ||
1286 | WWU(&o); | ||
1287 | WWAF(&t); | ||
1288 | } | ||
1289 | |||
1290 | static void ww_test_object_lock_stale_context(void) | ||
1291 | { | ||
1292 | WWAI(&t); | ||
1293 | o.ctx = &t2; | ||
1294 | WWL(&o, &t); | ||
1295 | } | ||
1296 | |||
1297 | static void ww_test_edeadlk_normal(void) | ||
1298 | { | ||
1299 | int ret; | ||
1300 | |||
1301 | mutex_lock(&o2.base); | ||
1302 | o2.ctx = &t2; | ||
1303 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1304 | |||
1305 | WWAI(&t); | ||
1306 | t2 = t; | ||
1307 | t2.stamp--; | ||
1308 | |||
1309 | ret = WWL(&o, &t); | ||
1310 | WARN_ON(ret); | ||
1311 | |||
1312 | ret = WWL(&o2, &t); | ||
1313 | WARN_ON(ret != -EDEADLK); | ||
1314 | |||
1315 | o2.ctx = NULL; | ||
1316 | mutex_acquire(&o2.base.dep_map, 0, 1, _THIS_IP_); | ||
1317 | mutex_unlock(&o2.base); | ||
1318 | WWU(&o); | ||
1319 | |||
1320 | WWL(&o2, &t); | ||
1321 | } | ||
1322 | |||
1323 | static void ww_test_edeadlk_normal_slow(void) | ||
1324 | { | ||
1325 | int ret; | ||
1326 | |||
1327 | mutex_lock(&o2.base); | ||
1328 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1329 | o2.ctx = &t2; | ||
1330 | |||
1331 | WWAI(&t); | ||
1332 | t2 = t; | ||
1333 | t2.stamp--; | ||
1334 | |||
1335 | ret = WWL(&o, &t); | ||
1336 | WARN_ON(ret); | ||
1337 | |||
1338 | ret = WWL(&o2, &t); | ||
1339 | WARN_ON(ret != -EDEADLK); | ||
1340 | |||
1341 | o2.ctx = NULL; | ||
1342 | mutex_acquire(&o2.base.dep_map, 0, 1, _THIS_IP_); | ||
1343 | mutex_unlock(&o2.base); | ||
1344 | WWU(&o); | ||
1345 | |||
1346 | ww_mutex_lock_slow(&o2, &t); | ||
1347 | } | ||
1348 | |||
1349 | static void ww_test_edeadlk_no_unlock(void) | ||
1350 | { | ||
1351 | int ret; | ||
1352 | |||
1353 | mutex_lock(&o2.base); | ||
1354 | o2.ctx = &t2; | ||
1355 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1356 | |||
1357 | WWAI(&t); | ||
1358 | t2 = t; | ||
1359 | t2.stamp--; | ||
1360 | |||
1361 | ret = WWL(&o, &t); | ||
1362 | WARN_ON(ret); | ||
1363 | |||
1364 | ret = WWL(&o2, &t); | ||
1365 | WARN_ON(ret != -EDEADLK); | ||
1366 | |||
1367 | o2.ctx = NULL; | ||
1368 | mutex_acquire(&o2.base.dep_map, 0, 1, _THIS_IP_); | ||
1369 | mutex_unlock(&o2.base); | ||
1370 | |||
1371 | WWL(&o2, &t); | ||
1372 | } | ||
1373 | |||
1374 | static void ww_test_edeadlk_no_unlock_slow(void) | ||
1375 | { | ||
1376 | int ret; | ||
1377 | |||
1378 | mutex_lock(&o2.base); | ||
1379 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1380 | o2.ctx = &t2; | ||
1381 | |||
1382 | WWAI(&t); | ||
1383 | t2 = t; | ||
1384 | t2.stamp--; | ||
1385 | |||
1386 | ret = WWL(&o, &t); | ||
1387 | WARN_ON(ret); | ||
1388 | |||
1389 | ret = WWL(&o2, &t); | ||
1390 | WARN_ON(ret != -EDEADLK); | ||
1391 | |||
1392 | o2.ctx = NULL; | ||
1393 | mutex_acquire(&o2.base.dep_map, 0, 1, _THIS_IP_); | ||
1394 | mutex_unlock(&o2.base); | ||
1395 | |||
1396 | ww_mutex_lock_slow(&o2, &t); | ||
1397 | } | ||
1398 | |||
1399 | static void ww_test_edeadlk_acquire_more(void) | ||
1400 | { | ||
1401 | int ret; | ||
1402 | |||
1403 | mutex_lock(&o2.base); | ||
1404 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1405 | o2.ctx = &t2; | ||
1406 | |||
1407 | WWAI(&t); | ||
1408 | t2 = t; | ||
1409 | t2.stamp--; | ||
1410 | |||
1411 | ret = WWL(&o, &t); | ||
1412 | WARN_ON(ret); | ||
1413 | |||
1414 | ret = WWL(&o2, &t); | ||
1415 | WARN_ON(ret != -EDEADLK); | ||
1416 | |||
1417 | ret = WWL(&o3, &t); | ||
1418 | } | ||
1419 | |||
1420 | static void ww_test_edeadlk_acquire_more_slow(void) | ||
1421 | { | ||
1422 | int ret; | ||
1423 | |||
1424 | mutex_lock(&o2.base); | ||
1425 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1426 | o2.ctx = &t2; | ||
1427 | |||
1428 | WWAI(&t); | ||
1429 | t2 = t; | ||
1430 | t2.stamp--; | ||
1431 | |||
1432 | ret = WWL(&o, &t); | ||
1433 | WARN_ON(ret); | ||
1434 | |||
1435 | ret = WWL(&o2, &t); | ||
1436 | WARN_ON(ret != -EDEADLK); | ||
1437 | |||
1438 | ww_mutex_lock_slow(&o3, &t); | ||
1439 | } | ||
1440 | |||
1441 | static void ww_test_edeadlk_acquire_more_edeadlk(void) | ||
1442 | { | ||
1443 | int ret; | ||
1444 | |||
1445 | mutex_lock(&o2.base); | ||
1446 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1447 | o2.ctx = &t2; | ||
1448 | |||
1449 | mutex_lock(&o3.base); | ||
1450 | mutex_release(&o3.base.dep_map, 1, _THIS_IP_); | ||
1451 | o3.ctx = &t2; | ||
1452 | |||
1453 | WWAI(&t); | ||
1454 | t2 = t; | ||
1455 | t2.stamp--; | ||
1456 | |||
1457 | ret = WWL(&o, &t); | ||
1458 | WARN_ON(ret); | ||
1459 | |||
1460 | ret = WWL(&o2, &t); | ||
1461 | WARN_ON(ret != -EDEADLK); | ||
1462 | |||
1463 | ret = WWL(&o3, &t); | ||
1464 | WARN_ON(ret != -EDEADLK); | ||
1465 | } | ||
1466 | |||
1467 | static void ww_test_edeadlk_acquire_more_edeadlk_slow(void) | ||
1468 | { | ||
1469 | int ret; | ||
1470 | |||
1471 | mutex_lock(&o2.base); | ||
1472 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1473 | o2.ctx = &t2; | ||
1474 | |||
1475 | mutex_lock(&o3.base); | ||
1476 | mutex_release(&o3.base.dep_map, 1, _THIS_IP_); | ||
1477 | o3.ctx = &t2; | ||
1478 | |||
1479 | WWAI(&t); | ||
1480 | t2 = t; | ||
1481 | t2.stamp--; | ||
1482 | |||
1483 | ret = WWL(&o, &t); | ||
1484 | WARN_ON(ret); | ||
1485 | |||
1486 | ret = WWL(&o2, &t); | ||
1487 | WARN_ON(ret != -EDEADLK); | ||
1488 | |||
1489 | ww_mutex_lock_slow(&o3, &t); | ||
1490 | } | ||
1491 | |||
1492 | static void ww_test_edeadlk_acquire_wrong(void) | ||
1493 | { | ||
1494 | int ret; | ||
1495 | |||
1496 | mutex_lock(&o2.base); | ||
1497 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1498 | o2.ctx = &t2; | ||
1499 | |||
1500 | WWAI(&t); | ||
1501 | t2 = t; | ||
1502 | t2.stamp--; | ||
1503 | |||
1504 | ret = WWL(&o, &t); | ||
1505 | WARN_ON(ret); | ||
1506 | |||
1507 | ret = WWL(&o2, &t); | ||
1508 | WARN_ON(ret != -EDEADLK); | ||
1509 | if (!ret) | ||
1510 | WWU(&o2); | ||
1511 | |||
1512 | WWU(&o); | ||
1513 | |||
1514 | ret = WWL(&o3, &t); | ||
1515 | } | ||
1516 | |||
1517 | static void ww_test_edeadlk_acquire_wrong_slow(void) | ||
1518 | { | ||
1519 | int ret; | ||
1520 | |||
1521 | mutex_lock(&o2.base); | ||
1522 | mutex_release(&o2.base.dep_map, 1, _THIS_IP_); | ||
1523 | o2.ctx = &t2; | ||
1524 | |||
1525 | WWAI(&t); | ||
1526 | t2 = t; | ||
1527 | t2.stamp--; | ||
1528 | |||
1529 | ret = WWL(&o, &t); | ||
1530 | WARN_ON(ret); | ||
1531 | |||
1532 | ret = WWL(&o2, &t); | ||
1533 | WARN_ON(ret != -EDEADLK); | ||
1534 | if (!ret) | ||
1535 | WWU(&o2); | ||
1536 | |||
1537 | WWU(&o); | ||
1538 | |||
1539 | ww_mutex_lock_slow(&o3, &t); | ||
1540 | } | ||
1541 | |||
1542 | static void ww_test_spin_nest_unlocked(void) | ||
1543 | { | ||
1544 | raw_spin_lock_nest_lock(&lock_A, &o.base); | ||
1545 | U(A); | ||
1546 | } | ||
1547 | |||
1548 | static void ww_test_unneeded_slow(void) | ||
1549 | { | ||
1550 | WWAI(&t); | ||
1551 | |||
1552 | ww_mutex_lock_slow(&o, &t); | ||
1553 | } | ||
1554 | |||
1555 | static void ww_test_context_block(void) | ||
1556 | { | ||
1557 | int ret; | ||
1558 | |||
1559 | WWAI(&t); | ||
1560 | |||
1561 | ret = WWL(&o, &t); | ||
1562 | WARN_ON(ret); | ||
1563 | WWL1(&o2); | ||
1564 | } | ||
1565 | |||
1566 | static void ww_test_context_try(void) | ||
1567 | { | ||
1568 | int ret; | ||
1569 | |||
1570 | WWAI(&t); | ||
1571 | |||
1572 | ret = WWL(&o, &t); | ||
1573 | WARN_ON(ret); | ||
1574 | |||
1575 | ret = WWT(&o2); | ||
1576 | WARN_ON(!ret); | ||
1577 | WWU(&o2); | ||
1578 | WWU(&o); | ||
1579 | } | ||
1580 | |||
1581 | static void ww_test_context_context(void) | ||
1582 | { | ||
1583 | int ret; | ||
1584 | |||
1585 | WWAI(&t); | ||
1586 | |||
1587 | ret = WWL(&o, &t); | ||
1588 | WARN_ON(ret); | ||
1589 | |||
1590 | ret = WWL(&o2, &t); | ||
1591 | WARN_ON(ret); | ||
1592 | |||
1593 | WWU(&o2); | ||
1594 | WWU(&o); | ||
1595 | } | ||
1596 | |||
1597 | static void ww_test_try_block(void) | ||
1598 | { | ||
1599 | bool ret; | ||
1600 | |||
1601 | ret = WWT(&o); | ||
1602 | WARN_ON(!ret); | ||
1603 | |||
1604 | WWL1(&o2); | ||
1605 | WWU(&o2); | ||
1606 | WWU(&o); | ||
1607 | } | ||
1608 | |||
1609 | static void ww_test_try_try(void) | ||
1610 | { | ||
1611 | bool ret; | ||
1612 | |||
1613 | ret = WWT(&o); | ||
1614 | WARN_ON(!ret); | ||
1615 | ret = WWT(&o2); | ||
1616 | WARN_ON(!ret); | ||
1617 | WWU(&o2); | ||
1618 | WWU(&o); | ||
1619 | } | ||
1620 | |||
1621 | static void ww_test_try_context(void) | ||
1622 | { | ||
1623 | int ret; | ||
1624 | |||
1625 | ret = WWT(&o); | ||
1626 | WARN_ON(!ret); | ||
1627 | |||
1628 | WWAI(&t); | ||
1629 | |||
1630 | ret = WWL(&o2, &t); | ||
1631 | WARN_ON(ret); | ||
1632 | } | ||
1633 | |||
1634 | static void ww_test_block_block(void) | ||
1635 | { | ||
1636 | WWL1(&o); | ||
1637 | WWL1(&o2); | ||
1638 | } | ||
1639 | |||
1640 | static void ww_test_block_try(void) | ||
1641 | { | ||
1642 | bool ret; | ||
1643 | |||
1644 | WWL1(&o); | ||
1645 | ret = WWT(&o2); | ||
1646 | WARN_ON(!ret); | ||
1647 | } | ||
1648 | |||
1649 | static void ww_test_block_context(void) | ||
1650 | { | ||
1651 | int ret; | ||
1652 | |||
1653 | WWL1(&o); | ||
1654 | WWAI(&t); | ||
1655 | |||
1656 | ret = WWL(&o2, &t); | ||
1657 | WARN_ON(ret); | ||
1658 | } | ||
1659 | |||
1660 | static void ww_test_spin_block(void) | ||
1661 | { | ||
1662 | L(A); | ||
1663 | U(A); | ||
1664 | |||
1665 | WWL1(&o); | ||
1666 | L(A); | ||
1667 | U(A); | ||
1668 | WWU(&o); | ||
1669 | |||
1670 | L(A); | ||
1671 | WWL1(&o); | ||
1672 | WWU(&o); | ||
1673 | U(A); | ||
1674 | } | ||
1675 | |||
1676 | static void ww_test_spin_try(void) | ||
1677 | { | ||
1678 | bool ret; | ||
1679 | |||
1680 | L(A); | ||
1681 | U(A); | ||
1682 | |||
1683 | ret = WWT(&o); | ||
1684 | WARN_ON(!ret); | ||
1685 | L(A); | ||
1686 | U(A); | ||
1687 | WWU(&o); | ||
1688 | |||
1689 | L(A); | ||
1690 | ret = WWT(&o); | ||
1691 | WARN_ON(!ret); | ||
1692 | WWU(&o); | ||
1693 | U(A); | ||
1694 | } | ||
1695 | |||
1696 | static void ww_test_spin_context(void) | ||
1697 | { | ||
1698 | int ret; | ||
1699 | |||
1700 | L(A); | ||
1701 | U(A); | ||
1702 | |||
1703 | WWAI(&t); | ||
1704 | |||
1705 | ret = WWL(&o, &t); | ||
1706 | WARN_ON(ret); | ||
1707 | L(A); | ||
1708 | U(A); | ||
1709 | WWU(&o); | ||
1710 | |||
1711 | L(A); | ||
1712 | ret = WWL(&o, &t); | ||
1713 | WARN_ON(ret); | ||
1714 | WWU(&o); | ||
1715 | U(A); | ||
1716 | } | ||
1717 | |||
1718 | static void ww_tests(void) | ||
1719 | { | ||
1720 | printk(" --------------------------------------------------------------------------\n"); | ||
1721 | printk(" | Wound/wait tests |\n"); | ||
1722 | printk(" ---------------------\n"); | ||
1723 | |||
1724 | print_testname("ww api failures"); | ||
1725 | dotest(ww_test_fail_acquire, SUCCESS, LOCKTYPE_WW); | ||
1726 | dotest(ww_test_normal, SUCCESS, LOCKTYPE_WW); | ||
1727 | dotest(ww_test_unneeded_slow, FAILURE, LOCKTYPE_WW); | ||
1728 | printk("\n"); | ||
1729 | |||
1730 | print_testname("ww contexts mixing"); | ||
1731 | dotest(ww_test_two_contexts, FAILURE, LOCKTYPE_WW); | ||
1732 | dotest(ww_test_diff_class, FAILURE, LOCKTYPE_WW); | ||
1733 | printk("\n"); | ||
1734 | |||
1735 | print_testname("finishing ww context"); | ||
1736 | dotest(ww_test_context_done_twice, FAILURE, LOCKTYPE_WW); | ||
1737 | dotest(ww_test_context_unlock_twice, FAILURE, LOCKTYPE_WW); | ||
1738 | dotest(ww_test_context_fini_early, FAILURE, LOCKTYPE_WW); | ||
1739 | dotest(ww_test_context_lock_after_done, FAILURE, LOCKTYPE_WW); | ||
1740 | printk("\n"); | ||
1741 | |||
1742 | print_testname("locking mismatches"); | ||
1743 | dotest(ww_test_object_unlock_twice, FAILURE, LOCKTYPE_WW); | ||
1744 | dotest(ww_test_object_lock_unbalanced, FAILURE, LOCKTYPE_WW); | ||
1745 | dotest(ww_test_object_lock_stale_context, FAILURE, LOCKTYPE_WW); | ||
1746 | printk("\n"); | ||
1747 | |||
1748 | print_testname("EDEADLK handling"); | ||
1749 | dotest(ww_test_edeadlk_normal, SUCCESS, LOCKTYPE_WW); | ||
1750 | dotest(ww_test_edeadlk_normal_slow, SUCCESS, LOCKTYPE_WW); | ||
1751 | dotest(ww_test_edeadlk_no_unlock, FAILURE, LOCKTYPE_WW); | ||
1752 | dotest(ww_test_edeadlk_no_unlock_slow, FAILURE, LOCKTYPE_WW); | ||
1753 | dotest(ww_test_edeadlk_acquire_more, FAILURE, LOCKTYPE_WW); | ||
1754 | dotest(ww_test_edeadlk_acquire_more_slow, FAILURE, LOCKTYPE_WW); | ||
1755 | dotest(ww_test_edeadlk_acquire_more_edeadlk, FAILURE, LOCKTYPE_WW); | ||
1756 | dotest(ww_test_edeadlk_acquire_more_edeadlk_slow, FAILURE, LOCKTYPE_WW); | ||
1757 | dotest(ww_test_edeadlk_acquire_wrong, FAILURE, LOCKTYPE_WW); | ||
1758 | dotest(ww_test_edeadlk_acquire_wrong_slow, FAILURE, LOCKTYPE_WW); | ||
1759 | printk("\n"); | ||
1760 | |||
1761 | print_testname("spinlock nest unlocked"); | ||
1762 | dotest(ww_test_spin_nest_unlocked, FAILURE, LOCKTYPE_WW); | ||
1763 | printk("\n"); | ||
1764 | |||
1765 | printk(" -----------------------------------------------------\n"); | ||
1766 | printk(" |block | try |context|\n"); | ||
1767 | printk(" -----------------------------------------------------\n"); | ||
1768 | |||
1769 | print_testname("context"); | ||
1770 | dotest(ww_test_context_block, FAILURE, LOCKTYPE_WW); | ||
1771 | dotest(ww_test_context_try, SUCCESS, LOCKTYPE_WW); | ||
1772 | dotest(ww_test_context_context, SUCCESS, LOCKTYPE_WW); | ||
1773 | printk("\n"); | ||
1774 | |||
1775 | print_testname("try"); | ||
1776 | dotest(ww_test_try_block, FAILURE, LOCKTYPE_WW); | ||
1777 | dotest(ww_test_try_try, SUCCESS, LOCKTYPE_WW); | ||
1778 | dotest(ww_test_try_context, FAILURE, LOCKTYPE_WW); | ||
1779 | printk("\n"); | ||
1780 | |||
1781 | print_testname("block"); | ||
1782 | dotest(ww_test_block_block, FAILURE, LOCKTYPE_WW); | ||
1783 | dotest(ww_test_block_try, SUCCESS, LOCKTYPE_WW); | ||
1784 | dotest(ww_test_block_context, FAILURE, LOCKTYPE_WW); | ||
1785 | printk("\n"); | ||
1786 | |||
1787 | print_testname("spinlock"); | ||
1788 | dotest(ww_test_spin_block, FAILURE, LOCKTYPE_WW); | ||
1789 | dotest(ww_test_spin_try, SUCCESS, LOCKTYPE_WW); | ||
1790 | dotest(ww_test_spin_context, FAILURE, LOCKTYPE_WW); | ||
1791 | printk("\n"); | ||
1792 | } | ||
1111 | 1793 | ||
1112 | void locking_selftest(void) | 1794 | void locking_selftest(void) |
1113 | { | 1795 | { |
@@ -1188,6 +1870,8 @@ void locking_selftest(void) | |||
1188 | DO_TESTCASE_6x2("irq read-recursion", irq_read_recursion); | 1870 | DO_TESTCASE_6x2("irq read-recursion", irq_read_recursion); |
1189 | // DO_TESTCASE_6x2B("irq read-recursion #2", irq_read_recursion2); | 1871 | // DO_TESTCASE_6x2B("irq read-recursion #2", irq_read_recursion2); |
1190 | 1872 | ||
1873 | ww_tests(); | ||
1874 | |||
1191 | if (unexpected_testcase_failures) { | 1875 | if (unexpected_testcase_failures) { |
1192 | printk("-----------------------------------------------------------------\n"); | 1876 | printk("-----------------------------------------------------------------\n"); |
1193 | debug_locks = 0; | 1877 | debug_locks = 0; |