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 // UNSUPPORTED: libcpp-has-no-incomplete-ranges
10*a8cf78c7SLouis Dionne
11*a8cf78c7SLouis Dionne // <span>
12*a8cf78c7SLouis Dionne
13*a8cf78c7SLouis Dionne // template<class R>
14*a8cf78c7SLouis Dionne // constexpr explicit(Extent != dynamic_extent) span(R&& r);
15*a8cf78c7SLouis Dionne
16*a8cf78c7SLouis Dionne
17*a8cf78c7SLouis Dionne #include <span>
18*a8cf78c7SLouis Dionne #include <cassert>
19*a8cf78c7SLouis Dionne #include <ranges>
20*a8cf78c7SLouis Dionne #include <string_view>
21*a8cf78c7SLouis Dionne #include <type_traits>
22*a8cf78c7SLouis Dionne #include <vector>
23*a8cf78c7SLouis Dionne
24*a8cf78c7SLouis Dionne #include "test_iterators.h"
25*a8cf78c7SLouis Dionne
26*a8cf78c7SLouis Dionne template <class T, size_t Extent>
test_from_range()27*a8cf78c7SLouis Dionne constexpr void test_from_range() {
28*a8cf78c7SLouis Dionne T val[3]{};
29*a8cf78c7SLouis Dionne std::span<T, Extent> s{val};
30*a8cf78c7SLouis Dionne assert(s.size() == std::size(val));
31*a8cf78c7SLouis Dionne assert(s.data() == std::data(val));
32*a8cf78c7SLouis Dionne }
33*a8cf78c7SLouis Dionne
34*a8cf78c7SLouis Dionne struct A {};
35*a8cf78c7SLouis Dionne
test()36*a8cf78c7SLouis Dionne constexpr bool test() {
37*a8cf78c7SLouis Dionne test_from_range<int, std::dynamic_extent>();
38*a8cf78c7SLouis Dionne test_from_range<int, 3>();
39*a8cf78c7SLouis Dionne test_from_range<A, std::dynamic_extent>();
40*a8cf78c7SLouis Dionne test_from_range<A, 3>();
41*a8cf78c7SLouis Dionne return true;
42*a8cf78c7SLouis Dionne }
43*a8cf78c7SLouis Dionne
44*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int>, std::vector<float>&>); // wrong type
45*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int, 3>, std::vector<float>&>); // wrong type
46*a8cf78c7SLouis Dionne
47*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<int>, std::vector<int>&>); // non-borrowed lvalue
48*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<int, 3>, std::vector<int>&>); // non-borrowed lvalue
49*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int>, std::vector<int>&>); // non-borrowed lvalue
50*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int, 3>, std::vector<int>&>); // non-borrowed lvalue
51*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int>, const std::vector<int>&>); // non-borrowed const lvalue
52*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int, 3>, const std::vector<int>&>); // non-borrowed const lvalue
53*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int>, const std::vector<int>&>); // non-borrowed const lvalue
54*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int, 3>, const std::vector<int>&>); // non-borrowed const lvalue
55*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int>, std::vector<int>>); // non-borrowed rvalue
56*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const int, 3>, std::vector<int>>); // non-borrowed rvalue
57*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int>, std::vector<int>&&>); // non-borrowed rvalue
58*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int, 3>, std::vector<int>&&>); // non-borrowed rvalue
59*a8cf78c7SLouis Dionne
60*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<int>, std::ranges::subrange<contiguous_iterator<int*>>>); // contiguous borrowed rvalue
61*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<int, 3>, std::ranges::subrange<contiguous_iterator<int*>>>); // contiguous borrowed rvalue
62*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int>, std::ranges::subrange<random_access_iterator<int*>>>); // non-contiguous borrowed rvalue
63*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<int, 3>, std::ranges::subrange<random_access_iterator<int*>>>); // non-contiguous borrowed rvalue
64*a8cf78c7SLouis Dionne
65*a8cf78c7SLouis Dionne using BorrowedContiguousSizedRange = std::string_view;
66*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const char>, BorrowedContiguousSizedRange>);
67*a8cf78c7SLouis Dionne static_assert(std::is_constructible_v<std::span<const char, 3>, BorrowedContiguousSizedRange>);
68*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<char>, BorrowedContiguousSizedRange>);
69*a8cf78c7SLouis Dionne static_assert(!std::is_constructible_v<std::span<char, 3>, BorrowedContiguousSizedRange>);
70*a8cf78c7SLouis Dionne
71*a8cf78c7SLouis Dionne static_assert(std::is_convertible_v<BorrowedContiguousSizedRange&, std::span<const char>>);
72*a8cf78c7SLouis Dionne static_assert(!std::is_convertible_v<BorrowedContiguousSizedRange&, std::span<const char, 3>>);
73*a8cf78c7SLouis Dionne static_assert(!std::is_convertible_v<BorrowedContiguousSizedRange&, std::span<char>>);
74*a8cf78c7SLouis Dionne static_assert(!std::is_convertible_v<BorrowedContiguousSizedRange&, std::span<char, 3>>);
75*a8cf78c7SLouis Dionne static_assert(std::is_convertible_v<const BorrowedContiguousSizedRange&, std::span<const char>>);
76*a8cf78c7SLouis Dionne static_assert(!std::is_convertible_v<const BorrowedContiguousSizedRange&, std::span<const char, 3>>);
77*a8cf78c7SLouis Dionne
main(int,char **)78*a8cf78c7SLouis Dionne int main(int, char**) {
79*a8cf78c7SLouis Dionne test();
80*a8cf78c7SLouis Dionne static_assert(test());
81*a8cf78c7SLouis Dionne
82*a8cf78c7SLouis Dionne return 0;
83*a8cf78c7SLouis Dionne }
84