2018-11-21 01:15:17 +08:00
|
|
|
|
|
|
|
|
|
#include <algorithm>
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
#include <cstdint>
|
2018-11-21 01:15:17 +08:00
|
|
|
|
#include <map>
|
|
|
|
|
#include <random>
|
|
|
|
|
#include <string>
|
|
|
|
|
#include <utility>
|
|
|
|
|
#include <vector>
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2019-08-21 09:59:12 +08:00
|
|
|
|
#include "CartesianBenchmarks.h"
|
|
|
|
|
#include "GenerateInput.h"
|
2018-11-21 01:15:17 +08:00
|
|
|
|
#include "benchmark/benchmark.h"
|
|
|
|
|
#include "test_macros.h"
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
enum class ValueType { Uint32, Uint64, Pair, Tuple, String };
|
|
|
|
|
struct AllValueTypes : EnumValuesAsTuple<AllValueTypes, ValueType, 5> {
|
|
|
|
|
static constexpr const char* Names[] = {
|
|
|
|
|
"uint32", "uint64", "pair<uint32, uint32>",
|
|
|
|
|
"tuple<uint32, uint64, uint32>", "string"};
|
2018-11-21 01:15:17 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <class V>
|
2020-07-07 06:29:38 +08:00
|
|
|
|
using Value = std::conditional_t<
|
|
|
|
|
V() == ValueType::Uint32, uint32_t,
|
|
|
|
|
std::conditional_t<
|
|
|
|
|
V() == ValueType::Uint64, uint64_t,
|
|
|
|
|
std::conditional_t<
|
|
|
|
|
V() == ValueType::Pair, std::pair<uint32_t, uint32_t>,
|
|
|
|
|
std::conditional_t<V() == ValueType::Tuple,
|
|
|
|
|
std::tuple<uint32_t, uint64_t, uint32_t>,
|
|
|
|
|
std::string> > > >;
|
2018-11-21 01:15:17 +08:00
|
|
|
|
|
|
|
|
|
enum class Order {
|
|
|
|
|
Random,
|
|
|
|
|
Ascending,
|
|
|
|
|
Descending,
|
|
|
|
|
SingleElement,
|
|
|
|
|
PipeOrgan,
|
2021-11-17 00:37:55 +08:00
|
|
|
|
Heap,
|
|
|
|
|
QuickSortAdversary,
|
2018-11-21 01:15:17 +08:00
|
|
|
|
};
|
2021-11-17 00:37:55 +08:00
|
|
|
|
struct AllOrders : EnumValuesAsTuple<AllOrders, Order, 7> {
|
2018-11-21 01:15:17 +08:00
|
|
|
|
static constexpr const char* Names[] = {"Random", "Ascending",
|
|
|
|
|
"Descending", "SingleElement",
|
2021-11-17 00:37:55 +08:00
|
|
|
|
"PipeOrgan", "Heap",
|
|
|
|
|
"QuickSortAdversary"};
|
2018-11-21 01:15:17 +08:00
|
|
|
|
};
|
|
|
|
|
|
2021-11-17 00:37:55 +08:00
|
|
|
|
// fillAdversarialQuickSortInput fills the input vector with N int-like values.
|
|
|
|
|
// These values are arranged in such a way that they would invoke O(N^2)
|
|
|
|
|
// behavior on any quick sort implementation that satisifies certain conditions.
|
|
|
|
|
// Details are available in the following paper:
|
|
|
|
|
// "A Killer Adversary for Quicksort", M. D. McIlroy, Software—Practice &
|
|
|
|
|
// ExperienceVolume 29 Issue 4 April 10, 1999 pp 341–344.
|
|
|
|
|
// https://dl.acm.org/doi/10.5555/311868.311871.
|
|
|
|
|
template <class T>
|
|
|
|
|
void fillAdversarialQuickSortInput(T& V, size_t N) {
|
|
|
|
|
assert(N > 0);
|
|
|
|
|
// If an element is equal to gas, it indicates that the value of the element
|
|
|
|
|
// is still to be decided and may change over the course of time.
|
|
|
|
|
const int gas = N - 1;
|
|
|
|
|
V.resize(N);
|
|
|
|
|
for (int i = 0; i < N; ++i) {
|
|
|
|
|
V[i] = gas;
|
|
|
|
|
}
|
|
|
|
|
// Candidate for the pivot position.
|
|
|
|
|
int candidate = 0;
|
|
|
|
|
int nsolid = 0;
|
|
|
|
|
// Populate all positions in the generated input to gas.
|
|
|
|
|
std::vector<int> ascVals(V.size());
|
|
|
|
|
// Fill up with ascending values from 0 to V.size()-1. These will act as
|
|
|
|
|
// indices into V.
|
|
|
|
|
std::iota(ascVals.begin(), ascVals.end(), 0);
|
|
|
|
|
std::sort(ascVals.begin(), ascVals.end(), [&](int x, int y) {
|
|
|
|
|
if (V[x] == gas && V[y] == gas) {
|
|
|
|
|
// We are comparing two inputs whose value is still to be decided.
|
|
|
|
|
if (x == candidate) {
|
|
|
|
|
V[x] = nsolid++;
|
|
|
|
|
} else {
|
|
|
|
|
V[y] = nsolid++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (V[x] == gas) {
|
|
|
|
|
candidate = x;
|
|
|
|
|
} else if (V[y] == gas) {
|
|
|
|
|
candidate = y;
|
|
|
|
|
}
|
|
|
|
|
return V[x] < V[y];
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
template <typename T>
|
|
|
|
|
void fillValues(std::vector<T>& V, size_t N, Order O) {
|
2018-11-21 01:15:17 +08:00
|
|
|
|
if (O == Order::SingleElement) {
|
|
|
|
|
V.resize(N, 0);
|
2021-11-17 00:37:55 +08:00
|
|
|
|
} else if (O == Order::QuickSortAdversary) {
|
|
|
|
|
fillAdversarialQuickSortInput(V, N);
|
2018-11-21 01:15:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
while (V.size() < N)
|
|
|
|
|
V.push_back(V.size());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
template <typename T>
|
|
|
|
|
void fillValues(std::vector<std::pair<T, T> >& V, size_t N, Order O) {
|
|
|
|
|
if (O == Order::SingleElement) {
|
|
|
|
|
V.resize(N, std::make_pair(0, 0));
|
|
|
|
|
} else {
|
|
|
|
|
while (V.size() < N)
|
|
|
|
|
// Half of array will have the same first element.
|
|
|
|
|
if (V.size() % 2) {
|
|
|
|
|
V.push_back(std::make_pair(V.size(), V.size()));
|
|
|
|
|
} else {
|
|
|
|
|
V.push_back(std::make_pair(0, V.size()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-11-21 01:15:17 +08:00
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
template <typename T1, typename T2, typename T3>
|
|
|
|
|
void fillValues(std::vector<std::tuple<T1, T2, T3> >& V, size_t N, Order O) {
|
|
|
|
|
if (O == Order::SingleElement) {
|
|
|
|
|
V.resize(N, std::make_tuple(0, 0, 0));
|
|
|
|
|
} else {
|
|
|
|
|
while (V.size() < N)
|
|
|
|
|
// One third of array will have the same first element.
|
|
|
|
|
// One third of array will have the same first element and the same second element.
|
|
|
|
|
switch (V.size() % 3) {
|
|
|
|
|
case 0:
|
|
|
|
|
V.push_back(std::make_tuple(V.size(), V.size(), V.size()));
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
V.push_back(std::make_tuple(0, V.size(), V.size()));
|
|
|
|
|
break;
|
|
|
|
|
case 2:
|
|
|
|
|
V.push_back(std::make_tuple(0, 0, V.size()));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void fillValues(std::vector<std::string>& V, size_t N, Order O) {
|
2018-11-21 01:15:17 +08:00
|
|
|
|
if (O == Order::SingleElement) {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
V.resize(N, getRandomString(64));
|
2018-11-21 01:15:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
while (V.size() < N)
|
2020-07-07 06:29:38 +08:00
|
|
|
|
V.push_back(getRandomString(64));
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
|
void sortValues(T& V, Order O) {
|
|
|
|
|
assert(std::is_sorted(V.begin(), V.end()));
|
|
|
|
|
switch (O) {
|
|
|
|
|
case Order::Random: {
|
|
|
|
|
std::random_device R;
|
|
|
|
|
std::mt19937 M(R());
|
|
|
|
|
std::shuffle(V.begin(), V.end(), M);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case Order::Ascending:
|
|
|
|
|
std::sort(V.begin(), V.end());
|
|
|
|
|
break;
|
|
|
|
|
case Order::Descending:
|
|
|
|
|
std::sort(V.begin(), V.end(), std::greater<>());
|
|
|
|
|
break;
|
|
|
|
|
case Order::SingleElement:
|
|
|
|
|
// Nothing to do
|
|
|
|
|
break;
|
|
|
|
|
case Order::PipeOrgan:
|
|
|
|
|
std::sort(V.begin(), V.end());
|
|
|
|
|
std::reverse(V.begin() + V.size() / 2, V.end());
|
|
|
|
|
break;
|
|
|
|
|
case Order::Heap:
|
|
|
|
|
std::make_heap(V.begin(), V.end());
|
|
|
|
|
break;
|
2021-11-17 00:37:55 +08:00
|
|
|
|
case Order::QuickSortAdversary:
|
|
|
|
|
// Nothing to do
|
|
|
|
|
break;
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
constexpr size_t TestSetElements =
|
|
|
|
|
#if !TEST_HAS_FEATURE(memory_sanitizer)
|
|
|
|
|
1 << 18;
|
|
|
|
|
#else
|
|
|
|
|
1 << 14;
|
|
|
|
|
#endif
|
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType>
|
|
|
|
|
std::vector<std::vector<Value<ValueType> > > makeOrderedValues(size_t N,
|
|
|
|
|
Order O) {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
std::vector<std::vector<Value<ValueType> > > Ret;
|
|
|
|
|
const size_t NumCopies = std::max(size_t{1}, TestSetElements / N);
|
|
|
|
|
Ret.resize(NumCopies);
|
|
|
|
|
for (auto& V : Ret) {
|
|
|
|
|
fillValues(V, N, O);
|
|
|
|
|
sortValues(V, O);
|
|
|
|
|
}
|
|
|
|
|
return Ret;
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <class T, class U>
|
|
|
|
|
TEST_ALWAYS_INLINE void resetCopies(benchmark::State& state, T& Copies,
|
|
|
|
|
U& Orig) {
|
|
|
|
|
state.PauseTiming();
|
|
|
|
|
for (auto& Copy : Copies)
|
|
|
|
|
Copy = Orig;
|
|
|
|
|
state.ResumeTiming();
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
enum class BatchSize {
|
|
|
|
|
CountElements,
|
|
|
|
|
CountBatch,
|
|
|
|
|
};
|
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType, class F>
|
|
|
|
|
void runOpOnCopies(benchmark::State& state, size_t Quantity, Order O,
|
2020-07-07 06:29:38 +08:00
|
|
|
|
BatchSize Count, F Body) {
|
2018-11-21 01:15:17 +08:00
|
|
|
|
auto Copies = makeOrderedValues<ValueType>(Quantity, O);
|
2020-07-07 06:29:38 +08:00
|
|
|
|
auto Orig = Copies;
|
2018-11-21 01:15:17 +08:00
|
|
|
|
|
2020-07-07 06:29:38 +08:00
|
|
|
|
const size_t Batch = Count == BatchSize::CountElements
|
|
|
|
|
? Copies.size() * Quantity
|
|
|
|
|
: Copies.size();
|
2018-11-21 01:15:17 +08:00
|
|
|
|
while (state.KeepRunningBatch(Batch)) {
|
|
|
|
|
for (auto& Copy : Copies) {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
Body(Copy);
|
2018-11-21 01:15:17 +08:00
|
|
|
|
benchmark::DoNotOptimize(Copy);
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
}
|
2020-07-07 06:29:38 +08:00
|
|
|
|
state.PauseTiming();
|
|
|
|
|
Copies = Orig;
|
|
|
|
|
state.ResumeTiming();
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType, class Order>
|
|
|
|
|
struct Sort {
|
|
|
|
|
size_t Quantity;
|
|
|
|
|
|
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(
|
|
|
|
|
state, Quantity, Order(), BatchSize::CountElements,
|
|
|
|
|
[](auto& Copy) { std::sort(Copy.begin(), Copy.end()); });
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool skip() const { return Order() == ::Order::Heap; }
|
|
|
|
|
|
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_Sort" + ValueType::name() + Order::name() + "_" +
|
|
|
|
|
std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <class ValueType, class Order>
|
|
|
|
|
struct StableSort {
|
|
|
|
|
size_t Quantity;
|
|
|
|
|
|
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(
|
|
|
|
|
state, Quantity, Order(), BatchSize::CountElements,
|
|
|
|
|
[](auto& Copy) { std::stable_sort(Copy.begin(), Copy.end()); });
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool skip() const { return Order() == ::Order::Heap; }
|
|
|
|
|
|
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_StableSort" + ValueType::name() + Order::name() + "_" +
|
|
|
|
|
std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <class ValueType, class Order>
|
|
|
|
|
struct MakeHeap {
|
|
|
|
|
size_t Quantity;
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(
|
|
|
|
|
state, Quantity, Order(), BatchSize::CountElements,
|
|
|
|
|
[](auto& Copy) { std::make_heap(Copy.begin(), Copy.end()); });
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_MakeHeap" + ValueType::name() + Order::name() + "_" +
|
|
|
|
|
std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType>
|
|
|
|
|
struct SortHeap {
|
|
|
|
|
size_t Quantity;
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
void run(benchmark::State& state) const {
|
|
|
|
|
runOpOnCopies<ValueType>(
|
2020-07-07 06:29:38 +08:00
|
|
|
|
state, Quantity, Order::Heap, BatchSize::CountElements,
|
2018-11-21 01:15:17 +08:00
|
|
|
|
[](auto& Copy) { std::sort_heap(Copy.begin(), Copy.end()); });
|
|
|
|
|
}
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_SortHeap" + ValueType::name() + "_" + std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType, class Order>
|
|
|
|
|
struct MakeThenSortHeap {
|
|
|
|
|
size_t Quantity;
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(state, Quantity, Order(), BatchSize::CountElements,
|
|
|
|
|
[](auto& Copy) {
|
|
|
|
|
std::make_heap(Copy.begin(), Copy.end());
|
|
|
|
|
std::sort_heap(Copy.begin(), Copy.end());
|
|
|
|
|
});
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_MakeThenSortHeap" + ValueType::name() + Order::name() + "_" +
|
|
|
|
|
std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
template <class ValueType, class Order>
|
|
|
|
|
struct PushHeap {
|
|
|
|
|
size_t Quantity;
|
[libcxx] Add support for benchmark tests using Google Benchmark.
Summary:
This patch does the following:
1. Checks in a copy of the Google Benchmark library into the libc++ repo under `utils/google-benchmark`.
2. Teaches libc++ how to build Google Benchmark against both (A) in-tree libc++ and (B) the platforms native STL.
3. Allows performance benchmarks to be built as part of the libc++ build.
Building the benchmarks (and Google Benchmark) is off by default. It must be enabled using the CMake option `-DLIBCXX_INCLUDE_BENCHMARKS=ON`. When this option is enabled the tests under `libcxx/benchmarks` can be built using the `libcxx-benchmarks` target.
On Linux platforms where libstdc++ is the default STL the CMake option `-DLIBCXX_BUILD_BENCHMARKS_NATIVE_STDLIB=ON` can be used to build each benchmark test against libstdc++ as well. This is useful for comparing performance between standard libraries.
Support for benchmarks is currently very minimal. They must be manually run by the user and there is no mechanism for detecting performance regressions.
Known Issues:
* `-DLIBCXX_INCLUDE_BENCHMARKS=ON` is only supported for Clang, and not GCC, since the `-stdlib=libc++` option is needed to build Google Benchmark.
Reviewers: danalbert, dberlin, chandlerc, mclow.lists, jroelofs
Subscribers: chandlerc, dberlin, tberghammer, danalbert, srhines, hfinkel
Differential Revision: https://reviews.llvm.org/D22240
llvm-svn: 276049
2016-07-20 07:07:03 +08:00
|
|
|
|
|
2018-11-21 01:15:17 +08:00
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(
|
|
|
|
|
state, Quantity, Order(), BatchSize::CountElements, [](auto& Copy) {
|
|
|
|
|
for (auto I = Copy.begin(), E = Copy.end(); I != E; ++I) {
|
|
|
|
|
std::push_heap(Copy.begin(), I + 1);
|
|
|
|
|
}
|
|
|
|
|
});
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool skip() const { return Order() == ::Order::Heap; }
|
|
|
|
|
|
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_PushHeap" + ValueType::name() + Order::name() + "_" +
|
|
|
|
|
std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <class ValueType>
|
|
|
|
|
struct PopHeap {
|
|
|
|
|
size_t Quantity;
|
|
|
|
|
|
|
|
|
|
void run(benchmark::State& state) const {
|
2020-07-07 06:29:38 +08:00
|
|
|
|
runOpOnCopies<ValueType>(
|
|
|
|
|
state, Quantity, Order(), BatchSize::CountElements, [](auto& Copy) {
|
|
|
|
|
for (auto B = Copy.begin(), I = Copy.end(); I != B; --I) {
|
|
|
|
|
std::pop_heap(B, I);
|
|
|
|
|
}
|
|
|
|
|
});
|
2018-11-21 01:15:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string name() const {
|
|
|
|
|
return "BM_PopHeap" + ValueType::name() + "_" + std::to_string(Quantity);
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
|
benchmark::Initialize(&argc, argv);
|
|
|
|
|
if (benchmark::ReportUnrecognizedArguments(argc, argv))
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
|
|
const std::vector<size_t> Quantities = {1 << 0, 1 << 2, 1 << 4, 1 << 6,
|
2019-08-21 08:14:48 +08:00
|
|
|
|
1 << 8, 1 << 10, 1 << 14,
|
|
|
|
|
// Running each benchmark in parallel consumes too much memory with MSAN
|
|
|
|
|
// and can lead to the test process being killed.
|
|
|
|
|
#if !TEST_HAS_FEATURE(memory_sanitizer)
|
|
|
|
|
1 << 18
|
|
|
|
|
#endif
|
|
|
|
|
};
|
2018-11-21 01:15:17 +08:00
|
|
|
|
makeCartesianProductBenchmark<Sort, AllValueTypes, AllOrders>(Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<StableSort, AllValueTypes, AllOrders>(
|
|
|
|
|
Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<MakeHeap, AllValueTypes, AllOrders>(Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<SortHeap, AllValueTypes>(Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<MakeThenSortHeap, AllValueTypes, AllOrders>(
|
|
|
|
|
Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<PushHeap, AllValueTypes, AllOrders>(Quantities);
|
|
|
|
|
makeCartesianProductBenchmark<PopHeap, AllValueTypes>(Quantities);
|
|
|
|
|
benchmark::RunSpecifiedBenchmarks();
|
2021-10-23 23:45:29 +08:00
|
|
|
|
}
|