0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-11-27 01:02:46 +01:00
construct/include/ircd/util/bswap.h

145 lines
2.5 KiB
C++

// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.
#pragma once
#define HAVE_IRCD_UTIL_BSWAP_H
namespace ircd::util
{
template<class T> T &bswap(T *const &);
template<class T> T bswap(T);
// Host <-> network endian
template<class T> T &hton(T *const &);
template<class T> T &ntoh(T *const &);
template<class T> T hton(T);
template<class T> T ntoh(T);
// Convenience host to network interface for arpa/inet.h compat
uint32_t htonl(const uint32_t &);
uint16_t htons(const uint16_t &);
double htond(const double &);
float htonf(const float &);
// Convenience network to host interface for arpa/inet.h compat
uint32_t ntohl(const uint32_t &);
uint16_t ntohs(const uint16_t &);
double ntohd(const double &);
float ntohf(const float &);
}
inline float
ircd::util::ntohf(const float &a)
{
return ntoh(a);
}
inline double
ircd::util::ntohd(const double &a)
{
return ntoh(a);
}
inline uint32_t
ircd::util::ntohl(const uint32_t &a)
{
return ntoh(a);
}
inline uint16_t
ircd::util::ntohs(const uint16_t &a)
{
return ntoh(a);
}
inline float
ircd::util::htonf(const float &a)
{
return hton(a);
}
inline double
ircd::util::htond(const double &a)
{
return hton(a);
}
inline uint32_t
ircd::util::htonl(const uint32_t &a)
{
return hton(a);
}
inline uint16_t
ircd::util::htons(const uint16_t &a)
{
return hton(a);
}
template<class T>
T
ircd::util::hton(T a)
{
return hton<T>(&a);
}
template<class T>
T
ircd::util::ntoh(T a)
{
return ntoh<T>(&a);
}
template<class T>
T &
ircd::util::hton(T *const &a)
{
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
return bswap(a);
#else
assert(a);
return *a;
#endif
}
template<class T>
T &
ircd::util::ntoh(T *const &a)
{
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
return bswap(a);
#else
assert(a);
return *a;
#endif
}
/// Reverse endian of T returning value copy
template<class T>
T
ircd::util::bswap(T ret)
{
return bswap(&ret);
}
/// Reverse endian of data pointed to; return reference
template<class T>
T &
ircd::util::bswap(T *const &val)
{
assert(val != nullptr);
const auto &ptr
{
reinterpret_cast<uint8_t *const &>(val)
};
std::reverse(ptr, ptr + sizeof(T));
return *val;
}