is_convertible<_Up&&, _Tp>> = true>
constexpr
optional(_Up&& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, _Up>)
: _Base(std::in_place, std::forward<_Up>(__t)) { }
template<typename _Up = _Tp,
__not_<is_convertible<_Up&&, _Tp>>> = false>
explicit constexpr
optional(_Up&& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, _Up>)
: _Base(std::in_place, std::forward<_Up>(__t)) { }
template<typename _Up,
__not_<__converts_from_optional<_Tp, _Up>>> = true>
constexpr
optional(const optional<_Up>& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, const _Up&>)
{
if (__t)
emplace(*__t);
__not_<__converts_from_optional<_Tp, _Up>>> = false>
explicit constexpr
optional(const optional<_Up>& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, const _Up&>)
{
if (__t)
emplace(*__t);
__not_<__converts_from_optional<_Tp, _Up>>> = true>
constexpr
optional(optional<_Up>&& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, _Up>)
{
if (__t)
emplace(std::move(*__t));
__not_<__converts_from_optional<_Tp, _Up>>> = false>
explicit constexpr
optional(optional<_Up>&& __t)
+ noexcept(is_nothrow_constructible_v<_Tp, _Up>)
{
if (__t)
emplace(std::move(*__t));
_Requires<is_constructible<_Tp, _Args&&...>> = false>
explicit constexpr
optional(in_place_t, _Args&&... __args)
+ noexcept(is_nothrow_constructible_v<_Tp, _Args...>)
: _Base(std::in_place, std::forward<_Args>(__args)...) { }
template<typename _Up, typename... _Args,
_Args&&...>> = false>
explicit constexpr
optional(in_place_t, initializer_list<_Up> __il, _Args&&... __args)
+ noexcept(is_nothrow_constructible_v<_Tp, initializer_list<_Up>&,
+ _Args...>)
: _Base(std::in_place, __il, std::forward<_Args>(__args)...) { }
+
// Assignment operators.
optional&
operator=(nullopt_t) noexcept
is_assignable<_Tp&, _Up>>,
optional&>
operator=(_Up&& __u)
+ noexcept(__and_v<is_nothrow_constructible<_Tp, _Up>,
+ is_nothrow_assignable<_Tp&, _Up>>)
{
if (this->_M_is_engaged())
this->_M_get() = std::forward<_Up>(__u);
__not_<__assigns_from_optional<_Tp, _Up>>>,
optional&>
operator=(const optional<_Up>& __u)
+ noexcept(__and_v<is_nothrow_constructible<_Tp, const _Up&>,
+ is_nothrow_assignable<_Tp&, const _Up&>>)
{
if (__u)
{
__not_<__assigns_from_optional<_Tp, _Up>>>,
optional&>
operator=(optional<_Up>&& __u)
+ noexcept(__and_v<is_nothrow_constructible<_Tp, _Up>,
+ is_nothrow_assignable<_Tp&, _Up>>)
{
if (__u)
{
template<typename... _Args>
enable_if_t<is_constructible_v<_Tp, _Args&&...>, _Tp&>
emplace(_Args&&... __args)
+ noexcept(is_nothrow_constructible_v<_Tp, _Args...>)
{
this->_M_reset();
this->_M_construct(std::forward<_Args>(__args)...);
enable_if_t<is_constructible_v<_Tp, initializer_list<_Up>&,
_Args&&...>, _Tp&>
emplace(initializer_list<_Up> __il, _Args&&... __args)
+ noexcept(is_nothrow_constructible_v<_Tp, initializer_list<_Up>&,
+ _Args...>)
{
this->_M_reset();
this->_M_construct(__il, std::forward<_Args>(__args)...);
// Observers.
constexpr const _Tp*
- operator->() const
+ operator->() const noexcept
{ return std::__addressof(this->_M_get()); }
constexpr _Tp*
- operator->()
+ operator->() noexcept
{ return std::__addressof(this->_M_get()); }
constexpr const _Tp&
- operator*() const&
+ operator*() const& noexcept
{ return this->_M_get(); }
constexpr _Tp&
- operator*()&
+ operator*()& noexcept
{ return this->_M_get(); }
constexpr _Tp&&
- operator*()&&
+ operator*()&& noexcept
{ return std::move(this->_M_get()); }
constexpr const _Tp&&
- operator*() const&&
+ operator*() const&& noexcept
{ return std::move(this->_M_get()); }
constexpr explicit operator bool() const noexcept
--- /dev/null
+// Copyright (C) 2020 Free Software Foundation, Inc.
+//
+// This file is part of the GNU ISO C++ Library. This library is free
+// software; you can redistribute it and/or modify it under the
+// terms of the GNU General Public License as published by the
+// Free Software Foundation; either version 3, or (at your option)
+// any later version.
+
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License along
+// with this library; see the file COPYING3. If not see
+// <http://www.gnu.org/licenses/>.
+
+// { dg-do compile { target c++17 } }
+
+#include <optional>
+
+template<bool B>
+struct X
+{
+ X() noexcept(B);
+
+ X(const X&) noexcept(B);
+ X& operator=(const X&) noexcept(B);
+
+ X(int) noexcept(B);
+ X& operator=(int) noexcept(false);
+
+ X(void*) noexcept(true);
+ X& operator=(void*) noexcept(B);
+
+ X(const X<!B>&) noexcept(B);
+ X& operator=(const X<!B>&) noexcept(B);
+
+ X(std::initializer_list<int>, int) noexcept(B);
+};
+
+using std::is_nothrow_assignable_v;
+
+using Xyes = X<true>;
+using Xno = X<false>;
+using Oyes = std::optional<Xyes>;
+using Ono = std::optional<Xno>;
+
+static_assert( is_nothrow_assignable_v<Oyes, std::nullopt_t> );
+static_assert( is_nothrow_assignable_v<Oyes, const Xyes&> );
+static_assert( is_nothrow_assignable_v<Oyes, Xyes> );
+static_assert( ! is_nothrow_assignable_v<Oyes, int> );
+static_assert( is_nothrow_assignable_v<Oyes, void*> );
+static_assert( is_nothrow_assignable_v<Oyes, const Ono&> );
+static_assert( is_nothrow_assignable_v<Oyes, Ono> );
+
+static_assert( is_nothrow_assignable_v<Ono, std::nullopt_t> );
+static_assert( ! is_nothrow_assignable_v<Ono, const Xno&> );
+static_assert( ! is_nothrow_assignable_v<Ono, Xno> );
+static_assert( ! is_nothrow_assignable_v<Ono, int> );
+static_assert( ! is_nothrow_assignable_v<Ono, void*> );
+static_assert( ! is_nothrow_assignable_v<Ono, const Xyes&> );
+static_assert( ! is_nothrow_assignable_v<Ono, Xyes> );
+
+Xyes xyes;
+Xno xno;
+Oyes oyes;
+Ono ono;
+static_assert( noexcept(oyes.emplace()) );
+static_assert( noexcept(oyes.emplace(xyes)) );
+static_assert( noexcept(oyes.emplace(1)) );
+static_assert( noexcept(oyes.emplace(nullptr)) );
+static_assert( noexcept(oyes.emplace(xno)) );
+static_assert( noexcept(oyes.emplace({1,2,3}, 1)) );
+
+static_assert( ! noexcept(ono.emplace()) );
+static_assert( ! noexcept(ono.emplace(xno)) );
+static_assert( ! noexcept(ono.emplace(1)) );
+static_assert( noexcept(ono.emplace(nullptr)) );
+static_assert( ! noexcept(ono.emplace(xyes)) );
+static_assert( ! noexcept(ono.emplace({1,2,3}, 1)) );
--- /dev/null
+// Copyright (C) 2020 Free Software Foundation, Inc.
+//
+// This file is part of the GNU ISO C++ Library. This library is free
+// software; you can redistribute it and/or modify it under the
+// terms of the GNU General Public License as published by the
+// Free Software Foundation; either version 3, or (at your option)
+// any later version.
+
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+
+// You should have received a copy of the GNU General Public License along
+// with this library; see the file COPYING3. If not see
+// <http://www.gnu.org/licenses/>.
+
+// { dg-do compile { target c++17 } }
+
+#include <optional>
+
+template<bool B>
+struct X
+{
+ X() noexcept(B);
+ X(const X&) noexcept(B);
+
+ X(int) noexcept(B);
+ X(std::initializer_list<int>, int) noexcept(B);
+
+ X(const X<!B>&) noexcept(B);
+
+ X& operator=(const X&) noexcept(false);
+};
+
+using std::is_nothrow_constructible_v;
+using std::in_place_t;
+
+using Xyes = X<true>;
+using Xno = X<false>;
+using Oyes = std::optional<Xyes>;
+using Ono = std::optional<Xno>;
+
+static_assert( is_nothrow_constructible_v<Oyes> );
+static_assert( is_nothrow_constructible_v<Oyes, std::nullopt_t> );
+static_assert( is_nothrow_constructible_v<Oyes, const Xyes&> );
+static_assert( is_nothrow_constructible_v<Oyes, Xyes> );
+static_assert( is_nothrow_constructible_v<Oyes, in_place_t, short> );
+static_assert( is_nothrow_constructible_v<Oyes, in_place_t,
+ std::initializer_list<int>,
+ long> );
+static_assert( is_nothrow_constructible_v<Oyes, const Ono&> );
+static_assert( is_nothrow_constructible_v<Oyes, Ono> );
+
+static_assert( is_nothrow_constructible_v<Ono> );
+static_assert( is_nothrow_constructible_v<Ono, std::nullopt_t> );
+static_assert( ! is_nothrow_constructible_v<Ono, const Xno&> );
+static_assert( ! is_nothrow_constructible_v<Ono, Xno> );
+static_assert( ! is_nothrow_constructible_v<Ono, in_place_t, short> );
+static_assert( ! is_nothrow_constructible_v<Ono, in_place_t,
+ std::initializer_list<int>,
+ long> );
+static_assert( ! is_nothrow_constructible_v<Ono, const Xyes&> );
+static_assert( ! is_nothrow_constructible_v<Ono, Xyes> );