1*042dc3c4SHui Xie //===----------------------------------------------------------------------===//
2*042dc3c4SHui Xie //
3*042dc3c4SHui Xie // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*042dc3c4SHui Xie // See https://llvm.org/LICENSE.txt for license information.
5*042dc3c4SHui Xie // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*042dc3c4SHui Xie //
7*042dc3c4SHui Xie //===----------------------------------------------------------------------===//
8*042dc3c4SHui Xie 
9*042dc3c4SHui Xie // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20
10*042dc3c4SHui Xie // UNSUPPORTED: libcpp-has-no-incomplete-ranges
11*042dc3c4SHui Xie 
12*042dc3c4SHui Xie // zip_view() = default;
13*042dc3c4SHui Xie 
14*042dc3c4SHui Xie #include <ranges>
15*042dc3c4SHui Xie 
16*042dc3c4SHui Xie #include <cassert>
17*042dc3c4SHui Xie #include <type_traits>
18*042dc3c4SHui Xie #include <utility>
19*042dc3c4SHui Xie 
20*042dc3c4SHui Xie constexpr int buff[] = {1, 2, 3};
21*042dc3c4SHui Xie 
22*042dc3c4SHui Xie struct DefaultConstructibleView : std::ranges::view_base {
DefaultConstructibleViewDefaultConstructibleView23*042dc3c4SHui Xie   constexpr DefaultConstructibleView() : begin_(buff), end_(buff + 3) {}
beginDefaultConstructibleView24*042dc3c4SHui Xie   constexpr int const* begin() const { return begin_; }
endDefaultConstructibleView25*042dc3c4SHui Xie   constexpr int const* end() const { return end_; }
26*042dc3c4SHui Xie 
27*042dc3c4SHui Xie private:
28*042dc3c4SHui Xie   int const* begin_;
29*042dc3c4SHui Xie   int const* end_;
30*042dc3c4SHui Xie };
31*042dc3c4SHui Xie 
32*042dc3c4SHui Xie struct NoDefaultCtrView : std::ranges::view_base {
33*042dc3c4SHui Xie   NoDefaultCtrView() = delete;
34*042dc3c4SHui Xie   int* begin() const;
35*042dc3c4SHui Xie   int* end() const;
36*042dc3c4SHui Xie };
37*042dc3c4SHui Xie 
38*042dc3c4SHui Xie // The default constructor requires all underlying views to be default constructible.
39*042dc3c4SHui Xie // It is implicitly required by the tuple's constructor. If any of the iterators are
40*042dc3c4SHui Xie // not default constructible, zip iterator's =default would be implicitly deleted.
41*042dc3c4SHui Xie static_assert(std::is_default_constructible_v<std::ranges::zip_view<DefaultConstructibleView>>);
42*042dc3c4SHui Xie static_assert(
43*042dc3c4SHui Xie     std::is_default_constructible_v<std::ranges::zip_view<DefaultConstructibleView, DefaultConstructibleView>>);
44*042dc3c4SHui Xie static_assert(!std::is_default_constructible_v<std::ranges::zip_view<DefaultConstructibleView, NoDefaultCtrView>>);
45*042dc3c4SHui Xie static_assert(!std::is_default_constructible_v<std::ranges::zip_view<NoDefaultCtrView, NoDefaultCtrView>>);
46*042dc3c4SHui Xie static_assert(!std::is_default_constructible_v<std::ranges::zip_view<NoDefaultCtrView>>);
47*042dc3c4SHui Xie 
test()48*042dc3c4SHui Xie constexpr bool test() {
49*042dc3c4SHui Xie   {
50*042dc3c4SHui Xie     using View = std::ranges::zip_view<DefaultConstructibleView, DefaultConstructibleView>;
51*042dc3c4SHui Xie     View v = View(); // the default constructor is not explicit
52*042dc3c4SHui Xie     assert(v.size() == 3);
53*042dc3c4SHui Xie     auto it = v.begin();
54*042dc3c4SHui Xie     using Pair = std::pair<const int&, const int&>;
55*042dc3c4SHui Xie     assert(*it++ == Pair(buff[0], buff[0]));
56*042dc3c4SHui Xie     assert(*it++ == Pair(buff[1], buff[1]));
57*042dc3c4SHui Xie     assert(*it == Pair(buff[2], buff[2]));
58*042dc3c4SHui Xie   }
59*042dc3c4SHui Xie 
60*042dc3c4SHui Xie   return true;
61*042dc3c4SHui Xie }
62*042dc3c4SHui Xie 
main(int,char **)63*042dc3c4SHui Xie int main(int, char**) {
64*042dc3c4SHui Xie   test();
65*042dc3c4SHui Xie   static_assert(test());
66*042dc3c4SHui Xie 
67*042dc3c4SHui Xie   return 0;
68*042dc3c4SHui Xie }
69