416 lines
22 KiB
C++
416 lines
22 KiB
C++
|
/* 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
|
||
|
|
||
|
#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
|
||
|
|
||
|
/* 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); }
|
||
|
|
||
|
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}); }
|
||
|
} // 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
|