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