1*065dc485SJakub Mazurkiewicz //===----------------------------------------------------------------------===//
2*065dc485SJakub Mazurkiewicz //
3*065dc485SJakub Mazurkiewicz // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*065dc485SJakub Mazurkiewicz // See https://llvm.org/LICENSE.txt for license information.
5*065dc485SJakub Mazurkiewicz // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*065dc485SJakub Mazurkiewicz //
7*065dc485SJakub Mazurkiewicz //===----------------------------------------------------------------------===//
8*065dc485SJakub Mazurkiewicz
9*065dc485SJakub Mazurkiewicz // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20
10*065dc485SJakub Mazurkiewicz
11*065dc485SJakub Mazurkiewicz // <ranges>
12*065dc485SJakub Mazurkiewicz
13*065dc485SJakub Mazurkiewicz // constexpr View base() const& requires copy_constructible<View>;
14*065dc485SJakub Mazurkiewicz // constexpr View base() &&;
15*065dc485SJakub Mazurkiewicz
16*065dc485SJakub Mazurkiewicz #include <ranges>
17*065dc485SJakub Mazurkiewicz
18*065dc485SJakub Mazurkiewicz #include <cassert>
19*065dc485SJakub Mazurkiewicz #include <concepts>
20*065dc485SJakub Mazurkiewicz #include <utility>
21*065dc485SJakub Mazurkiewicz
22*065dc485SJakub Mazurkiewicz struct Range : std::ranges::view_base {
RangeRange23*065dc485SJakub Mazurkiewicz constexpr explicit Range(int* b, int* e) : begin_(b), end_(e) {}
RangeRange24*065dc485SJakub Mazurkiewicz constexpr Range(Range const& other) : begin_(other.begin_), end_(other.end_), wasCopyInitialized(true) {}
RangeRange25*065dc485SJakub Mazurkiewicz constexpr Range(Range&& other) : begin_(other.begin_), end_(other.end_), wasMoveInitialized(true) {}
26*065dc485SJakub Mazurkiewicz Range& operator=(Range const&) = default;
27*065dc485SJakub Mazurkiewicz Range& operator=(Range&&) = default;
beginRange28*065dc485SJakub Mazurkiewicz constexpr int* begin() const { return begin_; }
endRange29*065dc485SJakub Mazurkiewicz constexpr int* end() const { return end_; }
30*065dc485SJakub Mazurkiewicz
31*065dc485SJakub Mazurkiewicz int* begin_;
32*065dc485SJakub Mazurkiewicz int* end_;
33*065dc485SJakub Mazurkiewicz bool wasCopyInitialized = false;
34*065dc485SJakub Mazurkiewicz bool wasMoveInitialized = false;
35*065dc485SJakub Mazurkiewicz };
36*065dc485SJakub Mazurkiewicz
37*065dc485SJakub Mazurkiewicz static_assert(std::ranges::view<Range>);
38*065dc485SJakub Mazurkiewicz static_assert(std::ranges::forward_range<Range>);
39*065dc485SJakub Mazurkiewicz
40*065dc485SJakub Mazurkiewicz struct Pred {
41*065dc485SJakub Mazurkiewicz bool operator()(int, int) const;
42*065dc485SJakub Mazurkiewicz };
43*065dc485SJakub Mazurkiewicz
44*065dc485SJakub Mazurkiewicz struct NonCopyableRange : std::ranges::view_base {
45*065dc485SJakub Mazurkiewicz explicit NonCopyableRange(int*, int*);
46*065dc485SJakub Mazurkiewicz NonCopyableRange(NonCopyableRange const&) = delete;
47*065dc485SJakub Mazurkiewicz NonCopyableRange(NonCopyableRange&&) = default;
48*065dc485SJakub Mazurkiewicz NonCopyableRange& operator=(NonCopyableRange const&) = default;
49*065dc485SJakub Mazurkiewicz NonCopyableRange& operator=(NonCopyableRange&&) = default;
50*065dc485SJakub Mazurkiewicz int* begin() const;
51*065dc485SJakub Mazurkiewicz int* end() const;
52*065dc485SJakub Mazurkiewicz };
53*065dc485SJakub Mazurkiewicz
54*065dc485SJakub Mazurkiewicz static_assert(!std::copy_constructible<NonCopyableRange>);
55*065dc485SJakub Mazurkiewicz
56*065dc485SJakub Mazurkiewicz template <typename T>
57*065dc485SJakub Mazurkiewicz concept CanCallBaseOn = requires(T t) { std::forward<T>(t).base(); };
58*065dc485SJakub Mazurkiewicz
test()59*065dc485SJakub Mazurkiewicz constexpr bool test() {
60*065dc485SJakub Mazurkiewicz int buff[] = {1, 2, 3, 4};
61*065dc485SJakub Mazurkiewicz
62*065dc485SJakub Mazurkiewicz // Check the const& overload
63*065dc485SJakub Mazurkiewicz {
64*065dc485SJakub Mazurkiewicz Range range(buff, buff + 4);
65*065dc485SJakub Mazurkiewicz std::ranges::chunk_by_view<Range, Pred> const view(range, Pred{});
66*065dc485SJakub Mazurkiewicz std::same_as<Range> decltype(auto) result = view.base();
67*065dc485SJakub Mazurkiewicz assert(result.wasCopyInitialized);
68*065dc485SJakub Mazurkiewicz assert(result.begin() == buff);
69*065dc485SJakub Mazurkiewicz assert(result.end() == buff + 4);
70*065dc485SJakub Mazurkiewicz }
71*065dc485SJakub Mazurkiewicz
72*065dc485SJakub Mazurkiewicz // Check the && overload
73*065dc485SJakub Mazurkiewicz {
74*065dc485SJakub Mazurkiewicz Range range(buff, buff + 4);
75*065dc485SJakub Mazurkiewicz std::ranges::chunk_by_view<Range, Pred> view(range, Pred{});
76*065dc485SJakub Mazurkiewicz std::same_as<Range> decltype(auto) result = std::move(view).base();
77*065dc485SJakub Mazurkiewicz assert(result.wasMoveInitialized);
78*065dc485SJakub Mazurkiewicz assert(result.begin() == buff);
79*065dc485SJakub Mazurkiewicz assert(result.end() == buff + 4);
80*065dc485SJakub Mazurkiewicz }
81*065dc485SJakub Mazurkiewicz
82*065dc485SJakub Mazurkiewicz // Ensure the const& overload is not considered when the base is not copy-constructible
83*065dc485SJakub Mazurkiewicz {
84*065dc485SJakub Mazurkiewicz static_assert(!CanCallBaseOn<std::ranges::chunk_by_view<NonCopyableRange, Pred> const&>);
85*065dc485SJakub Mazurkiewicz static_assert(!CanCallBaseOn<std::ranges::chunk_by_view<NonCopyableRange, Pred>&>);
86*065dc485SJakub Mazurkiewicz static_assert(!CanCallBaseOn<std::ranges::chunk_by_view<NonCopyableRange, Pred> const&&>);
87*065dc485SJakub Mazurkiewicz static_assert(CanCallBaseOn<std::ranges::chunk_by_view<NonCopyableRange, Pred>&&>);
88*065dc485SJakub Mazurkiewicz static_assert(CanCallBaseOn<std::ranges::chunk_by_view<NonCopyableRange, Pred>>);
89*065dc485SJakub Mazurkiewicz }
90*065dc485SJakub Mazurkiewicz
91*065dc485SJakub Mazurkiewicz return true;
92*065dc485SJakub Mazurkiewicz }
93*065dc485SJakub Mazurkiewicz
main(int,char **)94*065dc485SJakub Mazurkiewicz int main(int, char**) {
95*065dc485SJakub Mazurkiewicz test();
96*065dc485SJakub Mazurkiewicz static_assert(test());
97*065dc485SJakub Mazurkiewicz
98*065dc485SJakub Mazurkiewicz return 0;
99*065dc485SJakub Mazurkiewicz }
100