diff options
Diffstat (limited to 'storage/tokudb/PerconaFT/src/tests/test3529.cc')
-rw-r--r-- | storage/tokudb/PerconaFT/src/tests/test3529.cc | 210 |
1 files changed, 210 insertions, 0 deletions
diff --git a/storage/tokudb/PerconaFT/src/tests/test3529.cc b/storage/tokudb/PerconaFT/src/tests/test3529.cc new file mode 100644 index 00000000000..34f67a806aa --- /dev/null +++ b/storage/tokudb/PerconaFT/src/tests/test3529.cc @@ -0,0 +1,210 @@ +/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */ +// vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4: +#ident "$Id$" +/*====== +This file is part of PerconaFT. + + +Copyright (c) 2006, 2015, Percona and/or its affiliates. All rights reserved. + + PerconaFT is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License, version 2, + as published by the Free Software Foundation. + + PerconaFT is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with PerconaFT. If not, see <http://www.gnu.org/licenses/>. + +---------------------------------------- + + PerconaFT is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License, version 3, + as published by the Free Software Foundation. + + PerconaFT is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with PerconaFT. If not, see <http://www.gnu.org/licenses/>. +======= */ + +#ident "Copyright (c) 2006, 2015, Percona and/or its affiliates. All rights reserved." + +/* Test for #3522. Show that even with DB_TRYAGAIN, isolation still works. + * Strategy: + * 1. Create a tree (with relatively small nodes so things happen quickly, and relatively large compared to the cache). + * 2. Start two transactions YY and XX. + * 3. Force XX to precede YY (e.g., XX reads the last row, and then YY deletes it, under MVCC). + * 4. YY, in a single transaction: deletes everything + * 5. YY does do a DB_FIRST. + * Set things up so that while YY is doing it's retries, XX inserts a row at the beginning. + * Make the test terminate by capturing the calls to pread(). */ + +#include "test.h" +#include <portability/toku_atomic.h> + +static DB_ENV *env; +static DB *db; +const int N = 1000; +static DB_TXN *XX, *YY; + +long do_XX_on_pread = -1; +const int n_preads_limit = 1000; +long n_preads = 0; + +static void insert(int i, DB_TXN *txn); + +static ssize_t my_pread (int fd, void *buf, size_t count, off_t offset) { + long n_read_so_far = toku_sync_fetch_and_add(&n_preads, 1); + if (do_XX_on_pread==n_read_so_far && XX != NULL) { + // we're supposed to do the XX operation now. Insert a row. + printf("Did XX\n"); + insert(0, XX); + } + if (n_read_so_far > n_preads_limit) { + if (verbose) fprintf(stderr, "Apparent infinite loop detected\n"); + abort(); + } + return pread(fd, buf, count, offset); +} + +static void +insert(int i, DB_TXN *txn) +{ + char hello[30], there[30]; + snprintf(hello, sizeof(hello), "hello%04d", i); + snprintf(there, sizeof(there), "there%d", i); + DBT key, val; + if (db) { + int r=db->put(db, txn, + dbt_init(&key, hello, strlen(hello)+1), + dbt_init(&val, there, strlen(there)+1), + 0); + CKERR(r); + } +} + +static void op_delete (int i, DB_TXN *x) { + char hello[30]; + DBT key; + if (verbose>1) printf("op_delete %d\n", i); + snprintf(hello, sizeof(hello), "hello%04d", i); + int r = db->del(db, x, + dbt_init(&key, hello, strlen(hello)+1), + 0); + CKERR(r); +} + +int did_nothing = 0; + +static int +do_nothing(DBT const *UU(a), DBT const *UU(b), void *UU(c)) { + did_nothing++; + return 0; +} + + +static void +setup (void) { + db = NULL; + db_env_set_func_pread(my_pread); + int r; + toku_os_recursive_delete(TOKU_TEST_FILENAME); + toku_os_mkdir(TOKU_TEST_FILENAME, S_IRWXU+S_IRWXG+S_IRWXO); + r = db_env_create(&env, 0); CKERR(r); + r = env->set_redzone(env, 0); CKERR(r); + r = env->set_cachesize(env, 0, 2*128*1024, 1); CKERR(r); + r = env->open(env, TOKU_TEST_FILENAME, DB_INIT_LOCK|DB_INIT_LOG|DB_INIT_MPOOL|DB_INIT_TXN|DB_CREATE|DB_PRIVATE, S_IRWXU+S_IRWXG+S_IRWXO); CKERR(r); + r = db_create(&db, env, 0); CKERR(r); + r = db->set_pagesize(db, 4096); CKERR(r); + { + DB_TXN *txn; + r = env->txn_begin(env, 0, &txn, 0); CKERR(r); + r = db->open(db, txn, "foo.db", 0, DB_BTREE, DB_CREATE, S_IRWXU+S_IRWXG+S_IRWXO); CKERR(r); + r = txn->commit(txn, 0); CKERR(r); + } + { + DB_TXN *txn; + r = env->txn_begin(env, 0, &txn, 0); CKERR(r); + for (int i=0; i<N; i++) insert(i+1, txn); + r = txn->commit(txn, 0); CKERR(r); + } + r = env->txn_begin(env, 0, &XX, DB_TXN_SNAPSHOT); CKERR(r); + r = env->txn_begin(env, 0, &YY, DB_TXN_SNAPSHOT); CKERR(r); + + // Force XX to preceed YY by making XX read something. (YY will op_delete everything in a moment). + { + DBC *cursor; + r = db->cursor(db, XX, &cursor, 0); CKERR(r); + did_nothing = 0; + //r = cursor->c_getf_next(cursor, 0, do_nothing, NULL); CKERR(r); + //assert(did_nothing==1); + did_nothing = 0; + r = cursor->c_close(cursor); CKERR(r); + } +} + +static void finish (void) { + int r; + r = YY->commit(YY, 0); CKERR(r); + YY = NULL; + r = XX->commit(XX, 0); CKERR(r); + XX = NULL; + r = db->close(db, 0); CKERR(r); + r = env->close(env, 0); CKERR(r); +} + + +static void run_del_next (void) { + DBC *cursor; + int r; + for (int i=0; i<N; i++) op_delete(i+1, YY); + + r = db->cursor(db, YY, &cursor, 0); CKERR(r); + if (verbose) printf("read_next\n"); + n_preads = 0; + do_XX_on_pread = 2; + printf("doing on %ld\n", do_XX_on_pread); + r = cursor->c_getf_next(cursor, 0, do_nothing, NULL); CKERR2(r, DB_NOTFOUND); + do_XX_on_pread = 0; + assert(did_nothing==0); + if (verbose) printf("n_preads=%ld\n", n_preads); + r = cursor->c_close(cursor); CKERR(r); +} + +static void run_del_prev (void) { + DBC *cursor; + int r; + for (int i=0; i<N; i++) op_delete(i+1, YY); + + r = db->cursor(db, YY, &cursor, 0); CKERR(r); + if (verbose) printf("read_prev\n"); + n_preads = 0; + r = cursor->c_getf_prev(cursor, 0, do_nothing, NULL); CKERR2(r, DB_NOTFOUND); + assert(did_nothing==0); + if (verbose) printf("n_preads=%ld\n", n_preads); + r = cursor->c_close(cursor); CKERR(r); +} + +static void run_test (void) { + setup(); + run_del_next(); + finish(); + + setup(); + run_del_prev(); + finish(); +} +int test_main (int argc, char*const argv[]) { + parse_args(argc, argv); + run_test(); + return 0; +} + + |