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<class OtherElementType, size_t OtherExtent>
13*a8cf78c7SLouis Dionne //    constexpr span(const span<OtherElementType, OtherExtent>& s) noexcept;
14*a8cf78c7SLouis Dionne //
15*a8cf78c7SLouis Dionne //  Remarks: This constructor shall not participate in overload resolution unless:
16*a8cf78c7SLouis Dionne //      Extent == dynamic_extent || Extent == OtherExtent is true, and
17*a8cf78c7SLouis Dionne //      OtherElementType(*)[] is convertible to ElementType(*)[].
18*a8cf78c7SLouis Dionne 
19*a8cf78c7SLouis Dionne 
20*a8cf78c7SLouis Dionne #include <span>
21*a8cf78c7SLouis Dionne #include <cassert>
22*a8cf78c7SLouis Dionne #include <string>
23*a8cf78c7SLouis Dionne 
24*a8cf78c7SLouis Dionne #include "test_macros.h"
25*a8cf78c7SLouis Dionne 
26*a8cf78c7SLouis Dionne template<class T, size_t extent, size_t otherExtent>
createImplicitSpan(std::span<T,otherExtent> s)27*a8cf78c7SLouis Dionne std::span<T, extent> createImplicitSpan(std::span<T, otherExtent> s) {
28*a8cf78c7SLouis Dionne     return {s}; // expected-error {{chosen constructor is explicit in copy-initialization}}
29*a8cf78c7SLouis Dionne }
30*a8cf78c7SLouis Dionne 
checkCV()31*a8cf78c7SLouis Dionne void checkCV ()
32*a8cf78c7SLouis Dionne {
33*a8cf78c7SLouis Dionne //  std::span<               int>   sp;
34*a8cf78c7SLouis Dionne     std::span<const          int>  csp;
35*a8cf78c7SLouis Dionne     std::span<      volatile int>  vsp;
36*a8cf78c7SLouis Dionne     std::span<const volatile int> cvsp;
37*a8cf78c7SLouis Dionne 
38*a8cf78c7SLouis Dionne //  std::span<               int, 0>   sp0;
39*a8cf78c7SLouis Dionne     std::span<const          int, 0>  csp0;
40*a8cf78c7SLouis Dionne     std::span<      volatile int, 0>  vsp0;
41*a8cf78c7SLouis Dionne     std::span<const volatile int, 0> cvsp0;
42*a8cf78c7SLouis Dionne 
43*a8cf78c7SLouis Dionne //  Try to remove const and/or volatile (dynamic -> dynamic)
44*a8cf78c7SLouis Dionne     {
45*a8cf78c7SLouis Dionne     std::span<               int> s1{ csp}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
46*a8cf78c7SLouis Dionne     std::span<               int> s2{ vsp}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
47*a8cf78c7SLouis Dionne     std::span<               int> s3{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
48*a8cf78c7SLouis Dionne 
49*a8cf78c7SLouis Dionne     std::span<const          int> s4{ vsp}; // expected-error {{no matching constructor for initialization of 'std::span<const int>'}}
50*a8cf78c7SLouis Dionne     std::span<const          int> s5{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<const int>'}}
51*a8cf78c7SLouis Dionne 
52*a8cf78c7SLouis Dionne     std::span<      volatile int> s6{ csp}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int>'}}
53*a8cf78c7SLouis Dionne     std::span<      volatile int> s7{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int>'}}
54*a8cf78c7SLouis Dionne     }
55*a8cf78c7SLouis Dionne 
56*a8cf78c7SLouis Dionne //  Try to remove const and/or volatile (static -> static)
57*a8cf78c7SLouis Dionne     {
58*a8cf78c7SLouis Dionne     std::span<               int, 0> s1{ csp0}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
59*a8cf78c7SLouis Dionne     std::span<               int, 0> s2{ vsp0}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
60*a8cf78c7SLouis Dionne     std::span<               int, 0> s3{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
61*a8cf78c7SLouis Dionne 
62*a8cf78c7SLouis Dionne     std::span<const          int, 0> s4{ vsp0}; // expected-error {{no matching constructor for initialization of 'std::span<const int, 0>'}}
63*a8cf78c7SLouis Dionne     std::span<const          int, 0> s5{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<const int, 0>'}}
64*a8cf78c7SLouis Dionne 
65*a8cf78c7SLouis Dionne     std::span<      volatile int, 0> s6{ csp0}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int, 0>'}}
66*a8cf78c7SLouis Dionne     std::span<      volatile int, 0> s7{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int, 0>'}}
67*a8cf78c7SLouis Dionne     }
68*a8cf78c7SLouis Dionne 
69*a8cf78c7SLouis Dionne //  Try to remove const and/or volatile (static -> dynamic)
70*a8cf78c7SLouis Dionne     {
71*a8cf78c7SLouis Dionne     std::span<               int> s1{ csp0}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
72*a8cf78c7SLouis Dionne     std::span<               int> s2{ vsp0}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
73*a8cf78c7SLouis Dionne     std::span<               int> s3{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<int>'}}
74*a8cf78c7SLouis Dionne 
75*a8cf78c7SLouis Dionne     std::span<const          int> s4{ vsp0}; // expected-error {{no matching constructor for initialization of 'std::span<const int>'}}
76*a8cf78c7SLouis Dionne     std::span<const          int> s5{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<const int>'}}
77*a8cf78c7SLouis Dionne 
78*a8cf78c7SLouis Dionne     std::span<      volatile int> s6{ csp0}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int>'}}
79*a8cf78c7SLouis Dionne     std::span<      volatile int> s7{cvsp0}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int>'}}
80*a8cf78c7SLouis Dionne     }
81*a8cf78c7SLouis Dionne 
82*a8cf78c7SLouis Dionne //  Try to remove const and/or volatile (static -> static)
83*a8cf78c7SLouis Dionne     {
84*a8cf78c7SLouis Dionne     std::span<               int, 0> s1{ csp}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
85*a8cf78c7SLouis Dionne     std::span<               int, 0> s2{ vsp}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
86*a8cf78c7SLouis Dionne     std::span<               int, 0> s3{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<int, 0>'}}
87*a8cf78c7SLouis Dionne 
88*a8cf78c7SLouis Dionne     std::span<const          int, 0> s4{ vsp}; // expected-error {{no matching constructor for initialization of 'std::span<const int, 0>'}}
89*a8cf78c7SLouis Dionne     std::span<const          int, 0> s5{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<const int, 0>'}}
90*a8cf78c7SLouis Dionne 
91*a8cf78c7SLouis Dionne     std::span<      volatile int, 0> s6{ csp}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int, 0>'}}
92*a8cf78c7SLouis Dionne     std::span<      volatile int, 0> s7{cvsp}; // expected-error {{no matching constructor for initialization of 'std::span<volatile int, 0>'}}
93*a8cf78c7SLouis Dionne     }
94*a8cf78c7SLouis Dionne }
95*a8cf78c7SLouis Dionne 
main(int,char **)96*a8cf78c7SLouis Dionne int main(int, char**)
97*a8cf78c7SLouis Dionne {
98*a8cf78c7SLouis Dionne     std::span<int>      sp;
99*a8cf78c7SLouis Dionne     std::span<int, 0>   sp0;
100*a8cf78c7SLouis Dionne 
101*a8cf78c7SLouis Dionne     std::span<float> s1{sp};    // expected-error {{no matching constructor for initialization of 'std::span<float>'}}
102*a8cf78c7SLouis Dionne     std::span<float> s2{sp0};   // expected-error {{no matching constructor for initialization of 'std::span<float>'}}
103*a8cf78c7SLouis Dionne     std::span<float, 0> s3{sp}; // expected-error {{no matching constructor for initialization of 'std::span<float, 0>'}}
104*a8cf78c7SLouis Dionne     std::span<float, 0> s4{sp0};    // expected-error {{no matching constructor for initialization of 'std::span<float, 0>'}}
105*a8cf78c7SLouis Dionne 
106*a8cf78c7SLouis Dionne     checkCV();
107*a8cf78c7SLouis Dionne 
108*a8cf78c7SLouis Dionne     // explicit constructor necessary
109*a8cf78c7SLouis Dionne     {
110*a8cf78c7SLouis Dionne     createImplicitSpan<int, 1>(sp);
111*a8cf78c7SLouis Dionne     }
112*a8cf78c7SLouis Dionne 
113*a8cf78c7SLouis Dionne   return 0;
114*a8cf78c7SLouis Dionne }
115