forked from BLepers/KVell
-
Notifications
You must be signed in to change notification settings - Fork 1
/
workload-ycsb.c
153 lines (142 loc) · 4.21 KB
/
workload-ycsb.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
/*
* YCSB Workload
*/
#include "headers.h"
#include "workload-common.h"
static char *_create_unique_item_ycsb(uint64_t uid) {
size_t item_size = 1024;
// size_t item_size = sizeof(struct item_metadata) + 2*sizeof(size_t);
return create_unique_item(item_size, uid);
}
static char *create_unique_item_ycsb(uint64_t uid, uint64_t max_uid) {
return _create_unique_item_ycsb(uid);
}
/* Is the current request a get or a put? */
static int random_get_put(int test) {
long random = uniform_next() % 100;
switch (test) {
case 0: // A
return random >= 50;
case 1: // B
return random >= 95;
case 2: // C
return 0;
case 3: // E
return random >= 95;
}
die("Not a valid test\n");
}
/* YCSB A (or D), B, C */
static void _launch_ycsb(int test, int nb_requests, int zipfian) {
declare_periodic_count;
for (size_t i = 0; i < nb_requests; i++) {
struct slab_callback *cb = bench_cb();
if (zipfian)
cb->item = _create_unique_item_ycsb(zipf_next());
else
cb->item = _create_unique_item_ycsb(uniform_next());
if (random_get_put(
test)) { // In these tests we update with a given probability
kv_update_async(cb);
} else { // or we read
kv_read_async(cb);
}
periodic_count(1000, "YCSB Load Injector (%lu%%)", i * 100LU / nb_requests);
}
}
/* YCSB E */
static void _launch_ycsb_e(int test, int nb_requests, int zipfian) {
declare_periodic_count;
random_gen_t rand_next = zipfian ? zipf_next : uniform_next;
for (size_t i = 0; i < nb_requests; i++) {
if (random_get_put(
test)) { // In this test we update with a given probability
struct slab_callback *cb = bench_cb();
cb->item = _create_unique_item_ycsb(rand_next());
kv_update_async(cb);
} else { // or we scan
char *item = _create_unique_item_ycsb(rand_next());
tree_scan_res_t scan_res = kv_init_scan(item, uniform_next() % 99 + 1);
free(item);
for (size_t j = 0; j < scan_res.nb_entries; j++) {
struct slab_callback *cb = bench_cb();
cb->item = _create_unique_item_ycsb(scan_res.hashes[j]);
kv_read_async_no_lookup(cb, scan_res.entries[j].slab,
scan_res.entries[j].slab_idx);
}
free(scan_res.hashes);
free(scan_res.entries);
}
periodic_count(1000, "YCSB Load Injector (scans) (%lu%%)",
i * 100LU / nb_requests);
}
}
/* Generic interface */
static void launch_ycsb(struct workload *w, bench_t b) {
switch (b) {
case ycsb_a_uniform:
return _launch_ycsb(0, w->nb_requests_per_thread, 0);
case ycsb_b_uniform:
return _launch_ycsb(1, w->nb_requests_per_thread, 0);
case ycsb_c_uniform:
return _launch_ycsb(2, w->nb_requests_per_thread, 0);
case ycsb_e_uniform:
return _launch_ycsb_e(3, w->nb_requests_per_thread, 0);
case ycsb_a_zipfian:
return _launch_ycsb(0, w->nb_requests_per_thread, 1);
case ycsb_b_zipfian:
return _launch_ycsb(1, w->nb_requests_per_thread, 1);
case ycsb_c_zipfian:
return _launch_ycsb(2, w->nb_requests_per_thread, 1);
case ycsb_e_zipfian:
return _launch_ycsb_e(3, w->nb_requests_per_thread, 1);
default:
die("Unsupported workload\n");
}
}
/* Pretty printing */
static const char *name_ycsb(bench_t w) {
switch (w) {
case ycsb_a_uniform:
return "YCSB A - Uniform";
case ycsb_b_uniform:
return "YCSB B - Uniform";
case ycsb_c_uniform:
return "YCSB C - Uniform";
case ycsb_e_uniform:
return "YCSB E - Uniform";
case ycsb_a_zipfian:
return "YCSB A - Zipf";
case ycsb_b_zipfian:
return "YCSB B - Zipf";
case ycsb_c_zipfian:
return "YCSB C - Zipf";
case ycsb_e_zipfian:
return "YCSB E - Zipf";
default:
return "??";
}
}
static int handles_ycsb(bench_t w) {
switch (w) {
case ycsb_a_uniform:
case ycsb_b_uniform:
case ycsb_c_uniform:
case ycsb_e_uniform:
case ycsb_a_zipfian:
case ycsb_b_zipfian:
case ycsb_c_zipfian:
case ycsb_e_zipfian:
return 1;
default:
return 0;
}
}
static const char *api_name_ycsb(void) { return "YCSB"; }
struct workload_api YCSB = {
.handles = handles_ycsb,
.launch = launch_ycsb,
.api_name = api_name_ycsb,
.name = name_ycsb,
.create_unique_item = create_unique_item_ycsb,
};