]> git.ipfire.org Git - thirdparty/gcc.git/blob - libstdc++-v3/testsuite/24_iterators/associated_types/incrementable.traits.cc
Update copyright years.
[thirdparty/gcc.git] / libstdc++-v3 / testsuite / 24_iterators / associated_types / incrementable.traits.cc
1 // Copyright (C) 2019-2024 Free Software Foundation, Inc.
2 //
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)
7 // any later version.
8
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.
13
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/>.
17
18 // { dg-do compile { target c++20 } }
19
20 #include <iterator>
21
22 struct none;
23
24 template<typename T>
25 concept has_inc_traits_type
26 = requires { typename std::incrementable_traits<T>::difference_type; };
27
28 // Check std::incrementable_traits<T>::difference_type is U (or doesn't exist).
29 template<typename T, typename U>
30 concept check_inc_traits = (has_inc_traits_type<T> != std::same_as<U, none>);
31
32 static_assert( check_inc_traits<void, none> );
33 static_assert( check_inc_traits<const void, none> );
34 static_assert( check_inc_traits<void*, none> );
35 static_assert( check_inc_traits<const void*, none> );
36
37 static_assert( check_inc_traits<int, int> );
38 static_assert( check_inc_traits<const int, int> );
39
40 static_assert( check_inc_traits<int*, std::ptrdiff_t> );
41 static_assert( check_inc_traits<const int*, std::ptrdiff_t> );
42 static_assert( check_inc_traits<int[2], std::ptrdiff_t> );
43 static_assert( check_inc_traits<const int[2], std::ptrdiff_t> );
44
45 struct A { using difference_type = int; };
46 static_assert( check_inc_traits<A, int> );
47 static_assert( check_inc_traits<const A, int> );
48 struct B : private A { };
49 static_assert( check_inc_traits<B, none> );
50
51 struct C { };
52 short operator-(C, C) { return 0; }
53 static_assert( check_inc_traits<C, short> );
54 static_assert( check_inc_traits<const C, short> );
55
56 struct D { };
57 unsigned short operator-(D, D) { return 0; }
58 static_assert( check_inc_traits<D, short> );
59 static_assert( check_inc_traits<const D, short> );
60
61 struct E { };
62 template<>
63 struct std::incrementable_traits<E> { using difference_type = long; };
64 static_assert( check_inc_traits<E, long> );
65 static_assert( check_inc_traits<const E, long> );
66
67 template<typename T>
68 concept has_alias = requires { typename std::iter_difference_t<T>; };
69
70 // Check std::iter_difference_t<T> is U (or doesn't exist).
71 template<typename T, typename U>
72 concept check_alias = (has_alias<T> != std::same_as<U, none>);
73
74 static_assert( check_alias<void, none> );
75 static_assert( check_alias<const void, none> );
76 static_assert( check_alias<void*, none> );
77 static_assert( check_alias<const void*, none> );
78
79 static_assert( check_alias<int, int> );
80 static_assert( check_alias<const int, int> );
81 static_assert( check_alias<int*, std::ptrdiff_t> );
82 static_assert( check_alias<const int*, std::ptrdiff_t> );
83 static_assert( check_alias<int[2], std::ptrdiff_t> );
84 static_assert( check_alias<const int[2], std::ptrdiff_t> );
85
86 static_assert( check_alias<A, int> );
87 static_assert( check_alias<const A, int> );
88 static_assert( check_alias<B, none> );
89 static_assert( check_alias<C, short> );
90 static_assert( check_alias<const C, short> );
91 static_assert( check_alias<D, short> );
92 static_assert( check_alias<const D, short> );
93 static_assert( check_alias<E, long> );
94 static_assert( check_alias<const E, long> );
95
96 struct F { };
97 template<>
98 struct std::iterator_traits<F> { using difference_type = F; };
99 // iterator_traits<F> is specialized, so use its difference_type.
100 static_assert( check_alias<F, std::iterator_traits<F>::difference_type> );
101
102 struct G { };
103 template<>
104 struct std::incrementable_traits<G> { using difference_type = G; };
105 template<>
106 struct std::iterator_traits<G> { using difference_type = int; };
107 // iterator_traits<G> is specialized, so use its difference_type.
108 static_assert( check_alias<G, std::iterator_traits<G>::difference_type> );
109
110 struct H { };
111 template<>
112 struct std::incrementable_traits<H> { using difference_type = H; };
113 template<>
114 struct std::iterator_traits<H>
115 {
116 using iterator_category = input_iterator_tag;
117 using difference_type = int;
118 using value_type = char;
119 using reference = value_type&;
120 };
121 // iterator_traits<H> is specialized, so use its difference_type.
122 static_assert( check_alias<H, std::iterator_traits<H>::difference_type> );
123
124 struct I
125 {
126 using difference_type = I;
127 };
128 // iterator_traits<I> is not specialized, and no standard specialization
129 // matches, so use incrementable_traits.
130 static_assert( check_alias<I, std::incrementable_traits<I>::difference_type> );
131
132 struct J
133 {
134 using iterator_category = std::input_iterator_tag;
135 using difference_type = int;
136 using value_type = char;
137 using reference = value_type&;
138 };
139 // iterator_traits<J> matches constrained specialization in the library,
140 // so use its difference_type.
141 static_assert( check_alias<J, int> );