-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathfillFlush.cpp
92 lines (80 loc) · 3.14 KB
/
fillFlush.cpp
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
#include <algorithm>
#include <atomic>
#include <cstdint>
#include <cstdlib>
#include <random>
#include <thread>
#include <vector>
#include "gflags/gflags.h"
// Flags that control single-threaded behavior
DEFINE_int32(batch_size, 1000, "Number of pointers owned by a thread at a time");
DEFINE_int32(batch_thread_migrations, 200, "Number of pointers to free in a run");
DEFINE_int32(batch_frees, 200, "Number of pointers in a batch to free");
DEFINE_int32(batch_sleep_ms, 1, "Number of milliseconds to sleep between batch free and alloc");
// Flags that control cross-thread behavior
DEFINE_int32(num_threads, 1, "Number of threads to run the test");
DEFINE_int32(shared_buffer_size, 10 * 1000, "Shared buffer size");
// Control parameters
DEFINE_int32(num_runs, -1, "Number of runs to perform (or -1 to loop forever)");
DEFINE_int32(malloc_size, 32, "Size of the allocations");
DEFINE_int32(randseed, 12345, "Random seed, for gesture in the direction of reproducibility");
typedef std::minstd_rand URNG;
std::vector<std::atomic<void*>> createSharedBuffer(URNG& urng) {
std::vector<void*> resultNonAtomic(FLAGS_shared_buffer_size);
for (int i = 0; i < FLAGS_shared_buffer_size; ++i) {
resultNonAtomic[i] = std::malloc(FLAGS_malloc_size);
}
std::shuffle(resultNonAtomic.begin(), resultNonAtomic.end(), urng);
std::vector<std::atomic<void*>> result(resultNonAtomic.begin(), resultNonAtomic.end());
return result;
}
void doThreadMigrations(
URNG& urng,
std::vector<void*>& batch,
std::vector<std::atomic<void*>>& sharedBuffer) {
std::uniform_int_distribution<int> sharedDist(0, sharedBuffer.size() - 1);
std::uniform_int_distribution<int> localDist(0, batch.size() - 1);
for (int i = 0; i < FLAGS_batch_thread_migrations; ++i) {
int localIndex = localDist(urng);
int sharedIndex = sharedDist(urng);
void* oldLocal = batch[localIndex];
void* newLocal = sharedBuffer[sharedIndex].exchange(oldLocal);
batch[localIndex] = newLocal;
}
}
void doFrees(URNG& urng, std::vector<void*>& batch) {
for (int i = 0; i < FLAGS_batch_frees; ++i) {
std::free(batch[i]);
}
std::this_thread::sleep_for(std::chrono::milliseconds(FLAGS_batch_sleep_ms));
for (int i = 0; i < FLAGS_batch_frees; ++i) {
batch[i] = std::malloc(FLAGS_malloc_size);
}
std::shuffle(batch.begin(), batch.end(), urng);
}
void doThread(unsigned initSeed, std::vector<std::atomic<void*>>& sharedBuffer) {
std::vector<void*> batch(FLAGS_batch_size);
for (int i = 0; i < FLAGS_batch_size; ++i) {
batch[i] = std::malloc(FLAGS_malloc_size);
}
URNG urng(initSeed);
for (unsigned i = 0; i < (unsigned) FLAGS_num_runs || FLAGS_num_runs == -1; ++i) {
doThreadMigrations(urng, batch, sharedBuffer);
doFrees(urng, batch);
}
}
int main(int argc, char** argv) {
gflags::ParseCommandLineFlags(&argc, &argv, true);
URNG urng(FLAGS_randseed);
auto sharedBuffer = createSharedBuffer(urng);
std::vector<std::thread> threads;
for (unsigned i = 0; i < FLAGS_num_threads; ++i) {
unsigned seed = (unsigned)urng() + i;
threads.emplace_back([&, seed]() {
doThread(seed, sharedBuffer);
});
}
for (auto& thread : threads) {
thread.join();
}
}