Connector Switch c06ae43e26
[libcxx] Optimize std::generate_n for segmented iterators (#164266)
Part of #102817.

This is a natural follow-up to #163006. We are forwarding
`std::generate_n` to `std::__for_each_n` (`std::for_each_n` needs
c++17), resulting in improved performance for segmented iterators.

before:

```
std::generate_n(deque<int>)/32          17.5 ns         17.3 ns     40727273
std::generate_n(deque<int>)/50          25.7 ns         25.5 ns     26352941
std::generate_n(deque<int>)/1024         490 ns          487 ns      1445161
std::generate_n(deque<int>)/8192        3908 ns         3924 ns       179200
```

after:

```
std::generate_n(deque<int>)/32          11.1 ns         11.0 ns     64000000
std::generate_n(deque<int>)/50          16.1 ns         16.0 ns     44800000
std::generate_n(deque<int>)/1024         291 ns          292 ns      2357895
std::generate_n(deque<int>)/8192        2269 ns         2250 ns       298667
```
2025-10-21 12:01:36 +02:00

98 lines
2.5 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<class Iter, IntegralLike Size, Callable Generator>
// requires OutputIterator<Iter, Generator::result_type>
// && CopyConstructible<Generator>
// constexpr void // constexpr after c++17
// generate_n(Iter first, Size n, Generator gen);
#include <algorithm>
#include <cassert>
#include <deque>
#include "test_iterators.h"
#include "test_macros.h"
#include "user_defined_integral.h"
TEST_MSVC_DIAGNOSTIC_IGNORED(4244) // conversion from 'const double' to 'int', possible loss of data
struct gen_test
{
TEST_CONSTEXPR int operator()() const {return 2;}
};
#if TEST_STD_VER > 17
TEST_CONSTEXPR bool test_constexpr() {
const std::size_t N = 5;
int ib[] = {0, 0, 0, 0, 0, 0}; // one bigger than N
auto it = std::generate_n(std::begin(ib), N, gen_test());
return it == (std::begin(ib) + N)
&& std::all_of(std::begin(ib), it, [](int x) { return x == 2; })
&& *it == 0 // don't overwrite the last value in the output array
;
}
#endif
template <class Iter, class Size>
void
test2()
{
const unsigned n = 4;
int ia[n] = {0};
assert(std::generate_n(Iter(ia), Size(n), gen_test()) == Iter(ia+n));
assert(ia[0] == 2);
assert(ia[1] == 2);
assert(ia[2] == 2);
assert(ia[3] == 2);
}
template <class Iter>
void
test()
{
test2<Iter, int>();
test2<Iter, unsigned int>();
test2<Iter, long>();
test2<Iter, unsigned long>();
test2<Iter, UserDefinedIntegral<unsigned> >();
test2<Iter, float>();
test2<Iter, double>(); // this is PR#35498
test2<Iter, long double>();
}
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_n(d.begin(), size, gen_test());
assert(std::all_of(d.begin(), d.end(), [](int x) { return x == 2; }));
}
}
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;
}