1*a8cf78c7SLouis Dionne //===----------------------------------------------------------------------===//
2*a8cf78c7SLouis Dionne //
3*a8cf78c7SLouis Dionne // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*a8cf78c7SLouis Dionne // See https://llvm.org/LICENSE.txt for license information.
5*a8cf78c7SLouis Dionne // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*a8cf78c7SLouis Dionne //
7*a8cf78c7SLouis Dionne //===----------------------------------------------------------------------===//
8*a8cf78c7SLouis Dionne // UNSUPPORTED: c++03, c++11, c++14, c++17
9*a8cf78c7SLouis Dionne
10*a8cf78c7SLouis Dionne // <span>
11*a8cf78c7SLouis Dionne
12*a8cf78c7SLouis Dionne // constexpr reverse_iterator rbegin() const noexcept;
13*a8cf78c7SLouis Dionne // constexpr const_reverse_iterator crbegin() const noexcept;
14*a8cf78c7SLouis Dionne
15*a8cf78c7SLouis Dionne #include <span>
16*a8cf78c7SLouis Dionne #include <cassert>
17*a8cf78c7SLouis Dionne #include <string>
18*a8cf78c7SLouis Dionne
19*a8cf78c7SLouis Dionne #include "test_macros.h"
20*a8cf78c7SLouis Dionne
21*a8cf78c7SLouis Dionne template <class Span>
testConstexprSpan(Span s)22*a8cf78c7SLouis Dionne constexpr bool testConstexprSpan(Span s)
23*a8cf78c7SLouis Dionne {
24*a8cf78c7SLouis Dionne bool ret = true;
25*a8cf78c7SLouis Dionne typename Span::reverse_iterator b = s.rbegin();
26*a8cf78c7SLouis Dionne if (s.empty())
27*a8cf78c7SLouis Dionne {
28*a8cf78c7SLouis Dionne ret = ret && ( b == s.rend());
29*a8cf78c7SLouis Dionne }
30*a8cf78c7SLouis Dionne else
31*a8cf78c7SLouis Dionne {
32*a8cf78c7SLouis Dionne const typename Span::size_type last = s.size() - 1;
33*a8cf78c7SLouis Dionne ret = ret && ( *b == s[last]);
34*a8cf78c7SLouis Dionne ret = ret && (&*b == &s[last]);
35*a8cf78c7SLouis Dionne }
36*a8cf78c7SLouis Dionne return ret;
37*a8cf78c7SLouis Dionne }
38*a8cf78c7SLouis Dionne
39*a8cf78c7SLouis Dionne
40*a8cf78c7SLouis Dionne template <class Span>
testRuntimeSpan(Span s)41*a8cf78c7SLouis Dionne void testRuntimeSpan(Span s)
42*a8cf78c7SLouis Dionne {
43*a8cf78c7SLouis Dionne typename Span::reverse_iterator b = s.rbegin();
44*a8cf78c7SLouis Dionne if (s.empty())
45*a8cf78c7SLouis Dionne {
46*a8cf78c7SLouis Dionne assert(b == s.rend());
47*a8cf78c7SLouis Dionne }
48*a8cf78c7SLouis Dionne else
49*a8cf78c7SLouis Dionne {
50*a8cf78c7SLouis Dionne const typename Span::size_type last = s.size() - 1;
51*a8cf78c7SLouis Dionne assert( *b == s[last]);
52*a8cf78c7SLouis Dionne assert(&*b == &s[last]);
53*a8cf78c7SLouis Dionne }
54*a8cf78c7SLouis Dionne }
55*a8cf78c7SLouis Dionne
56*a8cf78c7SLouis Dionne
57*a8cf78c7SLouis Dionne struct A{};
operator ==(A,A)58*a8cf78c7SLouis Dionne bool operator==(A, A) {return true;}
59*a8cf78c7SLouis Dionne
60*a8cf78c7SLouis Dionne constexpr int iArr1[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
61*a8cf78c7SLouis Dionne int iArr2[] = {10, 11, 12, 13, 14, 15, 16, 17, 18, 19};
62*a8cf78c7SLouis Dionne
63*a8cf78c7SLouis Dionne
main(int,char **)64*a8cf78c7SLouis Dionne int main(int, char**)
65*a8cf78c7SLouis Dionne {
66*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<int>()), "");
67*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<long>()), "");
68*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<double>()), "");
69*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<A>()), "");
70*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<std::string>()), "");
71*a8cf78c7SLouis Dionne
72*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<int, 0>()), "");
73*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<long, 0>()), "");
74*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<double, 0>()), "");
75*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<A, 0>()), "");
76*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<std::string, 0>()), "");
77*a8cf78c7SLouis Dionne
78*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 1)), "");
79*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 2)), "");
80*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 3)), "");
81*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 4)), "");
82*a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 5)), "");
83*a8cf78c7SLouis Dionne
84*a8cf78c7SLouis Dionne
85*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int> ());
86*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<long> ());
87*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<double> ());
88*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<A> ());
89*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>());
90*a8cf78c7SLouis Dionne
91*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int, 0> ());
92*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<long, 0> ());
93*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<double, 0> ());
94*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<A, 0> ());
95*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string, 0>());
96*a8cf78c7SLouis Dionne
97*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 1));
98*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 2));
99*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 3));
100*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 4));
101*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 5));
102*a8cf78c7SLouis Dionne
103*a8cf78c7SLouis Dionne std::string s;
104*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>(&s, static_cast<std::size_t>(0)));
105*a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>(&s, 1));
106*a8cf78c7SLouis Dionne
107*a8cf78c7SLouis Dionne return 0;
108*a8cf78c7SLouis Dionne }
109