forked from BLepers/KVell
-
Notifications
You must be signed in to change notification settings - Fork 1
/
stats.c
123 lines (107 loc) · 2.58 KB
/
stats.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
#include "headers.h"
#include "slab.h"
#include "utils.h"
#define MAX_STATS 10000000LU
struct stats {
uint64_t *timing_time;
uint64_t *timing_value;
size_t timing_idx;
size_t max_timing_idx;
} stats;
void add_timing_stat(uint64_t elapsed) {
if (!stats.timing_value) {
stats.timing_time = malloc(MAX_STATS * sizeof(*stats.timing_time));
stats.timing_value = malloc(MAX_STATS * sizeof(*stats.timing_value));
stats.timing_idx = 0;
stats.max_timing_idx = MAX_STATS;
}
if (stats.timing_idx >= stats.max_timing_idx)
return;
// die("Cannot collect all stats, buffer is full!\n");
rdtscll(stats.timing_time[stats.timing_idx]);
stats.timing_value[stats.timing_idx] = elapsed;
stats.timing_idx++;
}
int cmp_uint(const void *_a, const void *_b) {
uint64_t a = *(uint64_t *)_a;
uint64_t b = *(uint64_t *)_b;
if (a > b)
return 1;
else if (a < b)
return -1;
else
return 0;
}
void print_stats(void) {
uint64_t avg = 0;
if (stats.timing_idx == 0) {
printf("#No stat has been collected\n");
return;
}
size_t last = stats.timing_idx;
qsort(stats.timing_value, last, sizeof(*stats.timing_value), cmp_uint);
for (size_t i = 0; i < last; i++)
avg += stats.timing_value[i];
printf("#Latency:\n#\tAVG - %lu us\n#\t99p - %lu us\n#\tmax - %lu us\n",
cycles_to_us(avg / last),
cycles_to_us(stats.timing_value[last * 99 / 100]),
cycles_to_us(stats.timing_value[last - 1]));
stats.timing_idx = 0;
}
struct timing_s {
size_t origin;
size_t time;
};
void *allocate_payload(void) {
#if DEBUG
return calloc(20, sizeof(struct timing_s));
#else
return NULL;
#endif
}
void add_time_in_payload(struct slab_callback *c, size_t origin) {
#if DEBUG
struct timing_s *payload = c->payload;
if (!payload)
return;
uint64_t t, pos = 0;
rdtscll(t);
while (pos < 20 && payload[pos].time)
pos++;
if (pos == 20)
die("Too many times added!\n");
payload[pos].time = t;
payload[pos].origin = origin;
#else
if (origin != 0)
return;
uint64_t t;
rdtscll(t);
c->payload = (void *)t;
#endif
}
uint64_t get_origin_from_payload(struct slab_callback *c, size_t pos) {
#if DEBUG
struct timing_s *payload = c->payload;
if (!payload)
return 0;
return payload[pos].origin;
#else
return 0;
#endif
}
uint64_t get_time_from_payload(struct slab_callback *c, size_t pos) {
#if DEBUG
struct timing_s *payload = c->payload;
if (!payload)
return 0;
return payload[pos].time;
#else
return (uint64_t)c->payload;
#endif
}
void free_payload(struct slab_callback *c) {
#if DEBUG
free(c->payload);
#endif
}