mirror of
https://github.com/MariaDB/server.git
synced 2025-01-27 01:04:19 +01:00
9f99932a76
git-svn-id: file:///svn/toku/tokudb@55577 c7de825b-a66e-492c-adef-691d508d4ae1
1233 lines
48 KiB
C++
1233 lines
48 KiB
C++
/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */
|
|
// vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4:
|
|
#ident "$Id$"
|
|
/*
|
|
COPYING CONDITIONS NOTICE:
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of version 2 of the GNU General Public License as
|
|
published by the Free Software Foundation, and provided that the
|
|
following conditions are met:
|
|
|
|
* Redistributions of source code must retain this COPYING
|
|
CONDITIONS NOTICE, the COPYRIGHT NOTICE (below), the
|
|
DISCLAIMER (below), the UNIVERSITY PATENT NOTICE (below), the
|
|
PATENT MARKING NOTICE (below), and the PATENT RIGHTS
|
|
GRANT (below).
|
|
|
|
* Redistributions in binary form must reproduce this COPYING
|
|
CONDITIONS NOTICE, the COPYRIGHT NOTICE (below), the
|
|
DISCLAIMER (below), the UNIVERSITY PATENT NOTICE (below), the
|
|
PATENT MARKING NOTICE (below), and the PATENT RIGHTS
|
|
GRANT (below) in the documentation and/or other materials
|
|
provided with the distribution.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
02110-1301, USA.
|
|
|
|
COPYRIGHT NOTICE:
|
|
|
|
TokuDB, Tokutek Fractal Tree Indexing Library.
|
|
Copyright (C) 2007-2013 Tokutek, Inc.
|
|
|
|
DISCLAIMER:
|
|
|
|
This program 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.
|
|
|
|
UNIVERSITY PATENT NOTICE:
|
|
|
|
The technology is licensed by the Massachusetts Institute of
|
|
Technology, Rutgers State University of New Jersey, and the Research
|
|
Foundation of State University of New York at Stony Brook under
|
|
United States of America Serial No. 11/760379 and to the patents
|
|
and/or patent applications resulting from it.
|
|
|
|
PATENT MARKING NOTICE:
|
|
|
|
This software is covered by US Patent No. 8,185,551.
|
|
|
|
PATENT RIGHTS GRANT:
|
|
|
|
"THIS IMPLEMENTATION" means the copyrightable works distributed by
|
|
Tokutek as part of the Fractal Tree project.
|
|
|
|
"PATENT CLAIMS" means the claims of patents that are owned or
|
|
licensable by Tokutek, both currently or in the future; and that in
|
|
the absence of this license would be infringed by THIS
|
|
IMPLEMENTATION or by using or running THIS IMPLEMENTATION.
|
|
|
|
"PATENT CHALLENGE" shall mean a challenge to the validity,
|
|
patentability, enforceability and/or non-infringement of any of the
|
|
PATENT CLAIMS or otherwise opposing any of the PATENT CLAIMS.
|
|
|
|
Tokutek hereby grants to you, for the term and geographical scope of
|
|
the PATENT CLAIMS, a non-exclusive, no-charge, royalty-free,
|
|
irrevocable (except as stated in this section) patent license to
|
|
make, have made, use, offer to sell, sell, import, transfer, and
|
|
otherwise run, modify, and propagate the contents of THIS
|
|
IMPLEMENTATION, where such license applies only to the PATENT
|
|
CLAIMS. This grant does not include claims that would be infringed
|
|
only as a consequence of further modifications of THIS
|
|
IMPLEMENTATION. If you or your agent or licensee institute or order
|
|
or agree to the institution of patent litigation against any entity
|
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
THIS IMPLEMENTATION constitutes direct or contributory patent
|
|
infringement, or inducement of patent infringement, then any rights
|
|
granted to you under this License shall terminate as of the date
|
|
such litigation is filed. If you or your agent or exclusive
|
|
licensee institute or order or agree to the institution of a PATENT
|
|
CHALLENGE, then Tokutek may terminate any rights granted to you
|
|
under this License.
|
|
*/
|
|
|
|
#ident "Copyright (c) 2007-2013 Tokutek Inc. All rights reserved."
|
|
|
|
#include "test.h"
|
|
|
|
|
|
#include "ule.h"
|
|
|
|
static TOKUTXN const null_txn = 0;
|
|
static DB * const null_db = 0;
|
|
static const char *fname = TOKU_TEST_FILENAME;
|
|
|
|
static int dummy_cmp(DB *db __attribute__((unused)),
|
|
const DBT *a, const DBT *b) {
|
|
int c;
|
|
if (a->size > b->size) {
|
|
c = memcmp(a->data, b->data, b->size);
|
|
} else if (a->size < b->size) {
|
|
c = memcmp(a->data, b->data, a->size);
|
|
} else {
|
|
return memcmp(a->data, b->data, a->size);
|
|
}
|
|
if (c == 0) {
|
|
c = a->size - b->size;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
// generate size random bytes into dest
|
|
static void
|
|
rand_bytes(void *dest, int size)
|
|
{
|
|
long *l;
|
|
for (CAST_FROM_VOIDP(l, dest); (unsigned int) size >= (sizeof *l); ++l, size -= (sizeof *l)) {
|
|
*l = random();
|
|
}
|
|
for (char *c = (char *) l; size > 0; ++c, --size) {
|
|
*c = random() & 0xff;
|
|
}
|
|
}
|
|
|
|
// generate size random bytes into dest, with a lot less entropy (every
|
|
// group of 4 bytes is the same)
|
|
static void
|
|
rand_bytes_limited(void *dest, int size)
|
|
{
|
|
long *l;
|
|
for (CAST_FROM_VOIDP(l, dest); (size_t) size >= (sizeof *l); ++l, size -= (sizeof *l)) {
|
|
char c = random() & 0xff;
|
|
for (char *p = (char *) l; (size_t) (p - (char *) l) < (sizeof *l); ++p) {
|
|
*p = c;
|
|
}
|
|
}
|
|
char c = random() & 0xff;
|
|
for (char *p = (char *) l; size > 0; ++p, --size) {
|
|
*p = c;
|
|
}
|
|
}
|
|
|
|
// generate a random message with xids and a key starting with pfx, insert
|
|
// it in bnc, and save it in output params save and is_fresh_out
|
|
static void
|
|
insert_random_message(NONLEAF_CHILDINFO bnc, FT_MSG_S **save, bool *is_fresh_out, XIDS xids, int pfx)
|
|
{
|
|
int keylen = (random() % 128) + 16;
|
|
int vallen = (random() % 128) + 16;
|
|
void *key = toku_xmalloc(keylen + (sizeof pfx));
|
|
void *val = toku_xmalloc(vallen);
|
|
*(int *) key = pfx;
|
|
rand_bytes((char *) key + (sizeof pfx), keylen);
|
|
rand_bytes(val, vallen);
|
|
MSN msn = next_dummymsn();
|
|
bool is_fresh = (random() & 0x100) == 0;
|
|
|
|
DBT *keydbt, *valdbt;
|
|
XMALLOC(keydbt);
|
|
XMALLOC(valdbt);
|
|
toku_fill_dbt(keydbt, key, keylen + (sizeof pfx));
|
|
toku_fill_dbt(valdbt, val, vallen);
|
|
FT_MSG_S *XMALLOC(result);
|
|
result->type = FT_INSERT;
|
|
result->msn = msn;
|
|
result->xids = xids;
|
|
result->u.id.key = keydbt;
|
|
result->u.id.val = valdbt;
|
|
*save = result;
|
|
*is_fresh_out = is_fresh;
|
|
|
|
toku_bnc_insert_msg(bnc, key, keylen + (sizeof pfx), val, vallen,
|
|
FT_INSERT, msn, xids, is_fresh,
|
|
NULL, dummy_cmp);
|
|
}
|
|
|
|
// generate a random message with xids and a key starting with pfx, insert
|
|
// it into blb, and save it in output param save
|
|
static void
|
|
insert_random_message_to_bn(FT_HANDLE t, BASEMENTNODE blb, LEAFENTRY *save, XIDS xids, int pfx)
|
|
{
|
|
int keylen = (random() % 16) + 16;
|
|
int vallen = (random() % 128) + 16;
|
|
uint32_t *pfxp;
|
|
char key[(sizeof *pfxp) + keylen];
|
|
char val[vallen];
|
|
pfxp = (uint32_t *) &key[0];
|
|
*pfxp = pfx;
|
|
char *randkeyp = &key[sizeof *pfxp];
|
|
rand_bytes_limited(randkeyp, keylen);
|
|
rand_bytes(val, vallen);
|
|
MSN msn = next_dummymsn();
|
|
|
|
DBT keydbt_s, *keydbt, valdbt_s, *valdbt;
|
|
keydbt = &keydbt_s;
|
|
valdbt = &valdbt_s;
|
|
toku_fill_dbt(keydbt, key, (sizeof *pfxp) + keylen);
|
|
toku_fill_dbt(valdbt, val, vallen);
|
|
FT_MSG_S msg;
|
|
msg.type = FT_INSERT;
|
|
msg.msn = msn;
|
|
msg.xids = xids;
|
|
msg.u.id.key = keydbt;
|
|
msg.u.id.val = valdbt;
|
|
size_t memsize;
|
|
int64_t numbytes;
|
|
toku_le_apply_msg(&msg, NULL, TXNID_NONE, make_gc_info(false), &memsize, save, NULL, NULL, NULL, &numbytes);
|
|
toku_ft_bn_apply_cmd(t->ft->compare_fun, t->ft->update_fun, NULL, blb, &msg, TXNID_NONE, make_gc_info(false), NULL, NULL);
|
|
if (msn.msn > blb->max_msn_applied.msn) {
|
|
blb->max_msn_applied = msn;
|
|
}
|
|
}
|
|
|
|
// generate a random message with xids and a key starting with pfx, insert
|
|
// it into blb1 and also into blb2, and save it in output param save
|
|
//
|
|
// used for making two leaf nodes the same in order to compare the result
|
|
// of 'maybe_apply' and a normal buffer flush
|
|
static void
|
|
insert_same_message_to_bns(FT_HANDLE t, BASEMENTNODE blb1, BASEMENTNODE blb2, LEAFENTRY *save, XIDS xids, int pfx)
|
|
{
|
|
int keylen = (random() % 16) + 16;
|
|
int vallen = (random() % 128) + 16;
|
|
uint32_t *pfxp;
|
|
char key[(sizeof *pfxp) + keylen];
|
|
char val[vallen];
|
|
pfxp = (uint32_t *) &key[0];
|
|
*pfxp = pfx;
|
|
char *randkeyp = &key[sizeof *pfxp];
|
|
rand_bytes_limited(randkeyp, keylen);
|
|
rand_bytes(val, vallen);
|
|
MSN msn = next_dummymsn();
|
|
|
|
DBT keydbt_s, *keydbt, valdbt_s, *valdbt;
|
|
keydbt = &keydbt_s;
|
|
valdbt = &valdbt_s;
|
|
toku_fill_dbt(keydbt, key, (sizeof *pfxp) + keylen);
|
|
toku_fill_dbt(valdbt, val, vallen);
|
|
FT_MSG_S msg;
|
|
msg.type = FT_INSERT;
|
|
msg.msn = msn;
|
|
msg.xids = xids;
|
|
msg.u.id.key = keydbt;
|
|
msg.u.id.val = valdbt;
|
|
size_t memsize;
|
|
int64_t numbytes;
|
|
toku_le_apply_msg(&msg, NULL, TXNID_NONE, make_gc_info(false), &memsize, save, NULL, NULL, NULL, &numbytes);
|
|
toku_ft_bn_apply_cmd(t->ft->compare_fun, t->ft->update_fun, NULL, blb1, &msg, TXNID_NONE, make_gc_info(false), NULL, NULL);
|
|
if (msn.msn > blb1->max_msn_applied.msn) {
|
|
blb1->max_msn_applied = msn;
|
|
}
|
|
toku_ft_bn_apply_cmd(t->ft->compare_fun, t->ft->update_fun, NULL, blb2, &msg, TXNID_NONE, make_gc_info(false), NULL, NULL);
|
|
if (msn.msn > blb2->max_msn_applied.msn) {
|
|
blb2->max_msn_applied = msn;
|
|
}
|
|
}
|
|
|
|
struct orthopush_flush_update_fun_extra {
|
|
DBT new_val;
|
|
int *num_applications;
|
|
};
|
|
|
|
static int
|
|
orthopush_flush_update_fun(DB * UU(db), const DBT *UU(key), const DBT *UU(old_val), const DBT *extra,
|
|
void (*set_val)(const DBT *new_val, void *set_extra), void *set_extra) {
|
|
struct orthopush_flush_update_fun_extra *CAST_FROM_VOIDP(e, extra->data);
|
|
(*e->num_applications)++;
|
|
set_val(&e->new_val, set_extra);
|
|
return 0;
|
|
}
|
|
|
|
// generate a random update message with xids and a key starting with pfx,
|
|
// insert it into blb, and save it in output param save, and update the
|
|
// max msn so far in max_msn
|
|
//
|
|
// the update message will overwrite the value with something generated
|
|
// here, and add one to the int pointed to by applied
|
|
static void
|
|
insert_random_update_message(NONLEAF_CHILDINFO bnc, FT_MSG_S **save, bool is_fresh, XIDS xids, int pfx, int *applied, MSN *max_msn)
|
|
{
|
|
int keylen = (random() % 16) + 16;
|
|
int vallen = (random() % 16) + 16;
|
|
void *key = toku_xmalloc(keylen + (sizeof pfx));
|
|
struct orthopush_flush_update_fun_extra *XMALLOC(update_extra);
|
|
*(int *) key = pfx;
|
|
rand_bytes_limited((char *) key + (sizeof pfx), keylen);
|
|
toku_fill_dbt(&update_extra->new_val, toku_xmalloc(vallen), vallen);
|
|
rand_bytes(update_extra->new_val.data, vallen);
|
|
update_extra->num_applications = applied;
|
|
MSN msn = next_dummymsn();
|
|
|
|
DBT *keydbt, *valdbt;
|
|
XMALLOC(keydbt);
|
|
XMALLOC(valdbt);
|
|
toku_fill_dbt(keydbt, key, keylen + (sizeof pfx));
|
|
toku_fill_dbt(valdbt, update_extra, sizeof *update_extra);
|
|
FT_MSG_S *XMALLOC(result);
|
|
result->type = FT_UPDATE;
|
|
result->msn = msn;
|
|
result->xids = xids;
|
|
result->u.id.key = keydbt;
|
|
result->u.id.val = valdbt;
|
|
*save = result;
|
|
|
|
toku_bnc_insert_msg(bnc, key, keylen + (sizeof pfx),
|
|
update_extra, sizeof *update_extra,
|
|
FT_UPDATE, msn, xids, is_fresh,
|
|
NULL, dummy_cmp);
|
|
if (msn.msn > max_msn->msn) {
|
|
*max_msn = msn;
|
|
}
|
|
}
|
|
|
|
const int M = 1024 * 1024;
|
|
|
|
// flush from one internal node to another, where both only have one
|
|
// buffer
|
|
static void
|
|
flush_to_internal(FT_HANDLE t) {
|
|
int r;
|
|
|
|
FT_MSG_S **MALLOC_N(4096,parent_messages); // 128k / 32 = 4096
|
|
FT_MSG_S **MALLOC_N(4096,child_messages);
|
|
bool *MALLOC_N(4096,parent_messages_is_fresh);
|
|
bool *MALLOC_N(4096,child_messages_is_fresh);
|
|
memset(parent_messages_is_fresh, 0, 4096*(sizeof parent_messages_is_fresh[0]));
|
|
memset(child_messages_is_fresh, 0, 4096*(sizeof child_messages_is_fresh[0]));
|
|
|
|
XIDS xids_0 = xids_get_root_xids();
|
|
XIDS xids_123, xids_234;
|
|
r = xids_create_child(xids_0, &xids_123, (TXNID)123);
|
|
CKERR(r);
|
|
r = xids_create_child(xids_0, &xids_234, (TXNID)234);
|
|
CKERR(r);
|
|
|
|
NONLEAF_CHILDINFO child_bnc = toku_create_empty_nl();
|
|
int i;
|
|
for (i = 0; toku_bnc_memory_used(child_bnc) < 128*1024; ++i) {
|
|
insert_random_message(child_bnc, &child_messages[i], &child_messages_is_fresh[i], xids_123, 0);
|
|
}
|
|
int num_child_messages = i;
|
|
|
|
NONLEAF_CHILDINFO parent_bnc = toku_create_empty_nl();
|
|
for (i = 0; toku_bnc_memory_used(parent_bnc) < 128*1024; ++i) {
|
|
insert_random_message(parent_bnc, &parent_messages[i], &parent_messages_is_fresh[i], xids_234, 0);
|
|
}
|
|
int num_parent_messages = i;
|
|
|
|
FTNODE XMALLOC(child);
|
|
BLOCKNUM blocknum = { 42 };
|
|
toku_initialize_empty_ftnode(child, blocknum, 1, 1, FT_LAYOUT_VERSION, 0);
|
|
destroy_nonleaf_childinfo(BNC(child, 0));
|
|
set_BNC(child, 0, child_bnc);
|
|
BP_STATE(child, 0) = PT_AVAIL;
|
|
|
|
toku_bnc_flush_to_child(t->ft, parent_bnc, child, TXNID_NONE);
|
|
|
|
int parent_messages_present[num_parent_messages];
|
|
int child_messages_present[num_child_messages];
|
|
memset(parent_messages_present, 0, sizeof parent_messages_present);
|
|
memset(child_messages_present, 0, sizeof child_messages_present);
|
|
|
|
FIFO_ITERATE(child_bnc->buffer, key, keylen, val, vallen, type, msn, xids, is_fresh,
|
|
{
|
|
DBT keydbt;
|
|
DBT valdbt;
|
|
toku_fill_dbt(&keydbt, key, keylen);
|
|
toku_fill_dbt(&valdbt, val, vallen);
|
|
int found = 0;
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (dummy_cmp(NULL, &keydbt, parent_messages[i]->u.id.key) == 0 &&
|
|
msn.msn == parent_messages[i]->msn.msn) {
|
|
assert(parent_messages_present[i] == 0);
|
|
assert(found == 0);
|
|
assert(dummy_cmp(NULL, &valdbt, parent_messages[i]->u.id.val) == 0);
|
|
assert(type == parent_messages[i]->type);
|
|
assert(xids_get_innermost_xid(xids) == xids_get_innermost_xid(parent_messages[i]->xids));
|
|
assert(parent_messages_is_fresh[i] == is_fresh);
|
|
parent_messages_present[i]++;
|
|
found++;
|
|
}
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
if (dummy_cmp(NULL, &keydbt, child_messages[i]->u.id.key) == 0 &&
|
|
msn.msn == child_messages[i]->msn.msn) {
|
|
assert(child_messages_present[i] == 0);
|
|
assert(found == 0);
|
|
assert(dummy_cmp(NULL, &valdbt, child_messages[i]->u.id.val) == 0);
|
|
assert(type == child_messages[i]->type);
|
|
assert(xids_get_innermost_xid(xids) == xids_get_innermost_xid(child_messages[i]->xids));
|
|
assert(child_messages_is_fresh[i] == is_fresh);
|
|
child_messages_present[i]++;
|
|
found++;
|
|
}
|
|
}
|
|
assert(found == 1);
|
|
});
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
assert(parent_messages_present[i] == 1);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
assert(child_messages_present[i] == 1);
|
|
}
|
|
|
|
xids_destroy(&xids_0);
|
|
xids_destroy(&xids_123);
|
|
xids_destroy(&xids_234);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
toku_free(parent_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.key);
|
|
toku_free(parent_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.val);
|
|
toku_free(parent_messages[i]);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
toku_free(child_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) child_messages[i]->u.id.key);
|
|
toku_free(child_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) child_messages[i]->u.id.val);
|
|
toku_free(child_messages[i]);
|
|
}
|
|
destroy_nonleaf_childinfo(parent_bnc);
|
|
toku_ftnode_free(&child);
|
|
toku_free(parent_messages);
|
|
toku_free(child_messages);
|
|
toku_free(parent_messages_is_fresh);
|
|
toku_free(child_messages_is_fresh);
|
|
}
|
|
|
|
// flush from one internal node to another, where the child has 8 buffers
|
|
static void
|
|
flush_to_internal_multiple(FT_HANDLE t) {
|
|
int r;
|
|
|
|
FT_MSG_S **MALLOC_N(4096,parent_messages); // 128k / 32 = 4096
|
|
FT_MSG_S **MALLOC_N(4096,child_messages);
|
|
bool *MALLOC_N(4096,parent_messages_is_fresh);
|
|
bool *MALLOC_N(4096,child_messages_is_fresh);
|
|
memset(parent_messages_is_fresh, 0, 4096*(sizeof parent_messages_is_fresh[0]));
|
|
memset(child_messages_is_fresh, 0, 4096*(sizeof child_messages_is_fresh[0]));
|
|
|
|
XIDS xids_0 = xids_get_root_xids();
|
|
XIDS xids_123, xids_234;
|
|
r = xids_create_child(xids_0, &xids_123, (TXNID)123);
|
|
CKERR(r);
|
|
r = xids_create_child(xids_0, &xids_234, (TXNID)234);
|
|
CKERR(r);
|
|
|
|
NONLEAF_CHILDINFO child_bncs[8];
|
|
FT_MSG childkeys[7];
|
|
int i;
|
|
for (i = 0; i < 8; ++i) {
|
|
child_bncs[i] = toku_create_empty_nl();
|
|
if (i < 7) {
|
|
childkeys[i] = NULL;
|
|
}
|
|
}
|
|
int total_size = 0;
|
|
for (i = 0; total_size < 128*1024; ++i) {
|
|
total_size -= toku_bnc_memory_used(child_bncs[i%8]);
|
|
insert_random_message(child_bncs[i%8], &child_messages[i], &child_messages_is_fresh[i], xids_123, i%8);
|
|
total_size += toku_bnc_memory_used(child_bncs[i%8]);
|
|
if (i % 8 < 7) {
|
|
if (childkeys[i%8] == NULL || dummy_cmp(NULL, child_messages[i]->u.id.key, childkeys[i%8]->u.id.key) > 0) {
|
|
childkeys[i%8] = child_messages[i];
|
|
}
|
|
}
|
|
}
|
|
int num_child_messages = i;
|
|
|
|
NONLEAF_CHILDINFO parent_bnc = toku_create_empty_nl();
|
|
for (i = 0; toku_bnc_memory_used(parent_bnc) < 128*1024; ++i) {
|
|
insert_random_message(parent_bnc, &parent_messages[i], &parent_messages_is_fresh[i], xids_234, 0);
|
|
}
|
|
int num_parent_messages = i;
|
|
|
|
FTNODE XMALLOC(child);
|
|
BLOCKNUM blocknum = { 42 };
|
|
toku_initialize_empty_ftnode(child, blocknum, 1, 8, FT_LAYOUT_VERSION, 0);
|
|
for (i = 0; i < 8; ++i) {
|
|
destroy_nonleaf_childinfo(BNC(child, i));
|
|
set_BNC(child, i, child_bncs[i]);
|
|
BP_STATE(child, i) = PT_AVAIL;
|
|
if (i < 7) {
|
|
toku_clone_dbt(&child->childkeys[i], *childkeys[i]->u.id.key);
|
|
}
|
|
}
|
|
|
|
toku_bnc_flush_to_child(t->ft, parent_bnc, child, TXNID_NONE);
|
|
|
|
int total_messages = 0;
|
|
for (i = 0; i < 8; ++i) {
|
|
total_messages += toku_bnc_n_entries(BNC(child, i));
|
|
}
|
|
assert(total_messages == num_parent_messages + num_child_messages);
|
|
int parent_messages_present[num_parent_messages];
|
|
int child_messages_present[num_child_messages];
|
|
memset(parent_messages_present, 0, sizeof parent_messages_present);
|
|
memset(child_messages_present, 0, sizeof child_messages_present);
|
|
|
|
for (int j = 0; j < 8; ++j) {
|
|
FIFO_ITERATE(child_bncs[j]->buffer, key, keylen, val, vallen, type, msn, xids, is_fresh,
|
|
{
|
|
DBT keydbt;
|
|
DBT valdbt;
|
|
toku_fill_dbt(&keydbt, key, keylen);
|
|
toku_fill_dbt(&valdbt, val, vallen);
|
|
int found = 0;
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (dummy_cmp(NULL, &keydbt, parent_messages[i]->u.id.key) == 0 &&
|
|
msn.msn == parent_messages[i]->msn.msn) {
|
|
assert(parent_messages_present[i] == 0);
|
|
assert(found == 0);
|
|
assert(dummy_cmp(NULL, &valdbt, parent_messages[i]->u.id.val) == 0);
|
|
assert(type == parent_messages[i]->type);
|
|
assert(xids_get_innermost_xid(xids) == xids_get_innermost_xid(parent_messages[i]->xids));
|
|
assert(parent_messages_is_fresh[i] == is_fresh);
|
|
parent_messages_present[i]++;
|
|
found++;
|
|
}
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
if (dummy_cmp(NULL, &keydbt, child_messages[i]->u.id.key) == 0 &&
|
|
msn.msn == child_messages[i]->msn.msn) {
|
|
assert(child_messages_present[i] == 0);
|
|
assert(found == 0);
|
|
assert(dummy_cmp(NULL, &valdbt, child_messages[i]->u.id.val) == 0);
|
|
assert(type == child_messages[i]->type);
|
|
assert(xids_get_innermost_xid(xids) == xids_get_innermost_xid(child_messages[i]->xids));
|
|
assert(child_messages_is_fresh[i] == is_fresh);
|
|
child_messages_present[i]++;
|
|
found++;
|
|
}
|
|
}
|
|
assert(found == 1);
|
|
});
|
|
}
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
assert(parent_messages_present[i] == 1);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
assert(child_messages_present[i] == 1);
|
|
}
|
|
|
|
xids_destroy(&xids_0);
|
|
xids_destroy(&xids_123);
|
|
xids_destroy(&xids_234);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
toku_free(parent_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.key);
|
|
toku_free(parent_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.val);
|
|
toku_free(parent_messages[i]);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
toku_free(child_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) child_messages[i]->u.id.key);
|
|
toku_free(child_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) child_messages[i]->u.id.val);
|
|
toku_free(child_messages[i]);
|
|
}
|
|
destroy_nonleaf_childinfo(parent_bnc);
|
|
toku_ftnode_free(&child);
|
|
toku_free(parent_messages);
|
|
toku_free(child_messages);
|
|
toku_free(parent_messages_is_fresh);
|
|
toku_free(child_messages_is_fresh);
|
|
}
|
|
|
|
// flush from one internal node to a leaf node, which has 8 basement
|
|
// nodes
|
|
//
|
|
// if make_leaf_up_to_date is true, then apply the messages that are stale
|
|
// in the parent to the leaf before doing the flush, otherwise assume the
|
|
// leaf was just read off disk
|
|
//
|
|
// if use_flush is true, use a buffer flush, otherwise, use maybe_apply
|
|
static void
|
|
flush_to_leaf(FT_HANDLE t, bool make_leaf_up_to_date, bool use_flush) {
|
|
int r;
|
|
|
|
FT_MSG_S **MALLOC_N(4096,parent_messages); // 128k / 32 = 4096
|
|
LEAFENTRY *MALLOC_N(4096,child_messages);
|
|
bool *MALLOC_N(4096,parent_messages_is_fresh);
|
|
memset(parent_messages_is_fresh, 0, 4096*(sizeof parent_messages_is_fresh[0]));
|
|
int *MALLOC_N(4096,parent_messages_applied);
|
|
memset(parent_messages_applied, 0, 4096*(sizeof parent_messages_applied[0]));
|
|
|
|
XIDS xids_0 = xids_get_root_xids();
|
|
XIDS xids_123, xids_234;
|
|
r = xids_create_child(xids_0, &xids_123, (TXNID)123);
|
|
CKERR(r);
|
|
r = xids_create_child(xids_0, &xids_234, (TXNID)234);
|
|
CKERR(r);
|
|
|
|
BASEMENTNODE child_blbs[8];
|
|
DBT childkeys[7];
|
|
int i;
|
|
for (i = 0; i < 8; ++i) {
|
|
child_blbs[i] = toku_create_empty_bn();
|
|
if (i < 7) {
|
|
toku_init_dbt(&childkeys[i]);
|
|
}
|
|
}
|
|
|
|
FTNODE XMALLOC(child);
|
|
BLOCKNUM blocknum = { 42 };
|
|
toku_initialize_empty_ftnode(child, blocknum, 0, 8, FT_LAYOUT_VERSION, 0);
|
|
for (i = 0; i < 8; ++i) {
|
|
destroy_basement_node(BLB(child, i));
|
|
set_BLB(child, i, child_blbs[i]);
|
|
BP_STATE(child, i) = PT_AVAIL;
|
|
}
|
|
|
|
int total_size = 0;
|
|
for (i = 0; total_size < 128*1024; ++i) {
|
|
total_size -= child_blbs[i%8]->n_bytes_in_buffer;
|
|
insert_random_message_to_bn(t, child_blbs[i%8], &child_messages[i], xids_123, i%8);
|
|
total_size += child_blbs[i%8]->n_bytes_in_buffer;
|
|
if (i % 8 < 7) {
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
if (childkeys[i%8].size == 0 || dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &childkeys[i%8]) > 0) {
|
|
toku_fill_dbt(&childkeys[i%8], key, keylen);
|
|
}
|
|
}
|
|
}
|
|
int num_child_messages = i;
|
|
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
if (i % 8 < 7) {
|
|
assert(dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &childkeys[i%8]) <= 0);
|
|
}
|
|
}
|
|
|
|
{
|
|
int num_stale = random() % 2000;
|
|
memset(&parent_messages_is_fresh[num_stale], true, (4096 - num_stale) * (sizeof parent_messages_is_fresh[0]));
|
|
}
|
|
NONLEAF_CHILDINFO parent_bnc = toku_create_empty_nl();
|
|
MSN max_parent_msn = MIN_MSN;
|
|
for (i = 0; toku_bnc_memory_used(parent_bnc) < 128*1024; ++i) {
|
|
insert_random_update_message(parent_bnc, &parent_messages[i], parent_messages_is_fresh[i], xids_234, i%8, &parent_messages_applied[i], &max_parent_msn);
|
|
}
|
|
int num_parent_messages = i;
|
|
|
|
for (i = 0; i < 7; ++i) {
|
|
toku_clone_dbt(&child->childkeys[i], childkeys[i]);
|
|
}
|
|
|
|
if (make_leaf_up_to_date) {
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (!parent_messages_is_fresh[i]) {
|
|
toku_ft_leaf_apply_cmd(t->ft->compare_fun, t->ft->update_fun, &t->ft->descriptor, child, -1, parent_messages[i], make_gc_info(false), NULL, NULL);
|
|
}
|
|
}
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child, i)->stale_ancestor_messages_applied = true;
|
|
}
|
|
} else {
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child, i)->stale_ancestor_messages_applied = false;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (make_leaf_up_to_date && !parent_messages_is_fresh[i]) {
|
|
assert(parent_messages_applied[i] == 1);
|
|
} else {
|
|
assert(parent_messages_applied[i] == 0);
|
|
}
|
|
}
|
|
|
|
if (use_flush) {
|
|
toku_bnc_flush_to_child(t->ft, parent_bnc, child, TXNID_NONE);
|
|
destroy_nonleaf_childinfo(parent_bnc);
|
|
} else {
|
|
FTNODE XMALLOC(parentnode);
|
|
BLOCKNUM parentblocknum = { 17 };
|
|
toku_initialize_empty_ftnode(parentnode, parentblocknum, 1, 1, FT_LAYOUT_VERSION, 0);
|
|
destroy_nonleaf_childinfo(BNC(parentnode, 0));
|
|
set_BNC(parentnode, 0, parent_bnc);
|
|
BP_STATE(parentnode, 0) = PT_AVAIL;
|
|
parentnode->max_msn_applied_to_node_on_disk = max_parent_msn;
|
|
struct ancestors ancestors = { .node = parentnode, .childnum = 0, .next = NULL };
|
|
const struct pivot_bounds infinite_bounds = { .lower_bound_exclusive = NULL, .upper_bound_inclusive = NULL };
|
|
bool msgs_applied;
|
|
toku_apply_ancestors_messages_to_node(t, child, &ancestors, &infinite_bounds, &msgs_applied);
|
|
|
|
FIFO_ITERATE(parent_bnc->buffer, key, keylen, val, vallen, type, msn, xids, is_fresh,
|
|
{
|
|
key = key; keylen = keylen; val = val; vallen = vallen; type = type; msn = msn; xids = xids;
|
|
assert(!is_fresh);
|
|
});
|
|
invariant(parent_bnc->fresh_message_tree.size() + parent_bnc->stale_message_tree.size()
|
|
== (uint32_t) num_parent_messages);
|
|
|
|
toku_ftnode_free(&parentnode);
|
|
}
|
|
|
|
int total_messages = 0;
|
|
for (i = 0; i < 8; ++i) {
|
|
total_messages += toku_omt_size(BLB_BUFFER(child, i));
|
|
}
|
|
assert(total_messages <= num_parent_messages + num_child_messages);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
assert(parent_messages_applied[i] == 1);
|
|
}
|
|
|
|
int parent_messages_present[num_parent_messages];
|
|
int child_messages_present[num_child_messages];
|
|
memset(parent_messages_present, 0, sizeof parent_messages_present);
|
|
memset(child_messages_present, 0, sizeof child_messages_present);
|
|
for (int j = 0; j < 8; ++j) {
|
|
OMT omt = BLB_BUFFER(child, j);
|
|
uint32_t len = toku_omt_size(omt);
|
|
for (uint32_t idx = 0; idx < len; ++idx) {
|
|
LEAFENTRY le;
|
|
DBT keydbt, valdbt;
|
|
{
|
|
OMTVALUE v;
|
|
r = toku_omt_fetch(omt, idx, &v);
|
|
assert_zero(r);
|
|
CAST_FROM_VOIDP(le, v);
|
|
uint32_t keylen, vallen;
|
|
void *keyp = le_key_and_len(le, &keylen);
|
|
void *valp = le_latest_val_and_len(le, &vallen);
|
|
toku_fill_dbt(&keydbt, keyp, keylen);
|
|
toku_fill_dbt(&valdbt, valp, vallen);
|
|
}
|
|
int found = 0;
|
|
for (i = num_parent_messages - 1; i >= 0; --i) {
|
|
if (dummy_cmp(NULL, &keydbt, parent_messages[i]->u.id.key) == 0) {
|
|
if (found == 0) {
|
|
struct orthopush_flush_update_fun_extra *CAST_FROM_VOIDP(e, parent_messages[i]->u.id.val->data);
|
|
assert(dummy_cmp(NULL, &valdbt, &e->new_val) == 0);
|
|
found++;
|
|
}
|
|
assert(parent_messages_present[i] == 0);
|
|
parent_messages_present[i]++;
|
|
}
|
|
}
|
|
for (i = j + (~7 & (num_child_messages - 1)); i >= 0; i -= 8) {
|
|
if (i >= num_child_messages) { continue; }
|
|
DBT childkeydbt, childvaldbt;
|
|
{
|
|
uint32_t keylen, vallen;
|
|
void *keyp = le_key_and_len(child_messages[i], &keylen);
|
|
void *valp = le_latest_val_and_len(child_messages[i], &vallen);
|
|
toku_fill_dbt(&childkeydbt, keyp, keylen);
|
|
toku_fill_dbt(&childvaldbt, valp, vallen);
|
|
}
|
|
if (dummy_cmp(NULL, &keydbt, &childkeydbt) == 0) {
|
|
if (found == 0) {
|
|
assert(dummy_cmp(NULL, &valdbt, &childvaldbt) == 0);
|
|
found++;
|
|
}
|
|
assert(child_messages_present[i] == 0);
|
|
child_messages_present[i]++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
assert(parent_messages_present[i] == 1);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
assert(child_messages_present[i] == 1);
|
|
}
|
|
|
|
xids_destroy(&xids_0);
|
|
xids_destroy(&xids_123);
|
|
xids_destroy(&xids_234);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
toku_free(parent_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.key);
|
|
struct orthopush_flush_update_fun_extra *CAST_FROM_VOIDP(extra, parent_messages[i]->u.id.val->data);
|
|
toku_free(extra->new_val.data);
|
|
toku_free(parent_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.val);
|
|
toku_free(parent_messages[i]);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
toku_free(child_messages[i]);
|
|
}
|
|
toku_ftnode_free(&child);
|
|
toku_free(parent_messages);
|
|
toku_free(child_messages);
|
|
toku_free(parent_messages_is_fresh);
|
|
toku_free(parent_messages_applied);
|
|
}
|
|
|
|
// flush from one internal node to a leaf node, which has 8 basement
|
|
// nodes, but only using maybe_apply, and with actual pivot bounds
|
|
//
|
|
// if make_leaf_up_to_date is true, then apply the messages that are stale
|
|
// in the parent to the leaf before doing the flush, otherwise assume the
|
|
// leaf was just read off disk
|
|
static void
|
|
flush_to_leaf_with_keyrange(FT_HANDLE t, bool make_leaf_up_to_date) {
|
|
int r;
|
|
|
|
FT_MSG_S **MALLOC_N(4096,parent_messages); // 128k / 32 = 4k
|
|
LEAFENTRY *MALLOC_N(4096,child_messages);
|
|
bool *MALLOC_N(4096,parent_messages_is_fresh);
|
|
memset(parent_messages_is_fresh, 0, 4096*(sizeof parent_messages_is_fresh[0]));
|
|
int *MALLOC_N(4096,parent_messages_applied);
|
|
memset(parent_messages_applied, 0, 4096*(sizeof parent_messages_applied[0]));
|
|
|
|
XIDS xids_0 = xids_get_root_xids();
|
|
XIDS xids_123, xids_234;
|
|
r = xids_create_child(xids_0, &xids_123, (TXNID)123);
|
|
CKERR(r);
|
|
r = xids_create_child(xids_0, &xids_234, (TXNID)234);
|
|
CKERR(r);
|
|
|
|
BASEMENTNODE child_blbs[8];
|
|
DBT childkeys[8];
|
|
int i;
|
|
for (i = 0; i < 8; ++i) {
|
|
child_blbs[i] = toku_create_empty_bn();
|
|
toku_init_dbt(&childkeys[i]);
|
|
}
|
|
|
|
FTNODE XMALLOC(child);
|
|
BLOCKNUM blocknum = { 42 };
|
|
toku_initialize_empty_ftnode(child, blocknum, 0, 8, FT_LAYOUT_VERSION, 0);
|
|
for (i = 0; i < 8; ++i) {
|
|
destroy_basement_node(BLB(child, i));
|
|
set_BLB(child, i, child_blbs[i]);
|
|
BP_STATE(child, i) = PT_AVAIL;
|
|
}
|
|
|
|
int total_size = 0;
|
|
for (i = 0; total_size < 128*1024; ++i) {
|
|
total_size -= child_blbs[i%8]->n_bytes_in_buffer;
|
|
insert_random_message_to_bn(t, child_blbs[i%8], &child_messages[i], xids_123, i%8);
|
|
total_size += child_blbs[i%8]->n_bytes_in_buffer;
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
if (childkeys[i%8].size == 0 || dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &childkeys[i%8]) > 0) {
|
|
toku_fill_dbt(&childkeys[i%8], key, keylen);
|
|
}
|
|
}
|
|
int num_child_messages = i;
|
|
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
assert(dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &childkeys[i%8]) <= 0);
|
|
}
|
|
|
|
{
|
|
int num_stale = random() % 2000;
|
|
memset(&parent_messages_is_fresh[num_stale], true, (4096 - num_stale) * (sizeof parent_messages_is_fresh[0]));
|
|
}
|
|
NONLEAF_CHILDINFO parent_bnc = toku_create_empty_nl();
|
|
MSN max_parent_msn = MIN_MSN;
|
|
for (i = 0; toku_bnc_memory_used(parent_bnc) < 128*1024; ++i) {
|
|
insert_random_update_message(parent_bnc, &parent_messages[i], parent_messages_is_fresh[i], xids_234, i%8, &parent_messages_applied[i], &max_parent_msn);
|
|
}
|
|
int num_parent_messages = i;
|
|
|
|
for (i = 0; i < 7; ++i) {
|
|
toku_clone_dbt(&child->childkeys[i], childkeys[i]);
|
|
}
|
|
|
|
if (make_leaf_up_to_date) {
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (dummy_cmp(NULL, parent_messages[i]->u.id.key, &childkeys[7]) <= 0 &&
|
|
!parent_messages_is_fresh[i]) {
|
|
toku_ft_leaf_apply_cmd(t->ft->compare_fun, t->ft->update_fun, &t->ft->descriptor, child, -1, parent_messages[i], make_gc_info(false), NULL, NULL);
|
|
}
|
|
}
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child, i)->stale_ancestor_messages_applied = true;
|
|
}
|
|
} else {
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child, i)->stale_ancestor_messages_applied = false;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (make_leaf_up_to_date &&
|
|
dummy_cmp(NULL, parent_messages[i]->u.id.key, &childkeys[7]) <= 0 &&
|
|
!parent_messages_is_fresh[i]) {
|
|
assert(parent_messages_applied[i] == 1);
|
|
} else {
|
|
assert(parent_messages_applied[i] == 0);
|
|
}
|
|
}
|
|
|
|
FTNODE XMALLOC(parentnode);
|
|
BLOCKNUM parentblocknum = { 17 };
|
|
toku_initialize_empty_ftnode(parentnode, parentblocknum, 1, 1, FT_LAYOUT_VERSION, 0);
|
|
destroy_nonleaf_childinfo(BNC(parentnode, 0));
|
|
set_BNC(parentnode, 0, parent_bnc);
|
|
BP_STATE(parentnode, 0) = PT_AVAIL;
|
|
parentnode->max_msn_applied_to_node_on_disk = max_parent_msn;
|
|
struct ancestors ancestors = { .node = parentnode, .childnum = 0, .next = NULL };
|
|
DBT lbe, ubi;
|
|
const struct pivot_bounds bounds = {
|
|
.lower_bound_exclusive = toku_init_dbt(&lbe),
|
|
.upper_bound_inclusive = toku_clone_dbt(&ubi, childkeys[7])
|
|
};
|
|
bool msgs_applied;
|
|
toku_apply_ancestors_messages_to_node(t, child, &ancestors, &bounds, &msgs_applied);
|
|
|
|
FIFO_ITERATE(parent_bnc->buffer, key, keylen, val, vallen, type, msn, xids, is_fresh,
|
|
{
|
|
val = val; vallen = vallen; type = type; msn = msn; xids = xids;
|
|
DBT keydbt;
|
|
toku_fill_dbt(&keydbt, key, keylen);
|
|
if (dummy_cmp(NULL, &keydbt, &childkeys[7]) > 0) {
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (dummy_cmp(NULL, &keydbt, parent_messages[i]->u.id.key) == 0 &&
|
|
msn.msn == parent_messages[i]->msn.msn) {
|
|
assert(is_fresh == parent_messages_is_fresh[i]);
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
assert(!is_fresh);
|
|
}
|
|
});
|
|
|
|
toku_ftnode_free(&parentnode);
|
|
|
|
int total_messages = 0;
|
|
for (i = 0; i < 8; ++i) {
|
|
total_messages += toku_omt_size(BLB_BUFFER(child, i));
|
|
}
|
|
assert(total_messages <= num_parent_messages + num_child_messages);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (dummy_cmp(NULL, parent_messages[i]->u.id.key, &childkeys[7]) <= 0) {
|
|
assert(parent_messages_applied[i] == 1);
|
|
} else {
|
|
assert(parent_messages_applied[i] == 0);
|
|
}
|
|
}
|
|
|
|
xids_destroy(&xids_0);
|
|
xids_destroy(&xids_123);
|
|
xids_destroy(&xids_234);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
toku_free(parent_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.key);
|
|
struct orthopush_flush_update_fun_extra *CAST_FROM_VOIDP(extra, parent_messages[i]->u.id.val->data);
|
|
toku_free(extra->new_val.data);
|
|
toku_free(parent_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.val);
|
|
toku_free(parent_messages[i]);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
toku_free(child_messages[i]);
|
|
}
|
|
toku_free(ubi.data);
|
|
toku_ftnode_free(&child);
|
|
toku_free(parent_messages);
|
|
toku_free(child_messages);
|
|
toku_free(parent_messages_is_fresh);
|
|
toku_free(parent_messages_applied);
|
|
}
|
|
|
|
// create identical leaf nodes and then buffer flush to one and
|
|
// maybe_apply to the other, and compare the results, they should be the
|
|
// same.
|
|
//
|
|
// if make_leaf_up_to_date is true, then apply the messages that are stale
|
|
// in the parent to the leaf before doing the flush, otherwise assume the
|
|
// leaf was just read off disk
|
|
static void
|
|
compare_apply_and_flush(FT_HANDLE t, bool make_leaf_up_to_date) {
|
|
int r;
|
|
|
|
FT_MSG_S **MALLOC_N(4096,parent_messages); // 128k / 32 = 4k
|
|
LEAFENTRY *MALLOC_N(4096,child_messages);
|
|
bool *MALLOC_N(4096,parent_messages_is_fresh);
|
|
memset(parent_messages_is_fresh, 0, 4096*(sizeof parent_messages_is_fresh[0]));
|
|
int *MALLOC_N(4096,parent_messages_applied);
|
|
memset(parent_messages_applied, 0, 4096*(sizeof parent_messages_applied[0]));
|
|
|
|
XIDS xids_0 = xids_get_root_xids();
|
|
XIDS xids_123, xids_234;
|
|
r = xids_create_child(xids_0, &xids_123, (TXNID)123);
|
|
CKERR(r);
|
|
r = xids_create_child(xids_0, &xids_234, (TXNID)234);
|
|
CKERR(r);
|
|
|
|
BASEMENTNODE child1_blbs[8], child2_blbs[8];
|
|
DBT child1keys[7], child2keys[7];
|
|
int i;
|
|
for (i = 0; i < 8; ++i) {
|
|
child1_blbs[i] = toku_create_empty_bn();
|
|
child2_blbs[i] = toku_create_empty_bn();
|
|
if (i < 7) {
|
|
toku_init_dbt(&child1keys[i]);
|
|
toku_init_dbt(&child2keys[i]);
|
|
}
|
|
}
|
|
|
|
FTNODE XMALLOC(child1), XMALLOC(child2);
|
|
BLOCKNUM blocknum = { 42 };
|
|
toku_initialize_empty_ftnode(child1, blocknum, 0, 8, FT_LAYOUT_VERSION, 0);
|
|
toku_initialize_empty_ftnode(child2, blocknum, 0, 8, FT_LAYOUT_VERSION, 0);
|
|
for (i = 0; i < 8; ++i) {
|
|
destroy_basement_node(BLB(child1, i));
|
|
set_BLB(child1, i, child1_blbs[i]);
|
|
BP_STATE(child1, i) = PT_AVAIL;
|
|
destroy_basement_node(BLB(child2, i));
|
|
set_BLB(child2, i, child2_blbs[i]);
|
|
BP_STATE(child2, i) = PT_AVAIL;
|
|
}
|
|
|
|
int total_size = 0;
|
|
for (i = 0; total_size < 128*1024; ++i) {
|
|
total_size -= child1_blbs[i%8]->n_bytes_in_buffer;
|
|
insert_same_message_to_bns(t, child1_blbs[i%8], child2_blbs[i%8], &child_messages[i], xids_123, i%8);
|
|
total_size += child1_blbs[i%8]->n_bytes_in_buffer;
|
|
if (i % 8 < 7) {
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
if (child1keys[i%8].size == 0 || dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &child1keys[i%8]) > 0) {
|
|
toku_fill_dbt(&child1keys[i%8], key, keylen);
|
|
toku_fill_dbt(&child2keys[i%8], key, keylen);
|
|
}
|
|
}
|
|
}
|
|
int num_child_messages = i;
|
|
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
uint32_t keylen;
|
|
char *CAST_FROM_VOIDP(key, le_key_and_len(child_messages[i], &keylen));
|
|
DBT keydbt;
|
|
if (i % 8 < 7) {
|
|
assert(dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &child1keys[i%8]) <= 0);
|
|
assert(dummy_cmp(NULL, toku_fill_dbt(&keydbt, key, keylen), &child2keys[i%8]) <= 0);
|
|
}
|
|
}
|
|
|
|
{
|
|
int num_stale = random() % 2000;
|
|
memset(&parent_messages_is_fresh[num_stale], true, (4096 - num_stale) * (sizeof parent_messages_is_fresh[0]));
|
|
}
|
|
NONLEAF_CHILDINFO parent_bnc = toku_create_empty_nl();
|
|
MSN max_parent_msn = MIN_MSN;
|
|
for (i = 0; toku_bnc_memory_used(parent_bnc) < 128*1024; ++i) {
|
|
insert_random_update_message(parent_bnc, &parent_messages[i], parent_messages_is_fresh[i], xids_234, i%8, &parent_messages_applied[i], &max_parent_msn);
|
|
}
|
|
int num_parent_messages = i;
|
|
|
|
for (i = 0; i < 7; ++i) {
|
|
toku_clone_dbt(&child1->childkeys[i], child1keys[i]);
|
|
toku_clone_dbt(&child2->childkeys[i], child2keys[i]);
|
|
}
|
|
|
|
if (make_leaf_up_to_date) {
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
if (!parent_messages_is_fresh[i]) {
|
|
toku_ft_leaf_apply_cmd(t->ft->compare_fun, t->ft->update_fun, &t->ft->descriptor, child1, -1, parent_messages[i], make_gc_info(false), NULL, NULL);
|
|
toku_ft_leaf_apply_cmd(t->ft->compare_fun, t->ft->update_fun, &t->ft->descriptor, child2, -1, parent_messages[i], make_gc_info(false), NULL, NULL);
|
|
}
|
|
}
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child1, i)->stale_ancestor_messages_applied = true;
|
|
BLB(child2, i)->stale_ancestor_messages_applied = true;
|
|
}
|
|
} else {
|
|
for (i = 0; i < 8; ++i) {
|
|
BLB(child1, i)->stale_ancestor_messages_applied = false;
|
|
BLB(child2, i)->stale_ancestor_messages_applied = false;
|
|
}
|
|
}
|
|
|
|
toku_bnc_flush_to_child(t->ft, parent_bnc, child1, TXNID_NONE);
|
|
|
|
FTNODE XMALLOC(parentnode);
|
|
BLOCKNUM parentblocknum = { 17 };
|
|
toku_initialize_empty_ftnode(parentnode, parentblocknum, 1, 1, FT_LAYOUT_VERSION, 0);
|
|
destroy_nonleaf_childinfo(BNC(parentnode, 0));
|
|
set_BNC(parentnode, 0, parent_bnc);
|
|
BP_STATE(parentnode, 0) = PT_AVAIL;
|
|
parentnode->max_msn_applied_to_node_on_disk = max_parent_msn;
|
|
struct ancestors ancestors = { .node = parentnode, .childnum = 0, .next = NULL };
|
|
const struct pivot_bounds infinite_bounds = { .lower_bound_exclusive = NULL, .upper_bound_inclusive = NULL };
|
|
bool msgs_applied;
|
|
toku_apply_ancestors_messages_to_node(t, child2, &ancestors, &infinite_bounds, &msgs_applied);
|
|
|
|
FIFO_ITERATE(parent_bnc->buffer, key, keylen, val, vallen, type, msn, xids, is_fresh,
|
|
{
|
|
key = key; keylen = keylen; val = val; vallen = vallen; type = type; msn = msn; xids = xids;
|
|
assert(!is_fresh);
|
|
});
|
|
invariant(parent_bnc->fresh_message_tree.size() + parent_bnc->stale_message_tree.size()
|
|
== (uint32_t) num_parent_messages);
|
|
|
|
toku_ftnode_free(&parentnode);
|
|
|
|
for (int j = 0; j < 8; ++j) {
|
|
OMT omt1 = BLB_BUFFER(child1, j);
|
|
OMT omt2 = BLB_BUFFER(child2, j);
|
|
uint32_t len = toku_omt_size(omt1);
|
|
assert(len == toku_omt_size(omt2));
|
|
for (uint32_t idx = 0; idx < len; ++idx) {
|
|
LEAFENTRY le1, le2;
|
|
DBT key1dbt, val1dbt, key2dbt, val2dbt;
|
|
{
|
|
OMTVALUE v;
|
|
r = toku_omt_fetch(omt1, idx, &v);
|
|
assert_zero(r);
|
|
CAST_FROM_VOIDP(le1, v);
|
|
uint32_t keylen, vallen;
|
|
void *keyp = le_key_and_len(le1, &keylen);
|
|
void *valp = le_latest_val_and_len(le1, &vallen);
|
|
toku_fill_dbt(&key1dbt, keyp, keylen);
|
|
toku_fill_dbt(&val1dbt, valp, vallen);
|
|
}
|
|
{
|
|
OMTVALUE v;
|
|
r = toku_omt_fetch(omt2, idx, &v);
|
|
assert_zero(r);
|
|
CAST_FROM_VOIDP(le2, v);
|
|
uint32_t keylen, vallen;
|
|
void *keyp = le_key_and_len(le2, &keylen);
|
|
void *valp = le_latest_val_and_len(le2, &vallen);
|
|
toku_fill_dbt(&key2dbt, keyp, keylen);
|
|
toku_fill_dbt(&val2dbt, valp, vallen);
|
|
}
|
|
assert(dummy_cmp(NULL, &key1dbt, &key2dbt) == 0);
|
|
assert(dummy_cmp(NULL, &val1dbt, &val2dbt) == 0);
|
|
}
|
|
}
|
|
|
|
xids_destroy(&xids_0);
|
|
xids_destroy(&xids_123);
|
|
xids_destroy(&xids_234);
|
|
|
|
for (i = 0; i < num_parent_messages; ++i) {
|
|
toku_free(parent_messages[i]->u.id.key->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.key);
|
|
struct orthopush_flush_update_fun_extra *CAST_FROM_VOIDP(extra, parent_messages[i]->u.id.val->data);
|
|
toku_free(extra->new_val.data);
|
|
toku_free(parent_messages[i]->u.id.val->data);
|
|
toku_free((DBT *) parent_messages[i]->u.id.val);
|
|
toku_free(parent_messages[i]);
|
|
}
|
|
for (i = 0; i < num_child_messages; ++i) {
|
|
toku_free(child_messages[i]);
|
|
}
|
|
toku_ftnode_free(&child1);
|
|
toku_ftnode_free(&child2);
|
|
toku_free(parent_messages);
|
|
toku_free(child_messages);
|
|
toku_free(parent_messages_is_fresh);
|
|
toku_free(parent_messages_applied);
|
|
}
|
|
|
|
static void
|
|
parse_args(int argc, const char *argv[]) {
|
|
const char *progname=argv[0];
|
|
argc--; argv++;
|
|
while (argc>0) {
|
|
if (strcmp(argv[0],"-v")==0) {
|
|
verbose=1;
|
|
} else if (strcmp(argv[0],"-q")==0) {
|
|
verbose=0;
|
|
} else {
|
|
fprintf(stderr, "Usage:\n %s [-v] [-q]\n", progname);
|
|
exit(1);
|
|
}
|
|
argc--; argv++;
|
|
}
|
|
}
|
|
|
|
int
|
|
test_main (int argc, const char *argv[]) {
|
|
parse_args(argc, argv);
|
|
|
|
initialize_dummymsn();
|
|
int r;
|
|
CACHETABLE ct;
|
|
toku_cachetable_create(&ct, 0, ZERO_LSN, NULL_LOGGER);
|
|
unlink(fname);
|
|
FT_HANDLE t;
|
|
r = toku_open_ft_handle(fname, 1, &t, 128*1024, 4096, TOKU_DEFAULT_COMPRESSION_METHOD, ct, null_txn, toku_builtin_compare_fun); assert(r==0);
|
|
toku_ft_set_update(t, orthopush_flush_update_fun);
|
|
// HACK
|
|
t->ft->update_fun = orthopush_flush_update_fun;
|
|
|
|
for (int i = 0; i < 10; ++i) {
|
|
flush_to_internal(t);
|
|
}
|
|
for (int i = 0; i < 10; ++i) {
|
|
flush_to_internal_multiple(t);
|
|
}
|
|
for (int i = 0; i < 3; ++i) {
|
|
flush_to_leaf(t, false, false);
|
|
flush_to_leaf(t, false, true);
|
|
flush_to_leaf(t, true, false);
|
|
flush_to_leaf(t, true, true);
|
|
}
|
|
for (int i = 0; i < 10; ++i) {
|
|
flush_to_leaf_with_keyrange(t, false);
|
|
flush_to_leaf_with_keyrange(t, true);
|
|
compare_apply_and_flush(t, false);
|
|
compare_apply_and_flush(t, true);
|
|
}
|
|
|
|
r = toku_close_ft_handle_nolsn(t, 0); assert(r==0);
|
|
toku_cachetable_close(&ct);
|
|
|
|
return 0;
|
|
}
|