//===----------------------------------------------------------------------===// // // 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 // template // constexpr explicit(!is_convertible_v) expected(U&& v); // // Constraints: // - is_same_v, in_place_t> is false; and // - is_same_v> is false; and // - remove_cvref_t is not a specialization of unexpected; and // - is_constructible_v is true. // // Effects: Direct-non-list-initializes val with std::forward(v). // // Postconditions: has_value() is true. // // Throws: Any exception thrown by the initialization of val. #include #include #include #include #include "MoveOnly.h" #include "test_macros.h" #include "../../types.h" // Test Constraints: static_assert(std::is_constructible_v, int>); // is_same_v, in_place_t> struct FromJustInplace { FromJustInplace(std::in_place_t); }; static_assert(!std::is_constructible_v, std::in_place_t>); static_assert(!std::is_constructible_v, std::in_place_t const&>); // is_same_v> // Note that result is true because it is covered by the constructors that take expected static_assert(std::is_constructible_v, std::expected&>); // remove_cvref_t is a specialization of unexpected // Note that result is true because it is covered by the constructors that take unexpected static_assert(std::is_constructible_v, std::unexpected&>); // !is_constructible_v struct foo {}; static_assert(!std::is_constructible_v, foo>); // test explicit(!is_convertible_v) struct NotConvertible { explicit NotConvertible(int); }; static_assert(std::is_convertible_v>); static_assert(!std::is_convertible_v>); struct CopyOnly { int i; constexpr CopyOnly(int ii) : i(ii) {} CopyOnly(const CopyOnly&) = default; CopyOnly(CopyOnly&&) = delete; friend constexpr bool operator==(const CopyOnly& mi, int ii) { return mi.i == ii; } }; struct BaseError {}; struct DerivedError : BaseError {}; template constexpr void testInt() { std::expected e(5); assert(e.has_value()); assert(e.value() == 5); } template constexpr void testLValue() { T t(5); std::expected e(t); assert(e.has_value()); assert(e.value() == 5); } template constexpr void testRValue() { std::expected e(T(5)); assert(e.has_value()); assert(e.value() == 5); } constexpr bool test() { testInt(); testInt(); testInt(); testInt, bool>(); testLValue(); testLValue(); testLValue, bool>(); testRValue(); testRValue(); testRValue, bool>(); // Test default template argument. // Without it, the template parameter cannot be deduced from an initializer list { struct Bar { int i; int j; constexpr Bar(int ii, int jj) : i(ii), j(jj) {} }; std::expected e({5, 6}); assert(e.value().i == 5); assert(e.value().j == 6); } // https://cplusplus.github.io/LWG/issue3836 // Test & { std::expected e1(false); std::expected e2(e1); assert(e2.has_value()); assert(!e2.value()); // yes, e2 holds "false" since LWG3836 } // Test && { std::expected e1(false); std::expected e2(std::move(e1)); assert(e2.has_value()); assert(!e2.value()); // yes, e2 holds "false" since LWG3836 } // Test const& { const std::expected e1(false); std::expected e2(e1); assert(e2.has_value()); assert(!e2.value()); // yes, e2 holds "false" since LWG3836 } // Test const&& { const std::expected e1(false); std::expected e2(std::move(e1)); assert(e2.has_value()); assert(!e2.value()); // yes, e2 holds "false" since LWG3836 } return true; } void testException() { #ifndef TEST_HAS_NO_EXCEPTIONS struct Throwing { Throwing(int) { throw Except{}; }; }; try { std::expected u(5); assert(false); } catch (Except) { } #endif // TEST_HAS_NO_EXCEPTIONS } int main(int, char**) { test(); static_assert(test()); testException(); return 0; }