memgraph/tests/benchmark/storage_v2_gc.cpp
Marin Tomic 8414479abe [StorageV2] Implement GC
Summary:
Here are some numbers from the benchmark:

```
(TOOLCHAIN) mtomic@poso:~/memgraph/build_release$ tests/benchmark/storage_v2_gc --num-threads 8
Config: NoGc, Time: 25.9836
Config: OnFinishGc, Time: 49.012
Config: 100msPeriodicGc, Time: 45.9856
Config: 1000msPeriodicGc, Time: 40.3094
```

```
(TOOLCHAIN) mtomic@poso:~/memgraph/build_release$ tests/benchmark/storage_v2_gc --num-threads 7
Config: NoGc, Time: 20.4256
Config: OnFinishGc, Time: 39.6669
Config: 100msPeriodicGc, Time: 30.7956
Config: 1000msPeriodicGc, Time: 35.128
```

It is not that bad if there is a core dedicated to doing garbage collection.

Reviewers: mferencevic, teon.banek

Reviewed By: mferencevic, teon.banek

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D2168
2019-07-10 16:12:08 +02:00

87 lines
2.8 KiB
C++

#include <iostream>
#include <gflags/gflags.h>
#include "storage/v2/storage.hpp"
#include "utils/timer.hpp"
// This benchmark should be run for a fixed amount of time that is
// large compared to GC interval to make the output relevant.
const int kNumIterations = 5000000;
const int kNumVertices = 1000000;
DEFINE_int32(num_threads, 4, "number of threads");
DEFINE_int32(num_vertices, kNumVertices, "number of vertices");
DEFINE_int32(num_iterations, kNumIterations, "number of iterations");
std::pair<std::string, storage::StorageGcConfig> TestConfigurations[] = {
{"NoGc",
storage::StorageGcConfig{.type = storage::StorageGcConfig::Type::NONE}},
{"OnFinishGc",
storage::StorageGcConfig{.type =
storage::StorageGcConfig::Type::ON_FINISH}},
{"100msPeriodicGc",
storage::StorageGcConfig{.type = storage::StorageGcConfig::Type::PERIODIC,
.interval = std::chrono::milliseconds(100)}},
{"1000msPeriodicGc",
storage::StorageGcConfig{.type = storage::StorageGcConfig::Type::ON_FINISH,
.interval = std::chrono::milliseconds(1000)}}};
void UpdateLabelFunc(int thread_id, storage::Storage *storage,
const std::vector<storage::Gid> &vertices,
int num_iterations) {
std::mt19937 gen(thread_id);
std::uniform_int_distribution<uint64_t> vertex_dist(0, vertices.size() - 1);
std::uniform_int_distribution<uint64_t> label_dist(0, 100);
utils::Timer timer;
for (int iter = 0; iter < num_iterations; ++iter) {
auto acc = storage->Access();
storage::Gid gid = vertices.at(vertex_dist(gen));
std::optional<storage::VertexAccessor> vertex =
acc.FindVertex(gid, storage::View::OLD);
CHECK(vertex.has_value())
<< "Vertex with GID " << gid.AsUint() << " doesn't exist";
if (vertex->AddLabel(label_dist(gen)).IsReturn()) {
acc.Commit();
} else {
acc.Abort();
}
}
}
int main(int argc, char *argv[]) {
gflags::ParseCommandLineFlags(&argc, &argv, true);
for (const auto &config : TestConfigurations) {
storage::Storage storage(config.second);
std::vector<storage::Gid> vertices;
{
auto acc = storage.Access();
for (int i = 0; i < FLAGS_num_vertices; ++i) {
vertices.push_back(acc.CreateVertex().Gid());
}
acc.Commit();
}
utils::Timer timer;
std::vector<std::thread> threads;
threads.reserve(FLAGS_num_threads);
for (int i = 0; i < FLAGS_num_threads; ++i) {
threads.emplace_back(UpdateLabelFunc, i, &storage, vertices,
FLAGS_num_iterations);
}
for (int i = 0; i < FLAGS_num_threads; ++i) {
threads[i].join();
}
std::cout << "Config: " << config.first
<< ", Time: " << timer.Elapsed().count() << std::endl;
}
return 0;
}