From 0a5a45ff8ae437ddf1c3b8de425f30a44e6f5580 Mon Sep 17 00:00:00 2001 From: David Robillard Date: Sun, 26 Aug 2018 23:16:00 +0200 Subject: Add benchmarks --- Makefile | 15 ++- benchmarks/plot.py | 155 ++++++++++++++++++++++ test/bench_bitvec.cpp | 339 +++++++++++++++++++++++++++++++++++++++++++++++++ test/bench_hilbert.cpp | 111 ++++++++++++++++ test/bench_utils.hpp | 64 ++++++++++ 5 files changed, 681 insertions(+), 3 deletions(-) create mode 100755 benchmarks/plot.py create mode 100644 test/bench_bitvec.cpp create mode 100644 test/bench_hilbert.cpp create mode 100644 test/bench_utils.hpp diff --git a/Makefile b/Makefile index e397515..ed93b0a 100644 --- a/Makefile +++ b/Makefile @@ -40,6 +40,7 @@ endif # DEBUG HEADERS = $(wildcard chilbert/*.hpp) $(wildcard chilbert/*.ipp) DETAIL_HEADERS = $(wildcard chilbert/detail/*.hpp) TESTS = test/test_bitvec test/test_gray_code_rank test/test_hilbert +BENCHMARKS = test/bench_bitvec test/bench_hilbert PROGS = bin/chilbert_obj bin/chilbert_svg ALL_HEADERS = $(HEADERS) $(DETAIL_HEADERS) @@ -49,6 +50,8 @@ all: $(PROGS) tests: $(TESTS) +benchmarks: $(BENCHMARKS) + test/%: test/%.cpp $(ALL_HEADERS) $(CXX) $(CXXFLAGS) -o $@ $@.cpp @@ -66,12 +69,18 @@ bin/%: bin/%.cpp $(ALL_HEADERS) FORCE: clean: - rm -f $(TESTS) $(PROGS) + rm -f $(PROGS) $(TESTS) $(BENCHMARKS) -%.run: tests +%.run: $* -test: $(addsuffix .run, $(TESTS)) FORCE +test: tests $(addsuffix .run, $(TESTS)) FORCE + +bench: $(BENCHMARKS) + mkdir -p benchmarks + cd benchmarks && ../test/bench_bitvec + cd benchmarks && ../test/bench_hilbert + cd benchmarks && ./plot.py install: FORCE install -d $(DESTDIR)$(includedir)/chilbert/ diff --git a/benchmarks/plot.py b/benchmarks/plot.py new file mode 100755 index 0000000..b329ed4 --- /dev/null +++ b/benchmarks/plot.py @@ -0,0 +1,155 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +import csv +import glob +import itertools +import math +import matplotlib.pyplot as plt +import sys + + +def get_dashes(): + """Generator for plot line dash patterns.""" + dash = 2.0 + space = dot = 0.75 + + yield [] # Solid + yield [dash, space] # Dashed + yield [dot, space] # Dotted + + # Dash-dots, with increasing number of dots for each line + for i in itertools.count(2): + yield [dash, space] + [dot, space] * (i - 1) + + +def plot(in_file, out_filename, x_label, y_max=None): + fig_height = 4.0 + dashes = get_dashes() + markers = itertools.cycle( + ['o', 's', 'v', 'D', '*', 'p', 'P', 'h', 'X', '+', 'x']) + + reader = csv.reader(in_file, delimiter='\t') + header = next(reader) + cols = [x for x in zip(*list(reader))] + + plt.clf() + fig = plt.figure(figsize=(fig_height * math.sqrt(2), fig_height)) + ax = fig.add_subplot(111) + + ax.set_xlabel(x_label) + + ax.set_ylabel(u'Time (μs)') + if y_max is not None: + ax.set_ylim([0.0, y_max]) + + ax.grid(linewidth=0.25, linestyle=':', color='0', dashes=[0.2, 1.6]) + ax.tick_params(axis='both', width=0.75) + + x = cols[0] + for i, y in enumerate(cols[1::]): + ax.plot(x, + list(map(float, y)), + label=header[i + 1], + marker=next(markers), + dashes=next(dashes), + markersize=3.0, + linewidth=1.0) + + plt.legend(loc='upper left') + plt.savefig(out_filename, bbox_inches='tight', pad_inches=0.025) + plt.close() + sys.stderr.write('wrote {}\n'.format(out_filename)) + + +def plot_section(x_label, plots): + # Get maximum value for all Y axes + y_max = 0.0 + for p in plots: + name = p[0] + filename = name + '.txt' + + with open(filename, 'r') as in_file: + reader = csv.reader(in_file, delimiter='\t') + next(reader) + for row in reader: + y_max = max(y_max, *map(float, row[1::])) + + html = '' + for p in plots: + name = p[0] + caption = p[1] + filename = name + '.txt' + out_filename = filename.replace('.txt', '.svg') + + with open(filename, 'r') as in_file: + plot(in_file, filename.replace('.txt', '.svg'), x_label, y_max) + + html += ''.join(['
', + ''.format(out_filename), + '
{}
'.format(caption), + '
\n']) + + return html + + +if __name__ == '__main__': + files = glob.glob('*.txt') + + html = ''' + + Hilbert Benchmarks + + + +

CHilbert Benchmarks

+''' + + html += '

Bit Vector Operations

' + html += plot_section( + 'Bits', + [('bitvec_and', 'Bitwise AND'), + ('bitvec_or', 'Bitwise OR'), + ('bitvec_xor', 'Bitwise XOR'), + ('bitvec_none', 'Zero test'), + ('bitvec_set_one', 'Set bit'), + ('bitvec_set_all', 'Set all'), + ('bitvec_reset_one', 'Reset bit'), + ('bitvec_reset_all', 'Reset all'), + ('bitvec_flip_one', 'Flip bit'), + ('bitvec_flip_all', 'Flip all'), + ('bitvec_count', 'Count'), + ('bitvec_find_first', 'Find first'), + ('bitvec_left_shift', 'Left shift'), + ('bitvec_right_shift', 'Right shift'), + ('bitvec_left_rotate', 'Left rotate'), + ('bitvec_right_rotate', 'Right rotate'), + ('bitvec_gray_code', 'Gray code'), + ('bitvec_gray_code_inv', 'Gray code inverse'), + ('bitvec_comparison', 'Comparison'), + ('bitvec_iteration', 'Iteration')]) + + html += '

Hilbert Mapping

\n' + html += plot_section( + 'Dimensions', + [('hilbert_coords_to_index', 'Coordinates to index'), + ('hilbert_index_to_coords', 'Index to coordinates')]) + + html += '\n\n' + with open('benchmarks.html', 'w') as index_file: + index_file.write(html) + + sys.stderr.write('wrote benchmarks.html\n') diff --git a/test/bench_bitvec.cpp b/test/bench_bitvec.cpp new file mode 100644 index 0000000..741b2d2 --- /dev/null +++ b/test/bench_bitvec.cpp @@ -0,0 +1,339 @@ +/* + Copyright (C) 2018 David Robillard + + This program is free software: you can redistribute it and/or modify it under + the terms of the GNU General Public License as published by the Free Software + Foundation, either version 2 of the License, or (at your option) any later + version. + + This program is distributed in the hope that it will be useful, but WITHOUT + ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS + FOR A PARTICULAR PURPOSE. See the GNU General Public License for more + details. + + You should have received a copy of the GNU General Public License along with + this program. If not, see . +*/ + +#include "bench_utils.hpp" + +#include "chilbert/BoundedBitVec.hpp" +#include "chilbert/DynamicBitVec.hpp" +#include "chilbert/SmallBitVec.hpp" +#include "chilbert/StaticBitVec.hpp" + +#include + +template +struct BenchAnd +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = make_random_bitvec(ctx); + + return run_bench([&](auto) { r &= v; }); + } +}; + +template +struct BenchOr +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = make_random_bitvec(ctx); + + return run_bench([&](auto) { r |= v; }); + } +}; + +template +struct BenchXor +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = make_random_bitvec(ctx); + + return run_bench([&](auto) { r ^= v; }); + } +}; + +template +struct BenchSetOne +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](const auto i) { v.set(i % N); }); + } +}; + +template +struct BenchSetAll +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](auto) { v.set(); }); + } +}; + +template +struct BenchResetOne +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](const auto i) { v.reset(i % N); }); + } +}; + +template +struct BenchResetAll +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](auto) { v.reset(); }); + } +}; + +template +struct BenchFlipOne +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](const auto i) { v.flip(i % N); }); + } +}; + +template +struct BenchFlipAll +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](auto) { v.flip(); }); + } +}; + +template +struct BenchNone +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + bool r = false; + return run_bench([&](auto) { r = v.none(); }); + } +}; + +template +struct BenchCount +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + volatile size_t r = 0; + return run_bench([&](auto) { r = v.count(); }); + } +}; + +template +struct BenchLeftShift +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = v; + return run_bench([&](const auto i) { r <<= (i % N); }); + } +}; + +template +struct BenchRightShift +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = v; + return run_bench([&](const auto i) { r >>= (i % N); }); + } +}; + +template +struct BenchLeftRotate +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + return run_bench([&](const auto i) { v.rotl(i % N); }); + } +}; + +template +struct BenchRightRotate +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + volatile bool r = false; + return run_bench([&](const auto i) { + v.rotr(i % N); + r = v.test(0); + }); + } +}; + +template +struct BenchFindFirst +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + volatile size_t r = 0; + return run_bench([&](auto) { r = v.find_first(); }); + } +}; + +template +struct BenchGrayCode +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = v; + + return run_bench([&](auto) { chilbert::detail::gray_code(r); }); + } +}; + +template +struct BenchGrayCodeInv +{ + Duration operator()(Context& ctx) + { + const T v = make_random_bitvec(ctx); + T r = v; + + return run_bench([&](auto) { chilbert::detail::gray_code_inv(r); }); + } +}; + +template +struct BenchComparison +{ + Duration operator()(Context& ctx) + { + std::vector vecs; + for (size_t i = 0; i < 32; ++i) { + vecs.emplace_back(make_random_bitvec(ctx)); + } + + volatile bool r = false; + + return run_bench([&](const auto i) { + r |= vecs[i % vecs.size()] < vecs[(i + 1) % vecs.size()]; + }); + } +}; + +template +struct BenchIteration +{ + Duration operator()(Context& ctx) + { + T v = make_random_bitvec(ctx); + auto i = v.begin(); + volatile bool r = false; + return run_bench([&](const auto) { + r = (++i == v.end()); + if (r) { + i = v.begin(); + } + }); + } +}; + +/// Run benchmark for size N +template