diff --git a/perf/graaflib/bron_kerbosch_benchmark.cpp b/perf/graaflib/bron_kerbosch_benchmark.cpp new file mode 100644 index 00000000..c27c8fbc --- /dev/null +++ b/perf/graaflib/bron_kerbosch_benchmark.cpp @@ -0,0 +1,173 @@ +#include +#include +#include + +#include +#include + +namespace { +// Generating random number of vertices (1- 30) for a clique +std::random_device dev; +std::mt19937 rng(dev()); +std::uniform_int_distribution random_clique_size(1, + 30); +template +[[nodiscard]] std::vector create_vertices( + graaf::undirected_graph& graph, size_t n) { + std::vector vertices{}; + vertices.reserve(n); + + for (size_t i{0}; i < n; ++i) { + vertices.push_back(graph.add_vertex(i)); + } + + return vertices; +} + +template +void add_clique(graaf::undirected_graph& graph, + const std::vector& vertices, + size_t start_vertex, size_t clique_size) { + // We are in range of vector of vertices + size_t end_vertex = std::min(start_vertex + clique_size, vertices.size()); + + // Constructing a clique + for (size_t i{start_vertex}; i < end_vertex; ++i) { + for (size_t j{i + 1}; j < end_vertex; ++j) { + graph.add_edge(vertices[i], vertices[j], 1); + } + } +} +} // namespace + +static void bron_kerbosh_cliques(benchmark::State& state) { + const auto number_of_edges{static_cast(state.range(0))}; + + graaf::undirected_graph graph{}; + + // We create enough vertices to construct the requested number of edges + const auto number_of_vertices{number_of_edges + 1}; + const auto vertices{create_vertices(graph, number_of_vertices)}; + const auto clique_size = static_cast(state.range(1)); + + for (size_t i{0}; i < number_of_edges; i += clique_size) { + add_clique(graph, vertices, i, clique_size); + } + + for (auto _ : state) { + auto result = graaf::algorithm::bron_kerbosch(graph); + benchmark::DoNotOptimize(result); + } + + state.SetComplexityN(state.range(1)); +} + +static void bron_kerbosh_connected_cliques(benchmark::State& state) { + const auto number_of_edges{static_cast(state.range(0))}; + + graaf::undirected_graph graph{}; + + // We create enough vertices to construct the requested number of edges + const auto number_of_vertices{number_of_edges + 1}; + const auto vertices{create_vertices(graph, number_of_vertices)}; + const auto clique_size = static_cast(state.range(1)); + + // Connecting all cliques + for (size_t i{0}; i + clique_size < number_of_edges; i += clique_size) { + graph.add_edge(vertices[i], vertices[i + clique_size], 1); + } + + for (size_t i{0}; i < number_of_edges; i += clique_size) { + add_clique(graph, vertices, i, clique_size); + } + + for (auto _ : state) { + auto result = graaf::algorithm::bron_kerbosch(graph); + benchmark::DoNotOptimize(result); + } + + state.SetComplexityN(state.range(1)); +} + +static void bron_kerbosh_random_cliques(benchmark::State& state) { + const auto number_of_edges{static_cast(state.range(0))}; + + graaf::undirected_graph graph{}; + + // We create enough vertices to construct the requested number of edges + const auto number_of_vertices{number_of_edges + 1}; + const auto vertices{create_vertices(graph, number_of_vertices)}; + size_t clique_size = random_clique_size(rng); + + for (size_t i{0}; i + clique_size < number_of_edges; i += clique_size) { + add_clique(graph, vertices, i, clique_size++); + clique_size = random_clique_size(rng); + } + + for (auto _ : state) { + auto result = graaf::algorithm::bron_kerbosch(graph); + benchmark::DoNotOptimize(result); + } + + state.SetComplexityN(state.range(0)); +} + +static void bron_kerbosh_connected_random_cliques(benchmark::State& state) { + const auto number_of_edges{static_cast(state.range(0))}; + + graaf::undirected_graph graph{}; + + // We create enough vertices to construct the requested number of edges + const auto number_of_vertices{number_of_edges + 1}; + const auto vertices{create_vertices(graph, number_of_vertices)}; + size_t clique_size = random_clique_size(rng); + + // Connecting all cliques + for (size_t i{0}; i < number_of_edges; ++i) { + graph.add_edge(vertices[i], vertices[i + 1], 1); + } + + for (size_t i{0}; i + clique_size < number_of_edges; i += clique_size) { + add_clique(graph, vertices, i, clique_size++); + clique_size = random_clique_size(rng); + } + + for (auto _ : state) { + auto result = graaf::algorithm::bron_kerbosch(graph); + benchmark::DoNotOptimize(result); + } + + state.SetComplexityN(state.range(0)); +} + +BENCHMARK(bron_kerbosh_cliques) + ->Ranges({{100, 10000}, {2, 32}}) /* clique size 2 ^ n */ + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK(bron_kerbosh_connected_cliques) + ->Ranges({{100, 10000}, {2, 32}}) /* clique size 2 ^ n */ + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK(bron_kerbosh_cliques) + ->Ranges({{100, 1000}, + {1000, 1000}}) /* Dense graph, number of vertices are + min(state.range(1), vertex.size()) */ + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK(bron_kerbosh_cliques) + ->Ranges({{100, 10000}, {10, 60}}) /* large cliques */ + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK(bron_kerbosh_connected_cliques) + ->Ranges({{100, 10000}, {10, 60}}) /* large cliques */ + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK(bron_kerbosh_random_cliques) /* random clique size between 1-30 */ + ->Range(100, 10000) + ->Unit(benchmark::kMillisecond) + ->Complexity(); +BENCHMARK( + bron_kerbosh_connected_random_cliques) /* random clique size between 1-30 */ + ->Range(100, 10000) + ->Unit(benchmark::kMillisecond) + ->Complexity(); \ No newline at end of file