//===----------------------------------------------------------------------===// // 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, c++20 // constexpr expected(const expected& rhs); // // Effects: If rhs.has_value() is true, direct-non-list-initializes val with *rhs. // Otherwise, direct-non-list-initializes unex with rhs.error(). // // Postconditions: rhs.has_value() == this->has_value(). // // Throws: Any exception thrown by the initialization of val or unex. // // Remarks: This constructor is defined as deleted unless // - is_copy_constructible_v is true and // - is_copy_constructible_v is true. // // This constructor is trivial if // - is_trivially_copy_constructible_v is true and // - is_trivially_copy_constructible_v is true. #include #include #include #include #include "test_macros.h" struct NonCopyable { NonCopyable(const NonCopyable&) = delete; }; struct CopyableNonTrivial { int i; constexpr CopyableNonTrivial(int ii) : i(ii) {} constexpr CopyableNonTrivial(const CopyableNonTrivial& o) { i = o.i; } friend constexpr bool operator==(const CopyableNonTrivial&, const CopyableNonTrivial&) = default; }; // Test: This constructor is defined as deleted unless // - is_copy_constructible_v is true and // - is_copy_constructible_v is true. static_assert(std::is_copy_constructible_v>); static_assert(std::is_copy_constructible_v>); static_assert(std::is_copy_constructible_v>); static_assert(std::is_copy_constructible_v>); static_assert(!std::is_copy_constructible_v>); static_assert(!std::is_copy_constructible_v>); static_assert(!std::is_copy_constructible_v>); // Test: This constructor is trivial if // - is_trivially_copy_constructible_v is true and // - is_trivially_copy_constructible_v is true. static_assert(std::is_trivially_copy_constructible_v>); static_assert(!std::is_trivially_copy_constructible_v>); static_assert(!std::is_trivially_copy_constructible_v>); static_assert(!std::is_trivially_copy_constructible_v>); constexpr bool test() { // copy the value non-trivial { const std::expected e1(5); auto e2 = e1; assert(e2.has_value()); assert(e2.value().i == 5); } // copy the error non-trivial { const std::expected e1(std::unexpect, 5); auto e2 = e1; assert(!e2.has_value()); assert(e2.error().i == 5); } // copy the value trivial { const std::expected e1(5); auto e2 = e1; assert(e2.has_value()); assert(e2.value() == 5); } // copy the error trivial { const std::expected e1(std::unexpect, 5); auto e2 = e1; assert(!e2.has_value()); assert(e2.error() == 5); } return true; } void testException() { #ifndef TEST_HAS_NO_EXCEPTIONS struct Except {}; struct Throwing { Throwing() = default; Throwing(const Throwing&) { throw Except{}; } }; // throw on copying value { const std::expected e1; try { [[maybe_unused]] auto e2 = e1; assert(false); } catch (Except) { } } // throw on copying error { const std::expected e1(std::unexpect); try { [[maybe_unused]] auto e2 = e1; assert(false); } catch (Except) { } } #endif // TEST_HAS_NO_EXCEPTIONS } int main(int, char**) { test(); static_assert(test()); testException(); return 0; }