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 // template<size_t N>
13*a8cf78c7SLouis Dionne //     constexpr span(element_type (&arr)[N]) noexcept;
14*a8cf78c7SLouis Dionne //
15*a8cf78c7SLouis Dionne // Remarks: These constructors shall not participate in overload resolution unless:
16*a8cf78c7SLouis Dionne //   — extent == dynamic_extent || N == extent is true, and
17*a8cf78c7SLouis Dionne //   — remove_pointer_t<decltype(data(arr))>(*)[] is convertible to ElementType(*)[].
18*a8cf78c7SLouis Dionne //
19*a8cf78c7SLouis Dionne 
20*a8cf78c7SLouis Dionne 
21*a8cf78c7SLouis Dionne #include <span>
22*a8cf78c7SLouis Dionne #include <cassert>
23*a8cf78c7SLouis Dionne #include <string>
24*a8cf78c7SLouis Dionne 
25*a8cf78c7SLouis Dionne #include "test_macros.h"
26*a8cf78c7SLouis Dionne 
27*a8cf78c7SLouis Dionne 
checkCV()28*a8cf78c7SLouis Dionne void checkCV()
29*a8cf78c7SLouis Dionne {
30*a8cf78c7SLouis Dionne                    int   arr[] = {1,2,3};
31*a8cf78c7SLouis Dionne     const          int  carr[] = {4,5,6};
32*a8cf78c7SLouis Dionne           volatile int  varr[] = {7,8,9};
33*a8cf78c7SLouis Dionne     const volatile int cvarr[] = {1,3,5};
34*a8cf78c7SLouis Dionne 
35*a8cf78c7SLouis Dionne //  Types the same (dynamic sized)
36*a8cf78c7SLouis Dionne     {
37*a8cf78c7SLouis Dionne     std::span<               int> s1{  arr};    // a span<               int> pointing at int.
38*a8cf78c7SLouis Dionne     std::span<const          int> s2{ carr};    // a span<const          int> pointing at const int.
39*a8cf78c7SLouis Dionne     std::span<      volatile int> s3{ varr};    // a span<      volatile int> pointing at volatile int.
40*a8cf78c7SLouis Dionne     std::span<const volatile int> s4{cvarr};    // a span<const volatile int> pointing at const volatile int.
41*a8cf78c7SLouis Dionne     assert(s1.size() + s2.size() + s3.size() + s4.size() == 12);
42*a8cf78c7SLouis Dionne     }
43*a8cf78c7SLouis Dionne 
44*a8cf78c7SLouis Dionne //  Types the same (static sized)
45*a8cf78c7SLouis Dionne     {
46*a8cf78c7SLouis Dionne     std::span<               int,3> s1{  arr};  // a span<               int> pointing at int.
47*a8cf78c7SLouis Dionne     std::span<const          int,3> s2{ carr};  // a span<const          int> pointing at const int.
48*a8cf78c7SLouis Dionne     std::span<      volatile int,3> s3{ varr};  // a span<      volatile int> pointing at volatile int.
49*a8cf78c7SLouis Dionne     std::span<const volatile int,3> s4{cvarr};  // a span<const volatile int> pointing at const volatile int.
50*a8cf78c7SLouis Dionne     assert(s1.size() + s2.size() + s3.size() + s4.size() == 12);
51*a8cf78c7SLouis Dionne     }
52*a8cf78c7SLouis Dionne 
53*a8cf78c7SLouis Dionne 
54*a8cf78c7SLouis Dionne //  types different (dynamic sized)
55*a8cf78c7SLouis Dionne     {
56*a8cf78c7SLouis Dionne     std::span<const          int> s1{ arr};     // a span<const          int> pointing at int.
57*a8cf78c7SLouis Dionne     std::span<      volatile int> s2{ arr};     // a span<      volatile int> pointing at int.
58*a8cf78c7SLouis Dionne     std::span<      volatile int> s3{ arr};     // a span<      volatile int> pointing at const int.
59*a8cf78c7SLouis Dionne     std::span<const volatile int> s4{ arr};     // a span<const volatile int> pointing at int.
60*a8cf78c7SLouis Dionne     std::span<const volatile int> s5{carr};     // a span<const volatile int> pointing at const int.
61*a8cf78c7SLouis Dionne     std::span<const volatile int> s6{varr};     // a span<const volatile int> pointing at volatile int.
62*a8cf78c7SLouis Dionne     assert(s1.size() + s2.size() + s3.size() + s4.size() + s5.size() + s6.size() == 18);
63*a8cf78c7SLouis Dionne     }
64*a8cf78c7SLouis Dionne 
65*a8cf78c7SLouis Dionne //  types different (static sized)
66*a8cf78c7SLouis Dionne     {
67*a8cf78c7SLouis Dionne     std::span<const          int,3> s1{ arr};   // a span<const          int> pointing at int.
68*a8cf78c7SLouis Dionne     std::span<      volatile int,3> s2{ arr};   // a span<      volatile int> pointing at int.
69*a8cf78c7SLouis Dionne     std::span<      volatile int,3> s3{ arr};   // a span<      volatile int> pointing at const int.
70*a8cf78c7SLouis Dionne     std::span<const volatile int,3> s4{ arr};   // a span<const volatile int> pointing at int.
71*a8cf78c7SLouis Dionne     std::span<const volatile int,3> s5{carr};   // a span<const volatile int> pointing at const int.
72*a8cf78c7SLouis Dionne     std::span<const volatile int,3> s6{varr};   // a span<const volatile int> pointing at volatile int.
73*a8cf78c7SLouis Dionne     assert(s1.size() + s2.size() + s3.size() + s4.size() + s5.size() + s6.size() == 18);
74*a8cf78c7SLouis Dionne     }
75*a8cf78c7SLouis Dionne }
76*a8cf78c7SLouis Dionne 
77*a8cf78c7SLouis Dionne template<class T>
testSpan()78*a8cf78c7SLouis Dionne constexpr bool testSpan()
79*a8cf78c7SLouis Dionne {
80*a8cf78c7SLouis Dionne     T val[2] = {};
81*a8cf78c7SLouis Dionne 
82*a8cf78c7SLouis Dionne     ASSERT_NOEXCEPT(std::span<T>{val});
83*a8cf78c7SLouis Dionne     ASSERT_NOEXCEPT(std::span<T, 2>{val});
84*a8cf78c7SLouis Dionne     ASSERT_NOEXCEPT(std::span<const T>{val});
85*a8cf78c7SLouis Dionne     ASSERT_NOEXCEPT(std::span<const T, 2>{val});
86*a8cf78c7SLouis Dionne 
87*a8cf78c7SLouis Dionne     std::span<T> s1 = val;
88*a8cf78c7SLouis Dionne     std::span<T, 2> s2 = val;
89*a8cf78c7SLouis Dionne     std::span<const T> s3 = val;
90*a8cf78c7SLouis Dionne     std::span<const T, 2> s4 = val;
91*a8cf78c7SLouis Dionne     assert(s1.data() == val && s1.size() == 2);
92*a8cf78c7SLouis Dionne     assert(s2.data() == val && s2.size() == 2);
93*a8cf78c7SLouis Dionne     assert(s3.data() == val && s3.size() == 2);
94*a8cf78c7SLouis Dionne     assert(s4.data() == val && s4.size() == 2);
95*a8cf78c7SLouis Dionne 
96*a8cf78c7SLouis Dionne     std::span<const int> s5 = {{1,2}};
97*a8cf78c7SLouis Dionne     std::span<const int, 2> s6 = {{1,2}};
98*a8cf78c7SLouis Dionne     assert(s5.size() == 2);  // and it dangles
99*a8cf78c7SLouis Dionne     assert(s6.size() == 2);  // and it dangles
100*a8cf78c7SLouis Dionne 
101*a8cf78c7SLouis Dionne     return true;
102*a8cf78c7SLouis Dionne }
103*a8cf78c7SLouis Dionne 
104*a8cf78c7SLouis Dionne 
105*a8cf78c7SLouis Dionne struct A {};
106*a8cf78c7SLouis Dionne 
main(int,char **)107*a8cf78c7SLouis Dionne int main(int, char**)
108*a8cf78c7SLouis Dionne {
109*a8cf78c7SLouis Dionne     testSpan<int>();
110*a8cf78c7SLouis Dionne     testSpan<double>();
111*a8cf78c7SLouis Dionne     testSpan<A>();
112*a8cf78c7SLouis Dionne     testSpan<std::string>();
113*a8cf78c7SLouis Dionne 
114*a8cf78c7SLouis Dionne     static_assert(testSpan<int>());
115*a8cf78c7SLouis Dionne     static_assert(testSpan<double>());
116*a8cf78c7SLouis Dionne     static_assert(testSpan<A>());
117*a8cf78c7SLouis Dionne 
118*a8cf78c7SLouis Dionne     checkCV();
119*a8cf78c7SLouis Dionne 
120*a8cf78c7SLouis Dionne     return 0;
121*a8cf78c7SLouis Dionne }
122