/* * Copyright (C) 2014 Cloudius Systems, Ltd. */ #ifndef BYTEORDER_HH_ #define BYTEORDER_HH_ #include // for ntohs() and friends #include #include inline uint64_t ntohq(uint64_t v) { return __builtin_bswap64(v); } inline uint64_t htonq(uint64_t v) { return __builtin_bswap64(v); } namespace net { inline void ntoh() {} inline void hton() {} inline uint16_t ntoh(uint16_t x) { return ntohs(x); } inline uint16_t hton(uint16_t x) { return htons(x); } inline uint32_t ntoh(uint32_t x) { return ntohl(x); } inline uint32_t hton(uint32_t x) { return htonl(x); } inline uint64_t ntoh(uint64_t x) { return ntohq(x); } inline uint64_t hton(uint64_t x) { return htonq(x); } // Wrapper around a primitive type to provide an unaligned version. // This is because gcc (correctly) doesn't allow binding an unaligned // scalar variable to a reference, and (unfortunately) doesn't allow // specifying unaligned references. // // So, packed& is our way of passing a reference (or pointer) // to a uint32_t around, without losing the knowledge about its alignment // or lack thereof. template struct packed { T raw; packed() = default; packed(T x) : raw(x) {} packed& operator=(const T& x) { raw = x; return *this; } operator T() const { return raw; } template void adjust_endianness(Adjuster a) { a(raw); } } __attribute__((packed)); template inline T ntoh(const packed& x) { T v = x; return ntoh(v); } template inline T hton(const packed& x) { T v = x; return hton(v); } template inline std::ostream& operator<<(std::ostream& os, const packed& v) { auto x = v.raw; return os << x; } inline void ntoh_inplace() {} inline void hton_inplace() {}; template inline void ntoh_inplace(First& first, Rest&... rest) { first = ntoh(first); ntoh_inplace(std::forward(rest)...); } template inline void hton_inplace(First& first, Rest&... rest) { first = hton(first); hton_inplace(std::forward(rest)...); } template inline T ntoh(const T& x) { T tmp = x; tmp.adjust_endianness([] (auto&&... what) { ntoh_inplace(std::forward(what)...); }); return tmp; } template inline T hton(const T& x) { T tmp = x; tmp.adjust_endianness([] (auto&&... what) { hton_inplace(std::forward(what)...); }); return tmp; } } #endif /* BYTEORDER_HH_ */