11#include <cosmos/types.hxx>
12#include <cosmos/utils.hxx>
21namespace cosmos::net {
35#if __BYTE_ORDER == __LITTLE_ENDIAN
38#elif __BYTE_ORDER == __BIG_ENDIAN
42# error "failed to determine endianness"
45constexpr auto foreign_endian = (
our_endian == Endian::LITTLE ? Endian::BIG : Endian::LITTLE);
47template <Endian endian>
53 static constexpr Endian other = Endian::BIG;
58 static constexpr Endian other = Endian::LITTLE;
72 return bswap_16(value);
77 return bswap_32(value);
82 return bswap_64(value);
111enum class RawLittleInt16 : uint16_t {};
112enum class RawLittleInt32 : uint32_t {};
113enum class RawLittleInt64 : uint64_t {};
114enum class RawBigInt16 : uint16_t {};
115enum class RawBigInt32 : uint32_t {};
116enum class RawBigInt64 : uint64_t {};
117using RawNetInt16 = RawBigInt16;
118using RawNetInt32 = RawBigInt32;
119using RawNetInt64 = RawBigInt64;
121template <
typename UINT, Endian endian>
127 using Int = RawLittleInt16;
131 using Int = RawLittleInt32;
135 using Int = RawLittleInt64;
139 using Int = RawBigInt16;
143 using Int = RawBigInt32;
147 using Int = RawBigInt64;
158template <
typename T, Endian endianness>
174 m_egg{to_integral(rint)} {
179 m_egg{toTargetEndianness(egg)} {
184 m_egg{toTargetEndianness(other.toHost())} {
187 void setFromHost(
const T egg) {
188 m_egg = toTargetEndianness(egg);
191 constexpr T toHost()
const {
192 if (our_endian == endianness) {
195 return swap_byte_order(m_egg);
204 return RawInt{m_egg};
209 constexpr static T toTargetEndianness(
const T egg) {
210 if (our_endian == endianness) {
222using LittleInt16 = EndianNumber<uint16_t, Endian::LITTLE>;
223using LittleInt32 = EndianNumber<uint32_t, Endian::LITTLE>;
224using LittleInt64 = EndianNumber<uint64_t, Endian::LITTLE>;
226using BigInt16 = EndianNumber<uint16_t, Endian::BIG>;
227using BigInt32 = EndianNumber<uint32_t, Endian::BIG>;
228using BigInt64 = EndianNumber<uint64_t, Endian::BIG>;
230using NetInt16 = BigInt16;
231using NetInt32 = BigInt32;
232using NetInt64 = BigInt64;
234using HostInt16 = EndianNumber<uint16_t, our_endian>;
235using HostInt32 = EndianNumber<uint32_t, our_endian>;
236using HostInt64 = EndianNumber<uint64_t, our_endian>;
T to_host_order(T network)
Return the host byte order version of network.
constexpr auto our_endian
The byte order setting for the current host.
Endian
Differentiation between different endianness storage format.
@ LITTLE
Little endian. Lower value bits are stored first.
@ BIG
Big endian. Higher value bits are stored first.
T to_network_order(T host)
Return the network byte order version of host.
uint16_t swap_byte_order(uint16_t value)
Return the reversed byte order for the given 16 bit value.
An endianness aware unsigned integer.
constexpr EndianNumber()
Construct a zero-initialized integer.
constexpr EndianNumber(const no_init_t)
Leave member undefined e.g. for use with placement new or for performance reasons.
constexpr EndianNumber(const EndianNumber< T, EndianTraits< endianness >::other > other)
Constructs the number from an EndianNumber of differing Endian type.
constexpr EndianNumber(const T egg)
Constructs the number from a native integer that will possibly be converted into the correct byte ord...
constexpr EndianNumber(const RawInt rint)
Construct the number from a raw integer in the correct byte order.
Type used to invoke constructors that explicitly don't zero-initialize low level data structures.