ft-serialize-benchmark.cc 10.1 KB
Newer Older
1 2
/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */
// vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4:
3 4 5 6 7 8 9 10
#ident "$Id$"
#ident "Copyright (c) 2007, 2008 Tokutek Inc.  All rights reserved."

#include <unistd.h>
#include <stdlib.h>
#include <sys/time.h>
#include "test.h"

11

12

13
#ifndef MIN
14
#define MIN(x, y) (((x) < (y)) ? (x) : (y))
15
#endif
16 17 18 19
const double USECS_PER_SEC = 1000000.0;

static int omt_cmp(OMTVALUE p, void *q)
{
20 21
    LEAFENTRY CAST_FROM_VOIDP(a, p);
    LEAFENTRY CAST_FROM_VOIDP(b, q);
22
    void *ak, *bk;
Yoni Fogel's avatar
Yoni Fogel committed
23
    uint32_t al, bl;
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
    ak = le_key_and_len(a, &al);
    bk = le_key_and_len(b, &bl);
    int l = MIN(al, bl);
    int c = memcmp(ak, bk, l);
    if (c < 0) { return -1; }
    if (c > 0) { return +1; }
    int d = al - bl;
    if (d < 0) { return -1; }
    if (d > 0) { return +1; }
    else       { return  0; }
}

static LEAFENTRY
le_fastmalloc(char *key, int keylen, char *val, int vallen)
{
39 40
    LEAFENTRY CAST_FROM_VOIDP(r, toku_malloc(sizeof(r->type) + sizeof(r->keylen) + sizeof(r->u.clean.vallen) +
                                             keylen + vallen));
41 42 43 44 45 46 47 48 49 50 51 52
    resource_assert(r);
    r->type = LE_CLEAN;
    r->keylen = keylen;
    r->u.clean.vallen = vallen;
    memcpy(&r->u.clean.key_val[0], key, keylen);
    memcpy(&r->u.clean.key_val[keylen], val, vallen);
    return r;
}

static int
long_key_cmp(DB *UU(e), const DBT *a, const DBT *b)
{
53 54
    const long *CAST_FROM_VOIDP(x, a->data);
    const long *CAST_FROM_VOIDP(y, b->data);
55 56 57 58 59
    return (*x > *y) - (*x < *y);
}

static void
test_serialize_leaf(int valsize, int nelts, double entropy) {
60
    //    struct ft_handle source_ft;
61
    struct ftnode *sn, *dn;
62

63
    int fd = open(__SRCFILE__ ".ft", O_RDWR|O_CREAT|O_BINARY, S_IRWXU|S_IRWXG|S_IRWXO); assert(fd >= 0);
64 65 66

    int r;

67 68 69 70 71 72 73 74 75 76
    XCALLOC(sn);

    sn->max_msn_applied_to_node_on_disk.msn = 0;
    sn->flags = 0x11223344;
    sn->thisnodename.b = 20;
    sn->layout_version = FT_LAYOUT_VERSION;
    sn->layout_version_original = FT_LAYOUT_VERSION;
    sn->height = 0;
    sn->n_children = 8;
    sn->dirty = 1;
77
    sn->oldest_referenced_xid_known = TXNID_NONE;
78 79 80 81 82 83
    MALLOC_N(sn->n_children, sn->bp);
    MALLOC_N(sn->n_children-1, sn->childkeys);
    sn->totalchildkeylens = 0;
    for (int i = 0; i < sn->n_children; ++i) {
        BP_STATE(sn,i) = PT_AVAIL;
        set_BLB(sn, i, toku_create_empty_bn());
84
    }
85
    int nperbn = nelts / sn->n_children;
86 87
    LEAFENTRY les[nelts];
    memset(les, 0, sizeof les);
88
    for (int ck = 0; ck < sn->n_children; ++ck) {
89 90 91 92 93 94
        long k;
        for (long i = 0; i < nperbn; ++i) {
            k = ck * nperbn + i;
            char buf[valsize];
            int c;
            for (c = 0; c < valsize * entropy; ) {
95 96 97
                int *p = (int *) &buf[c];
                *p = rand();
                c += sizeof(*p);
98 99 100
            }
            memset(&buf[c], 0, valsize - c);
            les[k] = le_fastmalloc((char *)&k, sizeof k, buf, sizeof buf);
101
            r = toku_omt_insert(BLB_BUFFER(sn, ck), les[k], omt_cmp, les[k], NULL); assert(r==0);
102
        }
103
        BLB_NBYTESINBUF(sn, ck) = nperbn*(KEY_VALUE_OVERHEAD+(sizeof(long)+valsize)) + toku_omt_size(BLB_BUFFER(sn, ck));
104
        if (ck < 7) {
105
            toku_memdup_dbt(&sn->childkeys[ck], &k, sizeof k);
106
            sn->totalchildkeylens += sizeof k;
107 108 109
        }
    }

110 111
    FT_HANDLE XMALLOC(brt);
    FT XCALLOC(brt_h);
112 113 114 115 116 117 118
    toku_ft_init(brt_h,
                 make_blocknum(0),
                 ZERO_LSN,
                 TXNID_NONE,
                 4*1024*1024,
                 128*1024,
                 TOKU_DEFAULT_COMPRESSION_METHOD);
119
    brt->ft = brt_h;
120
    
Zardosht Kasheff's avatar
Zardosht Kasheff committed
121
    brt_h->compare_fun = long_key_cmp;
122
    toku_blocktable_create_new(&brt_h->blocktable);
123
    { int r_truncate = ftruncate(fd, 0); CKERR(r_truncate); }
124 125 126 127 128 129 130 131 132 133
    //Want to use block #20
    BLOCKNUM b = make_blocknum(0);
    while (b.b < 20) {
        toku_allocate_blocknum(brt_h->blocktable, &b, brt_h);
    }
    assert(b.b == 20);

    {
        DISKOFF offset;
        DISKOFF size;
Yoni Fogel's avatar
Yoni Fogel committed
134
        toku_blocknum_realloc_on_disk(brt_h->blocktable, b, 100, &offset, brt_h, fd, false);
135 136 137 138 139 140 141 142 143
        assert(offset==BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);

        toku_translate_blocknum_to_offset_size(brt_h->blocktable, b, &offset, &size);
        assert(offset == BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);
        assert(size   == 100);
    }

    struct timeval t[2];
    gettimeofday(&t[0], NULL);
144
    FTNODE_DISK_DATA ndd = NULL;
Zardosht Kasheff's avatar
Zardosht Kasheff committed
145
    r = toku_serialize_ftnode_to(fd, make_blocknum(20), sn, &ndd, true, brt->ft, false);
146 147 148 149 150 151
    assert(r==0);
    gettimeofday(&t[1], NULL);
    double dt;
    dt = (t[1].tv_sec - t[0].tv_sec) + ((t[1].tv_usec - t[0].tv_usec) / USECS_PER_SEC);
    printf("serialize leaf:   %0.05lf\n", dt);

152
    struct ftnode_fetch_extra bfe;
Zardosht Kasheff's avatar
Zardosht Kasheff committed
153
    fill_bfe_for_full_read(&bfe, brt_h);
154
    gettimeofday(&t[0], NULL);
155 156
    FTNODE_DISK_DATA ndd2 = NULL;
    r = toku_deserialize_ftnode_from(fd, make_blocknum(20), 0/*pass zero for hash*/, &dn, &ndd2, &bfe);
157 158 159 160 161
    assert(r==0);
    gettimeofday(&t[1], NULL);
    dt = (t[1].tv_sec - t[0].tv_sec) + ((t[1].tv_usec - t[0].tv_usec) / USECS_PER_SEC);
    printf("deserialize leaf: %0.05lf\n", dt);

162
    toku_ftnode_free(&dn);
163 164

    for (int i = 0; i < nelts; ++i) {
165 166 167
        if (les[i]) {
            toku_free(les[i]);
        }
168
    }
169
    toku_ftnode_free(&sn);
170 171 172

    toku_block_free(brt_h->blocktable, BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);
    toku_blocktable_destroy(&brt_h->blocktable);
173
    toku_free(brt_h->h);
174 175
    toku_free(brt_h);
    toku_free(brt);
Zardosht Kasheff's avatar
Zardosht Kasheff committed
176 177
    toku_free(ndd);
    toku_free(ndd2);
178 179 180 181 182 183

    r = close(fd); assert(r != -1);
}

static void
test_serialize_nonleaf(int valsize, int nelts, double entropy) {
184 185
    //    struct ft_handle source_ft;
    struct ftnode sn, *dn;
186

187
    int fd = open(__SRCFILE__ ".ft", O_RDWR|O_CREAT|O_BINARY, S_IRWXU|S_IRWXG|S_IRWXO); assert(fd >= 0);
188 189 190

    int r;

191
    //    source_ft.fd=fd;
192 193 194
    sn.max_msn_applied_to_node_on_disk.msn = 0;
    sn.flags = 0x11223344;
    sn.thisnodename.b = 20;
195 196
    sn.layout_version = FT_LAYOUT_VERSION;
    sn.layout_version_original = FT_LAYOUT_VERSION;
197 198 199
    sn.height = 1;
    sn.n_children = 8;
    sn.dirty = 1;
200
    sn.oldest_referenced_xid_known = TXNID_NONE;
201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
    MALLOC_N(sn.n_children, sn.bp);
    MALLOC_N(sn.n_children-1, sn.childkeys);
    sn.totalchildkeylens = 0;
    for (int i = 0; i < sn.n_children; ++i) {
        BP_BLOCKNUM(&sn, i).b = 30 + (i*5);
        BP_STATE(&sn,i) = PT_AVAIL;
        set_BNC(&sn, i, toku_create_empty_nl());
    }
    //Create XIDS
    XIDS xids_0 = xids_get_root_xids();
    XIDS xids_123;
    r = xids_create_child(xids_0, &xids_123, (TXNID)123);
    CKERR(r);
    int nperchild = nelts / 8;
    for (int ck = 0; ck < sn.n_children; ++ck) {
        long k;
217
        NONLEAF_CHILDINFO bnc = BNC(&sn, ck);
218 219 220 221 222
        for (long i = 0; i < nperchild; ++i) {
            k = ck * nperchild + i;
            char buf[valsize];
            int c;
            for (c = 0; c < valsize * entropy; ) {
223 224 225
                int *p = (int *) &buf[c];
                *p = rand();
                c += sizeof(*p);
226 227
            }
            memset(&buf[c], 0, valsize - c);
228

229
            toku_bnc_insert_msg(bnc, &k, sizeof k, buf, valsize, FT_NONE, next_dummymsn(), xids_123, true, NULL, long_key_cmp);
230 231
        }
        if (ck < 7) {
232
            toku_memdup_dbt(&sn.childkeys[ck], &k, sizeof k);
233 234 235 236 237 238 239 240
            sn.totalchildkeylens += sizeof k;
        }
    }

    //Cleanup:
    xids_destroy(&xids_0);
    xids_destroy(&xids_123);

241 242
    FT_HANDLE XMALLOC(brt);
    FT XCALLOC(brt_h);
243 244 245 246 247 248 249
    toku_ft_init(brt_h,
                 make_blocknum(0),
                 ZERO_LSN,
                 TXNID_NONE,
                 4*1024*1024,
                 128*1024,
                 TOKU_DEFAULT_COMPRESSION_METHOD);
250
    brt->ft = brt_h;
251
    
Zardosht Kasheff's avatar
Zardosht Kasheff committed
252
    brt_h->compare_fun = long_key_cmp;
253
    toku_blocktable_create_new(&brt_h->blocktable);
254
    { int r_truncate = ftruncate(fd, 0); CKERR(r_truncate); }
255 256 257 258 259 260 261 262 263 264
    //Want to use block #20
    BLOCKNUM b = make_blocknum(0);
    while (b.b < 20) {
        toku_allocate_blocknum(brt_h->blocktable, &b, brt_h);
    }
    assert(b.b == 20);

    {
        DISKOFF offset;
        DISKOFF size;
Yoni Fogel's avatar
Yoni Fogel committed
265
        toku_blocknum_realloc_on_disk(brt_h->blocktable, b, 100, &offset, brt_h, fd, false);
266 267 268 269 270 271 272 273 274
        assert(offset==BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);

        toku_translate_blocknum_to_offset_size(brt_h->blocktable, b, &offset, &size);
        assert(offset == BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);
        assert(size   == 100);
    }

    struct timeval t[2];
    gettimeofday(&t[0], NULL);
275
    FTNODE_DISK_DATA ndd = NULL;
Zardosht Kasheff's avatar
Zardosht Kasheff committed
276
    r = toku_serialize_ftnode_to(fd, make_blocknum(20), &sn, &ndd, true, brt->ft, false);
277 278 279 280 281 282
    assert(r==0);
    gettimeofday(&t[1], NULL);
    double dt;
    dt = (t[1].tv_sec - t[0].tv_sec) + ((t[1].tv_usec - t[0].tv_usec) / USECS_PER_SEC);
    printf("serialize nonleaf:   %0.05lf\n", dt);

283
    struct ftnode_fetch_extra bfe;
Zardosht Kasheff's avatar
Zardosht Kasheff committed
284
    fill_bfe_for_full_read(&bfe, brt_h);
285
    gettimeofday(&t[0], NULL);
286 287
    FTNODE_DISK_DATA ndd2 = NULL;
    r = toku_deserialize_ftnode_from(fd, make_blocknum(20), 0/*pass zero for hash*/, &dn, &ndd2, &bfe);
288 289 290 291 292
    assert(r==0);
    gettimeofday(&t[1], NULL);
    dt = (t[1].tv_sec - t[0].tv_sec) + ((t[1].tv_usec - t[0].tv_usec) / USECS_PER_SEC);
    printf("deserialize nonleaf: %0.05lf\n", dt);

293
    toku_ftnode_free(&dn);
294 295

    for (int i = 0; i < sn.n_children-1; ++i) {
296
        toku_free(sn.childkeys[i].data);
297 298 299 300 301 302 303 304 305
    }
    for (int i = 0; i < sn.n_children; ++i) {
        destroy_nonleaf_childinfo(BNC(&sn, i));
    }
    toku_free(sn.bp);
    toku_free(sn.childkeys);

    toku_block_free(brt_h->blocktable, BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE);
    toku_blocktable_destroy(&brt_h->blocktable);
306
    toku_free(brt_h->h);
307 308
    toku_free(brt_h);
    toku_free(brt);
Zardosht Kasheff's avatar
Zardosht Kasheff committed
309 310
    toku_free(ndd);
    toku_free(ndd2);
311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326

    r = close(fd); assert(r != -1);
}

int
test_main (int argc __attribute__((__unused__)), const char *argv[] __attribute__((__unused__))) {
    long valsize, nelts;
    double entropy = 0.3;

    if (argc != 3) {
        fprintf(stderr, "Usage: %s <valsize> <nelts>\n", argv[0]);
        return 2;
    }
    valsize = strtol(argv[1], NULL, 0);
    nelts = strtol(argv[2], NULL, 0);

327
    initialize_dummymsn();
328 329 330 331 332
    test_serialize_leaf(valsize, nelts, entropy);
    test_serialize_nonleaf(valsize, nelts, entropy);

    return 0;
}