1 //===----------------------------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #ifndef SUPPORT_FROM_RANGE_CONTAINER_ADAPTORS_H
10 #define SUPPORT_FROM_RANGE_CONTAINER_ADAPTORS_H
20 #include "../exception_safety_helpers.h"
21 #include "../from_range_helpers.h"
23 #include "almost_satisfies_types.h"
24 #include "count_new.h"
25 #include "test_macros.h"
26 #include "unwrap_container_adaptor.h"
28 template <class Container
, class Range
>
29 concept HasFromRangeCtr
= requires (Range
&& range
) {
30 Container(std::from_range
, std::forward
<Range
>(range
));
31 Container(std::from_range
, std::forward
<Range
>(range
), std::allocator
<typename
Container::value_type
>());
34 template <template <class...> class Container
, class T
, class U
>
35 constexpr bool test_constraints() {
36 // Input range with the same value type.
37 static_assert(HasFromRangeCtr
<Container
<T
>, InputRange
<T
>>);
38 // Input range with a convertible value type.
39 static_assert(HasFromRangeCtr
<Container
<T
>, InputRange
<U
>>);
40 // Input range with a non-convertible value type.
41 static_assert(!HasFromRangeCtr
<Container
<T
>, InputRange
<Empty
>>);
42 // Not an input range.
43 static_assert(!HasFromRangeCtr
<Container
<T
>, InputRangeNotDerivedFrom
>);
44 static_assert(!HasFromRangeCtr
<Container
<T
>, InputRangeNotIndirectlyReadable
>);
45 static_assert(!HasFromRangeCtr
<Container
<T
>, InputRangeNotInputOrOutputIterator
>);
50 template <template <class ...> class Adaptor
,
51 template <class ...> class UnderlyingContainer
,
56 constexpr void test_container_adaptor_with_input(std::vector
<T
>&& input
) {
58 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
59 Adaptor
<T
> adaptor(std::from_range
, in
);
60 UnwrapAdaptor
<Adaptor
<T
>> unwrap_adaptor(std::move(adaptor
));
61 auto& c
= unwrap_adaptor
.get_container();
63 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
64 assert(std::ranges::equal(input
, c
));
65 LIBCPP_ASSERT(c
.__invariants());
68 { // (range, allocator)
69 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
70 using C
= UnderlyingContainer
<T
, Alloc
>;
72 Adaptor
<T
, C
> adaptor(std::from_range
, in
, alloc
);
73 UnwrapAdaptor
<Adaptor
<T
, C
>> unwrap_adaptor(std::move(adaptor
));
74 auto& c
= unwrap_adaptor
.get_container();
76 assert(c
.get_allocator() == alloc
);
77 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
78 assert(std::ranges::equal(input
, c
));
79 LIBCPP_ASSERT(c
.__invariants());
83 template <template <class ...> class UnderlyingContainer
,
89 constexpr void test_priority_queue_with_input(std::vector
<T
>&& input
) {
91 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
92 std::priority_queue
<T
> adaptor(std::from_range
, in
);
93 UnwrapAdaptor
<std::priority_queue
<T
>> unwrap_adaptor(std::move(adaptor
));
94 auto& c
= unwrap_adaptor
.get_container();
96 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
97 assert(std::ranges::is_permutation(input
, c
));
98 LIBCPP_ASSERT(c
.__invariants());
102 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
103 using C
= UnderlyingContainer
<T
>;
106 std::priority_queue
<T
, C
, Comp
> adaptor(std::from_range
, in
, comp
);
107 UnwrapAdaptor
<std::priority_queue
<T
, C
, Comp
>> unwrap_adaptor(std::move(adaptor
));
108 auto& c
= unwrap_adaptor
.get_container();
110 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
111 assert(std::ranges::is_permutation(input
, c
));
112 LIBCPP_ASSERT(c
.__invariants());
113 assert(unwrap_adaptor
.get_comparator() == comp
);
116 { // (range, allocator)
117 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
118 using C
= UnderlyingContainer
<T
, Alloc
>;
121 std::priority_queue
<T
, C
> adaptor(std::from_range
, in
, alloc
);
122 UnwrapAdaptor
<std::priority_queue
<T
, C
>> unwrap_adaptor(std::move(adaptor
));
123 auto& c
= unwrap_adaptor
.get_container();
125 assert(c
.get_allocator() == alloc
);
126 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
127 assert(std::ranges::is_permutation(input
, c
));
128 LIBCPP_ASSERT(c
.__invariants());
131 { // (range, comp, alloc)
132 std::ranges::subrange
in(Iter(input
.data()), Sent(Iter(input
.data() + input
.size())));
133 using C
= UnderlyingContainer
<T
, Alloc
>;
137 std::priority_queue
<T
, C
, Comp
> adaptor(std::from_range
, in
, comp
, alloc
);
138 UnwrapAdaptor
<std::priority_queue
<T
, C
, Comp
>> unwrap_adaptor(std::move(adaptor
));
139 auto& c
= unwrap_adaptor
.get_container();
141 assert(c
.get_allocator() == alloc
);
142 assert(c
.size() == static_cast<std::size_t>(std::distance(c
.begin(), c
.end())));
143 assert(std::ranges::is_permutation(input
, c
));
144 LIBCPP_ASSERT(c
.__invariants());
145 assert(unwrap_adaptor
.get_comparator() == comp
);
149 template <template <class ...> class Adaptor
,
150 template <class ...> class UnderlyingContainer
,
155 constexpr void test_container_adaptor() {
156 auto test_with_input
= &test_container_adaptor_with_input
<Adaptor
, UnderlyingContainer
, T
, Iter
, Sent
, Alloc
>;
159 test_with_input({0, 5, 12, 7, -1, 8, 26});
162 // Single-element input.
163 test_with_input({5});
166 template <template <class ...> class UnderlyingContainer
,
172 constexpr void test_priority_queue() {
173 auto test_with_input
= &test_priority_queue_with_input
<UnderlyingContainer
, T
, Iter
, Sent
, Comp
, Alloc
>;
176 test_with_input({0, 5, 12, 7, -1, 8, 26});
179 // Single-element input.
180 test_with_input({5});
183 template <template <class ...> class Container
>
184 constexpr void test_container_adaptor_move_only() {
186 std::ranges::subrange
in(std::move_iterator
{input
}, std::move_iterator
{input
+ 5});
188 [[maybe_unused
]] Container
<MoveOnly
> c(std::from_range
, in
);
191 template <template <class ...> class Adaptor
>
192 void test_exception_safety_throwing_copy() {
193 #if !defined(TEST_HAS_NO_EXCEPTIONS)
194 constexpr int ThrowOn
= 3;
195 using T
= ThrowingCopy
<ThrowOn
>;
196 test_exception_safety_throwing_copy
<ThrowOn
, /*Size=*/5>([](T
* from
, T
* to
) {
197 [[maybe_unused
]] Adaptor
<T
, std::vector
<T
>> c(std::from_range
, std::ranges::subrange(from
, to
));
202 template <template <class ...> class Adaptor
, class T
>
203 void test_exception_safety_throwing_allocator() {
204 #if !defined(TEST_HAS_NO_EXCEPTIONS)
208 using C
= std::vector
<T
, ThrowingAllocator
<T
>>;
209 ThrowingAllocator
<T
> alloc
;
211 globalMemCounter
.reset();
212 Adaptor
<T
, C
> c(std::from_range
, in
, alloc
);
213 assert(false); // The constructor call should throw.
216 assert(globalMemCounter
.new_called
== globalMemCounter
.delete_called
);
221 #endif // SUPPORT_FROM_RANGE_CONTAINER_ADAPTORS_H