/* -*- mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ // vim: expandtab:ts=8:sw=4:softtabstop=4: // T(A) gets W(L) // T(B) gets W(M) // T(A) tries W(M), gets blocked // T(B) tries W(L), gets deadlock // T(B) releases locks // T(A) granted W(M) // T(A) releases locks #include "test.h" int main(int argc, const char *argv[]) { int r; uint32_t max_locks = 2; uint64_t max_lock_memory = 4096; for (int i = 1; i < argc; i++) { if (strcmp(argv[i], "-v") == 0 || strcmp(argv[i], "--verbose") == 0) { verbose++; continue; } if (strcmp(argv[i], "-q") == 0 || strcmp(argv[i], "--quiet") == 0) { if (verbose > 0) verbose--; continue; } if (strcmp(argv[i], "--max_locks") == 0 && i+1 < argc) { max_locks = atoi(argv[++i]); continue; } if (strcmp(argv[i], "--max_lock_memory") == 0 && i+1 < argc) { max_lock_memory = atoi(argv[++i]); continue; } assert(0); } // setup toku_ltm *ltm = NULL; r = toku_ltm_create(<m, max_locks, max_lock_memory, dbpanic); assert(r == 0 && ltm); toku_lock_tree *lt = NULL; r = toku_ltm_get_lt(ltm, <, (DICTIONARY_ID){1}, NULL, dbcmp, NULL, NULL, NULL); assert(r == 0 && lt); const TXNID txn_a = 1; DBT key_l; dbt_init(&key_l, "L", 1); toku_lock_request a_w_l; toku_lock_request_init(&a_w_l, txn_a, &key_l, &key_l, LOCK_REQUEST_WRITE); r = toku_lock_request_start(&a_w_l, lt, false); assert(r == 0); assert(a_w_l.state == LOCK_REQUEST_COMPLETE && a_w_l.complete_r == 0); toku_lock_request_destroy(&a_w_l); const TXNID txn_b = 2; DBT key_m; dbt_init(&key_m, "M", 1); toku_lock_request b_w_m; toku_lock_request_init(&b_w_m, txn_b, &key_m, &key_m, LOCK_REQUEST_WRITE); r = toku_lock_request_start(&b_w_m, lt, false); assert(r == 0); assert(b_w_m.state == LOCK_REQUEST_COMPLETE && b_w_m.complete_r == 0); toku_lock_request_destroy(&b_w_m); toku_lock_request a_w_m; toku_lock_request_init(&a_w_m, txn_a, &key_m, &key_m, LOCK_REQUEST_WRITE); r = toku_lock_request_start(&a_w_m, lt, false); assert(r == DB_LOCK_NOTGRANTED); assert(a_w_m.state == LOCK_REQUEST_PENDING); toku_lock_request b_w_l; toku_lock_request_init(&b_w_l, txn_b, &key_l, &key_l, LOCK_REQUEST_WRITE); r = toku_lock_request_start(&b_w_l, lt, false); assert(r == DB_LOCK_DEADLOCK); assert(b_w_l.state == LOCK_REQUEST_COMPLETE && b_w_l.complete_r == DB_LOCK_DEADLOCK); r = toku_lt_unlock_txn(lt, txn_b); assert(r == 0); toku_lock_request_destroy(&b_w_l); assert(a_w_m.state == LOCK_REQUEST_COMPLETE && a_w_m.complete_r == 0 ); toku_lock_request_destroy(&a_w_m); r = toku_lt_unlock_txn(lt, txn_a); assert(r == 0); // shutdown toku_lt_remove_db_ref(lt); r = toku_ltm_close(ltm); assert(r == 0); return 0; }