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