forked from vmware/splinterdb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsplinter_test.c
908 lines (764 loc) · 31.4 KB
/
splinter_test.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
// Copyright 2022 VMware, Inc.
// SPDX-License-Identifier: Apache-2.0
/*
* -----------------------------------------------------------------------------
* splinter_test.c --
*
* Exercises internal interfaces, using private APIs.
*
* If you're writing new unit tests of the public API, please do not use this
* file as a template or example.
*
* NOTE: There is some duplication of the splinter_do_inserts() in the test
* cases which adds considerable execution times. The test_inserts() test case
* will run with the default test configuration, which is sufficiently large
* enough to trigger a compaction. The expectation is that this unit test case
* will be invoked on its own, with a reduced memtable capacity to invoke the
* lookups test case(s):
*
* $ bin/unit/splinter_test test_inserts
* $ bin/unit/splinter_test --memtable-capacity-mib 4 test_lookups
* -----------------------------------------------------------------------------
*/
#include "splinterdb/public_platform.h"
#include "trunk.h"
#include "clockcache.h"
#include "allocator.h"
#include "task.h"
#include "functional/test.h"
#include "functional/test_async.h"
#include "test_common.h"
#include "config.h"
#include "unit_tests.h"
#include "ctest.h" // This is required for all test-case files.
typedef struct shadow_entry {
uint64 key_offset;
uint64 key_length;
uint64 value_length;
} shadow_entry;
typedef struct trunk_shadow {
data_config *data_cfg;
bool32 sorted;
writable_buffer entries;
writable_buffer data;
} trunk_shadow;
/* Function prototypes */
static uint64
splinter_do_inserts(void *datap,
trunk_handle *spl,
bool32 verify,
trunk_shadow *shadow); // Out
static platform_status
test_lookup_by_range(void *datap,
trunk_handle *spl,
uint64 num_inserts,
trunk_shadow *shadow,
uint64 num_ranges);
// Verify consistency of data after so-many inserts
#define TEST_VERIFY_GRANULARITY 100000
/* Macro to show progress message as workload is running */
#define SHOW_PCT_PROGRESS(op_num, num_ops, msg) \
do { \
if (((op_num) % ((num_ops) / 100)) == 0) { \
platform_default_log(PLATFORM_CR msg, (op_num) / ((num_ops) / 100)); \
} \
} while (0)
/*
* Global data declaration macro:
*/
CTEST_DATA(splinter)
{
// Declare head handles for io, allocator, cache and splinter allocation.
platform_heap_id hid;
// Thread-related config parameters. These don't change for unit tests
uint32 num_insert_threads;
uint32 num_lookup_threads;
uint32 max_async_inflight;
int spl_num_tables;
// Config structs required, as per splinter_test() setup work.
io_config io_cfg;
task_system_config task_cfg;
allocator_config al_cfg;
shard_log_config log_cfg;
rc_allocator al;
// Following get setup pointing to allocated memory
trunk_config *splinter_cfg;
data_config *data_cfg;
clockcache_config *cache_cfg;
platform_io_handle *io;
clockcache *clock_cache;
task_system *tasks;
test_message_generator gen;
// Test execution related configuration
test_exec_config test_exec_cfg;
};
/*
* -------------------------------------------------------------------------
* Setup Splinter configuration:
* -------------------------------------------------------------------------
*/
// clang-format off
CTEST_SETUP(splinter)
{
bool use_shmem = config_parse_use_shmem(Ctest_argc, (char **)Ctest_argv);
// Defaults: For basic unit-tests, use single threads
data->num_insert_threads = 1;
data->num_lookup_threads = 1;
data->max_async_inflight = 64;
data->spl_num_tables = 1;
bool32 cache_per_table = FALSE;
int num_tables = data->spl_num_tables; // Cache, for re-use below
uint8 num_caches = (cache_per_table ? num_tables : 1);
uint64 heap_capacity = MAX(1024 * MiB * num_caches, 512 * MiB * num_tables);
heap_capacity = MIN(heap_capacity, UINT32_MAX);
heap_capacity = MAX(heap_capacity, 2 * GiB);
// Create a heap for io, allocator, cache and splinter
platform_status rc = platform_heap_create(platform_get_module_id(),
heap_capacity,
use_shmem,
&data->hid);
platform_assert_status_ok(rc);
// Allocate memory for global config structures
data->splinter_cfg = TYPED_ARRAY_MALLOC(data->hid, data->splinter_cfg,
num_tables);
data->cache_cfg = TYPED_ARRAY_MALLOC(data->hid, data->cache_cfg, num_tables);
ZERO_STRUCT(data->test_exec_cfg);
rc = test_parse_args_n(data->splinter_cfg,
&data->data_cfg,
&data->io_cfg,
&data->al_cfg,
data->cache_cfg,
&data->log_cfg,
&data->task_cfg,
&data->test_exec_cfg,
&data->gen,
num_tables,
Ctest_argc, // argc/argv globals setup by CTests
(char **)Ctest_argv);
platform_assert_status_ok(rc);
// Establish Max active threads
uint32 total_threads = data->num_lookup_threads;
if (total_threads < data->num_insert_threads) {
total_threads = data->num_insert_threads;
}
// Check if IO subsystem has enough reqs for max async IOs inflight
io_config * io_cfgp = &data->io_cfg;
if (io_cfgp->async_queue_size < total_threads * data->max_async_inflight) {
io_cfgp->async_queue_size = ROUNDUP(total_threads * data->max_async_inflight, 32);
CTEST_LOG_INFO("Bumped up IO queue size to %lu\n", io_cfgp->async_queue_size);
}
if (io_cfgp->kernel_queue_size < total_threads * data->max_async_inflight) {
io_cfgp->kernel_queue_size =
ROUNDUP(total_threads * data->max_async_inflight, 32);
CTEST_LOG_INFO("Bumped up IO queue size to %lu\n",
io_cfgp->kernel_queue_size);
}
// Allocate and initialize the IO sub-system.
data->io = TYPED_MALLOC(data->hid, data->io);
ASSERT_TRUE((data->io != NULL));
rc = io_handle_init(data->io, &data->io_cfg, data->hid);
data->tasks = NULL;
rc = test_init_task_system(data->hid, data->io, &data->tasks, &data->task_cfg);
ASSERT_TRUE(SUCCESS(rc),
"Failed to init splinter state: %s\n",
platform_status_to_string(rc));
rc_allocator_init(&data->al, &data->al_cfg, (io_handle *)data->io, data->hid,
platform_get_module_id());
data->clock_cache = TYPED_ARRAY_MALLOC(data->hid, data->clock_cache, num_caches);
ASSERT_TRUE((data->clock_cache != NULL));
for (uint8 idx = 0; idx < num_caches; idx++) {
rc = clockcache_init(&data->clock_cache[idx],
&data->cache_cfg[idx],
(io_handle *)data->io,
(allocator *)&data->al,
"test",
data->hid,
platform_get_module_id());
ASSERT_TRUE(SUCCESS(rc), "clockcache_init() failed for index=%d. ", idx);
}
}
// clang-format on
/*
* Tear down memory allocated for various sub-systems. Shutdown Splinter.
*/
CTEST_TEARDOWN(splinter)
{
clockcache_deinit(data->clock_cache);
platform_free(data->hid, data->clock_cache);
allocator *alp = (allocator *)&data->al;
allocator_assert_noleaks(alp);
rc_allocator_deinit(&data->al);
test_deinit_task_system(data->hid, &data->tasks);
io_handle_deinit(data->io);
platform_free(data->hid, data->io);
if (data->cache_cfg) {
platform_free(data->hid, data->cache_cfg);
}
if (data->splinter_cfg) {
platform_free(data->hid, data->splinter_cfg);
}
platform_heap_destroy(&data->hid);
}
/*
* **************************************************************************
* Basic test case to verify trunk_insert() API and validate a very large #
* of inserts. This test case is designed to insert enough rows to trigger
* compaction. (We don't, quite, actually verify that compaction has occurred
* but based on the default test configs, we expect that it would trigger.)
* **************************************************************************
*/
CTEST2(splinter, test_inserts)
{
allocator *alp = (allocator *)&data->al;
trunk_handle *spl = trunk_create(data->splinter_cfg,
alp,
(cache *)data->clock_cache,
data->tasks,
test_generate_allocator_root_id(),
data->hid);
ASSERT_TRUE(spl != NULL);
// TRUE : Also do verification-after-inserts
uint64 num_inserts = splinter_do_inserts(data, spl, TRUE, NULL);
ASSERT_NOT_EQUAL(0,
num_inserts,
"Expected to have inserted non-zero rows, num_inserts=%lu.",
num_inserts);
trunk_destroy(spl);
}
static void
trunk_shadow_init(trunk_shadow *shadow,
data_config *data_cfg,
platform_heap_id hid)
{
shadow->data_cfg = data_cfg;
shadow->sorted = TRUE;
writable_buffer_init(&shadow->entries, hid);
writable_buffer_init(&shadow->data, hid);
}
static void
trunk_shadow_deinit(trunk_shadow *shadow)
{
writable_buffer_deinit(&shadow->entries);
writable_buffer_deinit(&shadow->data);
}
static void
trunk_shadow_reinit(trunk_shadow *shadow)
{
shadow->sorted = TRUE;
writable_buffer_set_to_null(&shadow->entries);
writable_buffer_set_to_null(&shadow->data);
}
/*
* Copy the newly inserted key/value row to a shadow buffer. This set of
* rows will be used later during lookup-validation using range searches.
*/
static void
trunk_shadow_append(trunk_shadow *shadow, key tuple_key, message value)
{
platform_assert(message_class(value) == MESSAGE_TYPE_INSERT);
uint64 key_offset = writable_buffer_append(
&shadow->data, key_length(tuple_key), key_data(tuple_key));
writable_buffer_append(
&shadow->data, message_length(value), message_data(value));
shadow_entry new_entry = {.key_offset = key_offset,
.key_length = key_length(tuple_key),
.value_length = message_length(value)};
writable_buffer_append(&shadow->entries, sizeof(new_entry), &new_entry);
shadow->sorted = FALSE;
}
static key
shadow_entry_key(const shadow_entry *entry, char *data)
{
return key_create(entry->key_length, data + entry->key_offset);
}
static message
shadow_entry_value(const shadow_entry *entry, char *data)
{
return message_create(
MESSAGE_TYPE_INSERT,
slice_create(entry->value_length,
data + entry->key_offset + entry->key_length));
}
static int
compare_shadow_entries(const void *a, const void *b, void *arg)
{
trunk_shadow *shadow = (trunk_shadow *)arg;
char *data = writable_buffer_data(&shadow->data);
key akey = shadow_entry_key(a, data);
key bkey = shadow_entry_key(b, data);
return data_key_compare(shadow->data_cfg, akey, bkey);
}
static uint64
trunk_shadow_length(trunk_shadow *shadow)
{
return writable_buffer_length(&shadow->entries) / sizeof(shadow_entry);
}
static void
trunk_shadow_sort(trunk_shadow *shadow)
{
shadow_entry *entries = writable_buffer_data(&shadow->entries);
uint64 nentries = trunk_shadow_length(shadow);
shadow_entry temp;
platform_sort_slow(entries,
nentries,
sizeof(*entries),
compare_shadow_entries,
shadow,
&temp);
shadow->sorted = TRUE;
}
static void
trunk_shadow_get(trunk_shadow *shadow, uint64 i, key *tuple_key, message *value)
{
if (!shadow->sorted) {
trunk_shadow_sort(shadow);
}
shadow_entry *entries = writable_buffer_data(&shadow->entries);
debug_only uint64 nentries = trunk_shadow_length(shadow);
debug_assert(i < nentries);
shadow_entry *entry = &entries[i];
char *data = writable_buffer_data(&shadow->data);
*tuple_key = shadow_entry_key(entry, data);
*value = shadow_entry_value(entry, data);
}
static uint64
test_splinter_bsearch(trunk_shadow *shadow, key needle)
{
uint64 lo = 0;
uint64 hi = trunk_shadow_length(shadow);
while (lo < hi) {
// invariant: forall i | i < lo :: s[i] < key
// invariant: forall i | hi <= i :: key <= s[i]
key ckey;
message cvalue;
uint64 mid = (lo + hi) / 2;
trunk_shadow_get(shadow, mid, &ckey, &cvalue);
int cmp = data_key_compare(shadow->data_cfg, needle, ckey);
if (cmp <= 0) {
// key <= s[mid]
hi = mid;
} else {
// s[mid] < key
lo = mid + 1;
}
}
return lo;
}
/*
* **************************************************************************
* Test case to run a bunch of inserts into Splinter, and then perform
* different types of lookup-verification. As all lookups need an inserts
* step, this test case is really a set of multiple sub-test-cases for
* inserts, synchronous and async lookups, and lookups-by-range rolled into
* one.
* **************************************************************************
*/
CTEST2(splinter, test_lookups)
{
allocator *alp = (allocator *)&data->al;
trunk_handle *spl = trunk_create(data->splinter_cfg,
alp,
(cache *)data->clock_cache,
data->tasks,
test_generate_allocator_root_id(),
data->hid);
ASSERT_TRUE(spl != NULL);
trunk_shadow shadow;
trunk_shadow_init(&shadow, data->data_cfg, data->hid);
// FALSE : No need to do verification-after-inserts, as that functionality
// has been tested earlier in test_inserts() case.
uint64 num_inserts = splinter_do_inserts(data, spl, FALSE, &shadow);
ASSERT_NOT_EQUAL(0,
num_inserts,
"Expected to have inserted non-zero rows, num_inserts=%lu.",
num_inserts);
merge_accumulator qdata;
merge_accumulator_init(&qdata, spl->heap_id);
DECLARE_AUTO_KEY_BUFFER(keybuf, data->hid);
const size_t key_size = trunk_max_key_size(spl);
platform_status rc;
// **************************************************************************
// Test sub-case 1: Validate using synchronous trunk_lookup().
// Verify that all the keys inserted are found via lookup.
// **************************************************************************
uint64 start_time = platform_get_timestamp();
CTEST_LOG_INFO("\n");
for (uint64 insert_num = 0; insert_num < num_inserts; insert_num++) {
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS(
insert_num, num_inserts, "Verify positive lookups %3lu%% complete");
test_key(&keybuf, TEST_RANDOM, insert_num, 0, 0, key_size, 0);
merge_accumulator_set_to_null(&qdata);
rc = trunk_lookup(spl, key_buffer_key(&keybuf), &qdata);
ASSERT_TRUE(SUCCESS(rc),
"trunk_lookup() FAILURE, insert_num=%lu: %s\n",
insert_num,
platform_status_to_string(rc));
verify_tuple(spl,
&data->gen,
insert_num,
key_buffer_key(&keybuf),
merge_accumulator_to_message(&qdata),
TRUE);
}
uint64 elapsed_ns = platform_timestamp_elapsed(start_time);
CTEST_LOG_INFO(
" ... splinter positive lookup time %lu s, per tuple %lu ns\n",
NSEC_TO_SEC(elapsed_ns),
(elapsed_ns / num_inserts));
// **************************************************************************
// Test sub-case 2: Validate using synchronous trunk_lookup() that we
// do not find any keys outside the range of keys inserted.
// **************************************************************************
start_time = platform_get_timestamp();
for (uint64 insert_num = num_inserts; insert_num < 2 * num_inserts;
insert_num++)
{
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS((insert_num - num_inserts),
num_inserts,
"Verify negative lookups %3lu%% complete");
test_key(&keybuf, TEST_RANDOM, insert_num, 0, 0, key_size, 0);
rc = trunk_lookup(spl, key_buffer_key(&keybuf), &qdata);
ASSERT_TRUE(SUCCESS(rc),
"trunk_lookup() FAILURE, insert_num=%lu: %s\n",
insert_num,
platform_status_to_string(rc));
verify_tuple(spl,
&data->gen,
insert_num,
key_buffer_key(&keybuf),
merge_accumulator_to_message(&qdata),
FALSE);
}
elapsed_ns = platform_timestamp_elapsed(start_time);
CTEST_LOG_INFO(
" ... splinter negative lookup time %lu s, per tuple %lu ns\n",
NSEC_TO_SEC(elapsed_ns),
(elapsed_ns / num_inserts));
merge_accumulator_deinit(&qdata);
// **************************************************************************
// Test sub-case 3: Validate using binary searches across ranges for the
// keys inside the range of keys inserted.
// **************************************************************************
int niters = 3;
CTEST_LOG_INFO("Perform test_lookup_by_range() for %d iterations ...\n",
niters);
// Iterate thru small set of num_ranges for additional coverage.
trunk_shadow_sort(&shadow);
for (int ictr = 1; ictr <= 3; ictr++) {
uint64 num_ranges = (num_inserts / 128) * ictr;
// Range search uses the shadow-copy of the rows previously inserted while
// doing a binary-search.
rc = test_lookup_by_range(
(void *)data, spl, num_inserts, &shadow, num_ranges);
ASSERT_TRUE(SUCCESS(rc),
"test_lookup_by_range() FAILURE, num_ranges=%d: %s\n",
num_ranges,
platform_status_to_string(rc));
}
/*
** **********************************************
** **** Start of Async lookup sub-test-cases ****
** **********************************************
*/
// Setup Async-context sub-system for async lookups.
test_async_lookup *async_lookup;
async_ctxt_init(data->hid, data->max_async_inflight, &async_lookup);
test_async_ctxt *ctxt = NULL;
// **************************************************************************
// Test sub-case 4: Validate using asynchronous trunk_lookup().
// Verify that all the keys inserted are found via lookup.
// **************************************************************************
// Declare an expected data tuple that will be found.
verify_tuple_arg vtarg_true = {.expected_found = TRUE};
start_time = platform_get_timestamp();
for (uint64 insert_num = 0; insert_num < num_inserts; insert_num++) {
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS(insert_num,
num_inserts,
"Verify async positive lookups %3lu%% complete");
ctxt = test_async_ctxt_get(spl, async_lookup, &vtarg_true);
test_key(&ctxt->key, TEST_RANDOM, insert_num, 0, 0, key_size, 0);
ctxt->lookup_num = insert_num;
async_ctxt_process_one(
spl, async_lookup, ctxt, NULL, verify_tuple_callback, &vtarg_true);
}
test_wait_for_inflight(spl, async_lookup, &vtarg_true);
elapsed_ns = platform_timestamp_elapsed(start_time);
CTEST_LOG_INFO(
" ... splinter positive async lookup time %lu s, per tuple %lu ns\n",
NSEC_TO_SEC(elapsed_ns),
(elapsed_ns / num_inserts));
// **************************************************************************
// Test sub-case 5: Validate using asynchronous trunk_lookup() that we
// do not find any keys outside the range of keys inserted.
// **************************************************************************
// Declare a tuple that data will not be found.
verify_tuple_arg vtarg_false = {.expected_found = FALSE};
start_time = platform_get_timestamp();
for (uint64 insert_num = num_inserts; insert_num < 2 * num_inserts;
insert_num++)
{
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS((insert_num - num_inserts),
num_inserts,
"Verify async negative lookups %3lu%% complete");
ctxt = test_async_ctxt_get(spl, async_lookup, &vtarg_false);
test_key(&ctxt->key, TEST_RANDOM, insert_num, 0, 0, key_size, 0);
ctxt->lookup_num = insert_num;
async_ctxt_process_one(
spl, async_lookup, ctxt, NULL, verify_tuple_callback, &vtarg_false);
}
test_wait_for_inflight(spl, async_lookup, &vtarg_false);
elapsed_ns = platform_timestamp_elapsed(start_time);
CTEST_LOG_INFO(
" ... splinter negative async lookup time %lu s, per tuple %lu ns\n",
NSEC_TO_SEC(elapsed_ns),
(elapsed_ns / num_inserts));
// Cleanup memory allocated in this test case
if (async_lookup) {
async_ctxt_deinit(data->hid, async_lookup);
}
trunk_destroy(spl);
trunk_shadow_deinit(&shadow);
}
/*
* -----------------------------------------------------------------------------
* Simple test cases to exercise print / diagnostic functions provided
* by various sub-systems. Test is now readily useful as an educational tool.
*
* NOTE: This test case is mentioned in external docs. Be careful what
* changes you bring in here.
* -----------------------------------------------------------------------------
*/
CTEST2(splinter, test_splinter_print_diags)
{
set_log_streams_for_tests(MSG_LEVEL_DEBUG);
allocator *alp = (allocator *)&data->al;
trunk_handle *spl = trunk_create(data->splinter_cfg,
alp,
(cache *)data->clock_cache,
data->tasks,
test_generate_allocator_root_id(),
data->hid);
ASSERT_TRUE(spl != NULL);
uint64 num_inserts = splinter_do_inserts(data, spl, FALSE, NULL);
ASSERT_NOT_EQUAL(0,
num_inserts,
"Expected to have inserted non-zero rows, num_inserts=%lu",
num_inserts);
CTEST_LOG_INFO("**** Splinter Diagnostics ****\n"
"Generated by %s:%d:%s ****\n",
__FILE__,
__LINE__,
__func__);
trunk_print_super_block(Platform_default_log_handle, spl);
trunk_print_space_use(Platform_default_log_handle, spl);
CTEST_LOG_INFO("\n** trunk_print() **\n");
trunk_print(Platform_default_log_handle, spl);
CTEST_LOG_INFO("\n** Allocator stats **\n");
allocator_print_stats(alp);
allocator_print_allocated(alp);
set_log_streams_for_tests(MSG_LEVEL_INFO);
trunk_destroy(spl);
}
/*
* ----------------------------------
* Helper and minions live here.
* ----------------------------------
*/
/*
* Work-horse function to drive inserts into Splinter. # of inserts is
* determined by config parameters, and computed below.
*
* Parmeters:
* datap - Ptr to global data struct { }
* spl - Ptr to splinter handle, established by caller.
* verify - Boolean; periodically verify splinter tree consistency
* shadow - Ptr to shadow buffer, which will be re-initialized
* and filled-out in this function, if supplied.
*
* Returns the # of rows inserted.
*/
static uint64
splinter_do_inserts(void *datap,
trunk_handle *spl,
bool32 verify,
trunk_shadow *shadow) // Out
{
// Cast void * datap to ptr-to-CTEST_DATA() struct in use.
struct CTEST_IMPL_DATA_SNAME(splinter) *data =
(struct CTEST_IMPL_DATA_SNAME(splinter) *)datap;
// First see if test was invoked with --num-inserts execution parameter.
// (Override the default, which is some big value, like 12988800, with this
// hook for faster test execution.)
int num_inserts = data->test_exec_cfg.num_inserts;
// If not, derive total # of rows to be inserted
if (!num_inserts) {
trunk_config *splinter_cfg = data->splinter_cfg;
num_inserts = splinter_cfg[0].max_kv_bytes_per_node
* splinter_cfg[0].fanout / 2
/ generator_average_message_size(&data->gen);
}
CTEST_LOG_INFO("Splinter_cfg max_kv_bytes_per_node=%lu"
", fanout=%lu"
", max_extents_per_memtable=%lu, num_inserts=%d. ",
data->splinter_cfg[0].max_kv_bytes_per_node,
data->splinter_cfg[0].fanout,
data->splinter_cfg[0].mt_cfg.max_extents_per_memtable,
num_inserts);
uint64 start_time = platform_get_timestamp();
uint64 insert_num;
DECLARE_AUTO_KEY_BUFFER(keybuf, spl->heap_id);
const size_t key_size = trunk_max_key_size(spl);
// Allocate a large array for copying over shadow copies of rows
// inserted, if user has asked to return such an array.
if (shadow) {
trunk_shadow_reinit(shadow);
}
platform_status rc;
CTEST_LOG_INFO("trunk_insert() test with %d inserts %s ...\n",
num_inserts,
(verify ? "and verify" : ""));
merge_accumulator msg;
merge_accumulator_init(&msg, spl->heap_id);
for (insert_num = 0; insert_num < num_inserts; insert_num++) {
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS(insert_num, num_inserts, "inserting %3lu%% complete");
if (verify && (insert_num != 0)
&& (insert_num % TEST_VERIFY_GRANULARITY) == 0) {
bool32 result = trunk_verify_tree(spl);
ASSERT_TRUE(result,
"trunk_verify_tree() failed after %d inserts. ",
insert_num);
}
test_key(&keybuf, TEST_RANDOM, insert_num, 0, 0, key_size, 0);
generate_test_message(&data->gen, insert_num, &msg);
rc = trunk_insert(
spl, key_buffer_key(&keybuf), merge_accumulator_to_message(&msg));
ASSERT_TRUE(SUCCESS(rc),
"trunk_insert() FAILURE: %s\n",
platform_status_to_string(rc));
// Caller is interested in using a copy of the rows inserted for
// verification; e.g. by range-search lookups.
if (shadow) {
trunk_shadow_append(shadow,
key_buffer_key(&keybuf),
merge_accumulator_to_message(&msg));
}
}
uint64 elapsed_ns = platform_timestamp_elapsed(start_time);
uint64 elapsed_s = NSEC_TO_SEC(elapsed_ns);
// For small # of inserts, elapsed sec will be 0. Deal with it.
CTEST_LOG_INFO(
"... average tuple_size=%lu, splinter insert time %lu s, per "
"tuple %lu ns, %s%lu rows/sec. ",
key_size + generator_average_message_size(&data->gen),
elapsed_s,
(elapsed_ns / num_inserts),
(elapsed_s ? "" : "(n/a)"),
(elapsed_s ? (num_inserts / NSEC_TO_SEC(elapsed_ns)) : num_inserts));
platform_assert(trunk_verify_tree(spl));
cache_assert_free((cache *)data->clock_cache);
// Cleanup memory allocated in this test case
merge_accumulator_deinit(&msg);
return num_inserts;
}
typedef struct shadow_check_tuple_arg {
trunk_handle *spl;
trunk_shadow *shadow;
uint64 pos;
uint64 errors;
} shadow_check_tuple_arg;
static void
shadow_check_tuple_func(key returned_key, message value, void *varg)
{
shadow_check_tuple_arg *arg = varg;
key shadow_key;
message shadow_value;
trunk_shadow_get(arg->shadow, arg->pos, &shadow_key, &shadow_value);
if (data_key_compare(arg->spl->cfg.data_cfg, returned_key, shadow_key)
|| message_lex_cmp(value, shadow_value))
{
char expected_key[128];
char actual_key[128];
char expected_value[128];
char actual_value[128];
trunk_key_to_string(arg->spl, shadow_key, expected_key);
trunk_key_to_string(arg->spl, returned_key, actual_key);
trunk_message_to_string(arg->spl, shadow_value, expected_value);
trunk_message_to_string(arg->spl, value, actual_value);
CTEST_LOG_INFO("\nexpected: '%s' | '%s'\n", expected_key, expected_value);
CTEST_LOG_INFO("actual : '%s' | '%s'\n", actual_key, actual_value);
arg->errors++;
}
arg->pos++;
}
/*
* -----------------------------------------------------------------------------
* Driver routine to verify Splinter lookup by range searches.
*
* Parameters:
* datap - Ptr to global data struct
* spl - Ptr to trunk_handle
* num_inserts - # of inserts that we want ranges to span
* shadow - Shadow buffer allocated by caller, while inserting rows.
* num_ranges - # of range searches to do in this run
* -----------------------------------------------------------------------------
*/
static platform_status
test_lookup_by_range(void *datap,
trunk_handle *spl,
uint64 num_inserts,
trunk_shadow *shadow,
uint64 num_ranges)
{
const size_t key_size = trunk_max_key_size(spl);
uint64 start_time = platform_get_timestamp();
platform_status rc;
DECLARE_AUTO_KEY_BUFFER(start_key_buf, spl->heap_id);
for (uint64 range_num = 0; range_num != num_ranges; range_num++) {
// Show progress message in %age-completed to stdout
SHOW_PCT_PROGRESS(
range_num, num_ranges, "Verify range lookups %3lu%% complete");
test_key(&start_key_buf,
TEST_RANDOM,
num_inserts + range_num,
0,
0,
key_size,
0);
key start_key = key_buffer_key(&start_key_buf);
uint64 range_tuples = test_range(range_num, 1, 100);
uint64 start_idx = test_splinter_bsearch(shadow, start_key);
uint64 expected_returned_tuples = num_inserts - start_idx > range_tuples
? range_tuples
: num_inserts - start_idx;
shadow_check_tuple_arg arg = {
.spl = spl, .shadow = shadow, .pos = start_idx, .errors = 0};
rc = trunk_range(
spl, start_key, range_tuples, shadow_check_tuple_func, &arg);
ASSERT_TRUE(SUCCESS(rc));
ASSERT_TRUE(
arg.errors == 0, "trunk_range() found %lu mismatches", arg.errors);
ASSERT_TRUE(arg.pos == start_idx + expected_returned_tuples,
"trunk_range() saw wrong number of tuples: "
" expected_returned_tuples=%lu"
", returned_tuples=%lu"
", start_key='%.*s'"
", errors=%lu",
expected_returned_tuples,
arg.pos - start_idx,
key_size,
start_key,
arg.errors);
}
uint64 elapsed_ns = platform_timestamp_elapsed(start_time);
CTEST_LOG_INFO(" ... splinter range time %lu s, per operation %lu ns"
", %lu ranges\n",
NSEC_TO_SEC(elapsed_ns),
(elapsed_ns / num_ranges),
num_ranges);
return rc;
}