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