//===----------------------------------------------------------------------===// // // 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 // template // struct in_in_result; #include #include #include #include "MoveOnly.h" struct A { explicit A(int); }; // no implicit conversion static_assert(!std::is_constructible_v, std::ranges::in_in_result>); struct B { B(int); }; // implicit conversion static_assert(std::is_constructible_v, std::ranges::in_in_result>); static_assert(std::is_constructible_v, std::ranges::in_in_result&>); static_assert(std::is_constructible_v, const std::ranges::in_in_result>); static_assert(std::is_constructible_v, const std::ranges::in_in_result&>); struct C { C(int&); }; static_assert(!std::is_constructible_v, std::ranges::in_in_result&>); // has to be convertible via const& static_assert(std::is_convertible_v&, std::ranges::in_in_result>); static_assert(std::is_convertible_v&, std::ranges::in_in_result>); static_assert(std::is_convertible_v&&, std::ranges::in_in_result>); static_assert(std::is_convertible_v&&, std::ranges::in_in_result>); // should be move constructible static_assert(std::is_move_constructible_v>); static_assert(std::is_move_constructible_v>); struct NotConvertible {}; // conversions should not work if there is no conversion static_assert(!std::is_convertible_v, std::ranges::in_in_result>); static_assert(!std::is_convertible_v, std::ranges::in_in_result>); template struct ConstructibleFrom { constexpr ConstructibleFrom(T c) : content{c} {} T content; }; constexpr bool test() { // Checks that conversion operations are correct. { std::ranges::in_in_result res{10L, 0.}; assert(res.in1 == 10); assert(res.in2 == 0.); std::ranges::in_in_result, ConstructibleFrom> res2 = res; assert(res2.in1.content == 10); assert(res2.in2.content == 0.); } // Checks that conversions are possible when one of the types is move-only. { std::ranges::in_in_result res{MoveOnly{}, 0}; assert(res.in1.get() == 1); [[maybe_unused]] auto res2 = static_cast>(std::move(res)); assert(res.in1.get() == 0); } // Checks that structured bindings get the correct values. { auto [in1, in2] = std::ranges::in_in_result{1, 2}; assert(in1 == 1); assert(in2 == 2); } return true; } int main(int, char**) { test(); static_assert(test()); return 0; }