Part of #102817. This patch attempts to optimize the performance of `std::generate` for segmented iterators. Below are the benchmark numbers from `libcxx\test\benchmarks\algorithms\modifying\generate.bench.cpp`. Test cases that use segmented iterators have also been added. - before ``` std::generate(deque<int>)/32 194 ns 193 ns 3733333 std::generate(deque<int>)/50 276 ns 276 ns 2488889 std::generate(deque<int>)/1024 5096 ns 5022 ns 112000 std::generate(deque<int>)/8192 40806 ns 40806 ns 17231 ``` - after ``` std::generate(deque<int>)/32 106 ns 105 ns 6400000 std::generate(deque<int>)/50 139 ns 138 ns 4977778 std::generate(deque<int>)/1024 2713 ns 2699 ns 248889 std::generate(deque<int>)/8192 18983 ns 19252 ns 37333 ``` --------- Co-authored-by: A. Jiang <de34@live.cn>
78 lines
1.8 KiB
C++
78 lines
1.8 KiB
C++
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// <algorithm>
|
|
|
|
// template<ForwardIterator Iter, Callable Generator>
|
|
// requires OutputIterator<Iter, Generator::result_type>
|
|
// && CopyConstructible<Generator>
|
|
// constexpr void // constexpr after c++17
|
|
// generate(Iter first, Iter last, Generator gen);
|
|
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <deque>
|
|
|
|
#include "test_macros.h"
|
|
#include "test_iterators.h"
|
|
|
|
struct gen_test
|
|
{
|
|
TEST_CONSTEXPR int operator()() const {return 1;}
|
|
};
|
|
|
|
|
|
#if TEST_STD_VER > 17
|
|
TEST_CONSTEXPR bool test_constexpr() {
|
|
int ia[] = {0, 1, 2, 3, 4};
|
|
|
|
std::generate(std::begin(ia), std::end(ia), gen_test());
|
|
|
|
return std::all_of(std::begin(ia), std::end(ia), [](int x) { return x == 1; })
|
|
;
|
|
}
|
|
#endif
|
|
|
|
|
|
template <class Iter>
|
|
void
|
|
test()
|
|
{
|
|
const unsigned n = 4;
|
|
int ia[n] = {0};
|
|
std::generate(Iter(ia), Iter(ia+n), gen_test());
|
|
assert(ia[0] == 1);
|
|
assert(ia[1] == 1);
|
|
assert(ia[2] == 1);
|
|
assert(ia[3] == 1);
|
|
}
|
|
|
|
void deque_test() {
|
|
int sizes[] = {0, 1, 2, 1023, 1024, 1025, 2047, 2048, 2049};
|
|
for (const int size : sizes) {
|
|
std::deque<int> d(size);
|
|
std::generate(d.begin(), d.end(), gen_test());
|
|
assert(std::all_of(d.begin(), d.end(), [](int x) { return x == 1; }));
|
|
}
|
|
}
|
|
|
|
int main(int, char**)
|
|
{
|
|
test<forward_iterator<int*> >();
|
|
test<bidirectional_iterator<int*> >();
|
|
test<random_access_iterator<int*> >();
|
|
test<int*>();
|
|
deque_test();
|
|
|
|
#if TEST_STD_VER > 17
|
|
static_assert(test_constexpr());
|
|
#endif
|
|
|
|
return 0;
|
|
}
|