1 // Copyright (C) 2019-2024 Free Software Foundation, Inc.
3 // This file is part of the GNU ISO C++ Library. This library is free
4 // software; you can redistribute it and/or modify it under the
5 // terms of the GNU General Public License as published by the
6 // Free Software Foundation; either version 3, or (at your option)
9 // This library is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License along
15 // with this library; see the file COPYING3. If not see
16 // <http://www.gnu.org/licenses/>.
18 // { dg-do run { target c++20 } }
21 #include <testsuite_hooks.h>
26 using I
= std::common_iterator
<int*, const int*>;
27 static_assert( std::is_default_constructible_v
<I
> );
28 static_assert( std::is_copy_constructible_v
<I
> );
29 static_assert( std::is_move_constructible_v
<I
> );
30 static_assert( std::is_copy_assignable_v
<I
> );
31 static_assert( std::is_move_assignable_v
<I
> );
32 static_assert( std::is_constructible_v
<I
, int*> );
33 static_assert( std::is_constructible_v
<I
, const int*> );
35 static_assert( std::is_nothrow_copy_constructible_v
<I
> ); // GCC extension
36 static_assert( std::is_nothrow_move_constructible_v
<I
> ); // GCC extension
37 static_assert( std::is_nothrow_copy_assignable_v
<I
> ); // GCC extension
38 static_assert( std::is_nothrow_move_assignable_v
<I
> ); // GCC extension
40 struct sentinel
{ operator int*() const noexcept
{ return nullptr; } };
41 using K
= std::common_iterator
<int*, sentinel
>;
42 static_assert( std::is_constructible_v
<I
, const K
&> );
43 static_assert( std::is_assignable_v
<I
, const K
&> );
45 static_assert( std::is_nothrow_assignable_v
<I
&, const K
&> ); // GCC extension
47 struct sentinel_throwing
{ operator int*() const { return nullptr; } };
48 using K_throwing
= std::common_iterator
<int*, sentinel_throwing
>;
49 // Conversion is noexcept(false)
50 static_assert( ! std::is_nothrow_assignable_v
<I
&, const K_throwing
&> );
56 sentinel2(const int* p
= 0) : p(p
) { }
57 bool operator==(const int* p
) const { return p
== this->p
; }
60 using J
= std::common_iterator
<const int*, sentinel2
>;
61 static_assert( std::is_constructible_v
<J
, const I
&> );
62 static_assert( std::is_convertible_v
<const I
&, J
> );
64 static_assert( std::is_constructible_v
<J
, I
> );
65 static_assert( std::is_convertible_v
<I
, J
> );
67 // Constructor is noexcept(false)
68 static_assert( ! std::is_nothrow_constructible_v
<J
, I
> );
74 struct sentinel
{ int limit
; };
78 using iterator_category
= std::input_iterator_tag
;
79 using value_type
= int;
80 using difference_type
= std::ptrdiff_t;
81 using reference
= const int&;
83 const int& operator*() const { return counter
; }
85 iterator
& operator++() { ++counter
; return *this; }
87 iterator
operator++(int) { auto i
= *this; ++counter
; return i
; }
89 bool operator==(sentinel s
) const { return counter
== s
.limit
; }
94 static_assert( std::sentinel_for
<sentinel
, iterator
> );
97 std::common_iterator
<int*, const int*> obegin
= std::begin(out
);
98 std::common_iterator
<int*, const int*> oend
= std::cend(out
);
102 std::common_iterator
<iterator
, sentinel
> begin
= i
, end
= s
;
104 *obegin
++ = *begin
++;
106 VERIFY(obegin
== oend
);
108 VERIFY( i
== (&i
- out
) );
114 int arr
[2] = { 1, 2 };
115 std::common_iterator
<int*, const int*> i
= std::ranges::begin(arr
);
116 std::common_iterator
<int*, const int*> end
= std::ranges::cend(arr
);
118 VERIFY( (end
- i
) == 2 );
119 VERIFY( (i
- end
) == -2 );
122 VERIFY( (j
- i
) == 0 );
126 j
= std::ranges::next(i
);
129 VERIFY( (end
- j
) == 1 );
130 VERIFY( (j
- i
) == 1 );
131 VERIFY( (i
- j
) == -1 );
134 VERIFY( (end
- j
) == 0 );
137 VERIFY( (j
- end
) == -2 );
138 VERIFY( (j
- i
) == 0 );
140 if (std::is_constant_evaluated())
145 struct S
{ operator const int*() const { throw 1; } };
146 i
= std::common_iterator
<int*, S
>(S
{});
156 static_assert( test03() );
164 X(X
&& x
) : i(x
.i
) { x
.i
= -1; }
165 X
& operator=(X
&& x
) { i
= x
.i
; x
.i
= 0; return *this; }
170 std::common_iterator
<X
*, const X
*> i(arr
), j(arr
+1);
171 std::ranges::iter_swap(i
, j
);
172 VERIFY( arr
[0].i
== 2 );
173 VERIFY( arr
[1].i
== 1 );
175 X x
= std::ranges::iter_move(i
);
176 VERIFY( arr
[0].i
== -1 );
183 using C1
= std::common_iterator
<std::reverse_iterator
<int*>,
184 std::unreachable_sentinel_t
>;
185 using C2
= std::common_iterator
<std::reverse_iterator
<const int*>,
186 std::unreachable_sentinel_t
>;
194 static_assert( test_pr103992() );