123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086 |
- /* Essentially an internal optional implementation :)
- (C) 2017-2020 Niall Douglas <http://www.nedproductions.biz/> (24 commits)
- File Created: June 2017
- Boost Software License - Version 1.0 - August 17th, 2003
- Permission is hereby granted, free of charge, to any person or organization
- obtaining a copy of the software and accompanying documentation covered by
- this license (the "Software") to use, reproduce, display, distribute,
- execute, and transmit the Software, and to prepare derivative works of the
- Software, and to permit third-parties to whom the Software is furnished to
- do so, all subject to the following:
- The copyright notices in the Software and this entire statement, including
- the above license grant, this restriction and the following disclaimer,
- must be included in all copies of the Software, in whole or in part, and
- all derivative works of the Software, unless such copies or derivative
- works are solely in the form of machine-executable object code generated by
- a source language processor.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
- SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
- FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
- ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
- DEALINGS IN THE SOFTWARE.
- */
- #ifndef BOOST_OUTCOME_VALUE_STORAGE_HPP
- #define BOOST_OUTCOME_VALUE_STORAGE_HPP
- #include "../config.hpp"
- #include <cassert>
- BOOST_OUTCOME_V2_NAMESPACE_BEGIN
- namespace detail
- {
- template <class T, bool nothrow> struct strong_swap_impl
- {
- constexpr strong_swap_impl(bool &allgood, T &a, T &b)
- {
- allgood = true;
- using std::swap;
- swap(a, b);
- }
- };
- #ifndef BOOST_NO_EXCEPTIONS
- template <class T> struct strong_swap_impl<T, false>
- {
- strong_swap_impl(bool &allgood, T &a, T &b)
- {
- allgood = true;
- T v(static_cast<T &&>(a));
- try
- {
- a = static_cast<T &&>(b);
- }
- catch(...)
- {
- // Try to put back a
- try
- {
- a = static_cast<T &&>(v);
- // fall through as all good
- }
- catch(...)
- {
- // failed to completely restore
- allgood = false;
- // throw away second exception
- }
- throw; // rethrow original exception
- }
- // b has been moved to a, try to move v to b
- try
- {
- b = static_cast<T &&>(v);
- }
- catch(...)
- {
- // Try to restore a to b, and v to a
- try
- {
- b = static_cast<T &&>(a);
- a = static_cast<T &&>(v);
- // fall through as all good
- }
- catch(...)
- {
- // failed to completely restore
- allgood = false;
- // throw away second exception
- }
- throw; // rethrow original exception
- }
- }
- };
- #endif
- } // namespace detail
- /*!
- */
- BOOST_OUTCOME_TEMPLATE(class T)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(std::is_move_constructible<T>::value &&std::is_move_assignable<T>::value))
- constexpr inline void strong_swap(bool &allgood, T &a, T &b) noexcept(detail::is_nothrow_swappable<T>::value)
- {
- detail::strong_swap_impl<T, detail::is_nothrow_swappable<T>::value>(allgood, a, b);
- }
- namespace detail
- {
- template <class T>
- constexpr
- #ifdef _MSC_VER
- __declspec(noreturn)
- #elif defined(__GNUC__) || defined(__clang__)
- __attribute__((noreturn))
- #endif
- void make_ub(T && /*unused*/)
- {
- assert(false); // NOLINT
- #if defined(__GNUC__) || defined(__clang__)
- __builtin_unreachable();
- #elif defined(_MSC_VER)
- __assume(0);
- #endif
- }
- /* Outcome v1 used a C bitfield whose values were tracked by compiler optimisers nicely,
- but that produces ICEs when used in constexpr.
- Outcome v2.0-v2.1 used a 32 bit integer and manually set and cleared bits. Unfortunately
- only GCC's optimiser tracks bit values during constant folding, and only per byte, and
- even then unreliably. https://wg21.link/P1886 "Error speed benchmarking" showed just how
- poorly clang and MSVC fails to optimise outcome-using code, if you manually set bits.
- Outcome v2.2 therefore uses an enum with fixed values, and constexpr manipulation functions
- to change the value to one of the enum's values. This is stupid to look at in source code,
- but it make clang's optimiser do the right thing, so it's worth it.
- */
- #define BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS 0
- enum class status : uint16_t
- {
- // WARNING: These bits are not tracked by abi-dumper, but changing them will break ABI!
- none = 0,
- have_value = (1U << 0U),
- have_error = (1U << 1U),
- have_exception = (2U << 1U),
- have_error_exception = (3U << 1U),
- // failed to complete a strong swap
- have_lost_consistency = (1U << 3U),
- have_value_lost_consistency = (1U << 0U) | (1U << 3U),
- have_error_lost_consistency = (1U << 1U) | (1U << 3U),
- have_exception_lost_consistency = (2U << 1U) | (1U << 3U),
- have_error_exception_lost_consistency = (3U << 1U) | (1U << 3U),
- // can errno be set from this error?
- have_error_is_errno = (1U << 4U),
- have_error_error_is_errno = (1U << 1U) | (1U << 4U),
- have_error_exception_error_is_errno = (3U << 1U) | (1U << 4U),
- have_error_lost_consistency_error_is_errno = (1U << 1U) | (1U << 3U) | (1U << 4U),
- have_error_exception_lost_consistency_error_is_errno = (3U << 1U) | (1U << 3U) | (1U << 4U),
- // value has been moved from
- have_moved_from = (1U << 5U)
- };
- struct status_bitfield_type
- {
- status status_value{status::none};
- uint16_t spare_storage_value{0}; // hooks::spare_storage()
- constexpr status_bitfield_type() = default;
- constexpr status_bitfield_type(status v) noexcept
- : status_value(v)
- {
- } // NOLINT
- constexpr status_bitfield_type(status v, uint16_t s) noexcept
- : status_value(v)
- , spare_storage_value(s)
- {
- }
- constexpr status_bitfield_type(const status_bitfield_type &) = default;
- constexpr status_bitfield_type(status_bitfield_type &&) = default;
- constexpr status_bitfield_type &operator=(const status_bitfield_type &) = default;
- constexpr status_bitfield_type &operator=(status_bitfield_type &&) = default;
- //~status_bitfield_type() = default; // Do NOT uncomment this, it breaks older clangs!
- constexpr bool have_value() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- return (status_value == status::have_value) //
- || (status_value == status::have_value_lost_consistency) //
- ;
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_value)) != 0;
- #endif
- }
- constexpr bool have_error() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- return (status_value == status::have_error) //
- || (status_value == status::have_error_exception) //
- || (status_value == status::have_error_lost_consistency) //
- || (status_value == status::have_error_exception_lost_consistency) //
- || (status_value == status::have_error_error_is_errno) //
- || (status_value == status::have_error_exception_error_is_errno) //
- || (status_value == status::have_error_lost_consistency_error_is_errno) //
- || (status_value == status::have_error_exception_lost_consistency_error_is_errno) //
- ;
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_error)) != 0;
- #endif
- }
- constexpr bool have_exception() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- return (status_value == status::have_exception) //
- || (status_value == status::have_error_exception) //
- || (status_value == status::have_exception_lost_consistency) //
- || (status_value == status::have_error_exception_lost_consistency) //
- || (status_value == status::have_error_exception_error_is_errno) //
- || (status_value == status::have_error_exception_lost_consistency_error_is_errno) //
- ;
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_exception)) != 0;
- #endif
- }
- constexpr bool have_lost_consistency() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- return (status_value == status::have_value_lost_consistency) //
- || (status_value == status::have_error_lost_consistency) //
- || (status_value == status::have_exception_lost_consistency) //
- || (status_value == status::have_error_lost_consistency_error_is_errno) //
- || (status_value == status::have_error_exception_lost_consistency_error_is_errno) //
- ;
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_lost_consistency)) != 0;
- #endif
- }
- constexpr bool have_error_is_errno() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- return (status_value == status::have_error_error_is_errno) //
- || (status_value == status::have_error_exception_error_is_errno) //
- || (status_value == status::have_error_lost_consistency_error_is_errno) //
- || (status_value == status::have_error_exception_lost_consistency_error_is_errno) //
- ;
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_error_is_errno)) != 0;
- #endif
- }
- constexpr bool have_moved_from() const noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- #error Fixme
- #else
- return (static_cast<uint16_t>(status_value) & static_cast<uint16_t>(status::have_moved_from)) != 0;
- #endif
- }
- constexpr status_bitfield_type &set_have_value(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- switch(status_value)
- {
- case status::none:
- if(v)
- {
- status_value = status::have_value;
- }
- break;
- case status::have_value:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_exception:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_exception:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_value_lost_consistency:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error_lost_consistency:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_exception_lost_consistency:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_exception_lost_consistency:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_error_is_errno:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_exception_error_is_errno:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_lost_consistency_error_is_errno:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_exception_lost_consistency_error_is_errno:
- if(v)
- {
- make_ub(*this);
- }
- break;
- }
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_value)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_value)));
- #endif
- return *this;
- }
- constexpr status_bitfield_type &set_have_error(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- switch(status_value)
- {
- case status::none:
- if(v)
- {
- status_value = status::have_error;
- }
- break;
- case status::have_value:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_exception:
- if(v)
- {
- status_value = status::have_error_exception;
- }
- break;
- case status::have_error_exception:
- if(!v)
- {
- status_value = status::have_exception;
- }
- break;
- case status::have_value_lost_consistency:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_lost_consistency:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_exception_lost_consistency:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency;
- }
- break;
- case status::have_error_exception_lost_consistency:
- if(!v)
- {
- status_value = status::have_exception_lost_consistency;
- }
- break;
- case status::have_error_error_is_errno:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error_exception_error_is_errno:
- if(!v)
- {
- status_value = status::have_exception;
- }
- break;
- case status::have_error_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error_exception_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_exception_lost_consistency;
- }
- break;
- }
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_error)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_error)));
- #endif
- return *this;
- }
- constexpr status_bitfield_type &set_have_exception(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- switch(status_value)
- {
- case status::none:
- if(v)
- {
- status_value = status::have_exception;
- }
- break;
- case status::have_value:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error:
- if(v)
- {
- status_value = status::have_error_exception;
- }
- break;
- case status::have_exception:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error_exception:
- if(!v)
- {
- status_value = status::have_error;
- }
- break;
- case status::have_value_lost_consistency:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_error_lost_consistency:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency;
- }
- break;
- case status::have_exception_lost_consistency:
- if(!v)
- {
- status_value = status::none;
- }
- break;
- case status::have_error_exception_lost_consistency:
- if(!v)
- {
- status_value = status::have_error_lost_consistency;
- }
- break;
- case status::have_error_error_is_errno:
- if(v)
- {
- status_value = status::have_error_exception_error_is_errno;
- }
- break;
- case status::have_error_exception_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_error_is_errno;
- }
- break;
- case status::have_error_lost_consistency_error_is_errno:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency_error_is_errno;
- }
- break;
- case status::have_error_exception_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_lost_consistency_error_is_errno;
- }
- break;
- }
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_exception)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_exception)));
- #endif
- return *this;
- }
- constexpr status_bitfield_type &set_have_error_is_errno(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- switch(status_value)
- {
- case status::none:
- make_ub(*this);
- break;
- case status::have_value:
- make_ub(*this);
- break;
- case status::have_error:
- if(v)
- {
- status_value = status::have_error_error_is_errno;
- }
- break;
- case status::have_exception:
- make_ub(*this);
- break;
- case status::have_error_exception:
- if(v)
- {
- status_value = status::have_error_exception_error_is_errno;
- }
- break;
- case status::have_value_lost_consistency:
- make_ub(*this);
- break;
- case status::have_error_lost_consistency:
- if(v)
- {
- status_value = status::have_error_lost_consistency_error_is_errno;
- }
- break;
- case status::have_exception_lost_consistency:
- make_ub(*this);
- break;
- case status::have_error_exception_lost_consistency:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency_error_is_errno;
- }
- break;
- case status::have_error_error_is_errno:
- if(!v)
- {
- status_value = status::have_error;
- }
- break;
- case status::have_error_exception_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_exception;
- }
- break;
- case status::have_error_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_lost_consistency;
- }
- break;
- case status::have_error_exception_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_exception_lost_consistency;
- }
- break;
- }
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_error_is_errno)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_error_is_errno)));
- #endif
- return *this;
- }
- constexpr status_bitfield_type &set_have_lost_consistency(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- switch(status_value)
- {
- case status::none:
- if(v)
- {
- make_ub(*this);
- }
- break;
- case status::have_value:
- if(v)
- {
- status_value = status::have_value_lost_consistency;
- }
- break;
- case status::have_error:
- if(v)
- {
- status_value = status::have_error_lost_consistency;
- }
- break;
- case status::have_exception:
- if(v)
- {
- status_value = status::have_exception_lost_consistency;
- }
- break;
- case status::have_error_exception:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency;
- }
- break;
- case status::have_value_lost_consistency:
- if(!v)
- {
- status_value = status::have_value;
- }
- break;
- case status::have_error_lost_consistency:
- if(!v)
- {
- status_value = status::have_error;
- }
- break;
- case status::have_exception_lost_consistency:
- if(!v)
- {
- status_value = status::have_exception;
- }
- break;
- case status::have_error_exception_lost_consistency:
- if(!v)
- {
- status_value = status::have_error_exception;
- }
- break;
- case status::have_error_error_is_errno:
- if(v)
- {
- status_value = status::have_error_lost_consistency_error_is_errno;
- }
- break;
- case status::have_error_exception_error_is_errno:
- if(v)
- {
- status_value = status::have_error_exception_lost_consistency_error_is_errno;
- }
- break;
- case status::have_error_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_exception_error_is_errno;
- }
- break;
- case status::have_error_exception_lost_consistency_error_is_errno:
- if(!v)
- {
- status_value = status::have_error_exception_error_is_errno;
- }
- break;
- }
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_lost_consistency)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_lost_consistency)));
- #endif
- return *this;
- }
- constexpr status_bitfield_type &set_have_moved_from(bool v) noexcept
- {
- #if BOOST_OUTCOME_USE_CONSTEXPR_ENUM_STATUS
- #error Fixme
- #else
- status_value = static_cast<status>(v ? (static_cast<uint16_t>(status_value) | static_cast<uint16_t>(status::have_moved_from)) :
- (static_cast<uint16_t>(status_value) & ~static_cast<uint16_t>(status::have_moved_from)));
- #endif
- return *this;
- }
- };
- #if !defined(NDEBUG)
- // Check is trivial in all ways except default constructibility
- static_assert(sizeof(status_bitfield_type) == 4, "status_bitfield_type is not sized 4 bytes!");
- static_assert(std::is_trivially_copyable<status_bitfield_type>::value, "status_bitfield_type is not trivially copyable!");
- static_assert(std::is_trivially_assignable<status_bitfield_type, status_bitfield_type>::value, "status_bitfield_type is not trivially assignable!");
- static_assert(std::is_trivially_destructible<status_bitfield_type>::value, "status_bitfield_type is not trivially destructible!");
- static_assert(std::is_trivially_copy_constructible<status_bitfield_type>::value, "status_bitfield_type is not trivially copy constructible!");
- static_assert(std::is_trivially_move_constructible<status_bitfield_type>::value, "status_bitfield_type is not trivially move constructible!");
- static_assert(std::is_trivially_copy_assignable<status_bitfield_type>::value, "status_bitfield_type is not trivially copy assignable!");
- static_assert(std::is_trivially_move_assignable<status_bitfield_type>::value, "status_bitfield_type is not trivially move assignable!");
- // Also check is standard layout
- static_assert(std::is_standard_layout<status_bitfield_type>::value, "status_bitfield_type is not a standard layout type!");
- #endif
- // Used if T is trivial
- template <class T> struct value_storage_trivial
- {
- using value_type = T;
- union {
- empty_type _empty;
- devoid<T> _value;
- };
- status_bitfield_type _status;
- constexpr value_storage_trivial() noexcept
- : _empty{}
- {
- }
- // Special from-void catchall constructor, always constructs default T irrespective of whether void is valued or not (can do no better if T cannot be
- // copied)
- struct disable_void_catchall
- {
- };
- using void_value_storage_trivial = std::conditional_t<std::is_void<T>::value, disable_void_catchall, value_storage_trivial<void>>;
- explicit constexpr value_storage_trivial(const void_value_storage_trivial &o) noexcept(std::is_nothrow_default_constructible<value_type>::value)
- : _value()
- , _status(o._status)
- {
- }
- value_storage_trivial(const value_storage_trivial &) = default; // NOLINT
- value_storage_trivial(value_storage_trivial &&) = default; // NOLINT
- value_storage_trivial &operator=(const value_storage_trivial &) = default; // NOLINT
- value_storage_trivial &operator=(value_storage_trivial &&) = default; // NOLINT
- ~value_storage_trivial() = default;
- constexpr explicit value_storage_trivial(status_bitfield_type status)
- : _empty()
- , _status(status)
- {
- }
- template <class... Args>
- constexpr explicit value_storage_trivial(in_place_type_t<value_type> /*unused*/,
- Args &&... args) noexcept(std::is_nothrow_constructible<value_type, Args...>::value)
- : _value(static_cast<Args &&>(args)...)
- , _status(status::have_value)
- {
- }
- template <class U, class... Args>
- constexpr value_storage_trivial(in_place_type_t<value_type> /*unused*/, std::initializer_list<U> il,
- Args &&... args) noexcept(std::is_nothrow_constructible<value_type, std::initializer_list<U>, Args...>::value)
- : _value(il, static_cast<Args &&>(args)...)
- , _status(status::have_value)
- {
- }
- template <class U>
- static constexpr bool enable_converting_constructor = !std::is_same<std::decay_t<U>, value_type>::value && std::is_constructible<value_type, U>::value;
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_trivial(const value_storage_trivial<U> &o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_trivial(o._status.have_value() ? value_storage_trivial(in_place_type<value_type>, o._value) : value_storage_trivial()) // NOLINT
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_trivial(value_storage_trivial<U> &&o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_trivial(o._status.have_value() ? value_storage_trivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- value_storage_trivial()) // NOLINT
- {
- _status = o._status;
- }
- constexpr void swap(value_storage_trivial &o) noexcept
- {
- // storage is trivial, so just use assignment
- auto temp = static_cast<value_storage_trivial &&>(*this);
- *this = static_cast<value_storage_trivial &&>(o);
- o = static_cast<value_storage_trivial &&>(temp);
- }
- };
- // Used if T is non-trivial
- template <class T> struct value_storage_nontrivial
- {
- using value_type = T;
- union {
- empty_type _empty;
- value_type _value;
- };
- status_bitfield_type _status;
- value_storage_nontrivial() noexcept
- : _empty{}
- {
- }
- value_storage_nontrivial &operator=(const value_storage_nontrivial &) = default; // if reaches here, copy assignment is trivial
- value_storage_nontrivial &operator=(value_storage_nontrivial &&) = default; // NOLINT if reaches here, move assignment is trivial
- value_storage_nontrivial(value_storage_nontrivial &&o) noexcept(std::is_nothrow_move_constructible<value_type>::value) // NOLINT
- : _status(o._status)
- {
- if(this->_status.have_value())
- {
- this->_status.set_have_value(false);
- new(&_value) value_type(static_cast<value_type &&>(o._value)); // NOLINT
- _status = o._status;
- }
- }
- value_storage_nontrivial(const value_storage_nontrivial &o) noexcept(std::is_nothrow_copy_constructible<value_type>::value)
- : _status(o._status)
- {
- if(this->_status.have_value())
- {
- this->_status.set_have_value(false);
- new(&_value) value_type(o._value); // NOLINT
- _status = o._status;
- }
- }
- // Special from-void constructor, constructs default T if void valued
- explicit value_storage_nontrivial(const value_storage_trivial<void> &o) noexcept(std::is_nothrow_default_constructible<value_type>::value)
- : _status(o._status)
- {
- if(this->_status.have_value())
- {
- this->_status.set_have_value(false);
- new(&_value) value_type; // NOLINT
- _status = o._status;
- }
- }
- explicit value_storage_nontrivial(status_bitfield_type status)
- : _empty()
- , _status(status)
- {
- }
- template <class... Args>
- explicit value_storage_nontrivial(in_place_type_t<value_type> /*unused*/,
- Args &&... args) noexcept(std::is_nothrow_constructible<value_type, Args...>::value)
- : _value(static_cast<Args &&>(args)...) // NOLINT
- , _status(status::have_value)
- {
- }
- template <class U, class... Args>
- value_storage_nontrivial(in_place_type_t<value_type> /*unused*/, std::initializer_list<U> il,
- Args &&... args) noexcept(std::is_nothrow_constructible<value_type, std::initializer_list<U>, Args...>::value)
- : _value(il, static_cast<Args &&>(args)...)
- , _status(status::have_value)
- {
- }
- template <class U>
- static constexpr bool enable_converting_constructor = !std::is_same<std::decay_t<U>, value_type>::value && std::is_constructible<value_type, U>::value;
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(const value_storage_nontrivial<U> &o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_nontrivial(o._status.have_value() ? value_storage_nontrivial(in_place_type<value_type>, o._value) : value_storage_nontrivial())
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(const value_storage_trivial<U> &o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_nontrivial(o._status.have_value() ? value_storage_nontrivial(in_place_type<value_type>, o._value) : value_storage_nontrivial())
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(value_storage_nontrivial<U> &&o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_nontrivial(o._status.have_value() ? value_storage_nontrivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- value_storage_nontrivial())
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(value_storage_trivial<U> &&o) noexcept(std::is_nothrow_constructible<value_type, U>::value)
- : value_storage_nontrivial(o._status.have_value() ? value_storage_nontrivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- value_storage_nontrivial())
- {
- _status = o._status;
- }
- ~value_storage_nontrivial() noexcept(std::is_nothrow_destructible<T>::value)
- {
- if(this->_status.have_value())
- {
- this->_value.~value_type(); // NOLINT
- this->_status.set_have_value(false);
- }
- }
- constexpr void swap(value_storage_nontrivial &o) noexcept(detail::is_nothrow_swappable<value_type>::value)
- {
- using std::swap;
- if(!_status.have_value() && !o._status.have_value())
- {
- swap(_status, o._status);
- return;
- }
- if(_status.have_value() && o._status.have_value())
- {
- struct _
- {
- status_bitfield_type &a, &b;
- bool all_good{false};
- ~_()
- {
- if(!all_good)
- {
- // We lost one of the values
- a.set_have_lost_consistency(true);
- b.set_have_lost_consistency(true);
- }
- }
- } _{_status, o._status};
- strong_swap(_.all_good, _value, o._value);
- swap(_status, o._status);
- return;
- }
- // One must be empty and the other non-empty, so use move construction
- if(_status.have_value())
- {
- // Move construct me into other
- new(&o._value) value_type(static_cast<value_type &&>(_value)); // NOLINT
- this->_value.~value_type(); // NOLINT
- swap(_status, o._status);
- }
- else
- {
- // Move construct other into me
- new(&_value) value_type(static_cast<value_type &&>(o._value)); // NOLINT
- o._value.~value_type(); // NOLINT
- swap(_status, o._status);
- }
- }
- };
- template <class Base> struct value_storage_delete_copy_constructor : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_delete_copy_constructor() = default;
- value_storage_delete_copy_constructor(const value_storage_delete_copy_constructor &) = delete;
- value_storage_delete_copy_constructor(value_storage_delete_copy_constructor &&) = default; // NOLINT
- };
- template <class Base> struct value_storage_delete_copy_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_delete_copy_assignment() = default;
- value_storage_delete_copy_assignment(const value_storage_delete_copy_assignment &) = default;
- value_storage_delete_copy_assignment(value_storage_delete_copy_assignment &&) = default; // NOLINT
- value_storage_delete_copy_assignment &operator=(const value_storage_delete_copy_assignment &o) = delete;
- value_storage_delete_copy_assignment &operator=(value_storage_delete_copy_assignment &&o) = default; // NOLINT
- };
- template <class Base> struct value_storage_delete_move_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_delete_move_assignment() = default;
- value_storage_delete_move_assignment(const value_storage_delete_move_assignment &) = default;
- value_storage_delete_move_assignment(value_storage_delete_move_assignment &&) = default; // NOLINT
- value_storage_delete_move_assignment &operator=(const value_storage_delete_move_assignment &o) = default;
- value_storage_delete_move_assignment &operator=(value_storage_delete_move_assignment &&o) = delete;
- };
- template <class Base> struct value_storage_delete_move_constructor : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_delete_move_constructor() = default;
- value_storage_delete_move_constructor(const value_storage_delete_move_constructor &) = default;
- value_storage_delete_move_constructor(value_storage_delete_move_constructor &&) = delete;
- };
- template <class Base> struct value_storage_nontrivial_move_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_nontrivial_move_assignment() = default;
- value_storage_nontrivial_move_assignment(const value_storage_nontrivial_move_assignment &) = default;
- value_storage_nontrivial_move_assignment(value_storage_nontrivial_move_assignment &&) = default; // NOLINT
- value_storage_nontrivial_move_assignment &operator=(const value_storage_nontrivial_move_assignment &o) = default;
- value_storage_nontrivial_move_assignment &
- operator=(value_storage_nontrivial_move_assignment &&o) noexcept(std::is_nothrow_move_assignable<value_type>::value) // NOLINT
- {
- if(this->_status.have_value() && o._status.have_value())
- {
- this->_value = static_cast<value_type &&>(o._value); // NOLINT
- }
- else if(this->_status.have_value() && !o._status.have_value())
- {
- this->_value.~value_type(); // NOLINT
- }
- else if(!this->_status.have_value() && o._status.have_value())
- {
- new(&this->_value) value_type(static_cast<value_type &&>(o._value)); // NOLINT
- }
- this->_status = o._status;
- return *this;
- }
- };
- template <class Base> struct value_storage_nontrivial_copy_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- value_storage_nontrivial_copy_assignment() = default;
- value_storage_nontrivial_copy_assignment(const value_storage_nontrivial_copy_assignment &) = default;
- value_storage_nontrivial_copy_assignment(value_storage_nontrivial_copy_assignment &&) = default; // NOLINT
- value_storage_nontrivial_copy_assignment &operator=(value_storage_nontrivial_copy_assignment &&o) = default; // NOLINT
- value_storage_nontrivial_copy_assignment &
- operator=(const value_storage_nontrivial_copy_assignment &o) noexcept(std::is_nothrow_copy_assignable<value_type>::value)
- {
- if(this->_status.have_value() && o._status.have_value())
- {
- this->_value = o._value; // NOLINT
- }
- else if(this->_status.have_value() && !o._status.have_value())
- {
- this->_value.~value_type(); // NOLINT
- }
- else if(!this->_status.have_value() && o._status.have_value())
- {
- new(&this->_value) value_type(o._value); // NOLINT
- }
- this->_status = o._status;
- return *this;
- }
- };
- // We don't actually need all of std::is_trivial<>, std::is_trivially_copyable<> is sufficient
- template <class T>
- using value_storage_select_trivality =
- std::conditional_t<std::is_trivially_copyable<devoid<T>>::value, value_storage_trivial<T>, value_storage_nontrivial<T>>;
- template <class T>
- using value_storage_select_move_constructor = std::conditional_t<std::is_move_constructible<devoid<T>>::value, value_storage_select_trivality<T>,
- value_storage_delete_move_constructor<value_storage_select_trivality<T>>>;
- template <class T>
- using value_storage_select_copy_constructor = std::conditional_t<std::is_copy_constructible<devoid<T>>::value, value_storage_select_move_constructor<T>,
- value_storage_delete_copy_constructor<value_storage_select_move_constructor<T>>>;
- template <class T>
- using value_storage_select_move_assignment = std::conditional_t<
- std::is_trivially_move_assignable<devoid<T>>::value, value_storage_select_copy_constructor<T>,
- std::conditional_t<std::is_move_assignable<devoid<T>>::value, value_storage_nontrivial_move_assignment<value_storage_select_copy_constructor<T>>,
- value_storage_delete_copy_assignment<value_storage_select_copy_constructor<T>>>>;
- template <class T>
- using value_storage_select_copy_assignment = std::conditional_t<
- std::is_trivially_copy_assignable<devoid<T>>::value, value_storage_select_move_assignment<T>,
- std::conditional_t<std::is_copy_assignable<devoid<T>>::value, value_storage_nontrivial_copy_assignment<value_storage_select_move_assignment<T>>,
- value_storage_delete_copy_assignment<value_storage_select_move_assignment<T>>>>;
- template <class T> using value_storage_select_impl = value_storage_select_copy_assignment<T>;
- #ifndef NDEBUG
- // Check is trivial in all ways except default constructibility
- // static_assert(std::is_trivial<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivial!");
- // static_assert(std::is_trivially_default_constructible<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivially default
- // constructible!");
- static_assert(std::is_trivially_copyable<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivially copyable!");
- static_assert(std::is_trivially_assignable<value_storage_select_impl<int>, value_storage_select_impl<int>>::value,
- "value_storage_select_impl<int> is not trivially assignable!");
- static_assert(std::is_trivially_destructible<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivially destructible!");
- static_assert(std::is_trivially_copy_constructible<value_storage_select_impl<int>>::value,
- "value_storage_select_impl<int> is not trivially copy constructible!");
- static_assert(std::is_trivially_move_constructible<value_storage_select_impl<int>>::value,
- "value_storage_select_impl<int> is not trivially move constructible!");
- static_assert(std::is_trivially_copy_assignable<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivially copy assignable!");
- static_assert(std::is_trivially_move_assignable<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not trivially move assignable!");
- // Also check is standard layout
- static_assert(std::is_standard_layout<value_storage_select_impl<int>>::value, "value_storage_select_impl<int> is not a standard layout type!");
- #endif
- } // namespace detail
- BOOST_OUTCOME_V2_NAMESPACE_END
- #endif
|