From 6c9fb2428f651666cb96e3cd276637855432516e Mon Sep 17 00:00:00 2001 From: Nick Lewycky Date: Tue, 9 Feb 2016 04:47:58 +0000 Subject: [PATCH] Use std::forward to make ErrorOr constructible from a value that has a user-defined conversion to T. No functionality change intended. llvm-svn: 260196 --- include/llvm/Support/ErrorOr.h | 22 +++++----------- unittests/Support/ErrorOrTest.cpp | 42 ++++++++++++++++++++++++++++++- 2 files changed, 47 insertions(+), 17 deletions(-) diff --git a/include/llvm/Support/ErrorOr.h b/include/llvm/Support/ErrorOr.h index 7fe1d5a16e8..877f4063cd2 100644 --- a/include/llvm/Support/ErrorOr.h +++ b/include/llvm/Support/ErrorOr.h @@ -23,20 +23,6 @@ #include namespace llvm { -template -typename std::enable_if< std::is_constructible::value - , typename std::remove_reference::type>::type && - moveIfMoveConstructible(V &Val) { - return std::move(Val); -} - -template -typename std::enable_if< !std::is_constructible::value - , typename std::remove_reference::type>::type & -moveIfMoveConstructible(V &Val) { - return Val; -} - /// \brief Stores a reference that can be changed. template class ReferenceStorage { @@ -107,8 +93,12 @@ public: new (getErrorStorage()) std::error_code(EC); } - ErrorOr(T Val) : HasError(false) { - new (getStorage()) storage_type(moveIfMoveConstructible(Val)); + template + ErrorOr(OtherT &&Val, + typename std::enable_if::value>::type + * = nullptr) + : HasError(false) { + new (getStorage()) storage_type(std::forward(Val)); } ErrorOr(const ErrorOr &Other) { diff --git a/unittests/Support/ErrorOrTest.cpp b/unittests/Support/ErrorOrTest.cpp index 73d0e3f2bb7..87dcab7b181 100644 --- a/unittests/Support/ErrorOrTest.cpp +++ b/unittests/Support/ErrorOrTest.cpp @@ -16,7 +16,7 @@ using namespace llvm; namespace { -ErrorOr t1() {return 1;} +ErrorOr t1() { return 1; } ErrorOr t2() { return errc::invalid_argument; } TEST(ErrorOr, SimpleValue) { @@ -71,6 +71,46 @@ TEST(ErrorOr, Comparison) { EXPECT_EQ(x, errc::no_such_file_or_directory); } +TEST(ErrorOr, ImplicitConversion) { + ErrorOr x("string literal"); + EXPECT_TRUE(!!x); +} + +TEST(ErrorOr, ImplicitConversionCausesMove) { + struct Source {}; + struct Destination { + Destination(const Source&) {} + Destination(Source&&) = delete; + }; + Source s; + ErrorOr x = s; + EXPECT_TRUE(!!x); +} + +TEST(ErrorOr, ImplicitConversionNoAmbiguity) { + struct CastsToErrorCode { + CastsToErrorCode() = default; + CastsToErrorCode(std::error_code) {} + operator std::error_code() { return errc::invalid_argument; } + } casts_to_error_code; + ErrorOr x1(casts_to_error_code); + ErrorOr x2 = casts_to_error_code; + ErrorOr x3 = {casts_to_error_code}; + ErrorOr x4{casts_to_error_code}; + ErrorOr x5(errc::no_such_file_or_directory); + ErrorOr x6 = errc::no_such_file_or_directory; + ErrorOr x7 = {errc::no_such_file_or_directory}; + ErrorOr x8{errc::no_such_file_or_directory}; + EXPECT_TRUE(!!x1); + EXPECT_TRUE(!!x2); + EXPECT_TRUE(!!x3); + EXPECT_TRUE(!!x4); + EXPECT_FALSE(x5); + EXPECT_FALSE(x6); + EXPECT_FALSE(x7); + EXPECT_FALSE(x8); +} + // ErrorOr x(nullptr); // ErrorOr> y = x; // invalid conversion static_assert(