8bc8e867e4
Query allocator and evaluation allocator were different. After analysis, was determined they should be the same, this will help future development reduce TypeValue copies during queries. Changes: - Common allocator, PoolResource backed by MonotonicResource - Optimized Pool, now O(1) alloc/dealloc as all chunks in Pool form a single free list - 2nd PoolResource, using bin sizing, not as perfect for memory usage but O(1) bin selection - Now have jemalloc's background thread to make sure decay and return to OS happens - Optimized ProperyValue to be faster at destruction/copy/move - Less temporary memory allocations - CSV reader now maintains a common line buffer it reuses on line reads - Writing out bolt values, now reuses a values buffer - Evaluating an int no longer makes temporary strings for errors it most likely never throws - ExpandVariable will reuse existing edge list in frame it one existed
459 lines
14 KiB
C++
459 lines
14 KiB
C++
// Copyright 2024 Memgraph Ltd.
|
|
//
|
|
// Use of this software is governed by the Business Source License
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
//
|
|
// As of the Change Date specified in that file, in accordance with
|
|
// the Business Source License, use of this software will be governed
|
|
// by the Apache License, Version 2.0, included in the file
|
|
// licenses/APL.txt.
|
|
|
|
#include <chrono>
|
|
#include <iostream>
|
|
#include <map>
|
|
#include <set>
|
|
|
|
#include <benchmark/benchmark.h>
|
|
|
|
#include "utils/memory.hpp"
|
|
#include "utils/skip_list.hpp"
|
|
#include "utils/spin_lock.hpp"
|
|
|
|
const int kThreadsNum = 8;
|
|
const uint64_t kMaxNum = 10000000;
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// memgraph::utils::SkipList set Insert
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class SkipListSetInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
list = memgraph::utils::SkipList<uint64_t>();
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::SkipList<uint64_t> list;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(SkipListSetInsertFixture, Insert)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
auto acc = list.access();
|
|
if (acc.insert(dist(gen)).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SkipListSetInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// std::set Insert
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class StdSetInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
container = {};
|
|
}
|
|
}
|
|
|
|
protected:
|
|
std::set<uint64_t> container;
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdSetInsertFixture, Insert)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.insert(dist(gen)).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdSetInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
class StdSetWithPoolAllocatorInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
container.clear();
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::PoolResource memory_{128U /* max_blocks_per_chunk */, memgraph::utils::NewDeleteResource()};
|
|
std::set<uint64_t, std::less<>, memgraph::utils::Allocator<uint64_t>> container{&memory_};
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdSetWithPoolAllocatorInsertFixture, Insert)
|
|
(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.insert(dist(gen)).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdSetWithPoolAllocatorInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// memgraph::utils::SkipList set Find
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class SkipListSetFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && list.size() == 0) {
|
|
auto acc = list.access();
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
acc.insert(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::SkipList<uint64_t> list;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(SkipListSetFindFixture, Find)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
auto acc = list.access();
|
|
if (acc.find(dist(gen)) != acc.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SkipListSetFindFixture, Find)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// std::set Find
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class StdSetFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && container.size() == 0) {
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
container.insert(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
std::set<uint64_t> container;
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdSetFindFixture, Find)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.find(dist(gen)) != container.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdSetFindFixture, Find)->ThreadRange(1, kThreadsNum)->Unit(benchmark::kNanosecond)->UseRealTime();
|
|
|
|
class StdSetWithPoolAllocatorFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && container.size() == 0) {
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
container.insert(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::PoolResource memory_{128U /* max_blocks_per_chunk */, memgraph::utils::NewDeleteResource()};
|
|
std::set<uint64_t, std::less<>, memgraph::utils::Allocator<uint64_t>> container{&memory_};
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdSetWithPoolAllocatorFindFixture, Find)
|
|
(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.find(dist(gen)) != container.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdSetWithPoolAllocatorFindFixture, Find)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// Map tests common
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
struct MapObject {
|
|
uint64_t key;
|
|
uint64_t value;
|
|
};
|
|
|
|
bool operator==(const MapObject &a, const MapObject &b) { return a.key == b.key; }
|
|
bool operator<(const MapObject &a, const MapObject &b) { return a.key < b.key; }
|
|
bool operator==(const MapObject &a, uint64_t b) { return a.key == b; }
|
|
bool operator<(const MapObject &a, uint64_t b) { return a.key < b; }
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// memgraph::utils::SkipList map Insert
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class SkipListMapInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
list = memgraph::utils::SkipList<MapObject>();
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::SkipList<MapObject> list;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(SkipListMapInsertFixture, Insert)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
auto acc = list.access();
|
|
if (acc.insert({dist(gen), 0}).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SkipListMapInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// std::map Insert
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class StdMapInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
container = {};
|
|
}
|
|
}
|
|
|
|
protected:
|
|
std::map<uint64_t, uint64_t> container;
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdMapInsertFixture, Insert)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.insert({dist(gen), 0}).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdMapInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
class StdMapWithPoolAllocatorInsertFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0) {
|
|
container = {};
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::PoolResource memory_{128U /* max_blocks_per_chunk */, memgraph::utils::NewDeleteResource()};
|
|
std::map<uint64_t, uint64_t, std::less<>, memgraph::utils::Allocator<std::pair<const uint64_t, uint64_t>>> container{
|
|
&memory_};
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdMapWithPoolAllocatorInsertFixture, Insert)
|
|
(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.insert({dist(gen), 0}).second) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdMapWithPoolAllocatorInsertFixture, Insert)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// memgraph::utils::SkipList map Find
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class SkipListMapFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && list.size() == 0) {
|
|
auto acc = list.access();
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
acc.insert({i, 0});
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::SkipList<MapObject> list;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(SkipListMapFindFixture, Find)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
auto acc = list.access();
|
|
if (acc.find(dist(gen)) != acc.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(SkipListMapFindFixture, Find)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// std::map Find
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
class StdMapFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && container.size() == 0) {
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
container.insert({i, 0});
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
std::map<uint64_t, uint64_t> container;
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdMapFindFixture, Find)(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.find(dist(gen)) != container.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdMapFindFixture, Find)->ThreadRange(1, kThreadsNum)->Unit(benchmark::kNanosecond)->UseRealTime();
|
|
|
|
class StdMapWithPoolAllocatorFindFixture : public benchmark::Fixture {
|
|
protected:
|
|
void SetUp(const benchmark::State &state) override {
|
|
if (state.thread_index() == 0 && container.size() == 0) {
|
|
for (uint64_t i = 0; i < kMaxNum; ++i) {
|
|
container.insert({i, 0});
|
|
}
|
|
}
|
|
}
|
|
|
|
protected:
|
|
memgraph::utils::PoolResource memory_{128U /* max_blocks_per_chunk */, memgraph::utils::NewDeleteResource()};
|
|
std::map<uint64_t, uint64_t, std::less<>, memgraph::utils::Allocator<std::pair<const uint64_t, uint64_t>>> container{
|
|
&memory_};
|
|
memgraph::utils::SpinLock lock;
|
|
};
|
|
|
|
BENCHMARK_DEFINE_F(StdMapWithPoolAllocatorFindFixture, Find)
|
|
(benchmark::State &state) {
|
|
std::mt19937 gen(state.thread_index());
|
|
std::uniform_int_distribution<uint64_t> dist(0, kMaxNum);
|
|
uint64_t counter = 0;
|
|
while (state.KeepRunning()) {
|
|
std::lock_guard<memgraph::utils::SpinLock> guard(lock);
|
|
if (container.find(dist(gen)) != container.end()) {
|
|
++counter;
|
|
}
|
|
}
|
|
state.SetItemsProcessed(counter);
|
|
}
|
|
|
|
BENCHMARK_REGISTER_F(StdMapWithPoolAllocatorFindFixture, Find)
|
|
->ThreadRange(1, kThreadsNum)
|
|
->Unit(benchmark::kNanosecond)
|
|
->UseRealTime();
|
|
|
|
BENCHMARK_MAIN();
|