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 iterator end() const noexcept;
13a8cf78c7SLouis Dionne // constexpr const_iterator cend() 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::iterator e = s.end();
26a8cf78c7SLouis Dionne if (s.empty())
27a8cf78c7SLouis Dionne {
28a8cf78c7SLouis Dionne ret = ret && (e == s.begin());
29a8cf78c7SLouis Dionne }
30a8cf78c7SLouis Dionne else
31a8cf78c7SLouis Dionne {
32a8cf78c7SLouis Dionne typename Span::const_pointer last = &*(s.begin() + s.size() - 1);
33a8cf78c7SLouis Dionne ret = ret && (e != s.begin());
34a8cf78c7SLouis Dionne ret = ret && (&*( e-1) == last);
35a8cf78c7SLouis Dionne }
36a8cf78c7SLouis Dionne
37*fb855eb9SMark de Wever ret = ret && (static_cast<std::size_t>(e - s.begin()) == s.size());
38a8cf78c7SLouis Dionne return ret;
39a8cf78c7SLouis Dionne }
40a8cf78c7SLouis Dionne
41a8cf78c7SLouis Dionne template <class Span>
testRuntimeSpan(Span s)42a8cf78c7SLouis Dionne void testRuntimeSpan(Span s)
43a8cf78c7SLouis Dionne {
44a8cf78c7SLouis Dionne typename Span::iterator e = s.end();
45a8cf78c7SLouis Dionne if (s.empty())
46a8cf78c7SLouis Dionne {
47a8cf78c7SLouis Dionne assert(e == s.begin());
48a8cf78c7SLouis Dionne }
49a8cf78c7SLouis Dionne else
50a8cf78c7SLouis Dionne {
51a8cf78c7SLouis Dionne typename Span::const_pointer last = &*(s.begin() + s.size() - 1);
52a8cf78c7SLouis Dionne assert(e != s.begin());
53a8cf78c7SLouis Dionne assert(&*( e-1) == last);
54a8cf78c7SLouis Dionne }
55a8cf78c7SLouis Dionne
56*fb855eb9SMark de Wever assert(static_cast<std::size_t>(e - s.begin()) == s.size());
57a8cf78c7SLouis Dionne }
58a8cf78c7SLouis Dionne
59a8cf78c7SLouis Dionne
60a8cf78c7SLouis Dionne struct A{};
operator ==(A,A)61a8cf78c7SLouis Dionne bool operator==(A, A) {return true;}
62a8cf78c7SLouis Dionne
63a8cf78c7SLouis Dionne constexpr int iArr1[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
64a8cf78c7SLouis Dionne int iArr2[] = {10, 11, 12, 13, 14, 15, 16, 17, 18, 19};
65a8cf78c7SLouis Dionne
66a8cf78c7SLouis Dionne
main(int,char **)67a8cf78c7SLouis Dionne int main(int, char**)
68a8cf78c7SLouis Dionne {
69a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<int>()), "");
70a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<long>()), "");
71a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<double>()), "");
72a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<A>()), "");
73a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<std::string>()), "");
74a8cf78c7SLouis Dionne
75a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<int, 0>()), "");
76a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<long, 0>()), "");
77a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<double, 0>()), "");
78a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<A, 0>()), "");
79a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<std::string, 0>()), "");
80a8cf78c7SLouis Dionne
81a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 1)), "");
82a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 2)), "");
83a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 3)), "");
84a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 4)), "");
85a8cf78c7SLouis Dionne static_assert(testConstexprSpan(std::span<const int>(iArr1, 5)), "");
86a8cf78c7SLouis Dionne
87a8cf78c7SLouis Dionne
88a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int> ());
89a8cf78c7SLouis Dionne testRuntimeSpan(std::span<long> ());
90a8cf78c7SLouis Dionne testRuntimeSpan(std::span<double> ());
91a8cf78c7SLouis Dionne testRuntimeSpan(std::span<A> ());
92a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>());
93a8cf78c7SLouis Dionne
94a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int, 0> ());
95a8cf78c7SLouis Dionne testRuntimeSpan(std::span<long, 0> ());
96a8cf78c7SLouis Dionne testRuntimeSpan(std::span<double, 0> ());
97a8cf78c7SLouis Dionne testRuntimeSpan(std::span<A, 0> ());
98a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string, 0>());
99a8cf78c7SLouis Dionne
100a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 1));
101a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 2));
102a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 3));
103a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 4));
104a8cf78c7SLouis Dionne testRuntimeSpan(std::span<int>(iArr2, 5));
105a8cf78c7SLouis Dionne
106a8cf78c7SLouis Dionne std::string s;
107a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>(&s, (std::size_t) 0));
108a8cf78c7SLouis Dionne testRuntimeSpan(std::span<std::string>(&s, 1));
109a8cf78c7SLouis Dionne
110a8cf78c7SLouis Dionne return 0;
111a8cf78c7SLouis Dionne }
112