/* -*- 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) tries R(L), gets DB_LOCK_NOTGRANTED
// T(C) tries R(L), gets DB_LOCK_NOTGRANTED
// T(A) releases locks
// T(B) gets R(L)
// T(C) gets R(L)
// T(B) releases locks
// T(C) releases locks

#include "test.h"

static int read_lock(toku_lock_tree *lt, TXNID txnid, const char *k) {
    DBT key; dbt_init(&key, k, strlen(k));
    int r = toku_lt_acquire_read_lock(lt, txnid, &key);
    return r;
}

static int write_lock(toku_lock_tree *lt, TXNID txnid, const char *k) {
    DBT key; dbt_init(&key, k, strlen(k));
    int r = toku_lt_acquire_write_lock(lt, txnid, &key);
    return r;
}

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(&ltm, max_locks, max_lock_memory, dbpanic);
    assert(r == 0 && ltm);

    toku_lock_tree *lt = NULL;
    r = toku_ltm_get_lt(ltm, &lt, (DICTIONARY_ID){1}, NULL, dbcmp, NULL, NULL, NULL);
    assert(r == 0 && lt);

    const TXNID txn_a = 1;
    r = write_lock(lt, txn_a, "L"); assert(r == 0);

    const TXNID txn_b = 2;
    r = read_lock(lt, txn_b, "L"); assert(r == DB_LOCK_NOTGRANTED);

    const TXNID txn_c = 3;
    r = read_lock(lt, txn_c, "L"); assert(r == DB_LOCK_NOTGRANTED);

    r = toku_lt_unlock_txn(lt, txn_a); assert(r == 0);
    r = read_lock(lt, txn_b, "L"); assert(r == 0);
    r = read_lock(lt, txn_c, "L"); assert(r == 0);
    r = toku_lt_unlock_txn(lt, txn_b); assert(r == 0);
    r = toku_lt_unlock_txn(lt, txn_c); assert(r == 0);

    // shutdown 
    toku_lt_remove_db_ref(lt);
    r = toku_ltm_close(ltm); assert(r == 0);

    return 0;
}