123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734 |
- /* Essentially an internal optional implementation :)
- (C) 2017-2024 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"
- BOOST_OUTCOME_V2_NAMESPACE_EXPORT_BEGIN
- namespace detail
- {
- // Helpers for move assigning to empty storage
- template <class T, bool isCopyOrMoveConstructible = std::is_copy_constructible<T>::value || std::is_move_constructible<T>::value,
- bool isDefaultConstructibleAndCopyOrMoveAssignable =
- std::is_default_constructible<T>::value && (std::is_copy_assignable<T>::value || std::is_move_assignable<T>::value)>
- struct move_assign_to_empty;
- // Prefer to use move or copy construction
- template <class T> struct move_assign_to_empty<T, true, false>
- {
- move_assign_to_empty(T *dest, T *o) noexcept(std::is_nothrow_move_constructible<T>::value) { new(dest) T(static_cast<T &&>(*o)); }
- };
- template <class T> struct move_assign_to_empty<T, true, true>
- {
- move_assign_to_empty(T *dest, T *o) noexcept(std::is_nothrow_move_constructible<T>::value) { new(dest) T(static_cast<T &&>(*o)); }
- };
- // But fall back on default construction and move assign if necessary
- template <class T> struct move_assign_to_empty<T, false, true>
- {
- move_assign_to_empty(T *dest, T *o) noexcept(std::is_nothrow_default_constructible<T>::value && std::is_nothrow_move_assignable<T>::value)
- {
- new(dest) T;
- *dest = static_cast<T &&>(*o);
- }
- };
- // Void does nothing
- template <> struct move_assign_to_empty<void, false, false>
- {
- move_assign_to_empty(void *, void *) noexcept { /* nothing to assign */ }
- };
- template <> struct move_assign_to_empty<const void, false, false>
- {
- move_assign_to_empty(const void *, const void *) noexcept { /* nothing to assign */ }
- };
- // Helpers for copy assigning to empty storage
- template <class T, bool isCopyConstructible = std::is_copy_constructible<T>::value,
- bool isDefaultConstructibleAndCopyAssignable = std::is_default_constructible<T>::value && std::is_copy_assignable<T>::value>
- struct copy_assign_to_empty;
- // Prefer to use copy construction
- template <class T> struct copy_assign_to_empty<T, true, false>
- {
- copy_assign_to_empty(T *dest, const T *o) noexcept(std::is_nothrow_copy_constructible<T>::value) { new(dest) T(*o); }
- };
- template <class T> struct copy_assign_to_empty<T, true, true>
- {
- copy_assign_to_empty(T *dest, const T *o) noexcept(std::is_nothrow_copy_constructible<T>::value) { new(dest) T(*o); }
- };
- // But fall back on default construction and copy assign if necessary
- template <class T> struct copy_assign_to_empty<T, false, true>
- {
- copy_assign_to_empty(T *dest, const T *o) noexcept(std::is_nothrow_default_constructible<T>::value && std::is_nothrow_copy_assignable<T>::value)
- {
- new(dest) T;
- *dest = *o;
- }
- };
- // Void does nothing
- template <> struct copy_assign_to_empty<void, false, false>
- {
- copy_assign_to_empty(void *, void *) noexcept { /* nothing to assign */ }
- };
- template <> struct copy_assign_to_empty<const void, false, false>
- {
- copy_assign_to_empty(const void *, const void *) noexcept { /* nothing to assign */ }
- };
- 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);
- }
- };
- template <class T, bool nothrow> struct strong_placement_impl
- {
- template <class F> constexpr strong_placement_impl(bool &allgood, T *a, T *b, F &&f)
- {
- allgood = true;
- new(a) T(static_cast<T &&>(*b));
- b->~T();
- f();
- }
- };
- #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
- }
- }
- };
- template <class T> struct strong_placement_impl<T, false>
- {
- template <class F> strong_placement_impl(bool &allgood, T *a, T *b, F &&f)
- {
- new(a) T(static_cast<T &&>(*b));
- try
- {
- b->~T();
- f();
- }
- catch(...)
- {
- // Try to put back a, but only if we are still good
- if(allgood)
- {
- try
- {
- new(b) T(static_cast<T &&>(*a));
- // 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);
- }
- /*!
- */
- BOOST_OUTCOME_TEMPLATE(class T, class F)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(std::is_move_constructible<T>::value &&std::is_move_assignable<T>::value))
- constexpr inline void strong_placement(bool &allgood, T *a, T *b, F &&f) noexcept(std::is_nothrow_move_constructible<T>::value)
- {
- detail::strong_placement_impl<T, std::is_nothrow_move_constructible<T>::value>(allgood, a, b, static_cast<F &&>(f));
- }
- namespace detail
- {
- template <class T>
- constexpr
- #ifdef _MSC_VER
- __declspec(noreturn)
- #elif defined(__GNUC__) || defined(__clang__)
- __attribute__((noreturn))
- #endif
- void
- make_ub(T && /*unused*/)
- {
- BOOST_OUTCOME_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
- template <class State> constexpr inline void _set_error_is_errno(State & /*unused*/) {}
- #ifdef _MSC_VER
- #pragma warning(push)
- #pragma warning(disable : 4127) // conditional expression is constant
- #pragma warning(disable : 4624) // destructor was implicitly defined as deleted
- #endif
- // Used if both T and E are trivial
- template <class T, class E> struct value_storage_trivial
- {
- using value_type = T;
- using error_type = E;
- // Disable in place construction if they are the same type
- struct disable_in_place_value_type
- {
- };
- struct disable_in_place_error_type
- {
- };
- using _value_type = std::conditional_t<std::is_same<value_type, error_type>::value, disable_in_place_value_type, value_type>;
- using _error_type = std::conditional_t<std::is_same<value_type, error_type>::value, disable_in_place_error_type, error_type>;
- using _value_type_ = devoid<value_type>;
- using _error_type_ = devoid<error_type>;
- union
- {
- empty_type _empty;
- _value_type_ _value;
- _error_type_ _error;
- };
- status_bitfield_type _status;
- constexpr value_storage_trivial() noexcept
- : _empty{}
- {
- }
- 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(detail::is_nothrow_constructible<_value_type_, Args...>)
- : _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(detail::is_nothrow_constructible<_value_type_, std::initializer_list<U>, Args...>)
- : _value(il, static_cast<Args &&>(args)...)
- , _status(status::have_value)
- {
- }
- template <class... Args>
- constexpr explicit value_storage_trivial(in_place_type_t<_error_type> /*unused*/,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_error_type_, Args...>)
- : _error(static_cast<Args &&>(args)...)
- , _status(status::have_error)
- {
- _set_error_is_errno(*this);
- }
- template <class U, class... Args>
- constexpr value_storage_trivial(in_place_type_t<_error_type> /*unused*/, std::initializer_list<U> il,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_error_type_, std::initializer_list<U>, Args...>)
- : _error(il, static_cast<Args &&>(args)...)
- , _status(status::have_error)
- {
- _set_error_is_errno(*this);
- }
- struct nonvoid_converting_constructor_tag
- {
- };
- template <class U, class V>
- static constexpr bool enable_nonvoid_converting_constructor =
- !(std::is_same<std::decay_t<U>, value_type>::value && std::is_same<std::decay_t<V>, error_type>::value) //
- && detail::is_constructible<value_type, U> && detail::is_constructible<error_type, V>;
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_trivial(const value_storage_trivial<U, V> &o,
- nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(detail::is_nothrow_constructible<_value_type_, U> &&
- detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_trivial(o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>, o._value) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>, o._error) : value_storage_trivial())) // NOLINT
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_trivial(value_storage_trivial<U, V> &&o,
- nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(detail::is_nothrow_constructible<_value_type_, U> &&
- detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_trivial(
- o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>, static_cast<V &&>(o._error)) : value_storage_trivial())) // NOLINT
- {
- _status = o._status;
- }
- struct void_value_converting_constructor_tag
- {
- };
- template <class V>
- static constexpr bool enable_void_value_converting_constructor =
- std::is_default_constructible<value_type>::value && detail::is_constructible<error_type, V>;
- BOOST_OUTCOME_TEMPLATE(class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_value_converting_constructor<V>))
- constexpr explicit value_storage_trivial(const value_storage_trivial<void, V> &o, void_value_converting_constructor_tag /*unused*/ = {}) noexcept(
- std::is_nothrow_default_constructible<_value_type_>::value && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_trivial(o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>, o._error) : value_storage_trivial())) // NOLINT
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_value_converting_constructor<V>))
- constexpr explicit value_storage_trivial(value_storage_trivial<void, V> &&o, void_value_converting_constructor_tag /*unused*/ = {}) noexcept(
- std::is_nothrow_default_constructible<_value_type_>::value && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_trivial(
- o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>, static_cast<V &&>(o._error)) : value_storage_trivial())) // NOLINT
- {
- _status = o._status;
- }
- struct void_error_converting_constructor_tag
- {
- };
- template <class U>
- static constexpr bool enable_void_error_converting_constructor =
- std::is_default_constructible<error_type>::value && detail::is_constructible<value_type, U>;
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_error_converting_constructor<U>))
- constexpr explicit value_storage_trivial(const value_storage_trivial<U, void> &o, void_error_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && std::is_nothrow_default_constructible<_error_type_>::value)
- : value_storage_trivial(o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>, o._value) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>) : value_storage_trivial())) // NOLINT
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_error_converting_constructor<U>))
- constexpr explicit value_storage_trivial(value_storage_trivial<U, void> &&o, void_error_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && std::is_nothrow_default_constructible<_error_type_>::value)
- : value_storage_trivial(o._status.have_value() ?
- value_storage_trivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- (o._status.have_error() ? value_storage_trivial(in_place_type<error_type>) : 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 or E is non-trivial. The additional constexpr is injected in C++ 20 to enable Outcome to
- work in constexpr evaluation contexts in C++ 20 where non-trivial constexpr destructors are now allowed.
- */
- template <class T, class E> struct value_storage_nontrivial
- {
- using value_type = T;
- using error_type = E;
- struct disable_in_place_value_type
- {
- };
- struct disable_in_place_error_type
- {
- };
- using _value_type = std::conditional_t<std::is_same<value_type, error_type>::value, disable_in_place_value_type, value_type>;
- using _error_type = std::conditional_t<std::is_same<value_type, error_type>::value, disable_in_place_error_type, error_type>;
- using _value_type_ = devoid<value_type>;
- using _error_type_ = devoid<error_type>;
- union
- {
- empty_type _empty1;
- _value_type_ _value;
- };
- status_bitfield_type _status;
- union
- {
- empty_type _empty2;
- _error_type_ _error;
- };
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- value_storage_nontrivial() noexcept
- : _empty1{}
- , _empty2{}
- {
- }
- 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
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- value_storage_nontrivial(value_storage_nontrivial &&o) noexcept(std::is_nothrow_move_constructible<_value_type_>::value &&
- std::is_nothrow_move_constructible<_error_type_>::value) // NOLINT
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(static_cast<_value_type_ &&>(o._value)); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(static_cast<_error_type_ &&>(o._error)); // NOLINT
- }
- _status = o._status;
- o._status.set_have_moved_from(true);
- }
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- value_storage_nontrivial(const value_storage_nontrivial &o) noexcept(std::is_nothrow_copy_constructible<_value_type_>::value &&
- std::is_nothrow_copy_constructible<_error_type_>::value)
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(o._value); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(o._error); // NOLINT
- }
- _status = o._status;
- }
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- explicit value_storage_nontrivial(status_bitfield_type status)
- : _empty1()
- , _status(status)
- , _empty2()
- {
- }
- template <class... Args>
- constexpr explicit value_storage_nontrivial(in_place_type_t<_value_type> /*unused*/,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_value_type_, Args...>)
- : _value(static_cast<Args &&>(args)...) // NOLINT
- , _status(status::have_value)
- {
- }
- template <class U, class... Args>
- constexpr value_storage_nontrivial(in_place_type_t<_value_type> /*unused*/, std::initializer_list<U> il,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_value_type_, std::initializer_list<U>, Args...>)
- : _value(il, static_cast<Args &&>(args)...)
- , _status(status::have_value)
- {
- }
- template <class... Args>
- constexpr explicit value_storage_nontrivial(in_place_type_t<_error_type> /*unused*/,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_error_type_, Args...>)
- : _status(status::have_error)
- , _error(static_cast<Args &&>(args)...) // NOLINT
- {
- _set_error_is_errno(*this);
- }
- template <class U, class... Args>
- constexpr value_storage_nontrivial(in_place_type_t<_error_type> /*unused*/, std::initializer_list<U> il,
- Args &&...args) noexcept(detail::is_nothrow_constructible<_error_type_, std::initializer_list<U>, Args...>)
- : _status(status::have_error)
- , _error(il, static_cast<Args &&>(args)...)
- {
- _set_error_is_errno(*this);
- }
- struct nonvoid_converting_constructor_tag
- {
- };
- template <class U, class V>
- static constexpr bool enable_nonvoid_converting_constructor =
- !(std::is_same<std::decay_t<U>, value_type>::value && std::is_same<std::decay_t<V>, error_type>::value) //
- && detail::is_constructible<value_type, U> && detail::is_constructible<error_type, V>;
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_nontrivial(const value_storage_trivial<U, V> &o, nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_nontrivial(o._status.have_value() ?
- value_storage_nontrivial(in_place_type<value_type>, o._value) :
- (o._status.have_error() ? value_storage_nontrivial(in_place_type<error_type>, o._error) : value_storage_nontrivial()))
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_nontrivial(value_storage_trivial<U, V> &&o, nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_nontrivial(
- o._status.have_value() ?
- value_storage_nontrivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- (o._status.have_error() ? value_storage_nontrivial(in_place_type<error_type>, static_cast<V &&>(o._error)) : value_storage_nontrivial()))
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_nontrivial(const value_storage_nontrivial<U, V> &o, nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_nontrivial(o._status.have_value() ?
- value_storage_nontrivial(in_place_type<value_type>, o._value) :
- (o._status.have_error() ? value_storage_nontrivial(in_place_type<error_type>, o._error) : value_storage_nontrivial()))
- {
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U, class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_nonvoid_converting_constructor<U, V>))
- constexpr explicit value_storage_nontrivial(value_storage_nontrivial<U, V> &&o, nonvoid_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && detail::is_nothrow_constructible<_error_type_, V>)
- : value_storage_nontrivial(
- o._status.have_value() ?
- value_storage_nontrivial(in_place_type<value_type>, static_cast<U &&>(o._value)) :
- (o._status.have_error() ? value_storage_nontrivial(in_place_type<error_type>, static_cast<V &&>(o._error)) : value_storage_nontrivial()))
- {
- _status = o._status;
- }
- struct void_value_converting_constructor_tag
- {
- };
- template <class V>
- static constexpr bool enable_void_value_converting_constructor =
- std::is_default_constructible<value_type>::value && detail::is_constructible<error_type, V>;
- BOOST_OUTCOME_TEMPLATE(class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_value_converting_constructor<V>))
- constexpr explicit value_storage_nontrivial(const value_storage_trivial<void, V> &o, void_value_converting_constructor_tag /*unused*/ = {}) noexcept(
- std::is_nothrow_default_constructible<_value_type_>::value && detail::is_nothrow_constructible<_error_type_, V>)
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(o._error); // NOLINT
- }
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class V)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_value_converting_constructor<V>))
- constexpr explicit value_storage_nontrivial(value_storage_trivial<void, V> &&o, void_value_converting_constructor_tag /*unused*/ = {}) noexcept(
- std::is_nothrow_default_constructible<_value_type_>::value && detail::is_nothrow_constructible<_error_type_, V>)
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(static_cast<_error_type_ &&>(o._error)); // NOLINT
- }
- _status = o._status;
- o._status.set_have_moved_from(true);
- }
- struct void_error_converting_constructor_tag
- {
- };
- template <class U>
- static constexpr bool enable_void_error_converting_constructor =
- std::is_default_constructible<error_type>::value && detail::is_constructible<value_type, U>;
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_error_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(const value_storage_trivial<U, void> &o, void_error_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && std::is_nothrow_default_constructible<_error_type_>::value)
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(o._value); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(); // NOLINT
- }
- _status = o._status;
- }
- BOOST_OUTCOME_TEMPLATE(class U)
- BOOST_OUTCOME_TREQUIRES(BOOST_OUTCOME_TPRED(enable_void_error_converting_constructor<U>))
- constexpr explicit value_storage_nontrivial(value_storage_trivial<U, void> &&o, void_error_converting_constructor_tag /*unused*/ = {}) noexcept(
- detail::is_nothrow_constructible<_value_type_, U> && std::is_nothrow_default_constructible<_error_type_>::value)
- {
- if(o._status.have_value())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(static_cast<_value_type_ &&>(o._value)); // NOLINT
- }
- else if(o._status.have_error())
- {
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(); // NOLINT
- }
- _status = o._status;
- o._status.set_have_moved_from(true);
- }
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- ~value_storage_nontrivial() noexcept(std::is_nothrow_destructible<_value_type_>::value && std::is_nothrow_destructible<_error_type_>::value)
- {
- if(this->_status.have_value())
- {
- if(!trait::is_move_bitcopying<value_type>::value || !this->_status.have_moved_from())
- {
- this->_value.~_value_type_(); // NOLINT
- }
- this->_status.set_have_value(false);
- }
- else if(this->_status.have_error())
- {
- if(!trait::is_move_bitcopying<error_type>::value || !this->_status.have_moved_from())
- {
- this->_error.~_error_type_(); // NOLINT
- }
- this->_status.set_have_error(false);
- }
- }
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- void
- swap(value_storage_nontrivial &o) noexcept(detail::is_nothrow_swappable<_value_type_>::value && detail::is_nothrow_swappable<_error_type_>::value)
- {
- using std::swap;
- // empty/empty
- if(!_status.have_value() && !o._status.have_value() && !_status.have_error() && !o._status.have_error())
- {
- swap(_status, o._status);
- return;
- }
- // value/value
- if(_status.have_value() && o._status.have_value())
- {
- struct some_type
- {
- status_bitfield_type &a, &b;
- bool all_good{false};
- ~some_type()
- {
- if(!this->all_good)
- {
- // We lost one of the values
- this->a.set_have_lost_consistency(true);
- this->b.set_have_lost_consistency(true);
- }
- }
- } some_type_value{_status, o._status};
- strong_swap(some_type_value.all_good, _value, o._value);
- swap(_status, o._status);
- return;
- }
- // error/error
- if(_status.have_error() && o._status.have_error())
- {
- struct some_type
- {
- status_bitfield_type &a, &b;
- bool all_good{false};
- ~some_type()
- {
- if(!this->all_good)
- {
- // We lost one of the values
- this->a.set_have_lost_consistency(true);
- this->b.set_have_lost_consistency(true);
- }
- }
- } some_type_value{_status, o._status};
- strong_swap(some_type_value.all_good, _error, o._error);
- swap(_status, o._status);
- return;
- }
- // Could be value/empty, error/empty, etc
- if(_status.have_value() && !o._status.have_error())
- {
- // Move construct me into other
- new(BOOST_OUTCOME_ADDRESS_OF(o._value)) _value_type_(static_cast<_value_type_ &&>(_value)); // NOLINT
- if(!trait::is_move_bitcopying<value_type>::value)
- {
- this->_value.~value_type(); // NOLINT
- }
- swap(_status, o._status);
- return;
- }
- if(o._status.have_value() && !_status.have_error())
- {
- // Move construct other into me
- new(BOOST_OUTCOME_ADDRESS_OF(_value)) _value_type_(static_cast<_value_type_ &&>(o._value)); // NOLINT
- if(!trait::is_move_bitcopying<value_type>::value)
- {
- o._value.~value_type(); // NOLINT
- }
- swap(_status, o._status);
- return;
- }
- if(_status.have_error() && !o._status.have_value())
- {
- // Move construct me into other
- new(BOOST_OUTCOME_ADDRESS_OF(o._error)) _error_type_(static_cast<_error_type_ &&>(_error)); // NOLINT
- if(!trait::is_move_bitcopying<error_type>::value)
- {
- this->_error.~error_type(); // NOLINT
- }
- swap(_status, o._status);
- return;
- }
- if(o._status.have_error() && !_status.have_value())
- {
- // Move construct other into me
- new(BOOST_OUTCOME_ADDRESS_OF(_error)) _error_type_(static_cast<_error_type_ &&>(o._error)); // NOLINT
- if(!trait::is_move_bitcopying<error_type>::value)
- {
- o._error.~error_type(); // NOLINT
- }
- swap(_status, o._status);
- return;
- }
- // It can now only be value/error, or error/value
- struct some_type
- {
- status_bitfield_type &a, &b;
- _value_type_ *value, *o_value;
- _error_type_ *error, *o_error;
- bool all_good{true};
- ~some_type()
- {
- if(!this->all_good)
- {
- // We lost one of the values
- this->a.set_have_lost_consistency(true);
- this->b.set_have_lost_consistency(true);
- }
- }
- } some_type_value{_status, o._status, BOOST_OUTCOME_ADDRESS_OF(_value), BOOST_OUTCOME_ADDRESS_OF(o._value), BOOST_OUTCOME_ADDRESS_OF(_error), BOOST_OUTCOME_ADDRESS_OF(o._error)};
- if(_status.have_value() && o._status.have_error())
- {
- strong_placement(some_type_value.all_good, some_type_value.o_value, some_type_value.value, [&some_type_value] { //
- strong_placement(some_type_value.all_good, some_type_value.error, some_type_value.o_error, [&some_type_value] { //
- swap(some_type_value.a, some_type_value.b); //
- });
- });
- return;
- }
- if(_status.have_error() && o._status.have_value())
- {
- strong_placement(some_type_value.all_good, some_type_value.o_error, some_type_value.error, [&some_type_value] { //
- strong_placement(some_type_value.all_good, some_type_value.value, some_type_value.o_value, [&some_type_value] { //
- swap(some_type_value.a, some_type_value.b); //
- });
- });
- return;
- }
- // Should never reach here
- make_ub(_value);
- }
- };
- template <class Base> struct value_storage_delete_copy_constructor : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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
- value_storage_delete_copy_constructor &operator=(const value_storage_delete_copy_constructor &o) = default;
- value_storage_delete_copy_constructor &operator=(value_storage_delete_copy_constructor &&o) = default; // NOLINT
- ~value_storage_delete_copy_constructor() = default;
- };
- template <class Base> struct value_storage_delete_copy_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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
- ~value_storage_delete_copy_assignment() = default;
- };
- template <class Base> struct value_storage_delete_move_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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;
- ~value_storage_delete_move_assignment() = default;
- };
- template <class Base> struct value_storage_delete_move_constructor : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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;
- value_storage_delete_move_constructor &operator=(const value_storage_delete_move_constructor &o) = default;
- value_storage_delete_move_constructor &operator=(value_storage_delete_move_constructor &&o) = default;
- ~value_storage_delete_move_constructor() = default;
- };
- template <class Base> struct value_storage_nontrivial_move_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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() = default;
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- value_storage_nontrivial_move_assignment &
- operator=(value_storage_nontrivial_move_assignment &&o) noexcept(std::is_nothrow_move_assignable<value_type>::value &&
- std::is_nothrow_move_assignable<error_type>::value &&
- noexcept(move_assign_to_empty<value_type>(static_cast<value_type *>(nullptr),
- static_cast<value_type *>(nullptr))) &&
- noexcept(move_assign_to_empty<error_type>(static_cast<error_type *>(nullptr),
- static_cast<error_type *>(nullptr)))) // NOLINT
- {
- using _value_type_ = typename Base::_value_type_;
- using _error_type_ = typename Base::_error_type_;
- if(!this->_status.have_value() && !this->_status.have_error() && !o._status.have_value() && !o._status.have_error())
- {
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_value() && o._status.have_value())
- {
- this->_value = static_cast<_value_type_ &&>(o._value); // NOLINT
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_error() && o._status.have_error())
- {
- this->_error = static_cast<_error_type_ &&>(o._error); // NOLINT
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_value() && !o._status.have_value() && !o._status.have_error())
- {
- if(!trait::is_move_bitcopying<value_type>::value || this->_status.have_moved_from())
- {
- this->_value.~_value_type_(); // NOLINT
- }
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(!this->_status.have_value() && !this->_status.have_error() && o._status.have_value())
- {
- move_assign_to_empty<_value_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_value), BOOST_OUTCOME_ADDRESS_OF(o._value));
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_error() && !o._status.have_value() && !o._status.have_error())
- {
- if(!trait::is_move_bitcopying<error_type>::value || this->_status.have_moved_from())
- {
- this->_error.~_error_type_(); // NOLINT
- }
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(!this->_status.have_value() && !this->_status.have_error() && o._status.have_error())
- {
- move_assign_to_empty<_error_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_error), BOOST_OUTCOME_ADDRESS_OF(o._error));
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_value() && o._status.have_error())
- {
- if(!trait::is_move_bitcopying<value_type>::value || this->_status.have_moved_from())
- {
- this->_value.~_value_type_(); // NOLINT
- }
- move_assign_to_empty<_error_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_error), BOOST_OUTCOME_ADDRESS_OF(o._error));
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- if(this->_status.have_error() && o._status.have_value())
- {
- if(!trait::is_move_bitcopying<error_type>::value || this->_status.have_moved_from())
- {
- this->_error.~_error_type_(); // NOLINT
- }
- move_assign_to_empty<_value_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_value), BOOST_OUTCOME_ADDRESS_OF(o._value));
- this->_status = o._status;
- o._status.set_have_moved_from(true);
- return *this;
- }
- // Should never reach here
- make_ub(this->_value);
- }
- };
- template <class Base> struct value_storage_nontrivial_copy_assignment : Base // NOLINT
- {
- using Base::Base;
- using value_type = typename Base::value_type;
- using error_type = typename Base::error_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() = default;
- #if __cplusplus >= 202000L || _HAS_CXX20
- constexpr
- #endif
- value_storage_nontrivial_copy_assignment &
- operator=(const value_storage_nontrivial_copy_assignment &o) noexcept(
- std::is_nothrow_copy_assignable<value_type>::value && std::is_nothrow_copy_assignable<error_type>::value &&
- noexcept(copy_assign_to_empty<value_type>(static_cast<value_type *>(nullptr), static_cast<value_type *>(nullptr))) &&
- noexcept(copy_assign_to_empty<error_type>(static_cast<error_type *>(nullptr), static_cast<error_type *>(nullptr))))
- {
- using _value_type_ = typename Base::_value_type_;
- using _error_type_ = typename Base::_error_type_;
- if(!this->_status.have_value() && !this->_status.have_error() && !o._status.have_value() && !o._status.have_error())
- {
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_value() && o._status.have_value())
- {
- this->_value = o._value; // NOLINT
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_error() && o._status.have_error())
- {
- this->_error = o._error; // NOLINT
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_value() && !o._status.have_value() && !o._status.have_error())
- {
- if(!trait::is_move_bitcopying<value_type>::value || this->_status.have_moved_from())
- {
- this->_value.~_value_type_(); // NOLINT
- }
- this->_status = o._status;
- return *this;
- }
- if(!this->_status.have_value() && !this->_status.have_error() && o._status.have_value())
- {
- copy_assign_to_empty<_value_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_value), BOOST_OUTCOME_ADDRESS_OF(o._value));
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_error() && !o._status.have_value() && !o._status.have_error())
- {
- if(!trait::is_move_bitcopying<error_type>::value || this->_status.have_moved_from())
- {
- this->_error.~_error_type_(); // NOLINT
- }
- this->_status = o._status;
- return *this;
- }
- if(!this->_status.have_value() && !this->_status.have_error() && o._status.have_error())
- {
- copy_assign_to_empty<_error_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_error), BOOST_OUTCOME_ADDRESS_OF(o._error));
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_value() && o._status.have_error())
- {
- if(!trait::is_move_bitcopying<value_type>::value || this->_status.have_moved_from())
- {
- this->_value.~_value_type_(); // NOLINT
- }
- copy_assign_to_empty<_error_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_error), BOOST_OUTCOME_ADDRESS_OF(o._error));
- this->_status = o._status;
- return *this;
- }
- if(this->_status.have_error() && o._status.have_value())
- {
- if(!trait::is_move_bitcopying<error_type>::value || this->_status.have_moved_from())
- {
- this->_error.~_error_type_(); // NOLINT
- }
- copy_assign_to_empty<_value_type_>(BOOST_OUTCOME_ADDRESS_OF(this->_value), BOOST_OUTCOME_ADDRESS_OF(o._value));
- this->_status = o._status;
- return *this;
- }
- // Should never reach here
- make_ub(this->_value);
- }
- };
- #ifdef _MSC_VER
- #pragma warning(pop)
- #endif
- // is_trivially_copyable is true even if type is not copyable, so handle that here
- template <class T> struct is_storage_trivial
- {
- static constexpr bool value = std::is_void<T>::value || (std::is_trivially_copy_constructible<T>::value && std::is_trivially_copyable<T>::value);
- };
- // work around libstdc++ 7 bug
- template <> struct is_storage_trivial<void>
- {
- static constexpr bool value = true;
- };
- template <> struct is_storage_trivial<const void>
- {
- static constexpr bool value = true;
- };
- // Ability to do copy assigns needs more than just copy assignment
- template <class T> struct is_copy_assignable
- {
- static constexpr bool value = std::is_copy_assignable<T>::value && (std::is_copy_constructible<T>::value || std::is_default_constructible<T>::value);
- };
- // Ability to do move assigns needs more than just move assignment
- template <class T> struct is_move_assignable
- {
- static constexpr bool value = std::is_move_assignable<T>::value && (std::is_move_constructible<T>::value || std::is_default_constructible<T>::value);
- };
- template <class T, class E>
- using value_storage_select_trivality =
- std::conditional_t<is_storage_trivial<T>::value && is_storage_trivial<E>::value, value_storage_trivial<T, E>, value_storage_nontrivial<T, E>>;
- template <class T, class E>
- using value_storage_select_move_constructor =
- std::conditional_t<std::is_move_constructible<devoid<T>>::value && std::is_move_constructible<devoid<E>>::value, value_storage_select_trivality<T, E>,
- value_storage_delete_move_constructor<value_storage_select_trivality<T, E>>>;
- template <class T, class E>
- using value_storage_select_copy_constructor =
- std::conditional_t<std::is_copy_constructible<devoid<T>>::value && std::is_copy_constructible<devoid<E>>::value, value_storage_select_move_constructor<T, E>,
- value_storage_delete_copy_constructor<value_storage_select_move_constructor<T, E>>>;
- template <class T, class E>
- using value_storage_select_move_assignment =
- std::conditional_t<std::is_trivially_move_assignable<devoid<T>>::value && std::is_trivially_move_assignable<devoid<E>>::value,
- value_storage_select_copy_constructor<T, E>,
- std::conditional_t<is_move_assignable<devoid<T>>::value && is_move_assignable<devoid<E>>::value,
- value_storage_nontrivial_move_assignment<value_storage_select_copy_constructor<T, E>>,
- value_storage_delete_move_assignment<value_storage_select_copy_constructor<T, E>>>>;
- template <class T, class E>
- using value_storage_select_copy_assignment =
- std::conditional_t<std::is_trivially_copy_assignable<devoid<T>>::value && std::is_trivially_copy_assignable<devoid<E>>::value,
- value_storage_select_move_assignment<T, E>,
- std::conditional_t<is_copy_assignable<devoid<T>>::value && is_copy_assignable<devoid<E>>::value,
- value_storage_nontrivial_copy_assignment<value_storage_select_move_assignment<T, E>>,
- value_storage_delete_copy_assignment<value_storage_select_move_assignment<T, E>>>>;
- template <class T, class E> using value_storage_select_impl = value_storage_select_copy_assignment<T, E>;
- #ifndef NDEBUG
- // Check is trivial in all ways except default constructibility
- // static_assert(std::is_trivial<value_storage_select_impl<int, long>>::value, "value_storage_select_impl<int, long> is not trivial!");
- // static_assert(std::is_trivially_default_constructible<value_storage_select_impl<int, long>>::value, "value_storage_select_impl<int, long> is not
- // trivially default constructible!");
- static_assert(std::is_trivially_copyable<value_storage_select_impl<int, long>>::value, "value_storage_select_impl<int, long> is not trivially copyable!");
- static_assert(std::is_trivially_assignable<value_storage_select_impl<int, long>, value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially assignable!");
- static_assert(std::is_trivially_destructible<value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially destructible!");
- static_assert(std::is_trivially_copy_constructible<value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially copy constructible!");
- static_assert(std::is_trivially_move_constructible<value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially move constructible!");
- static_assert(std::is_trivially_copy_assignable<value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially copy assignable!");
- static_assert(std::is_trivially_move_assignable<value_storage_select_impl<int, long>>::value,
- "value_storage_select_impl<int, long> is not trivially move assignable!");
- // Also check is standard layout
- static_assert(std::is_standard_layout<value_storage_select_impl<int, long>>::value, "value_storage_select_impl<int, long> is not a standard layout type!");
- #endif
- } // namespace detail
- BOOST_OUTCOME_V2_NAMESPACE_END
- #endif
|