1 //===----------------------------------------------------------------------===//
2 //
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
6 //
7 //===----------------------------------------------------------------------===//
8 
9 // UNSUPPORTED: c++03, c++11, c++14, c++17
10 // UNSUPPORTED: libcpp-no-concepts
11 
12 // constexpr const I& base() const &;
13 // constexpr I base() &&;
14 
15 #include <iterator>
16 
17 #include "test_macros.h"
18 #include "test_iterators.h"
19 
20 struct InputOrOutputArchetype {
21   using difference_type = int;
22 
23   int *ptr;
24 
25   int operator*() { return *ptr; }
26   void operator++(int) { ++ptr; }
27   InputOrOutputArchetype& operator++() { ++ptr; return *this; }
28 };
29 
30 constexpr bool test() {
31   int buffer[8] = {1, 2, 3, 4, 5, 6, 7, 8};
32 
33   {
34     std::counted_iterator iter(cpp20_input_iterator<int*>{buffer}, 8);
35     assert(base(iter.base()) == buffer);
36     assert(base(std::move(iter).base()) == buffer);
37 
38     ASSERT_NOEXCEPT(iter.base());
39     ASSERT_SAME_TYPE(decltype(iter.base()), const cpp20_input_iterator<int*>&);
40     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), cpp20_input_iterator<int*>);
41   }
42 
43   {
44     std::counted_iterator iter(forward_iterator<int*>{buffer}, 8);
45     assert(base(iter.base()) == buffer);
46     assert(base(std::move(iter).base()) == buffer);
47 
48     ASSERT_SAME_TYPE(decltype(iter.base()), const forward_iterator<int*>&);
49     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), forward_iterator<int*>);
50   }
51 
52   {
53     std::counted_iterator iter(contiguous_iterator<int*>{buffer}, 8);
54     assert(base(iter.base()) == buffer);
55     assert(base(std::move(iter).base()) == buffer);
56 
57     ASSERT_SAME_TYPE(decltype(iter.base()), const contiguous_iterator<int*>&);
58     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), contiguous_iterator<int*>);
59   }
60 
61   {
62     std::counted_iterator iter(InputOrOutputArchetype{buffer}, 6);
63     assert(iter.base().ptr == buffer);
64     assert(std::move(iter).base().ptr == buffer);
65 
66     ASSERT_SAME_TYPE(decltype(iter.base()), const InputOrOutputArchetype&);
67     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), InputOrOutputArchetype);
68   }
69 
70   {
71     const std::counted_iterator iter(cpp20_input_iterator<int*>{buffer}, 8);
72     assert(base(iter.base()) == buffer);
73     assert(base(std::move(iter).base()) == buffer);
74 
75     ASSERT_SAME_TYPE(decltype(iter.base()), const cpp20_input_iterator<int*>&);
76     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), const cpp20_input_iterator<int*>&);
77   }
78 
79   {
80     const std::counted_iterator iter(forward_iterator<int*>{buffer}, 7);
81     assert(base(iter.base()) == buffer);
82     assert(base(std::move(iter).base()) == buffer);
83 
84     ASSERT_SAME_TYPE(decltype(iter.base()), const forward_iterator<int*>&);
85     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), const forward_iterator<int*>&);
86   }
87 
88   {
89     const std::counted_iterator iter(contiguous_iterator<int*>{buffer}, 6);
90     assert(base(iter.base()) == buffer);
91     assert(base(std::move(iter).base()) == buffer);
92 
93     ASSERT_SAME_TYPE(decltype(iter.base()), const contiguous_iterator<int*>&);
94     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), const contiguous_iterator<int*>&);
95   }
96 
97   {
98     const std::counted_iterator iter(InputOrOutputArchetype{buffer}, 6);
99     assert(iter.base().ptr == buffer);
100     assert(std::move(iter).base().ptr == buffer);
101 
102     ASSERT_SAME_TYPE(decltype(iter.base()), const InputOrOutputArchetype&);
103     ASSERT_SAME_TYPE(decltype(std::move(iter).base()), const InputOrOutputArchetype&);
104   }
105 
106   return true;
107 }
108 
109 int main(int, char**) {
110   test();
111   static_assert(test());
112 
113   return 0;
114 }
115