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