//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14, c++17 // constexpr V base() const & requires copy_constructible { return base_; } // constexpr V base() && { return std::move(base_); } #include #include #include #include #include "MoveOnly.h" struct View : std::ranges::view_base { int i; int* begin() const; int* end() const; }; struct MoveOnlyView : View { MoveOnly mo; }; template concept HasBase = requires(T&& t) { std::forward(t).base(); }; static_assert(HasBase const&>); static_assert(HasBase&&>); static_assert(!HasBase const&>); static_assert(HasBase&&>); constexpr bool test() { // const & { const std::ranges::split_view sv{View{{}, 5}, View{{}, 0}}; std::same_as decltype(auto) v = sv.base(); assert(v.i == 5); } // & { std::ranges::split_view sv{View{{}, 5}, View{{}, 0}}; std::same_as decltype(auto) v = sv.base(); assert(v.i == 5); } // && { std::ranges::split_view sv{View{{}, 5}, View{{}, 0}}; std::same_as decltype(auto) v = std::move(sv).base(); assert(v.i == 5); } // const && { std::ranges::split_view sv{View{{}, 5}, View{{}, 0}}; std::same_as decltype(auto) v = std::move(sv).base(); assert(v.i == 5); } // move only { std::ranges::split_view sv{MoveOnlyView{{}, 5}, View{{}, 0}}; std::same_as decltype(auto) v = std::move(sv).base(); assert(v.mo.get() == 5); } // LWG 3590 split_view::base() const & is overconstrained { struct CopyCtorButNotAssignable : std::ranges::view_base { int i; constexpr CopyCtorButNotAssignable(int ii) : i(ii) {} constexpr CopyCtorButNotAssignable(const CopyCtorButNotAssignable&) = default; constexpr CopyCtorButNotAssignable(CopyCtorButNotAssignable&&) = default; constexpr CopyCtorButNotAssignable& operator=(CopyCtorButNotAssignable&&) = default; constexpr CopyCtorButNotAssignable& operator=(const CopyCtorButNotAssignable&) = delete; constexpr int* begin() const { return nullptr; } constexpr int* end() const { return nullptr; } }; static_assert(std::copy_constructible); static_assert(!std::copyable); const std::ranges::split_view sv{ CopyCtorButNotAssignable{5}, CopyCtorButNotAssignable{5}}; std::same_as decltype(auto) v = sv.base(); assert(v.i == 5); } return true; } int main(int, char**) { test(); static_assert(test()); return 0; }