]> git.ipfire.org Git - thirdparty/gcc.git/blame - 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
CommitLineData
a945c346 1// Copyright (C) 2019-2024 Free Software Foundation, Inc.
6d0dff49
JW
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
762baaf0 18// { dg-do compile { target c++20 } }
6d0dff49
JW
19
20#include <iterator>
21
22struct none;
23
24template<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).
29template<typename T, typename U>
30 concept check_inc_traits = (has_inc_traits_type<T> != std::same_as<U, none>);
31
32static_assert( check_inc_traits<void, none> );
33static_assert( check_inc_traits<const void, none> );
34static_assert( check_inc_traits<void*, none> );
35static_assert( check_inc_traits<const void*, none> );
36
37static_assert( check_inc_traits<int, int> );
38static_assert( check_inc_traits<const int, int> );
39
40static_assert( check_inc_traits<int*, std::ptrdiff_t> );
41static_assert( check_inc_traits<const int*, std::ptrdiff_t> );
42static_assert( check_inc_traits<int[2], std::ptrdiff_t> );
43static_assert( check_inc_traits<const int[2], std::ptrdiff_t> );
44
45struct A { using difference_type = int; };
46static_assert( check_inc_traits<A, int> );
47static_assert( check_inc_traits<const A, int> );
48struct B : private A { };
49static_assert( check_inc_traits<B, none> );
50
51struct C { };
52short operator-(C, C) { return 0; }
53static_assert( check_inc_traits<C, short> );
54static_assert( check_inc_traits<const C, short> );
55
56struct D { };
57unsigned short operator-(D, D) { return 0; }
58static_assert( check_inc_traits<D, short> );
59static_assert( check_inc_traits<const D, short> );
60
61struct E { };
62template<>
63 struct std::incrementable_traits<E> { using difference_type = long; };
64static_assert( check_inc_traits<E, long> );
65static_assert( check_inc_traits<const E, long> );
66
67template<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).
71template<typename T, typename U>
72 concept check_alias = (has_alias<T> != std::same_as<U, none>);
73
74static_assert( check_alias<void, none> );
75static_assert( check_alias<const void, none> );
76static_assert( check_alias<void*, none> );
77static_assert( check_alias<const void*, none> );
78
79static_assert( check_alias<int, int> );
80static_assert( check_alias<const int, int> );
81static_assert( check_alias<int*, std::ptrdiff_t> );
82static_assert( check_alias<const int*, std::ptrdiff_t> );
83static_assert( check_alias<int[2], std::ptrdiff_t> );
84static_assert( check_alias<const int[2], std::ptrdiff_t> );
85
86static_assert( check_alias<A, int> );
87static_assert( check_alias<const A, int> );
88static_assert( check_alias<B, none> );
89static_assert( check_alias<C, short> );
90static_assert( check_alias<const C, short> );
91static_assert( check_alias<D, short> );
92static_assert( check_alias<const D, short> );
93static_assert( check_alias<E, long> );
94static_assert( check_alias<const E, long> );
95
96struct F { };
97template<>
98 struct std::iterator_traits<F> { using difference_type = F; };
99// iterator_traits<F> is specialized, so use its difference_type.
100static_assert( check_alias<F, std::iterator_traits<F>::difference_type> );
101
102struct G { };
103template<>
104 struct std::incrementable_traits<G> { using difference_type = G; };
105template<>
106 struct std::iterator_traits<G> { using difference_type = int; };
107// iterator_traits<G> is specialized, so use its difference_type.
108static_assert( check_alias<G, std::iterator_traits<G>::difference_type> );
109
110struct H { };
111template<>
112 struct std::incrementable_traits<H> { using difference_type = H; };
113template<>
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.
122static_assert( check_alias<H, std::iterator_traits<H>::difference_type> );
123
124struct 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.
130static_assert( check_alias<I, std::incrementable_traits<I>::difference_type> );
131
132struct 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.
141static_assert( check_alias<J, int> );