libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
byte_order.hxx
Go to the documentation of this file.
1#pragma once
2
3// glibc
4#include <endian.h>
5#include <byteswap.h>
6
7// Linux
8#include <arpa/inet.h>
9
10// cosmos
11#include <cosmos/types.hxx>
12#include <cosmos/utils.hxx>
13
21namespace cosmos::net {
22
24enum class Endian {
26 LITTLE,
28 BIG
29};
30
31// TODO: in C++20 the <bits> header provides language builtin support for
32// checking the endianness during compile time, so the preprocessor defines
33// will no longer be necessary.
34
35#if __BYTE_ORDER == __LITTLE_ENDIAN
37constexpr auto our_endian = Endian::LITTLE;
38#elif __BYTE_ORDER == __BIG_ENDIAN
40constexpr auto our_endian = Endian::BIG;
41#else
42# error "failed to determine endianness"
43#endif
44
45constexpr auto foreign_endian = (our_endian == Endian::LITTLE ? Endian::BIG : Endian::LITTLE);
46
47template <Endian endian>
49};
50
51template <>
53 static constexpr Endian other = Endian::BIG;
54};
55
56template <>
58 static constexpr Endian other = Endian::LITTLE;
59};
60
61// TODO: using C++20 std::is_constant_evaluated() we can perform byte order
62// swapping transparently during compile time already, if possible.
63//
64// a template recursion algorithm can be used to perform the byte order
65// swapping for compile time.
66//
67// NOTE: older gcc/gblic doesn't work with `constexpr` here, so drop it for
68// the time being.
69
71inline uint16_t swap_byte_order(uint16_t value) {
72 return bswap_16(value);
73}
74
76inline uint32_t swap_byte_order(uint32_t value) {
77 return bswap_32(value);
78}
79
81inline uint64_t swap_byte_order(uint64_t value) {
82 return bswap_64(value);
83}
84
86template <typename T>
87inline T to_network_order(T host) {
88 if (our_endian == Endian::BIG) {
89 return host;
90 } else {
91 return swap_byte_order(host);
92 }
93}
94
96template <typename T>
97inline T to_host_order(T network) {
98 if (our_endian == Endian::BIG) {
99 return network;
100 } else {
101 return swap_byte_order(network);
102 }
103}
104
105/*
106 * these enum types are used as strong types for unsigned integers that carry
107 * raw data potentially in a foreign endianness. This is used by
108 * EndianNumber::raw().
109 */
110
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;
120
121template <typename UINT, Endian endian>
123};
124
125template <>
126struct RawIntTraits<uint16_t, Endian::LITTLE> {
127 using Int = RawLittleInt16;
128};
129template <>
130struct RawIntTraits<uint32_t, Endian::LITTLE> {
131 using Int = RawLittleInt32;
132};
133template <>
134struct RawIntTraits<uint64_t, Endian::LITTLE> {
135 using Int = RawLittleInt64;
136};
137template <>
138struct RawIntTraits<uint16_t, Endian::BIG> {
139 using Int = RawBigInt16;
140};
141template <>
142struct RawIntTraits<uint32_t, Endian::BIG> {
143 using Int = RawBigInt32;
144};
145template <>
146struct RawIntTraits<uint64_t, Endian::BIG> {
147 using Int = RawBigInt64;
148};
149
151
158template <typename T, Endian endianness>
160public: // types
161
162 using RawInt = typename RawIntTraits<T, endianness>::Int;
163
164public: // functions
165
167 constexpr EndianNumber() : m_egg{} {}
168
170 constexpr EndianNumber(const no_init_t) {}
171
173 constexpr EndianNumber(const RawInt rint) :
174 m_egg{to_integral(rint)} {
175 }
176
178 constexpr EndianNumber(const T egg) :
179 m_egg{toTargetEndianness(egg)} {
180 }
181
184 m_egg{toTargetEndianness(other.toHost())} {
185 }
186
187 void setFromHost(const T egg) {
188 m_egg = toTargetEndianness(egg);
189 }
190
191 constexpr T toHost() const {
192 if (our_endian == endianness) {
193 return m_egg;
194 } else {
195 return swap_byte_order(m_egg);
196 }
197 };
198
199 operator T() const {
200 return toHost();
201 }
202
203 RawInt raw() const {
204 return RawInt{m_egg};
205 }
206
207protected: // functions
208
209 constexpr static T toTargetEndianness(const T egg) {
210 if (our_endian == endianness) {
211 return egg;
212 } else {
213 return swap_byte_order(egg);
214 }
215 }
216
217protected: // data
218
219 T m_egg;
220};
221
222using LittleInt16 = EndianNumber<uint16_t, Endian::LITTLE>;
223using LittleInt32 = EndianNumber<uint32_t, Endian::LITTLE>;
224using LittleInt64 = EndianNumber<uint64_t, Endian::LITTLE>;
225
226using BigInt16 = EndianNumber<uint16_t, Endian::BIG>;
227using BigInt32 = EndianNumber<uint32_t, Endian::BIG>;
228using BigInt64 = EndianNumber<uint64_t, Endian::BIG>;
229
230using NetInt16 = BigInt16;
231using NetInt32 = BigInt32;
232using NetInt64 = BigInt64;
233
234using HostInt16 = EndianNumber<uint16_t, our_endian>;
235using HostInt32 = EndianNumber<uint32_t, our_endian>;
236using HostInt64 = EndianNumber<uint64_t, our_endian>;
237
238} // end ns
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.
Definition types.hxx:37