Back to home page

EIC code displayed by LXR

 
 

    


File indexing completed on 2025-04-04 08:33:22

0001 /* Proposed SG14 status_code
0002 (C) 2018 - 2023 Niall Douglas <http://www.nedproductions.biz/> (5 commits)
0003 File Created: Feb 2018
0004 
0005 
0006 Licensed under the Apache License, Version 2.0 (the "License");
0007 you may not use this file except in compliance with the License.
0008 You may obtain a copy of the License in the accompanying file
0009 Licence.txt or at
0010 
0011 http://www.apache.org/licenses/LICENSE-2.0
0012 
0013 Unless required by applicable law or agreed to in writing, software
0014 distributed under the License is distributed on an "AS IS" BASIS,
0015 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
0016 See the License for the specific language governing permissions and
0017 limitations under the License.
0018 
0019 
0020 Distributed under the Boost Software License, Version 1.0.
0021 (See accompanying file Licence.txt or copy at
0022 http://www.boost.org/LICENSE_1_0.txt)
0023 */
0024 
0025 #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_HPP
0026 #define BOOST_OUTCOME_SYSTEM_ERROR2_STATUS_CODE_HPP
0027 
0028 #include "status_code_domain.hpp"
0029 
0030 #if(__cplusplus >= 201700 || _HAS_CXX17) && !defined(BOOST_OUTCOME_SYSTEM_ERROR2_DISABLE_STD_IN_PLACE)
0031 // 0.26
0032 #include <utility>  // for in_place
0033 
0034 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0035 using in_place_t = std::in_place_t;
0036 using std::in_place;
0037 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0038 
0039 #else
0040 
0041 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0042 //! Aliases `std::in_place_t` if on C++ 17 or later, else defined locally.
0043 struct in_place_t
0044 {
0045   explicit in_place_t() = default;
0046 };
0047 //! Aliases `std::in_place` if on C++ 17 or later, else defined locally.
0048 constexpr in_place_t in_place{};
0049 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0050 #endif
0051 
0052 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0053 
0054 //! Namespace for user injected mixins
0055 namespace mixins
0056 {
0057   template <class Base, class T> struct mixin : public Base
0058   {
0059     using Base::Base;
0060   };
0061 }  // namespace mixins
0062 
0063 namespace detail
0064 {
0065   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class ErasedType)  //
0066   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(traits::is_move_bitcopying<ErasedType>::value))
0067   struct erased
0068   {
0069     using value_type = ErasedType;
0070   };
0071 }  // namespace detail
0072 
0073 /*! The tag type used to specialise erased editions of `status_code<D>`.
0074 Available only if `ErasedType` satisfies `traits::is_move_bitcopying<ErasedType>::value`.
0075 */
0076 template <class ErasedType>  //
0077 using status_code_erased_tag_type = detail::erased<ErasedType>;
0078 
0079 /*! Specialise this template to quickly wrap a third party enumeration into a
0080 custom status code domain.
0081 
0082 Use like this:
0083 
0084 ```c++
0085 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
0086 template <> struct quick_status_code_from_enum<AnotherCode> : quick_status_code_from_enum_defaults<AnotherCode>
0087 {
0088   // Text name of the enum
0089   static constexpr const auto domain_name = "Another Code";
0090   // Unique UUID for the enum. PLEASE use https://www.random.org/cgi-bin/randbyte?nbytes=16&format=h
0091   static constexpr const auto domain_uuid = "{be201f65-3962-dd0e-1266-a72e63776a42}";
0092   // Map of each enum value to its text string, and list of semantically equivalent errc's
0093   static const std::initializer_list<mapping> &value_mappings()
0094   {
0095     static const std::initializer_list<mapping<AnotherCode>> v = {
0096     // Format is: { enum value, "string representation", { list of errc mappings ... } }
0097     {AnotherCode::success1, "Success 1", {errc::success}},        //
0098     {AnotherCode::goaway, "Go away", {errc::permission_denied}},  //
0099     {AnotherCode::success2, "Success 2", {errc::success}},        //
0100     {AnotherCode::error2, "Error 2", {}},                         //
0101     };
0102     return v;
0103   }
0104   // Completely optional definition of mixin for the status code synthesised from `Enum`. It can be omitted.
0105   template <class Base> struct mixin : Base
0106   {
0107     using Base::Base;
0108     constexpr int custom_method() const { return 42; }
0109   };
0110 };
0111 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0112 ```
0113 
0114 Note that if the `errc` mapping contains `errc::success`, then
0115 the enumeration value is considered to be a successful value.
0116 Otherwise it is considered to be a failure value.
0117 
0118 The first value in the `errc` mapping is the one chosen as the
0119 `generic_code` conversion. Other values are used during equivalence
0120 comparisons.
0121 */
0122 template <class Enum> struct quick_status_code_from_enum;
0123 
0124 namespace detail
0125 {
0126   template <class T> struct is_status_code
0127   {
0128     static constexpr bool value = false;
0129   };
0130   template <class T> struct is_status_code<status_code<T>>
0131   {
0132     static constexpr bool value = true;
0133   };
0134   template <class T> struct is_erased_status_code
0135   {
0136     static constexpr bool value = false;
0137   };
0138   template <class T> struct is_erased_status_code<status_code<detail::erased<T>>>
0139   {
0140     static constexpr bool value = true;
0141   };
0142 
0143 #if !defined(__GNUC__) || defined(__clang__) || __GNUC__ >= 8
0144   // From http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2015/n4436.pdf
0145   namespace impl
0146   {
0147     template <typename... Ts> struct make_void
0148     {
0149       using type = void;
0150     };
0151     template <typename... Ts> using void_t = typename make_void<Ts...>::type;
0152     template <class...> struct types
0153     {
0154       using type = types;
0155     };
0156     template <template <class...> class T, class types, class = void> struct test_apply
0157     {
0158       using type = void;
0159     };
0160     template <template <class...> class T, class... Ts> struct test_apply<T, types<Ts...>, void_t<T<Ts...>>>
0161     {
0162       using type = T<Ts...>;
0163     };
0164   }  // namespace impl
0165   template <template <class...> class T, class... Ts> using test_apply = impl::test_apply<T, impl::types<Ts...>>;
0166 
0167   template <class T, class... Args> using get_make_status_code_result = decltype(make_status_code(std::declval<T>(), std::declval<Args>()...));
0168   template <class... Args> using safe_get_make_status_code_result = test_apply<get_make_status_code_result, Args...>;
0169 
0170 #else
0171 
0172   // ICE avoidance form for GCCs before 8. Note this form doesn't prevent recursive make_status_code ADL instantation,
0173   // so in certain corner cases this will break. On the other hand, more useful than an ICE.
0174   namespace impl
0175   {
0176     template <typename... Ts> struct make_void
0177     {
0178       using type = void;
0179     };
0180     template <typename... Ts> using void_t = typename make_void<Ts...>::type;
0181     template <class...> struct types
0182     {
0183       using type = types;
0184     };
0185     template <typename Types, typename = void> struct make_status_code_rettype
0186     {
0187       using type = void;
0188     };
0189     template <typename... Args> using get_make_status_code_result = decltype(make_status_code(std::declval<Args>()...));
0190     template <typename... Args> struct make_status_code_rettype<types<Args...>, void_t<get_make_status_code_result<Args...>>>
0191     {
0192       using type = get_make_status_code_result<Args...>;
0193     };
0194   }  // namespace impl
0195   template <class... Args> struct safe_get_make_status_code_result
0196   {
0197     using type = typename impl::make_status_code_rettype<impl::types<Args...>>::type;
0198   };
0199 #endif
0200 }  // namespace detail
0201 
0202 //! Trait returning true if the type is a status code.
0203 template <class T> struct is_status_code
0204 {
0205   static constexpr bool value =
0206   detail::is_status_code<typename std::decay<T>::type>::value || detail::is_erased_status_code<typename std::decay<T>::type>::value;
0207 };
0208 
0209 /*! A type erased lightweight status code reflecting empty, success, or failure.
0210 Differs from `status_code<erased<>>` by being always available irrespective of
0211 the domain's value type, but cannot be copied, moved, nor destructed. Thus one
0212 always passes this around by const lvalue reference.
0213 */
0214 template <> class BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI status_code<void>
0215 {
0216   template <class T> friend class status_code;
0217 
0218 public:
0219   //! The type of the domain.
0220   using domain_type = void;
0221   //! The type of the status code.
0222   using value_type = void;
0223   //! The type of a reference to a message string.
0224   using string_ref = typename status_code_domain::string_ref;
0225 
0226 protected:
0227   const status_code_domain *_domain{nullptr};
0228 
0229 protected:
0230   //! No default construction at type erased level
0231   status_code() = default;
0232   //! No public copying at type erased level
0233   status_code(const status_code &) = default;
0234   //! No public moving at type erased level
0235   status_code(status_code &&) = default;
0236   //! No public assignment at type erased level
0237   status_code &operator=(const status_code &) = default;
0238   //! No public assignment at type erased level
0239   status_code &operator=(status_code &&) = default;
0240   //! No public destruction at type erased level
0241   ~status_code() = default;
0242 
0243   //! Used to construct a non-empty type erased status code
0244   constexpr explicit status_code(const status_code_domain *v) noexcept
0245       : _domain(v)
0246   {
0247   }
0248 
0249   // Used to work around triggering a ubsan failure. Do NOT remove!
0250   constexpr const status_code_domain *_domain_ptr() const noexcept { return _domain; }
0251 
0252 public:
0253   //! Return the status code domain.
0254   constexpr const status_code_domain &domain() const noexcept { return *_domain; }
0255   //! True if the status code is empty.
0256   BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD constexpr bool empty() const noexcept { return _domain == nullptr; }
0257 
0258   //! Return a reference to a string textually representing a code.
0259   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref message() const noexcept
0260   {
0261     // Avoid MSVC's buggy ternary operator for expensive to destruct things
0262     if(_domain != nullptr)
0263     {
0264       return _domain->_do_message(*this);
0265     }
0266     return string_ref("(empty)");
0267   }
0268   //! True if code means success.
0269   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 bool success() const noexcept { return (_domain != nullptr) ? !_domain->_do_failure(*this) : false; }
0270   //! True if code means failure.
0271   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 bool failure() const noexcept { return (_domain != nullptr) ? _domain->_do_failure(*this) : false; }
0272   /*! True if code is strictly (and potentially non-transitively) semantically equivalent to another code in another domain.
0273   Note that usually non-semantic i.e. pure value comparison is used when the other status code has the same domain.
0274   As `equivalent()` will try mapping to generic code, this usually captures when two codes have the same semantic
0275   meaning in `equivalent()`.
0276   */
0277   template <class T> BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 bool strictly_equivalent(const status_code<T> &o) const noexcept
0278   {
0279     if(_domain && o._domain)
0280     {
0281       return _domain->_do_equivalent(*this, o);
0282     }
0283     // If we are both empty, we are equivalent
0284     if(!_domain && !o._domain)
0285     {
0286       return true;  // NOLINT
0287     }
0288     // Otherwise not equivalent
0289     return false;
0290   }
0291   /*! True if code is equivalent, by any means, to another code in another domain (guaranteed transitive).
0292   Firstly `strictly_equivalent()` is run in both directions. If neither succeeds, each domain is asked
0293   for the equivalent generic code and those are compared.
0294   */
0295   template <class T> BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 inline bool equivalent(const status_code<T> &o) const noexcept;
0296 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0297   //! Throw a code as a C++ exception.
0298   BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN void throw_exception() const
0299   {
0300     _domain->_do_throw_exception(*this);
0301     abort();  // suppress buggy GCC warning
0302   }
0303 #endif
0304 };
0305 
0306 namespace detail
0307 {
0308   template <class DomainType> struct get_domain_value_type
0309   {
0310     using domain_type = DomainType;
0311     using value_type = typename domain_type::value_type;
0312   };
0313   template <class ErasedType> struct get_domain_value_type<erased<ErasedType>>
0314   {
0315     using domain_type = status_code_domain;
0316     using value_type = ErasedType;
0317   };
0318   template <class DomainType> class BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI status_code_storage : public status_code<void>
0319   {
0320     static_assert(!std::is_void<DomainType>::value, "status_code_storage<void> should never occur!");
0321     using _base = status_code<void>;
0322 
0323   public:
0324     //! The type of the domain.
0325     using domain_type = typename get_domain_value_type<DomainType>::domain_type;
0326     //! The type of the status code.
0327     using value_type = typename get_domain_value_type<DomainType>::value_type;
0328     //! The type of a reference to a message string.
0329     using string_ref = typename domain_type::string_ref;
0330 
0331 #ifndef NDEBUG
0332     static_assert(std::is_move_constructible<value_type>::value || std::is_copy_constructible<value_type>::value,
0333                   "DomainType::value_type is neither move nor copy constructible!");
0334     static_assert(!std::is_default_constructible<value_type>::value || std::is_nothrow_default_constructible<value_type>::value,
0335                   "DomainType::value_type is not nothrow default constructible!");
0336     static_assert(!std::is_move_constructible<value_type>::value || std::is_nothrow_move_constructible<value_type>::value,
0337                   "DomainType::value_type is not nothrow move constructible!");
0338     static_assert(std::is_nothrow_destructible<value_type>::value, "DomainType::value_type is not nothrow destructible!");
0339 #endif
0340 
0341     // Replace the type erased implementations with type aware implementations for better codegen
0342     //! Return the status code domain.
0343     constexpr const domain_type &domain() const noexcept { return *static_cast<const domain_type *>(this->_domain); }
0344 
0345     //! Reset the code to empty.
0346     BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 void clear() noexcept
0347     {
0348       this->_value.~value_type();
0349       this->_domain = nullptr;
0350       new(std::addressof(this->_value)) value_type();
0351     }
0352 
0353 #if __cplusplus >= 201400 || _MSC_VER >= 1910 /* VS2017 */
0354     //! Return a reference to the `value_type`.
0355     constexpr value_type &value() & noexcept { return this->_value; }
0356     //! Return a reference to the `value_type`.
0357     constexpr value_type &&value() && noexcept { return static_cast<value_type &&>(this->_value); }
0358 #endif
0359     //! Return a reference to the `value_type`.
0360     constexpr const value_type &value() const & noexcept { return this->_value; }
0361     //! Return a reference to the `value_type`.
0362     constexpr const value_type &&value() const && noexcept { return static_cast<const value_type &&>(this->_value); }
0363 
0364   protected:
0365     status_code_storage() = default;
0366     status_code_storage(const status_code_storage &) = default;
0367     BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 status_code_storage(status_code_storage &&o) noexcept
0368         : _base(static_cast<status_code_storage &&>(o))
0369         , _value(static_cast<status_code_storage &&>(o)._value)
0370     {
0371       o._domain = nullptr;
0372     }
0373     status_code_storage &operator=(const status_code_storage &) = default;
0374     BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 status_code_storage &operator=(status_code_storage &&o) noexcept
0375     {
0376       this->~status_code_storage();
0377       new(this) status_code_storage(static_cast<status_code_storage &&>(o));
0378       return *this;
0379     }
0380     ~status_code_storage() = default;
0381 
0382     value_type _value{};
0383     struct _value_type_constructor
0384     {
0385     };
0386     template <class... Args>
0387     constexpr status_code_storage(_value_type_constructor /*unused*/, const status_code_domain *v, Args &&...args)
0388         : _base(v)
0389         , _value(static_cast<Args &&>(args)...)
0390     {
0391     }
0392   };
0393 
0394   template <class DomainType> struct has_stateful_mixin
0395   {
0396     static constexpr bool value = (sizeof(status_code_storage<DomainType>) != sizeof(mixins::mixin<status_code_storage<DomainType>, DomainType>));
0397   };
0398 
0399   template <class ToDomain, class FromDomain> struct domain_value_type_erasure_is_safe
0400   {
0401     using to_value_type = typename get_domain_value_type<ToDomain>::value_type;
0402     using from_value_type = typename get_domain_value_type<FromDomain>::value_type;
0403     static constexpr bool value = traits::is_move_bitcopying<to_value_type>::value                                     //
0404                                   && traits::is_move_bitcopying<from_value_type>::value                                //
0405                                   && sizeof(status_code_storage<FromDomain>) <= sizeof(status_code_storage<ToDomain>)  //
0406                                   && !has_stateful_mixin<FromDomain>::value;
0407   };
0408   template <class ToDomain> struct domain_value_type_erasure_is_safe<ToDomain, void>
0409   {
0410     static constexpr bool value = false;
0411   };
0412 }  // namespace detail
0413 
0414 namespace traits
0415 {
0416   //! Determines whether the mixin contained in `StatusCode` contains non-static member variables.
0417   template <class StatusCode> using has_stateful_mixin = detail::has_stateful_mixin<typename detail::remove_cvref<StatusCode>::type::value_type>;
0418 
0419   //! Determines whether the status code `From` can be type erased into the status code `To`.
0420   template <class To, class From>
0421   using is_type_erasable_to =
0422   detail::domain_value_type_erasure_is_safe<typename detail::remove_cvref<To>::type::domain_type, typename detail::remove_cvref<From>::type::domain_type>;
0423 }  // namespace traits
0424 
0425 /*! A lightweight, typed, status code reflecting empty, success, or failure.
0426 This is the main workhorse of the system_error2 library. Its characteristics reflect the value type
0427 set by its domain type, so if that value type is move-only or trivial, so is this.
0428 
0429 An ADL discovered helper function `make_status_code(T, Args...)` is looked up by one of the constructors.
0430 If it is found, and it generates a status code compatible with this status code, implicit construction
0431 is made available.
0432 
0433 You may mix in custom member functions and member function overrides by injecting a specialisation of
0434 `mixins::mixin<Base, YourDomainType>`. Your mixin must inherit from `Base`. Your mixin can carry state,
0435 but if it does, it will no longer be possible to construct erased status codes from such unerased status
0436 codes.
0437 */
0438 template <class DomainType> class BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI status_code : public mixins::mixin<detail::status_code_storage<DomainType>, DomainType>
0439 {
0440   template <class T> friend class status_code;
0441   using _base = mixins::mixin<detail::status_code_storage<DomainType>, DomainType>;
0442 
0443 public:
0444   //! The type of the domain.
0445   using domain_type = DomainType;
0446   //! The type of the status code.
0447   using value_type = typename domain_type::value_type;
0448   //! The type of a reference to a message string.
0449   using string_ref = typename domain_type::string_ref;
0450 
0451 protected:
0452   using _base::_base;
0453 
0454 public:
0455   //! Default construction to empty
0456   status_code() = default;
0457   //! Copy constructor
0458   status_code(const status_code &) = default;
0459   //! Move constructor
0460   status_code(status_code &&) = default;  // NOLINT
0461   //! Copy assignment
0462   status_code &operator=(const status_code &) = default;
0463   //! Move assignment
0464   status_code &operator=(status_code &&) = default;  // NOLINT
0465   ~status_code() = default;
0466 
0467   //! Return a copy of the code.
0468   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 status_code clone() const { return *this; }
0469 
0470   /***** KEEP THESE IN SYNC WITH ERRORED_STATUS_CODE *****/
0471   //! Implicit construction from any type where an ADL discovered `make_status_code(T, Args ...)` returns a `status_code`.
0472   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(
0473   class T, class... Args,  //
0474   class MakeStatusCodeResult =
0475   typename detail::safe_get_make_status_code_result<T, Args...>::type)  // Safe ADL lookup of make_status_code(), returns void if not found
0476   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(!std::is_same<typename std::decay<T>::type, status_code>::value       // not copy/move of self
0477                                               && !std::is_same<typename std::decay<T>::type, in_place_t>::value     // not in_place_t
0478                                               && is_status_code<MakeStatusCodeResult>::value                        // ADL makes a status code
0479                                               && std::is_constructible<status_code, MakeStatusCodeResult>::value))  // ADLed status code is compatible
0480   constexpr status_code(T &&v, Args &&...args) noexcept(noexcept(make_status_code(std::declval<T>(), std::declval<Args>()...)))  // NOLINT
0481       : status_code(make_status_code(static_cast<T &&>(v), static_cast<Args &&>(args)...))
0482   {
0483   }
0484   //! Implicit construction from any `quick_status_code_from_enum<Enum>` enumerated type.
0485   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class Enum,                                                                                //
0486                          class QuickStatusCodeType = typename quick_status_code_from_enum<Enum>::code_type)         // Enumeration has been activated
0487   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(std::is_constructible<status_code, QuickStatusCodeType>::value))      // Its status code is compatible
0488   constexpr status_code(Enum &&v) noexcept(std::is_nothrow_constructible<status_code, QuickStatusCodeType>::value)  // NOLINT
0489       : status_code(QuickStatusCodeType(static_cast<Enum &&>(v)))
0490   {
0491   }
0492   //! Explicit in-place construction. Disables if `domain_type::get()` is not a valid expression.
0493   template <class... Args>
0494   constexpr explicit status_code(in_place_t /*unused */, Args &&...args) noexcept(std::is_nothrow_constructible<value_type, Args &&...>::value)
0495       : _base(typename _base::_value_type_constructor{}, &domain_type::get(), static_cast<Args &&>(args)...)
0496   {
0497   }
0498   //! Explicit in-place construction from initialiser list. Disables if `domain_type::get()` is not a valid expression.
0499   template <class T, class... Args>
0500   constexpr explicit status_code(in_place_t /*unused */, std::initializer_list<T> il,
0501                                  Args &&...args) noexcept(std::is_nothrow_constructible<value_type, std::initializer_list<T>, Args &&...>::value)
0502       : _base(typename _base::_value_type_constructor{}, &domain_type::get(), il, static_cast<Args &&>(args)...)
0503   {
0504   }
0505   //! Explicit copy construction from a `value_type`. Disables if `domain_type::get()` is not a valid expression.
0506   constexpr explicit status_code(const value_type &v) noexcept(std::is_nothrow_copy_constructible<value_type>::value)
0507       : _base(typename _base::_value_type_constructor{}, &domain_type::get(), v)
0508   {
0509   }
0510   //! Explicit move construction from a `value_type`. Disables if `domain_type::get()` is not a valid expression.
0511   constexpr explicit status_code(value_type &&v) noexcept(std::is_nothrow_move_constructible<value_type>::value)
0512       : _base(typename _base::_value_type_constructor{}, &domain_type::get(), static_cast<value_type &&>(v))
0513   {
0514   }
0515   /*! Explicit construction from an erased status code. Available only if
0516   `value_type` is trivially copyable or move bitcopying, and `sizeof(status_code) <= sizeof(status_code<erased<>>)`.
0517   Does not check if domains are equal.
0518   */
0519   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class ErasedType)  //
0520   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(detail::domain_value_type_erasure_is_safe<domain_type, detail::erased<ErasedType>>::value))
0521   constexpr explicit status_code(const status_code<detail::erased<ErasedType>> &v) noexcept(std::is_nothrow_copy_constructible<value_type>::value)
0522       : status_code(detail::erasure_cast<value_type>(v.value()))
0523   {
0524 #if __cplusplus >= 201400
0525     assert(v.domain() == this->domain());
0526 #endif
0527   }
0528 
0529   //! Return a reference to a string textually representing a code.
0530   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 string_ref message() const noexcept
0531   {
0532     // Avoid MSVC's buggy ternary operator for expensive to destruct things
0533     if(this->_domain != nullptr)
0534     {
0535       return string_ref(this->domain()._do_message(*this));
0536     }
0537     return string_ref("(empty)");
0538   }
0539 };
0540 
0541 namespace traits
0542 {
0543   template <class DomainType> struct is_move_bitcopying<status_code<DomainType>>
0544   {
0545     static constexpr bool value = is_move_bitcopying<typename DomainType::value_type>::value;
0546   };
0547 }  // namespace traits
0548 
0549 
0550 /*! Type erased, move-only status_code, unlike `status_code<void>` which cannot be moved nor destroyed. Available
0551 only if `erased<>` is available, which is when the domain's type is trivially
0552 copyable or is move relocatable, and if the size of the domain's typed error code is less than or equal to
0553 this erased error code. Copy construction is disabled, but if you want a copy call `.clone()`.
0554 
0555 An ADL discovered helper function `make_status_code(T, Args...)` is looked up by one of the constructors.
0556 If it is found, and it generates a status code compatible with this status code, implicit construction
0557 is made available.
0558 */
0559 template <class ErasedType>
0560 class BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI status_code<detail::erased<ErasedType>>
0561     : public mixins::mixin<detail::status_code_storage<detail::erased<ErasedType>>, detail::erased<ErasedType>>
0562 {
0563   template <class T> friend class status_code;
0564   using _base = mixins::mixin<detail::status_code_storage<detail::erased<ErasedType>>, detail::erased<ErasedType>>;
0565 
0566 public:
0567   //! The type of the domain (void, as it is erased).
0568   using domain_type = void;
0569   //! The type of the erased status code.
0570   using value_type = ErasedType;
0571   //! The type of a reference to a message string.
0572   using string_ref = typename _base::string_ref;
0573 
0574 public:
0575   //! Default construction to empty
0576   status_code() = default;
0577   //! Copy constructor
0578   status_code(const status_code &) = delete;
0579   //! Move constructor
0580   status_code(status_code &&) = default;  // NOLINT
0581   //! Copy assignment
0582   status_code &operator=(const status_code &) = delete;
0583   //! Move assignment
0584   status_code &operator=(status_code &&) = default;  // NOLINT
0585   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 ~status_code()
0586   {
0587     if(nullptr != this->_domain)
0588     {
0589       status_code_domain::payload_info_t info{sizeof(value_type), sizeof(status_code), alignof(status_code)};
0590       this->_domain->_do_erased_destroy(*this, info);
0591     }
0592   }
0593 
0594   //! Return a copy of the erased code by asking the domain to perform the erased copy.
0595   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 status_code clone() const
0596   {
0597     if(nullptr == this->_domain)
0598     {
0599       return {};
0600     }
0601     status_code x;
0602     if(!this->_domain->_do_erased_copy(x, *this, this->_domain->payload_info()))
0603     {
0604       abort();  // should not be possible
0605     }
0606     return x;
0607   }
0608 
0609   /***** KEEP THESE IN SYNC WITH ERRORED_STATUS_CODE *****/
0610   //! Implicit copy construction from any other status code if its value type is trivially copyable, it would fit into our storage, and it is not an erased
0611   //! status code.
0612   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class DomainType)  //
0613   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(detail::domain_value_type_erasure_is_safe<detail::erased<ErasedType>, DomainType>::value),
0614                           BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(!detail::is_erased_status_code<status_code<typename std::decay<DomainType>::type>>::value))
0615   constexpr status_code(const status_code<DomainType> &v) noexcept  // NOLINT
0616       : _base(typename _base::_value_type_constructor{}, v._domain_ptr(), detail::erasure_cast<value_type>(v.value()))
0617   {
0618   }
0619   //! Implicit move construction from any other status code if its value type is trivially copyable or move bitcopying and it would fit into our storage
0620   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class DomainType)  //
0621   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(detail::domain_value_type_erasure_is_safe<detail::erased<ErasedType>, DomainType>::value))
0622   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 status_code(status_code<DomainType> &&v) noexcept  // NOLINT
0623       : _base(typename _base::_value_type_constructor{}, v._domain_ptr(), detail::erasure_cast<value_type>(v.value()))
0624   {
0625     alignas(alignof(typename DomainType::value_type)) char buffer[sizeof(typename DomainType::value_type)];
0626     new(buffer) typename DomainType::value_type(static_cast<status_code<DomainType> &&>(v).value());
0627     // deliberately do not destruct value moved into buffer
0628     (void) buffer;
0629     v._domain = nullptr;
0630   }
0631   //! Implicit construction from any type where an ADL discovered `make_status_code(T, Args ...)` returns a `status_code`.
0632   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(
0633   class T, class... Args,  //
0634   class MakeStatusCodeResult =
0635   typename detail::safe_get_make_status_code_result<T, Args...>::type)  // Safe ADL lookup of make_status_code(), returns void if not found
0636   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(!std::is_same<typename std::decay<T>::type, status_code>::value       // not copy/move of self
0637                                               && !std::is_same<typename std::decay<T>::type, value_type>::value     // not copy/move of value type
0638                                               && is_status_code<MakeStatusCodeResult>::value                        // ADL makes a status code
0639                                               && std::is_constructible<status_code, MakeStatusCodeResult>::value))  // ADLed status code is compatible
0640   constexpr status_code(T &&v, Args &&...args) noexcept(noexcept(make_status_code(std::declval<T>(), std::declval<Args>()...)))  // NOLINT
0641       : status_code(make_status_code(static_cast<T &&>(v), static_cast<Args &&>(args)...))
0642   {
0643   }
0644 
0645   //! Implicit construction from any `quick_status_code_from_enum<Enum>` enumerated type.
0646   BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class Enum,                                                                                //
0647                          class QuickStatusCodeType = typename quick_status_code_from_enum<Enum>::code_type)         // Enumeration has been activated
0648   BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(std::is_constructible<status_code, QuickStatusCodeType>::value))      // Its status code is compatible
0649   constexpr status_code(Enum &&v) noexcept(std::is_nothrow_constructible<status_code, QuickStatusCodeType>::value)  // NOLINT
0650       : status_code(QuickStatusCodeType(static_cast<Enum &&>(v)))
0651   {
0652   }
0653 
0654 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0655   //! Explicit copy construction from an unknown status code. Note that this will throw an exception if its value type is not trivially copyable or would not
0656   //! fit into our storage or the source domain's `_do_erased_copy()` refused the copy.
0657   explicit BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 status_code(in_place_t, const status_code<void> &v)  // NOLINT
0658       : _base(typename _base::_value_type_constructor{}, v._domain_ptr(), value_type{})
0659   {
0660     status_code_domain::payload_info_t info{sizeof(value_type), sizeof(status_code), alignof(status_code)};
0661     if(this->_domain->_do_erased_copy(*this, v, info))
0662     {
0663       return;
0664     }
0665     struct _ final : public std::exception
0666     {
0667       virtual const char *what() const noexcept override { return "status_code: source domain's erased copy function returned failure or refusal"; }
0668     };
0669     throw _{};
0670   }
0671 #endif
0672   //! Tagged copy construction from an unknown status code. Note that this will be empty if its value type is not trivially copyable or would not fit into our
0673   //! storage or the source domain's `_do_erased_copy()` refused the copy.
0674   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 status_code(std::nothrow_t, const status_code<void> &v) noexcept  // NOLINT
0675       : _base(typename _base::_value_type_constructor{}, v._domain_ptr(), value_type{})
0676   {
0677 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0678     try
0679 #endif
0680     {
0681       status_code_domain::payload_info_t info{sizeof(value_type), sizeof(status_code), alignof(status_code)};
0682       if(this->_domain->_do_erased_copy(*this, v, info))
0683       {
0684         return;
0685       }
0686       this->_domain = nullptr;
0687     }
0688 #if defined(_CPPUNWIND) || defined(__EXCEPTIONS) || defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE)
0689     catch(...)
0690     {
0691       this->_domain = nullptr;
0692     }
0693 #endif
0694   }
0695 
0696   /**** By rights ought to be removed in any formal standard ****/
0697   //! Reset the code to empty.
0698   BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 void clear() noexcept { *this = status_code(); }
0699   //! Return the erased `value_type` by value.
0700   constexpr value_type value() const noexcept { return this->_value; }
0701 };
0702 
0703 /*! An erased type specialisation of `status_code<D>`.
0704 Available only if `ErasedType` satisfies `traits::is_move_bitcopying<ErasedType>::value`.
0705 */
0706 template <class ErasedType> using erased_status_code = status_code<detail::erased<ErasedType>>;
0707 
0708 namespace traits
0709 {
0710   template <class ErasedType> struct is_move_bitcopying<status_code<detail::erased<ErasedType>>>
0711   {
0712     static constexpr bool value = true;
0713   };
0714 }  // namespace traits
0715 
0716 BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
0717 
0718 #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_DISABLE_INLINE_GDB_PRETTY_PRINTERS
0719 #if defined(__ELF__)
0720 __asm__(
0721 ".pushsection \".debug_gdb_scripts\", \"MS\",@progbits,1\n"
0722 ".byte 4 /* Python Text */\n"
0723 ".ascii \"gdb.inlined-script\\n\"\n"
0724 ".ascii \"import gdb.printing\\n\"\n"
0725 ".ascii \"import gdb\\n\"\n"
0726 ".ascii \"import os\\n\"\n"
0727 
0728 ".ascii \"def synthesise_gdb_value_from_string(s):\\n\"\n"
0729 ".ascii \"    '''For when you want to return a synthetic string from children()'''\\n\"\n"
0730 ".ascii \"    return gdb.Value(s + '\\0').cast(gdb.lookup_type('char').pointer())\\n\"\n"
0731 
0732 ".ascii \"class StatusCodePrinter(object):\\n\"\n"
0733 ".ascii \"    '''Print a system_error2::status_code<T>'''\\n\"\n"
0734 
0735 ".ascii \"    def __init__(self, val):\\n\"\n"
0736 ".ascii \"        self.val = val\\n\"\n"
0737 
0738 ".ascii \"    def children(self):\\n\"\n"
0739 ".ascii \"        s = str(self.val['_domain'])\\n\"\n"
0740 ".ascii \"        if 'posix_code_domain' in s or 'generic_code_domain' in s:\\n\"\n"
0741 ".ascii \"            yield ('msg', synthesise_gdb_value_from_string(str(self.val['_value']) + ' (' + os.strerror(int(self.val['_value'])) + ')'))\\n\"\n"
0742 ".ascii \"        yield ('domain', self.val['_domain'])\\n\"\n"
0743 ".ascii \"        yield ('value', self.val['_value'])\\n\"\n"
0744 
0745 ".ascii \"    def display_hint(self):\\n\"\n"
0746 ".ascii \"        return None\\n\"\n"
0747 
0748 ".ascii \"    def to_string(self):\\n\"\n"
0749 ".ascii \"        s = str(self.val['_domain'])\\n\"\n"
0750 ".ascii \"        if 'posix_code_domain' in s or 'generic_code_domain' in s:\\n\"\n"
0751 ".ascii \"            return str(self.val['_value']) + ' (' + os.strerror(int(self.val['_value'])) + ')'\\n\"\n"
0752 ".ascii \"        else:\\n\"\n"
0753 ".ascii \"            return self.val['_value']\\n\"\n"
0754 
0755 ".ascii \"def build_pretty_printer():\\n\"\n"
0756 ".ascii \"    pp = gdb.printing.RegexpCollectionPrettyPrinter('system_error2')\\n\"\n"
0757 ".ascii \"    pp.add_printer('system_error2::status_code', '^(boost::)?system_error2::status_code<.*>$', StatusCodePrinter)\\n\"\n"
0758 ".ascii \"    return pp\\n\"\n"
0759 
0760 ".ascii \"def register_printers(obj = None):\\n\"\n"
0761 ".ascii \"    gdb.printing.register_pretty_printer(obj, build_pretty_printer(), replace = True)\\n\"\n"
0762 
0763 ".ascii \"register_printers(gdb.current_objfile())\\n\"\n"
0764 ".byte 0\n"
0765 ".popsection\n");
0766 #endif
0767 #endif
0768 
0769 #endif