123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459 |
- /* Proposed SG14 status_code
- (C) 2018 - 2021 Niall Douglas <http://www.nedproductions.biz/> (5 commits)
- File Created: Feb 2018
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License in the accompanying file
- Licence.txt or at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- Distributed under the Boost Software License, Version 1.0.
- (See accompanying file Licence.txt or copy at
- http://www.boost.org/LICENSE_1_0.txt)
- */
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_CONFIG_HPP
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CONFIG_HPP
- // < 0.1 each
- #include <cassert>
- #include <cstddef> // for size_t
- #include <cstdlib> // for free
- // 0.22
- #include <type_traits>
- // 0.29
- #include <atomic>
- // 0.28 (0.15 of which is exception_ptr)
- #include <exception> // for std::exception
- // <new> includes <exception>, <exception> includes <new>
- #include <new>
- // 0.01
- #include <initializer_list>
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST
- #ifdef __has_include
- #if __has_include(<bit>) && (__cplusplus >= 202002L || _HAS_CXX20)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST 1
- #endif
- #elif __cplusplus >= 202002L
- #define BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST 1
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST
- #define BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST 0
- #endif
- #endif
- #if BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST
- #include <bit>
- #if __cpp_lib_bit_cast < 201806L
- #undef BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST
- #define BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST 0
- #endif
- #endif
- #if BOOST_OUTCOME_SYSTEM_ERROR2_USE_STD_ADDRESSOF
- #include <memory> // for std::addressof
- #define BOOST_OUTCOME_SYSTEM_ERROR2_ADDRESS_OF(...) std::addressof(__VA_ARGS__)
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_ADDRESS_OF(...) (&__VA_ARGS__)
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14
- #if defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE) || __cplusplus >= 201400 || _MSC_VER >= 1910 /* VS2017 */
- //! Defined to be `constexpr` when on C++ 14 or better compilers. Usually automatic, can be overriden.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14 constexpr
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR14
- #endif
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20
- #if defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE) || __cplusplus >= 202000 || _HAS_CXX20
- //! Defined to be `constexpr` when on C++ 20 or better compilers. Usually automatic, can be overriden.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20 constexpr
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CONSTEXPR20
- #endif
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN
- #if defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE) || (_HAS_CXX17 && _MSC_VER >= 1911 /* VS2017.3 */)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN [[noreturn]]
- #endif
- #endif
- #if !defined(BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN)
- #ifdef __has_cpp_attribute
- #if __has_cpp_attribute(noreturn)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN [[noreturn]]
- #endif
- #endif
- #endif
- #if !defined(BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN)
- #if defined(_MSC_VER)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN __declspec(noreturn)
- #elif defined(__GNUC__)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN __attribute__((__noreturn__))
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN
- #endif
- #endif
- // GCCs before 7 don't grok [[noreturn]] virtual functions, and warn annoyingly
- #if defined(__GNUC__) && !defined(__clang__) && __GNUC__ < 7
- #undef BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NORETURN
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD
- #if defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE) || (_HAS_CXX17 && _MSC_VER >= 1911 /* VS2017.3 */)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD [[nodiscard]]
- #endif
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD
- #ifdef __has_cpp_attribute
- #if __has_cpp_attribute(nodiscard)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD [[nodiscard]]
- #endif
- #elif defined(__clang__)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD __attribute__((warn_unused_result))
- #elif defined(_MSC_VER)
- // _Must_inspect_result_ expands into this
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD \
- __declspec( \
- "SAL_name" \
- "(" \
- "\"_Must_inspect_result_\"" \
- "," \
- "\"\"" \
- "," \
- "\"2\"" \
- ")") __declspec("SAL_begin") __declspec("SAL_post") __declspec("SAL_mustInspect") __declspec("SAL_post") __declspec("SAL_checkReturn") __declspec("SAL_end")
- #endif
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NODISCARD
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI
- #if defined(BOOST_OUTCOME_STANDARDESE_IS_IN_THE_HOUSE) || (__clang_major__ >= 7 && !defined(__APPLE__))
- //! Defined to be `[[clang::trivial_abi]]` when on a new enough clang compiler. Usually automatic, can be overriden.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI [[clang::trivial_abi]]
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TRIVIAL_ABI
- #endif
- #endif
- #if defined(__cpp_concepts) && !defined(BOOST_OUTCOME_SYSTEM_ERROR2_DISABLE_CONCEPTS_SUPPORT)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_GLUE(x, y) x y
- #define BOOST_OUTCOME_SYSTEM_ERROR2_RETURN_ARG_COUNT(_1_, _2_, _3_, _4_, _5_, _6_, _7_, _8_, count, ...) count
- #define BOOST_OUTCOME_SYSTEM_ERROR2_EXPAND_ARGS(args) BOOST_OUTCOME_SYSTEM_ERROR2_RETURN_ARG_COUNT args
- #define BOOST_OUTCOME_SYSTEM_ERROR2_COUNT_ARGS_MAX8(...) BOOST_OUTCOME_SYSTEM_ERROR2_EXPAND_ARGS((__VA_ARGS__, 8, 7, 6, 5, 4, 3, 2, 1, 0))
- #define BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO2(name, count) name##count
- #define BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO1(name, count) BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO2(name, count)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO(name, count) BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO1(name, count)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_CALL_OVERLOAD(name, ...) BOOST_OUTCOME_SYSTEM_ERROR2_GLUE(BOOST_OUTCOME_SYSTEM_ERROR2_OVERLOAD_MACRO(name, BOOST_OUTCOME_SYSTEM_ERROR2_COUNT_ARGS_MAX8(__VA_ARGS__)), (__VA_ARGS__))
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND8(a, b, c, d, e, f, g, h) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND7(b, c, d, e, f, g, h)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND7(a, b, c, d, e, f, g) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND6(b, c, d, e, f, g)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND6(a, b, c, d, e, f) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND5(b, c, d, e, f)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND5(a, b, c, d, e) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND4(b, c, d, e)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND4(a, b, c, d) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND3(b, c, d)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND3(a, b, c) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND2(b, c)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND2(a, b) a &&BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND1(b)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND1(a) a
- //! Expands into a && b && c && ...
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(...) requires BOOST_OUTCOME_SYSTEM_ERROR2_CALL_OVERLOAD(BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES_EXPAND, __VA_ARGS__)
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(...) template <__VA_ARGS__>
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TEXPR(...) \
- requires { (__VA_ARGS__); }
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(...) (__VA_ARGS__)
- #if !defined(_MSC_VER) || _MSC_FULL_VER >= 192400000 // VS 2019 16.3 is broken here
- #define BOOST_OUTCOME_SYSTEM_ERROR2_REQUIRES(...) requires(__VA_ARGS__)
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_REQUIRES(...)
- #endif
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(...) template <__VA_ARGS__
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(...) , __VA_ARGS__ >
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TEXPR(...) typename = decltype(__VA_ARGS__)
- #ifdef _MSC_VER
- // MSVC gives an error if every specialisation of a template is always ill-formed, so
- // the more powerful SFINAE form below causes pukeage :(
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(...) typename = typename std::enable_if<(__VA_ARGS__)>::type
- #else
- #define BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(...) typename std::enable_if<(__VA_ARGS__), bool>::type = true
- #endif
- #define BOOST_OUTCOME_SYSTEM_ERROR2_REQUIRES(...)
- #endif
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE
- //! The system_error2 namespace name.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE system_error2
- //! Begins the system_error2 namespace.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN \
- namespace system_error2 \
- {
- //! Ends the system_error2 namespace.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END }
- #endif
- //! Namespace for the library
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
- //! Namespace for user specialised traits
- namespace traits
- {
- /*! Specialise to true if you guarantee that a type is move bitcopying (i.e.
- its move constructor equals copying bits from old to new, old is left in a
- default constructed state, and calling the destructor on a default constructed
- instance is trivial). All trivially copyable types are move bitcopying by
- definition, and that is the unspecialised implementation.
- */
- template <class T> struct is_move_bitcopying
- {
- static constexpr bool value = std::is_trivially_copyable<T>::value;
- };
- } // namespace traits
- namespace detail
- {
- #if __cplusplus >= 201400 || _MSC_VER >= 1910 /* VS2017 */
- inline constexpr size_t cstrlen(const char *str)
- {
- const char *end = nullptr;
- for(end = str; *end != 0; ++end) // NOLINT
- ;
- return end - str;
- }
- #else
- inline constexpr size_t cstrlen_(const char *str, size_t acc) { return (str[0] == 0) ? acc : cstrlen_(str + 1, acc + 1); }
- inline constexpr size_t cstrlen(const char *str) { return cstrlen_(str, 0); }
- #endif
- #if(__cplusplus >= 202002L || _MSVC_LANG >= 202002L) && __cpp_lib_remove_cvref >= 201711L
- template <class T> using remove_cvref = std::remove_cvref<T>;
- #else
- template <class T> struct remove_cvref
- {
- using type = typename std::remove_cv<typename std::remove_reference<T>::type>::type;
- };
- #endif
- /* A partially compliant implementation of C++20's std::bit_cast function contributed
- by Jesse Towner.
- Our bit_cast is only guaranteed to be constexpr when both the input and output
- arguments are either integrals or enums. However, this covers most use cases
- since the vast majority of status_codes have an underlying type that is either
- an integral or enum. We still attempt a constexpr union-based type pun for non-array
- input types, which some compilers accept. For array inputs, we fall back to
- non-constexpr memmove.
- */
- template <class T> using is_integral_or_enum = std::integral_constant<bool, std::is_integral<T>::value || std::is_enum<T>::value>;
- template <class To, class From> using is_static_castable = std::integral_constant<bool, is_integral_or_enum<To>::value && is_integral_or_enum<From>::value>;
- template <class To, class From>
- using is_union_castable = std::integral_constant<bool, !is_static_castable<To, From>::value && !std::is_array<To>::value && !std::is_array<From>::value>;
- template <class To, class From>
- using is_bit_castable =
- std::integral_constant<bool, sizeof(To) == sizeof(From) && traits::is_move_bitcopying<To>::value && traits::is_move_bitcopying<From>::value>;
- template <class To, class From> union bit_cast_union
- {
- From source;
- To target;
- };
- #if BOOST_OUTCOME_SYSTEM_ERROR2_HAVE_BIT_CAST
- using std::bit_cast; // available for all trivially copyable types
- // For move bit copying types
- template <class To, class From>
- requires(is_bit_castable<To, From>::value //
- &&is_union_castable<To, From>::value //
- && (!std::is_trivially_copyable_v<From> //
- || !std::is_trivially_copyable_v<To>) ) //
- constexpr To bit_cast(const From &from) noexcept
- {
- return bit_cast_union<To, From>{from}.target;
- }
- template <class To, class From>
- requires(is_bit_castable<To, From>::value //
- && !is_union_castable<To, From>::value //
- && (!std::is_trivially_copyable_v<From> //
- || !std::is_trivially_copyable_v<To>) ) //
- To bit_cast(const From &from)
- noexcept
- {
- bit_cast_union<To, From> ret;
- memmove(&ret.source, &from, sizeof(ret.source));
- return ret.target;
- }
- #else
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, int = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_bit_castable<To, From>::value //
- &&is_static_castable<To, From>::value //
- && !is_union_castable<To, From>::value))
- constexpr To bit_cast(const From &from) noexcept { return static_cast<To>(from); }
- #if defined(__GNUC__) && !defined(__clang__)
- #pragma GCC diagnostic push
- #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
- #endif
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, long = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_bit_castable<To, From>::value //
- && !is_static_castable<To, From>::value //
- && is_union_castable<To, From>::value))
- constexpr To bit_cast(const From &from) noexcept { return bit_cast_union<To, From>{from}.target; }
- #if defined(__GNUC__) && !defined(__clang__)
- #pragma GCC diagnostic pop
- #endif
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, short = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_bit_castable<To, From>::value //
- && !is_static_castable<To, From>::value //
- && !is_union_castable<To, From>::value))
- To bit_cast(const From &from) noexcept
- {
- bit_cast_union<To, From> ret;
- memmove(&ret.source, &from, sizeof(ret.source));
- return ret.target;
- }
- #endif
- /* erasure_cast performs a bit_cast with additional rules to handle types
- of differing sizes. For integral & enum types, it may perform a narrowing
- or widing conversion with static_cast if necessary, before doing the final
- conversion with bit_cast. When casting to or from non-integral, non-enum
- types it may insert the value into another object with extra padding bytes
- to satisfy bit_cast's preconditions that both types have the same size. */
- template <class To, class From>
- using is_erasure_castable = std::integral_constant<bool, traits::is_move_bitcopying<To>::value && traits::is_move_bitcopying<From>::value>;
- template <class T, bool = std::is_enum<T>::value> struct identity_or_underlying_type
- {
- using type = T;
- };
- template <class T> struct identity_or_underlying_type<T, true>
- {
- using type = typename std::underlying_type<T>::type;
- };
- template <class OfSize, class OfSign>
- using erasure_integer_type = typename std::conditional<std::is_signed<typename identity_or_underlying_type<OfSign>::type>::value,
- typename std::make_signed<typename identity_or_underlying_type<OfSize>::type>::type,
- typename std::make_unsigned<typename identity_or_underlying_type<OfSize>::type>::type>::type;
- template <class ErasedType, std::size_t N> struct padded_erasure_object
- {
- static_assert(traits::is_move_bitcopying<ErasedType>::value, "ErasedType must be TriviallyCopyable or MoveBitcopying");
- static_assert(alignof(ErasedType) <= sizeof(ErasedType), "ErasedType must not be over-aligned");
- ErasedType value;
- char padding[N];
- constexpr explicit padded_erasure_object(const ErasedType &v) noexcept
- : value(v)
- , padding{}
- {
- }
- };
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value && (sizeof(To) == sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<To>(from); }
- #if defined(_WIN32) || defined(__APPLE__) || __LITTLE_ENDIAN__ || (defined(__BYTE_ORDER) && defined(__LITTLE_ENDIAN) && __BYTE_ORDER == __LITTLE_ENDIAN)
- // We can avoid the type pun on little endian architectures which can aid optimisation
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, long = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value &&is_static_castable<To, From>::value && (sizeof(To) < sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return static_cast<To>(bit_cast<erasure_integer_type<From, To>>(from)); }
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, int = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value &&is_static_castable<To, From>::value && (sizeof(To) > sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<To>(static_cast<erasure_integer_type<To, From>>(from)); }
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, short = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value && !is_static_castable<To, From>::value && (sizeof(To) < sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<padded_erasure_object<To, sizeof(From) - sizeof(To)>>(from).value; }
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, char = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value && !is_static_castable<To, From>::value && (sizeof(To) > sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<To>(padded_erasure_object<From, sizeof(To) - sizeof(From)>{from}); }
- #else
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, short = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value && (sizeof(To) < sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<padded_erasure_object<To, sizeof(From) - sizeof(To)>>(from).value; }
- BOOST_OUTCOME_SYSTEM_ERROR2_TEMPLATE(class To, class From, char = 5)
- BOOST_OUTCOME_SYSTEM_ERROR2_TREQUIRES(BOOST_OUTCOME_SYSTEM_ERROR2_TPRED(is_erasure_castable<To, From>::value && (sizeof(To) > sizeof(From))))
- constexpr To erasure_cast(const From &from) noexcept { return bit_cast<To>(padded_erasure_object<From, sizeof(To) - sizeof(From)>{from}); }
- #endif
- } // namespace detail
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
- #ifndef BOOST_OUTCOME_SYSTEM_ERROR2_FATAL
- #ifdef BOOST_OUTCOME_SYSTEM_ERROR2_NOT_POSIX
- #error If BOOST_OUTCOME_SYSTEM_ERROR2_NOT_POSIX is defined, you must define your own BOOST_OUTCOME_SYSTEM_ERROR2_FATAL implementation!
- #endif
- #include <cstdlib> // for abort
- #ifdef __APPLE__
- #include <unistd.h> // for write
- #endif
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_BEGIN
- namespace detail
- {
- namespace avoid_stdio_include
- {
- #if !defined(__APPLE__) && !defined(_MSC_VER)
- extern "C" ptrdiff_t write(int, const void *, size_t);
- #elif defined(_MSC_VER)
- extern ptrdiff_t write(int, const void *, size_t);
- #if(defined(__x86_64__) || defined(_M_X64)) || (defined(__aarch64__) || defined(_M_ARM64)) || (defined(__arm__) || defined(_M_ARM))
- #pragma comment(linker, "/alternatename:?write@avoid_stdio_include@detail@system_error2@@YA_JHPEBX_K@Z=write")
- #elif defined(__x86__) || defined(_M_IX86) || defined(__i386__)
- #pragma comment(linker, "/alternatename:?write@avoid_stdio_include@detail@system_error2@@YAHHPBXI@Z=_write")
- #else
- #error Unknown architecture
- #endif
- #endif
- } // namespace avoid_stdio_include
- inline void do_fatal_exit(const char *msg)
- {
- using namespace avoid_stdio_include;
- write(2 /*stderr*/, msg, cstrlen(msg));
- write(2 /*stderr*/, "\n", 1);
- abort();
- }
- } // namespace detail
- BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE_END
- //! Prints msg to stderr, and calls `std::terminate()`. Can be overriden via predefinition.
- #define BOOST_OUTCOME_SYSTEM_ERROR2_FATAL(msg) ::BOOST_OUTCOME_SYSTEM_ERROR2_NAMESPACE::detail::do_fatal_exit(msg)
- #endif
- #endif
|