// Copyright 2021 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 #include #include #include #include #include "utils/memory.hpp" #include "utils/skip_list.hpp" #include "utils/spin_lock.hpp" const int kThreadsNum = 8; const uint64_t kMaxNum = 10000000; /////////////////////////////////////////////////////////////////////////////// // utils::SkipList set Insert /////////////////////////////////////////////////////////////////////////////// class SkipListSetInsertFixture : public benchmark::Fixture { protected: void SetUp(const benchmark::State &state) override { if (state.thread_index == 0) { list = utils::SkipList(); } } protected: utils::SkipList list; }; BENCHMARK_DEFINE_F(SkipListSetInsertFixture, Insert)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution 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 container; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdSetInsertFixture, Insert)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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: utils::PoolResource memory_{256U /* max_blocks_per_chunk */, 1024U /* max_block_size */, utils::NewDeleteResource()}; std::set, utils::Allocator> container{&memory_}; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdSetWithPoolAllocatorInsertFixture, Insert) (benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard guard(lock); if (container.insert(dist(gen)).second) { ++counter; } } state.SetItemsProcessed(counter); } BENCHMARK_REGISTER_F(StdSetWithPoolAllocatorInsertFixture, Insert) ->ThreadRange(1, kThreadsNum) ->Unit(benchmark::kNanosecond) ->UseRealTime(); /////////////////////////////////////////////////////////////////////////////// // 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: utils::SkipList list; }; BENCHMARK_DEFINE_F(SkipListSetFindFixture, Find)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution 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 container; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdSetFindFixture, Find)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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: utils::PoolResource memory_{256U /* max_blocks_per_chunk */, 1024U /* max_block_size */, utils::NewDeleteResource()}; std::set, utils::Allocator> container{&memory_}; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdSetWithPoolAllocatorFindFixture, Find) (benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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; } /////////////////////////////////////////////////////////////////////////////// // utils::SkipList map Insert /////////////////////////////////////////////////////////////////////////////// class SkipListMapInsertFixture : public benchmark::Fixture { protected: void SetUp(const benchmark::State &state) override { if (state.thread_index == 0) { list = utils::SkipList(); } } protected: utils::SkipList list; }; BENCHMARK_DEFINE_F(SkipListMapInsertFixture, Insert)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution 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 container; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdMapInsertFixture, Insert)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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: utils::PoolResource memory_{256U /* max_blocks_per_chunk */, 1024U /* max_block_size */, utils::NewDeleteResource()}; std::map, utils::Allocator>> container{&memory_}; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdMapWithPoolAllocatorInsertFixture, Insert) (benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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(); /////////////////////////////////////////////////////////////////////////////// // 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: utils::SkipList list; }; BENCHMARK_DEFINE_F(SkipListMapFindFixture, Find)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution 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 container; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdMapFindFixture, Find)(benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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: utils::PoolResource memory_{256U /* max_blocks_per_chunk */, 1024U /* max_block_size */, utils::NewDeleteResource()}; std::map, utils::Allocator>> container{&memory_}; utils::SpinLock lock; }; BENCHMARK_DEFINE_F(StdMapWithPoolAllocatorFindFixture, Find) (benchmark::State &state) { std::mt19937 gen(state.thread_index); std::uniform_int_distribution dist(0, kMaxNum); uint64_t counter = 0; while (state.KeepRunning()) { std::lock_guard 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();