1
0
mirror of https://github.com/RPCS3/rpcs3.git synced 2024-11-23 03:02:53 +01:00
rpcs3/Utilities/BEType.h

791 lines
16 KiB
C
Raw Normal View History

#pragma once
2016-02-01 22:55:43 +01:00
#include "types.h"
#include <cstring>
2016-08-13 15:36:04 +02:00
// 128-bit vector type and also se_storage<> storage type
2016-02-01 22:55:43 +01:00
union alignas(16) v128
{
2016-02-01 22:55:43 +01:00
char _bytes[16];
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t N, std::size_t M>
2016-02-01 22:55:43 +01:00
struct masked_array_t // array type accessed as (index ^ M)
2014-10-07 15:35:44 +02:00
{
T m_data[N];
2014-10-07 15:35:44 +02:00
public:
2016-08-13 15:36:04 +02:00
T& operator[](std::size_t index)
{
return m_data[index ^ M];
}
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
const T& operator[](std::size_t index) const
{
return m_data[index ^ M];
}
};
2014-10-07 15:35:44 +02:00
2016-02-01 22:55:43 +01:00
#if IS_LE_MACHINE == 1
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t N = 16 / sizeof(T)>
using normal_array_t = masked_array_t<T, N, 0>;
template <typename T, std::size_t N = 16 / sizeof(T)>
using reversed_array_t = masked_array_t<T, N, N - 1>;
#endif
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
normal_array_t<u64> _u64;
normal_array_t<s64> _s64;
reversed_array_t<u64> u64r;
reversed_array_t<s64> s64r;
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
normal_array_t<u32> _u32;
normal_array_t<s32> _s32;
reversed_array_t<u32> u32r;
reversed_array_t<s32> s32r;
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
normal_array_t<u16> _u16;
normal_array_t<s16> _s16;
reversed_array_t<u16> u16r;
reversed_array_t<s16> s16r;
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
normal_array_t<u8> _u8;
normal_array_t<s8> _s8;
reversed_array_t<u8> u8r;
reversed_array_t<s8> s8r;
2014-10-07 15:35:44 +02:00
2016-08-13 15:36:04 +02:00
normal_array_t<f32> _f;
normal_array_t<f64> _d;
reversed_array_t<f32> fr;
reversed_array_t<f64> dr;
2014-10-07 15:35:44 +02:00
2014-10-07 23:37:04 +02:00
__m128 vf;
__m128i vi;
2015-03-21 15:29:33 +01:00
__m128d vd;
2016-02-01 22:55:43 +01:00
struct bit_array_128
{
u64 m_data[2];
public:
class bit_element
{
u64& data;
const u64 mask;
public:
bit_element(u64& data, const u64 mask)
: data(data)
, mask(mask)
{
}
operator bool() const
{
return (data & mask) != 0;
}
2016-08-13 15:36:04 +02:00
bit_element& operator=(const bool right)
{
if (right)
{
data |= mask;
}
else
{
data &= ~mask;
}
return *this;
}
2016-08-13 15:36:04 +02:00
bit_element& operator=(const bit_element& right)
{
if (right)
{
data |= mask;
}
else
{
data &= ~mask;
}
return *this;
}
};
2015-02-04 16:29:34 +01:00
// Index 0 returns the MSB and index 127 returns the LSB
2016-08-13 15:36:04 +02:00
bit_element operator[](u32 index)
{
2016-02-01 22:55:43 +01:00
#if IS_LE_MACHINE == 1
return bit_element(m_data[1 - (index >> 6)], 0x8000000000000000ull >> (index & 0x3F));
2015-02-04 16:29:34 +01:00
#endif
}
2015-02-04 16:29:34 +01:00
// Index 0 returns the MSB and index 127 returns the LSB
2016-08-13 15:36:04 +02:00
bool operator[](u32 index) const
{
2016-02-01 22:55:43 +01:00
#if IS_LE_MACHINE == 1
return (m_data[1 - (index >> 6)] & (0x8000000000000000ull >> (index & 0x3F))) != 0;
2015-02-04 16:29:34 +01:00
#endif
}
2016-08-13 15:36:04 +02:00
} _bit;
static v128 from64(u64 _0, u64 _1 = 0)
{
v128 ret;
2014-09-15 00:17:24 +02:00
ret._u64[0] = _0;
ret._u64[1] = _1;
return ret;
}
static v128 from64r(u64 _1, u64 _0 = 0)
2014-10-07 15:35:44 +02:00
{
return from64(_0, _1);
}
static v128 from32(u32 _0, u32 _1 = 0, u32 _2 = 0, u32 _3 = 0)
{
v128 ret;
2014-09-15 00:17:24 +02:00
ret._u32[0] = _0;
ret._u32[1] = _1;
ret._u32[2] = _2;
ret._u32[3] = _3;
return ret;
}
static v128 from32r(u32 _3, u32 _2 = 0, u32 _1 = 0, u32 _0 = 0)
2014-10-02 12:29:20 +02:00
{
2014-10-07 15:35:44 +02:00
return from32(_0, _1, _2, _3);
2014-10-02 12:29:20 +02:00
}
static v128 from32p(u32 value)
2014-10-02 12:29:20 +02:00
{
v128 ret;
2015-03-29 13:00:10 +02:00
ret.vi = _mm_set1_epi32(static_cast<s32>(value));
return ret;
}
static v128 from16p(u16 value)
2015-03-29 13:00:10 +02:00
{
v128 ret;
2015-03-29 13:00:10 +02:00
ret.vi = _mm_set1_epi16(static_cast<s16>(value));
2014-10-07 23:37:04 +02:00
return ret;
}
static v128 from8p(u8 value)
2014-10-07 23:37:04 +02:00
{
v128 ret;
2015-03-29 13:00:10 +02:00
ret.vi = _mm_set1_epi8(static_cast<s8>(value));
2014-10-02 12:29:20 +02:00
return ret;
}
static v128 fromBit(u32 bit)
{
v128 ret = {};
ret._bit[bit] = true;
return ret;
}
static v128 fromV(__m128i value)
2014-10-07 23:37:04 +02:00
{
v128 ret;
2014-10-07 23:37:04 +02:00
ret.vi = value;
return ret;
}
static v128 fromF(__m128 value)
2015-03-21 00:36:05 +01:00
{
v128 ret;
2015-03-21 00:36:05 +01:00
ret.vf = value;
return ret;
}
static v128 fromD(__m128d value)
2015-03-21 15:29:33 +01:00
{
v128 ret;
2015-03-21 15:29:33 +01:00
ret.vd = value;
return ret;
}
static inline v128 add8(const v128& left, const v128& right)
{
2014-10-07 23:37:04 +02:00
return fromV(_mm_add_epi8(left.vi, right.vi));
}
static inline v128 add16(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromV(_mm_add_epi16(left.vi, right.vi));
}
static inline v128 add32(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromV(_mm_add_epi32(left.vi, right.vi));
}
static inline v128 addfs(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromF(_mm_add_ps(left.vf, right.vf));
}
static inline v128 addfd(const v128& left, const v128& right)
2015-03-21 15:29:33 +01:00
{
return fromD(_mm_add_pd(left.vd, right.vd));
}
static inline v128 sub8(const v128& left, const v128& right)
2014-10-07 23:37:04 +02:00
{
return fromV(_mm_sub_epi8(left.vi, right.vi));
}
static inline v128 sub16(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromV(_mm_sub_epi16(left.vi, right.vi));
}
static inline v128 sub32(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromV(_mm_sub_epi32(left.vi, right.vi));
}
static inline v128 subfs(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromF(_mm_sub_ps(left.vf, right.vf));
}
static inline v128 subfd(const v128& left, const v128& right)
2015-03-21 15:29:33 +01:00
{
return fromD(_mm_sub_pd(left.vd, right.vd));
}
static inline v128 maxu8(const v128& left, const v128& right)
2015-03-21 00:36:05 +01:00
{
return fromV(_mm_max_epu8(left.vi, right.vi));
}
static inline v128 minu8(const v128& left, const v128& right)
2014-10-07 23:37:04 +02:00
{
return fromV(_mm_min_epu8(left.vi, right.vi));
}
static inline v128 eq8(const v128& left, const v128& right)
{
2014-10-07 23:37:04 +02:00
return fromV(_mm_cmpeq_epi8(left.vi, right.vi));
2015-03-29 13:00:10 +02:00
}
static inline v128 eq16(const v128& left, const v128& right)
2015-03-29 13:00:10 +02:00
{
return fromV(_mm_cmpeq_epi16(left.vi, right.vi));
}
static inline v128 eq32(const v128& left, const v128& right)
2015-03-29 13:00:10 +02:00
{
return fromV(_mm_cmpeq_epi32(left.vi, right.vi));
}
2016-08-13 15:36:04 +02:00
bool operator==(const v128& right) const
{
return _u64[0] == right._u64[0] && _u64[1] == right._u64[1];
}
2016-08-13 15:36:04 +02:00
bool operator!=(const v128& right) const
{
return _u64[0] != right._u64[0] || _u64[1] != right._u64[1];
}
2014-10-07 23:37:04 +02:00
// result = (~left) & (right)
static inline v128 andnot(const v128& left, const v128& right)
2014-10-07 23:37:04 +02:00
{
return fromV(_mm_andnot_si128(left.vi, right.vi));
}
void clear()
{
_u64[0] = 0;
_u64[1] = 0;
}
};
template <typename T, std::size_t N, std::size_t M>
struct offset32_array<v128::masked_array_t<T, N, M>>
{
template <typename Arg>
static inline u32 index32(const Arg& arg)
{
2018-09-03 17:46:14 +02:00
return u32{sizeof(T)} * (static_cast<u32>(arg) ^ static_cast<u32>(M));
}
};
2016-08-13 15:36:04 +02:00
inline v128 operator|(const v128& left, const v128& right)
{
return v128::fromV(_mm_or_si128(left.vi, right.vi));
}
2016-08-13 15:36:04 +02:00
inline v128 operator&(const v128& left, const v128& right)
{
return v128::fromV(_mm_and_si128(left.vi, right.vi));
}
2016-08-13 15:36:04 +02:00
inline v128 operator^(const v128& left, const v128& right)
{
return v128::fromV(_mm_xor_si128(left.vi, right.vi));
}
2016-08-13 15:36:04 +02:00
inline v128 operator~(const v128& other)
{
return v128::from64(~other._u64[0], ~other._u64[1]);
}
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t Align, std::size_t Size>
2016-02-01 22:55:43 +01:00
struct se_storage
{
struct type
{
alignas(Align) std::byte data[Size];
};
2016-05-26 17:06:06 +02:00
// Unoptimized generic byteswap for unaligned data
static void reverse(u8* dst, const u8* src)
{
for (std::size_t i = 0; i < Size; i++)
{
dst[i] = src[Size - 1 - i];
}
}
static type to(const T& src)
{
type result;
reverse(reinterpret_cast<u8*>(&result), reinterpret_cast<const u8*>(&src));
return result;
}
static T from(const type& src)
{
T result;
reverse(reinterpret_cast<u8*>(&result), reinterpret_cast<const u8*>(&src));
return result;
}
};
2016-08-13 15:36:04 +02:00
template <typename T>
2016-05-26 17:06:06 +02:00
struct se_storage<T, 2, 2>
{
using type = u16;
2016-02-01 22:55:43 +01:00
static constexpr u16 swap(u16 src)
2015-11-26 09:13:33 +01:00
{
#if defined(__GNUG__)
return __builtin_bswap16(src);
#else
return _byteswap_ushort(src);
#endif
}
static inline u16 to(const T& src)
{
return swap(std::bit_cast<u16>(src));
}
static inline T from(u16 src)
{
return std::bit_cast<T, u16>(swap(src));
}
};
2016-08-13 15:36:04 +02:00
template <typename T>
2016-05-26 17:06:06 +02:00
struct se_storage<T, 4, 4>
{
using type = u32;
2016-02-01 22:55:43 +01:00
static constexpr u32 swap(u32 src)
2015-11-26 09:13:33 +01:00
{
#if defined(__GNUG__)
return __builtin_bswap32(src);
#else
return _byteswap_ulong(src);
#endif
}
static inline u32 to(const T& src)
{
return swap(std::bit_cast<u32>(src));
}
static inline T from(u32 src)
{
return std::bit_cast<T, u32>(swap(src));
}
};
2016-08-13 15:36:04 +02:00
template <typename T>
2016-05-26 17:06:06 +02:00
struct se_storage<T, 8, 8>
{
using type = u64;
2016-02-01 22:55:43 +01:00
static constexpr u64 swap(u64 src)
2015-11-26 09:13:33 +01:00
{
#if defined(__GNUG__)
return __builtin_bswap64(src);
#else
return _byteswap_uint64(src);
#endif
}
static inline u64 to(const T& src)
{
return swap(std::bit_cast<u64>(src));
}
static inline T from(u64 src)
{
return std::bit_cast<T, u64>(swap(src));
}
};
2016-08-13 15:36:04 +02:00
template <typename T>
2016-05-26 17:06:06 +02:00
struct se_storage<T, 16, 16>
{
using type = v128;
2016-02-01 22:55:43 +01:00
static inline v128 swap(const v128& src)
{
2017-12-05 21:48:01 +01:00
return v128::from64(se_storage<u64>::swap(src._u64[1]), se_storage<u64>::swap(src._u64[0]));
2016-02-01 22:55:43 +01:00
}
static inline v128 to(const T& src)
{
return swap(std::bit_cast<v128>(src));
}
static inline T from(const v128& src)
{
return std::bit_cast<T, v128>(swap(src));
}
};
2016-02-01 22:55:43 +01:00
// Switched endianness
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t Align>
2016-05-26 17:06:06 +02:00
class se_t<T, true, Align>
{
using type = typename std::remove_cv<T>::type;
2016-05-26 17:06:06 +02:00
using stype = typename se_storage<type, Align>::type;
using storage = se_storage<type, Align>;
stype m_data;
static_assert(!std::is_pointer<type>::value, "se_t<> error: invalid type (pointer)");
static_assert(!std::is_reference<type>::value, "se_t<> error: invalid type (reference)");
static_assert(!std::is_array<type>::value, "se_t<> error: invalid type (array)");
2016-02-01 22:55:43 +01:00
static_assert(sizeof(type) == alignof(type), "se_t<> error: unexpected alignment");
public:
se_t() = default;
2015-09-15 18:23:17 +02:00
se_t(const se_t& right) = default;
se_t(type value)
: m_data(storage::to(value))
{
}
2015-09-15 18:23:17 +02:00
type value() const
{
return storage::from(m_data);
}
stype& raw()
{
return m_data;
}
2016-08-13 15:36:04 +02:00
se_t& operator=(const se_t&) = default;
2016-08-13 15:36:04 +02:00
se_t& operator=(type value)
{
return m_data = storage::to(value), *this;
}
2016-02-01 22:55:43 +01:00
using simple_type = simple_t<T>;
operator type() const
{
return storage::from(m_data);
}
};
2016-02-01 22:55:43 +01:00
// Native endianness
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t Align>
2016-05-26 17:06:06 +02:00
class se_t<T, false, Align>
{
using type = typename std::remove_cv<T>::type;
2016-05-26 17:06:06 +02:00
using stype = typename se_storage<type, Align>::type;
using storage = se_storage<type, Align>;
static_assert(!std::is_pointer<type>::value, "se_t<> error: invalid type (pointer)");
static_assert(!std::is_reference<type>::value, "se_t<> error: invalid type (reference)");
static_assert(!std::is_array<type>::value, "se_t<> error: invalid type (array)");
2016-02-01 22:55:43 +01:00
static_assert(sizeof(type) == alignof(type), "se_t<> error: unexpected alignment");
2016-05-26 17:06:06 +02:00
stype m_data;
public:
se_t() = default;
2016-05-26 17:06:06 +02:00
se_t(type value)
: m_data(std::bit_cast<stype>(value))
{
}
2016-05-26 17:06:06 +02:00
type value() const
2016-02-01 22:55:43 +01:00
{
return std::bit_cast<type>(m_data);
2016-02-01 22:55:43 +01:00
}
stype& raw()
{
return m_data;
}
2016-08-13 15:36:04 +02:00
se_t& operator=(const se_t& value) = default;
2016-08-13 15:36:04 +02:00
se_t& operator=(type value)
{
m_data = std::bit_cast<stype>(value);
return *this;
}
2016-02-01 22:55:43 +01:00
using simple_type = simple_t<T>;
2016-05-26 17:06:06 +02:00
operator type() const
{
return value();
}
2015-09-15 18:23:17 +02:00
};
2016-08-13 15:36:04 +02:00
// se_t<> with native endianness
template <typename T, std::size_t Align = alignof(T)>
using nse_t = se_t<T, false, Align>;
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator+=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value += right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator-=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value -= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator*=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value *= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator/=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value /= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator%=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value %= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator&=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value &= right);
}
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator|=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value |= right);
}
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator^=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value ^= right);
}
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator<<=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value <<= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align, typename T1>
inline se_t<T, Se, Align>& operator>>=(se_t<T, Se, Align>& left, const T1& right)
{
auto value = left.value();
return left = (value >>= right);
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align>
inline se_t<T, Se, Align> operator++(se_t<T, Se, Align>& left, int)
{
auto value = left.value();
auto result = value++;
left = value;
return result;
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align>
inline se_t<T, Se, Align> operator--(se_t<T, Se, Align>& left, int)
{
auto value = left.value();
auto result = value--;
left = value;
return result;
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align>
inline se_t<T, Se, Align>& operator++(se_t<T, Se, Align>& right)
{
auto value = right.value();
return right = ++value;
}
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align>
inline se_t<T, Se, Align>& operator--(se_t<T, Se, Align>& right)
{
auto value = right.value();
return right = --value;
}
2016-02-01 22:55:43 +01:00
#if IS_LE_MACHINE == 1
2016-08-13 15:36:04 +02:00
template <typename T, std::size_t Align = alignof(T)>
using be_t = se_t<T, true, Align>;
template <typename T, std::size_t Align = alignof(T)>
using le_t = se_t<T, false, Align>;
#endif
2016-02-01 22:55:43 +01:00
// Type converter: converts native endianness arithmetic/enum types to appropriate se_t<> type
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, typename = void>
2016-02-01 22:55:43 +01:00
struct to_se
2014-09-01 18:16:44 +02:00
{
2016-08-13 15:36:04 +02:00
template <typename T2, typename = void>
2016-05-26 17:06:06 +02:00
struct to_se_
{
using type = T2;
};
2016-08-13 15:36:04 +02:00
template <typename T2>
2016-05-26 17:06:06 +02:00
struct to_se_<T2, std::enable_if_t<std::is_arithmetic<T2>::value || std::is_enum<T2>::value>>
{
using type = se_t<T2, Se>;
};
2016-02-01 22:55:43 +01:00
// Convert arithmetic and enum types
2016-05-26 17:06:06 +02:00
using type = typename to_se_<T>::type;
2015-01-28 13:59:16 +01:00
};
2014-09-01 18:16:44 +02:00
2016-08-13 15:36:04 +02:00
template <bool Se>
struct to_se<v128, Se>
{
using type = se_t<v128, Se>;
};
template <bool Se>
struct to_se<u128, Se>
{
using type = se_t<u128, Se>;
};
2016-02-01 22:55:43 +01:00
2016-08-13 15:36:04 +02:00
template <bool Se>
struct to_se<s128, Se>
{
using type = se_t<s128, Se>;
};
template <bool Se>
struct to_se<bool, Se>
{
using type = bool;
};
template <bool Se>
struct to_se<char, Se>
{
using type = char;
};
template <bool Se>
struct to_se<u8, Se>
{
using type = u8;
};
template <bool Se>
struct to_se<s8, Se>
{
using type = s8;
};
template <typename T, bool Se>
struct to_se<const T, Se, std::enable_if_t<!std::is_array<T>::value>>
2015-06-15 19:00:08 +02:00
{
2016-02-01 22:55:43 +01:00
// Move const qualifier
using type = const typename to_se<T, Se>::type;
2015-06-15 19:00:08 +02:00
};
2016-08-13 15:36:04 +02:00
template <typename T, bool Se>
2016-02-01 22:55:43 +01:00
struct to_se<volatile T, Se, std::enable_if_t<!std::is_array<T>::value && !std::is_const<T>::value>>
2015-06-15 19:00:08 +02:00
{
2016-02-01 22:55:43 +01:00
// Move volatile qualifier
using type = volatile typename to_se<T, Se>::type;
2015-06-15 19:00:08 +02:00
};
2016-08-13 15:36:04 +02:00
template <typename T, bool Se>
2016-02-01 22:55:43 +01:00
struct to_se<T[], Se>
2015-01-28 13:59:16 +01:00
{
2016-02-01 22:55:43 +01:00
// Move array qualifier
using type = typename to_se<T, Se>::type[];
2015-01-28 13:59:16 +01:00
};
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t N>
2016-02-01 22:55:43 +01:00
struct to_se<T[N], Se>
2015-01-28 13:59:16 +01:00
{
2016-02-01 22:55:43 +01:00
// Move array qualifier
using type = typename to_se<T, Se>::type[N];
2014-09-01 18:16:44 +02:00
};
2016-02-01 22:55:43 +01:00
// BE/LE aliases for to_se<>
#if IS_LE_MACHINE == 1
2016-08-13 15:36:04 +02:00
template <typename T>
using to_be_t = typename to_se<T, true>::type;
template <typename T>
using to_le_t = typename to_se<T, false>::type;
#endif
2016-02-01 22:55:43 +01:00
// BE/LE aliases for atomic_t
#if IS_LE_MACHINE == 1
2016-08-13 15:36:04 +02:00
template <typename T>
using atomic_be_t = atomic_t<be_t<T>>;
template <typename T>
using atomic_le_t = atomic_t<le_t<T>>;
2016-02-01 22:55:43 +01:00
#endif
2015-06-15 19:00:08 +02:00
2016-08-13 15:36:04 +02:00
template <typename T, bool Se, std::size_t Align>
struct fmt_unveil<se_t<T, Se, Align>, void>
{
using type = typename fmt_unveil<T>::type;
2016-08-07 15:59:46 +02:00
static inline auto get(const se_t<T, Se, Align>& arg)
2016-02-01 22:55:43 +01:00
{
return fmt_unveil<T>::get(arg);
}
};