123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174 |
- //===----------------------------------------------------------------------===//
- //
- // The LLVM Compiler Infrastructure
- //
- // This file is dual licensed under the MIT and the University of Illinois Open
- // Source Licenses. See LICENSE.TXT for details.
- //
- //===----------------------------------------------------------------------===//
- // UNSUPPORTED: c++98, c++03, c++11, c++14
- // <optional>
- // optional<T>& operator=(optional<T>&& rhs)
- // noexcept(is_nothrow_move_assignable<T>::value &&
- // is_nothrow_move_constructible<T>::value);
- #include <optional>
- #include <type_traits>
- #include <cassert>
- #include "test_macros.h"
- #include "archetypes.hpp"
- using std::optional;
- struct X
- {
- static bool throw_now;
- static int alive;
- X() { ++alive; }
- X(X&&)
- {
- if (throw_now)
- TEST_THROW(6);
- ++alive;
- }
- X& operator=(X&&)
- {
- if (throw_now)
- TEST_THROW(42);
- return *this;
- }
- ~X() { assert(alive > 0); --alive; }
- };
- struct Y {};
- bool X::throw_now = false;
- int X::alive = 0;
- int main()
- {
- {
- static_assert(std::is_nothrow_move_assignable<optional<int>>::value, "");
- optional<int> opt;
- constexpr optional<int> opt2;
- opt = std::move(opt2);
- static_assert(static_cast<bool>(opt2) == false, "");
- assert(static_cast<bool>(opt) == static_cast<bool>(opt2));
- }
- {
- optional<int> opt;
- constexpr optional<int> opt2(2);
- opt = std::move(opt2);
- static_assert(static_cast<bool>(opt2) == true, "");
- static_assert(*opt2 == 2, "");
- assert(static_cast<bool>(opt) == static_cast<bool>(opt2));
- assert(*opt == *opt2);
- }
- {
- optional<int> opt(3);
- constexpr optional<int> opt2;
- opt = std::move(opt2);
- static_assert(static_cast<bool>(opt2) == false, "");
- assert(static_cast<bool>(opt) == static_cast<bool>(opt2));
- }
- {
- using T = TestTypes::TestType;
- T::reset();
- optional<T> opt(3);
- optional<T> opt2;
- assert(T::alive == 1);
- opt = std::move(opt2);
- assert(T::alive == 0);
- assert(static_cast<bool>(opt2) == false);
- assert(static_cast<bool>(opt) == static_cast<bool>(opt2));
- }
- {
- optional<int> opt(3);
- constexpr optional<int> opt2(2);
- opt = std::move(opt2);
- static_assert(static_cast<bool>(opt2) == true, "");
- static_assert(*opt2 == 2, "");
- assert(static_cast<bool>(opt) == static_cast<bool>(opt2));
- assert(*opt == *opt2);
- }
- #ifndef TEST_HAS_NO_EXCEPTIONS
- {
- static_assert(!std::is_nothrow_move_assignable<optional<X>>::value, "");
- X::alive = 0;
- X::throw_now = false;
- optional<X> opt;
- optional<X> opt2(X{});
- assert(X::alive == 1);
- assert(static_cast<bool>(opt2) == true);
- try
- {
- X::throw_now = true;
- opt = std::move(opt2);
- assert(false);
- }
- catch (int i)
- {
- assert(i == 6);
- assert(static_cast<bool>(opt) == false);
- }
- assert(X::alive == 1);
- }
- assert(X::alive == 0);
- {
- static_assert(!std::is_nothrow_move_assignable<optional<X>>::value, "");
- X::throw_now = false;
- optional<X> opt(X{});
- optional<X> opt2(X{});
- assert(X::alive == 2);
- assert(static_cast<bool>(opt2) == true);
- try
- {
- X::throw_now = true;
- opt = std::move(opt2);
- assert(false);
- }
- catch (int i)
- {
- assert(i == 42);
- assert(static_cast<bool>(opt) == true);
- }
- assert(X::alive == 2);
- }
- assert(X::alive == 0);
- #endif // TEST_HAS_NO_EXCEPTIONS
- {
- static_assert(std::is_nothrow_move_assignable<optional<Y>>::value, "");
- }
- {
- struct ThrowsMove {
- ThrowsMove() noexcept {}
- ThrowsMove(ThrowsMove const&) noexcept {}
- ThrowsMove(ThrowsMove &&) noexcept(false) {}
- ThrowsMove& operator=(ThrowsMove const&) noexcept { return *this; }
- ThrowsMove& operator=(ThrowsMove &&) noexcept { return *this; }
- };
- static_assert(!std::is_nothrow_move_assignable<optional<ThrowsMove>>::value, "");
- struct ThrowsMoveAssign {
- ThrowsMoveAssign() noexcept {}
- ThrowsMoveAssign(ThrowsMoveAssign const&) noexcept {}
- ThrowsMoveAssign(ThrowsMoveAssign &&) noexcept {}
- ThrowsMoveAssign& operator=(ThrowsMoveAssign const&) noexcept { return *this; }
- ThrowsMoveAssign& operator=(ThrowsMoveAssign &&) noexcept(false) { return *this; }
- };
- static_assert(!std::is_nothrow_move_assignable<optional<ThrowsMoveAssign>>::value, "");
- struct NoThrowMove {
- NoThrowMove() noexcept(false) {}
- NoThrowMove(NoThrowMove const&) noexcept(false) {}
- NoThrowMove(NoThrowMove &&) noexcept {}
- NoThrowMove& operator=(NoThrowMove const&) noexcept { return *this; }
- NoThrowMove& operator=(NoThrowMove&&) noexcept { return *this; }
- };
- static_assert(std::is_nothrow_move_assignable<optional<NoThrowMove>>::value, "");
- }
- }
|