forked from vmware/splinterdb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
test_data.c
136 lines (121 loc) · 4.82 KB
/
test_data.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
// Copyright 2018-2021 VMware, Inc.
// SPDX-License-Identifier: Apache-2.0
#include "test_data.h"
#include "data_internal.h"
typedef struct data_test_config {
data_config super;
uint64 payload_size_limit;
} data_test_config;
static int
test_data_key_cmp(const data_config *cfg, slice key1, slice key2)
{
return slice_lex_cmp(key1, key2);
}
void
test_data_generate_message(const data_config *cfg,
message_type type,
uint8 ref_count,
merge_accumulator *msg)
{
uint64 payload_size = 0;
if (type == MESSAGE_TYPE_INSERT) {
const data_test_config *tdcfg = (const data_test_config *)cfg;
// A coupla good ol' random primes
payload_size = (253456363ULL + (uint64)ref_count * 750599937895091ULL)
% tdcfg->payload_size_limit;
}
merge_accumulator_set_class(msg, type);
merge_accumulator_resize(msg, sizeof(data_handle) + payload_size);
data_handle *dh = merge_accumulator_data(msg);
dh->ref_count = ref_count;
memset(dh->data, ref_count, payload_size);
}
/*
*-----------------------------------------------------------------------------
* data_merge_tuples --
*
* Given two data messages, merges them by decoding the type of messages.
* Returns the result in new_data.
*-----------------------------------------------------------------------------
*/
static int
test_data_merge_tuples(const data_config *cfg,
slice key,
message old_raw_message,
merge_accumulator *new_raw_message)
{
platform_assert(sizeof(data_handle) <= message_length(old_raw_message));
platform_assert(sizeof(data_handle)
<= merge_accumulator_length(new_raw_message));
platform_assert(merge_accumulator_message_class(new_raw_message)
== MESSAGE_TYPE_UPDATE);
platform_assert(message_class(old_raw_message) != MESSAGE_TYPE_DELETE);
const data_handle *old_data = message_data(old_raw_message);
data_handle *new_data = merge_accumulator_data(new_raw_message);
debug_assert(old_data != NULL);
debug_assert(new_data != NULL);
int8 result_ref_count = old_data->ref_count + new_data->ref_count;
message_type result_type = message_class(old_raw_message);
if (result_ref_count == 0 && result_type == MESSAGE_TYPE_INSERT) {
result_type = MESSAGE_TYPE_DELETE;
}
test_data_generate_message(
cfg, result_type, result_ref_count, new_raw_message);
return 0;
}
/*
*-----------------------------------------------------------------------------
* data_merge_tuples_final --
*
* Called for non-MESSAGE_TYPE_INSERT messages when they are determined to
*be the oldest message in the system.
*
* Can change data_class or contents. If necessary, update new_data.
*-----------------------------------------------------------------------------
*/
static int
test_data_merge_tuples_final(const data_config *cfg,
slice key,
merge_accumulator *oldest_raw_data) // IN/OUT
{
platform_assert(merge_accumulator_message_class(oldest_raw_data)
== MESSAGE_TYPE_UPDATE);
assert(sizeof(data_handle) <= merge_accumulator_length(oldest_raw_data));
data_handle *old_data = merge_accumulator_data(oldest_raw_data);
debug_assert(old_data != NULL);
test_data_generate_message(cfg,
(old_data->ref_count == 0) ? MESSAGE_TYPE_DELETE
: MESSAGE_TYPE_INSERT,
old_data->ref_count,
oldest_raw_data);
return 0;
}
static void
test_data_key_to_string(const data_config *cfg,
slice key,
char *str,
size_t len)
{
debug_hex_encode(str, len, slice_data(key), slice_length(key));
}
static void
test_data_message_to_string(const data_config *cfg,
message message,
char *str,
size_t len)
{
debug_hex_encode(str, len, message_data(message), message_length(message));
}
static data_test_config data_test_config_internal = {
.super =
{
.max_key_size = 24,
.key_compare = test_data_key_cmp,
.key_hash = platform_hash32,
.key_to_string = test_data_key_to_string,
.message_to_string = test_data_message_to_string,
.merge_tuples = test_data_merge_tuples,
.merge_tuples_final = test_data_merge_tuples_final,
},
.payload_size_limit = 24};
data_config *test_data_config = &data_test_config_internal.super;