xref: /llvm-project/libcxx/test/std/ranges/range.adaptors/range.split/base.pass.cpp (revision a2b3ab8f7786b9bb6e1b8bbb01b88d4bbe28af69)
1*a2b3ab8fSHui //===----------------------------------------------------------------------===//
2*a2b3ab8fSHui //
3*a2b3ab8fSHui // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*a2b3ab8fSHui // See https://llvm.org/LICENSE.txt for license information.
5*a2b3ab8fSHui // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*a2b3ab8fSHui //
7*a2b3ab8fSHui //===----------------------------------------------------------------------===//
8*a2b3ab8fSHui 
9*a2b3ab8fSHui // UNSUPPORTED: c++03, c++11, c++14, c++17
10*a2b3ab8fSHui 
11*a2b3ab8fSHui // constexpr V base() const & requires copy_constructible<V> { return base_; }
12*a2b3ab8fSHui // constexpr V base() && { return std::move(base_); }
13*a2b3ab8fSHui 
14*a2b3ab8fSHui #include <cassert>
15*a2b3ab8fSHui #include <ranges>
16*a2b3ab8fSHui #include <type_traits>
17*a2b3ab8fSHui #include <utility>
18*a2b3ab8fSHui 
19*a2b3ab8fSHui #include "MoveOnly.h"
20*a2b3ab8fSHui 
21*a2b3ab8fSHui struct View : std::ranges::view_base {
22*a2b3ab8fSHui   int i;
23*a2b3ab8fSHui   int* begin() const;
24*a2b3ab8fSHui   int* end() const;
25*a2b3ab8fSHui };
26*a2b3ab8fSHui 
27*a2b3ab8fSHui struct MoveOnlyView : View {
28*a2b3ab8fSHui   MoveOnly mo;
29*a2b3ab8fSHui };
30*a2b3ab8fSHui 
31*a2b3ab8fSHui template <class T>
32*a2b3ab8fSHui concept HasBase = requires(T&& t) { std::forward<T>(t).base(); };
33*a2b3ab8fSHui 
34*a2b3ab8fSHui static_assert(HasBase<std::ranges::split_view<View, View> const&>);
35*a2b3ab8fSHui static_assert(HasBase<std::ranges::split_view<View, View>&&>);
36*a2b3ab8fSHui 
37*a2b3ab8fSHui static_assert(!HasBase<std::ranges::split_view<MoveOnlyView, View> const&>);
38*a2b3ab8fSHui static_assert(HasBase<std::ranges::split_view<MoveOnlyView, View>&&>);
39*a2b3ab8fSHui 
test()40*a2b3ab8fSHui constexpr bool test() {
41*a2b3ab8fSHui   // const &
42*a2b3ab8fSHui   {
43*a2b3ab8fSHui     const std::ranges::split_view<View, View> sv{View{{}, 5}, View{{}, 0}};
44*a2b3ab8fSHui     std::same_as<View> decltype(auto) v = sv.base();
45*a2b3ab8fSHui     assert(v.i == 5);
46*a2b3ab8fSHui   }
47*a2b3ab8fSHui 
48*a2b3ab8fSHui   // &
49*a2b3ab8fSHui   {
50*a2b3ab8fSHui     std::ranges::split_view<View, View> sv{View{{}, 5}, View{{}, 0}};
51*a2b3ab8fSHui     std::same_as<View> decltype(auto) v = sv.base();
52*a2b3ab8fSHui     assert(v.i == 5);
53*a2b3ab8fSHui   }
54*a2b3ab8fSHui 
55*a2b3ab8fSHui   // &&
56*a2b3ab8fSHui   {
57*a2b3ab8fSHui     std::ranges::split_view<View, View> sv{View{{}, 5}, View{{}, 0}};
58*a2b3ab8fSHui     std::same_as<View> decltype(auto) v = std::move(sv).base();
59*a2b3ab8fSHui     assert(v.i == 5);
60*a2b3ab8fSHui   }
61*a2b3ab8fSHui 
62*a2b3ab8fSHui   // const &&
63*a2b3ab8fSHui   {
64*a2b3ab8fSHui     std::ranges::split_view<View, View> sv{View{{}, 5}, View{{}, 0}};
65*a2b3ab8fSHui     std::same_as<View> decltype(auto) v = std::move(sv).base();
66*a2b3ab8fSHui     assert(v.i == 5);
67*a2b3ab8fSHui   }
68*a2b3ab8fSHui 
69*a2b3ab8fSHui   // move only
70*a2b3ab8fSHui   {
71*a2b3ab8fSHui     std::ranges::split_view<MoveOnlyView, View> sv{MoveOnlyView{{}, 5}, View{{}, 0}};
72*a2b3ab8fSHui     std::same_as<MoveOnlyView> decltype(auto) v = std::move(sv).base();
73*a2b3ab8fSHui     assert(v.mo.get() == 5);
74*a2b3ab8fSHui   }
75*a2b3ab8fSHui 
76*a2b3ab8fSHui   // LWG 3590 split_view::base() const & is overconstrained
77*a2b3ab8fSHui   {
78*a2b3ab8fSHui     struct CopyCtorButNotAssignable : std::ranges::view_base {
79*a2b3ab8fSHui       int i;
80*a2b3ab8fSHui       constexpr CopyCtorButNotAssignable(int ii) : i(ii) {}
81*a2b3ab8fSHui       constexpr CopyCtorButNotAssignable(const CopyCtorButNotAssignable&)            = default;
82*a2b3ab8fSHui       constexpr CopyCtorButNotAssignable(CopyCtorButNotAssignable&&)                 = default;
83*a2b3ab8fSHui       constexpr CopyCtorButNotAssignable& operator=(CopyCtorButNotAssignable&&)      = default;
84*a2b3ab8fSHui       constexpr CopyCtorButNotAssignable& operator=(const CopyCtorButNotAssignable&) = delete;
85*a2b3ab8fSHui       constexpr int* begin() const { return nullptr; }
86*a2b3ab8fSHui       constexpr int* end() const { return nullptr; }
87*a2b3ab8fSHui     };
88*a2b3ab8fSHui     static_assert(std::copy_constructible<CopyCtorButNotAssignable>);
89*a2b3ab8fSHui     static_assert(!std::copyable<CopyCtorButNotAssignable>);
90*a2b3ab8fSHui     const std::ranges::split_view<CopyCtorButNotAssignable, CopyCtorButNotAssignable> sv{
91*a2b3ab8fSHui         CopyCtorButNotAssignable{5}, CopyCtorButNotAssignable{5}};
92*a2b3ab8fSHui     std::same_as<CopyCtorButNotAssignable> decltype(auto) v = sv.base();
93*a2b3ab8fSHui     assert(v.i == 5);
94*a2b3ab8fSHui   }
95*a2b3ab8fSHui 
96*a2b3ab8fSHui   return true;
97*a2b3ab8fSHui }
98*a2b3ab8fSHui 
main(int,char **)99*a2b3ab8fSHui int main(int, char**) {
100*a2b3ab8fSHui   test();
101*a2b3ab8fSHui   static_assert(test());
102*a2b3ab8fSHui   return 0;
103*a2b3ab8fSHui }
104