xref: /llvm-project/libcxx/test/std/numerics/numeric.ops/reduce/pstl.reduce.pass.cpp (revision bd3f5a4bd3d9d7ee8ae801c24c5081073b20abd4)
1*bd3f5a4bSLouis Dionne //===----------------------------------------------------------------------===//
2*bd3f5a4bSLouis Dionne //
3*bd3f5a4bSLouis Dionne // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*bd3f5a4bSLouis Dionne // See https://llvm.org/LICENSE.txt for license information.
5*bd3f5a4bSLouis Dionne // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*bd3f5a4bSLouis Dionne //
7*bd3f5a4bSLouis Dionne //===----------------------------------------------------------------------===//
8*bd3f5a4bSLouis Dionne 
9*bd3f5a4bSLouis Dionne // UNSUPPORTED: c++03, c++11, c++14
10*bd3f5a4bSLouis Dionne 
11*bd3f5a4bSLouis Dionne // UNSUPPORTED: libcpp-has-no-incomplete-pstl
12*bd3f5a4bSLouis Dionne 
13*bd3f5a4bSLouis Dionne // <numeric>
14*bd3f5a4bSLouis Dionne 
15*bd3f5a4bSLouis Dionne // template<class ExecutionPolicy, class ForwardIterator>
16*bd3f5a4bSLouis Dionne //   typename iterator_traits<ForwardIterator>::value_type
17*bd3f5a4bSLouis Dionne //     reduce(ExecutionPolicy&& exec,
18*bd3f5a4bSLouis Dionne //            ForwardIterator first, ForwardIterator last);
19*bd3f5a4bSLouis Dionne // template<class ExecutionPolicy, class ForwardIterator, class T, class BinaryOperation>
20*bd3f5a4bSLouis Dionne //   T reduce(ExecutionPolicy&& exec,
21*bd3f5a4bSLouis Dionne //            ForwardIterator first, ForwardIterator last, T init,
22*bd3f5a4bSLouis Dionne //            BinaryOperation binary_op);
23*bd3f5a4bSLouis Dionne 
24*bd3f5a4bSLouis Dionne #include <numeric>
25*bd3f5a4bSLouis Dionne #include <vector>
26*bd3f5a4bSLouis Dionne 
27*bd3f5a4bSLouis Dionne #include "MoveOnly.h"
28*bd3f5a4bSLouis Dionne #include "test_execution_policies.h"
29*bd3f5a4bSLouis Dionne #include "test_iterators.h"
30*bd3f5a4bSLouis Dionne #include "test_macros.h"
31*bd3f5a4bSLouis Dionne 
32*bd3f5a4bSLouis Dionne template <class Iter, class ValueT>
33*bd3f5a4bSLouis Dionne struct Test {
34*bd3f5a4bSLouis Dionne   template <class Policy>
operator ()Test35*bd3f5a4bSLouis Dionne   void operator()(Policy&& policy) {
36*bd3f5a4bSLouis Dionne     for (const auto& pair : {std::pair{0, 34}, {1, 36}, {2, 39}, {100, 5184}, {350, 61809}}) {
37*bd3f5a4bSLouis Dionne       auto [size, expected] = pair;
38*bd3f5a4bSLouis Dionne       std::vector<int> a(size);
39*bd3f5a4bSLouis Dionne       for (int i = 0; i != size; ++i)
40*bd3f5a4bSLouis Dionne         a[i] = i;
41*bd3f5a4bSLouis Dionne 
42*bd3f5a4bSLouis Dionne       {
43*bd3f5a4bSLouis Dionne         decltype(auto) ret = std::reduce(
44*bd3f5a4bSLouis Dionne             policy, Iter(std::data(a)), Iter(std::data(a) + std::size(a)), ValueT(34), [](ValueT i, ValueT j) {
45*bd3f5a4bSLouis Dionne               return i + j + 2;
46*bd3f5a4bSLouis Dionne             });
47*bd3f5a4bSLouis Dionne         static_assert(std::is_same_v<decltype(ret), ValueT>);
48*bd3f5a4bSLouis Dionne         assert(ret == ValueT(expected));
49*bd3f5a4bSLouis Dionne       }
50*bd3f5a4bSLouis Dionne       {
51*bd3f5a4bSLouis Dionne         decltype(auto) ret = std::reduce(policy, Iter(std::data(a)), Iter(std::data(a) + std::size(a)), ValueT(34));
52*bd3f5a4bSLouis Dionne         static_assert(std::is_same_v<decltype(ret), ValueT>);
53*bd3f5a4bSLouis Dionne         assert(ret == expected - 2 * size);
54*bd3f5a4bSLouis Dionne       }
55*bd3f5a4bSLouis Dionne       {
56*bd3f5a4bSLouis Dionne         decltype(auto) ret = std::reduce(policy, Iter(std::data(a)), Iter(std::data(a) + std::size(a)));
57*bd3f5a4bSLouis Dionne         static_assert(std::is_same_v<decltype(ret), typename std::iterator_traits<Iter>::value_type>);
58*bd3f5a4bSLouis Dionne         assert(ret == expected - 2 * size - 34);
59*bd3f5a4bSLouis Dionne       }
60*bd3f5a4bSLouis Dionne     }
61*bd3f5a4bSLouis Dionne   }
62*bd3f5a4bSLouis Dionne };
63*bd3f5a4bSLouis Dionne 
main(int,char **)64*bd3f5a4bSLouis Dionne int main(int, char**) {
65*bd3f5a4bSLouis Dionne   types::for_each(types::forward_iterator_list<int*>{}, types::apply_type_identity{[](auto v) {
66*bd3f5a4bSLouis Dionne                     using Iter = typename decltype(v)::type;
67*bd3f5a4bSLouis Dionne                     types::for_each(
68*bd3f5a4bSLouis Dionne                         types::type_list<int, MoveOnly>{},
69*bd3f5a4bSLouis Dionne                         TestIteratorWithPolicies<types::partial_instantiation<Test, Iter>::template apply>{});
70*bd3f5a4bSLouis Dionne                   }});
71*bd3f5a4bSLouis Dionne 
72*bd3f5a4bSLouis Dionne   return 0;
73*bd3f5a4bSLouis Dionne }
74