Sprout/sprout/net/endian.hpp

176 lines
6.4 KiB
C++
Raw Normal View History

2014-03-26 02:57:33 +00:00
/*=============================================================================
Copyright (c) 2011-2014 Bolero MURAKAMI
https://github.com/bolero-MURAKAMI/Sprout
Copyright (c) 2014 Chris KAY
https://github.com/cjkay-cpp-utils/Sprout
Distributed under the Boost Software License, Version 1.0. (See accompanying
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
=============================================================================*/
#ifndef SPROUT_NET_ENDIAN_HPP
#define SPROUT_NET_ENDIAN_HPP
#include <cstdint>
#include <sprout/config.hpp>
#include <boost/detail/endian.hpp>
namespace sprout {
namespace net {
namespace detail {
/*
* lshift_by
* Left shift 'val' by 'n' bytes
*/
template<typename T>
SPROUT_CONSTEXPR T lshift_by(T val, std::uint_fast8_t n) {
return val << (n * 8);
}
/*
* rshift_by
* Right shift 'val' by 'n' bytes
*/
template<typename T>
SPROUT_CONSTEXPR T rshift_by(T val, std::uint_fast8_t n) {
return val >> (n * 8);
}
/*
* mask_t
* Mask 'n'th byte
*/
template<typename T>
SPROUT_CONSTEXPR T mask_at(std::uint_fast8_t n) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::lshift_by(T(0xFF), n);
2014-03-26 02:57:33 +00:00
}
/*
* byte_at
* Get 'n'th byte from 'val'
*/
template<typename T>
SPROUT_CONSTEXPR T byte_at(T val, std::uint_fast8_t n) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::rshift_by((val & sprout::net::detail::mask_at<T>(n)), n);
2014-03-26 02:57:33 +00:00
}
/*
* replace_at
* Replace, in 'val', 'n'th byte with 'byte_val'
*/
template<typename T>
SPROUT_CONSTEXPR T replace_at(T val, std::uint8_t byte_val, std::uint_fast8_t n) {
2014-03-26 03:20:13 +00:00
return (val & ~sprout::net::detail::mask_at<T>(n)) | sprout::net::detail::lshift_by(T(byte_val), n);
2014-03-26 02:57:33 +00:00
}
/*
* copy
* Copy, in 'val', byte at index 'from' to byte at index 'to'
*/
template<typename T>
SPROUT_CONSTEXPR T copy(T val, std::uint_fast8_t from, std::uint_fast8_t to) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::replace_at(val, sprout::net::detail::byte_at(val, from), to);
2014-03-26 02:57:33 +00:00
}
/*
* swap
* Swap, in 'val', byte at index 'n1' with byte at index 'n2'
*/
template<typename T>
SPROUT_CONSTEXPR T swap(T val, std::uint_fast8_t n1, std::uint_fast8_t n2) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::replace_at(sprout::net::detail::copy(val, n1, n2), sprout::net::detail::byte_at(val, n2), n1);
2014-03-26 02:57:33 +00:00
}
/*
* reverse_impl
* Swap, in 'val', byte at index 'n1' with byte at index 'n2' in 'val', increment n1, decrement n2 until n1 > n2
*/
template<typename T>
SPROUT_CONSTEXPR T reverse_impl(T val, std::uint_fast8_t n1, std::uint_fast8_t n2) {
2014-03-26 03:20:13 +00:00
return n1 > n2 ? val : sprout::net::detail::reverse_impl(swap(val, n1, n2), n1 + 1, n2 - 1);
2014-03-26 02:57:33 +00:00
}
/*
* reverse
* Reverse 'val'
*/
template<typename T>
SPROUT_CONSTEXPR T reverse(T val) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse_impl(val, 0, sizeof(T) - 1);
2014-03-26 02:57:33 +00:00
}
/*
* reverse_words_impl
* Swap, in 'val', byte at index 'n' - 1 with byte at index 'n' - 2, decrement n by 2 until n == 0
*/
template<typename T>
SPROUT_CONSTEXPR T reverse_words_impl(T val, std::uint_fast8_t n) {
2014-03-26 03:20:13 +00:00
return n == 0 ? val : sprout::net::detail::reverse_words_impl(swap(val, n - 1, n - 2), n - 2);
2014-03-26 02:57:33 +00:00
}
/*
* reverse_words
* Reverse each word in 'val'
*/
template<typename T>
SPROUT_CONSTEXPR T reverse_words(T val) {
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse_words_impl(val, sizeof(T));
2014-03-26 02:57:33 +00:00
}
} // namespace detail
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// template<>
// constexpr unsigned-integral hton(unsigned-integral host)
template<typename T>
SPROUT_CONSTEXPR T hton(T host) {
#if defined(BOOST_BIG_ENDIAN)
return host;
#elif defined(BOOST_LITTLE_ENDIAN)
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse(host);
2014-03-26 02:57:33 +00:00
#elif defined(BOOST_PDP_ENDIAN)
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse_words(host);
2014-03-26 02:57:33 +00:00
#endif
}
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// template<>
// constexpr unsigned-integral ntoh(unsigned-integral net)
template<typename T>
SPROUT_CONSTEXPR T ntoh(T net) {
#if defined(BOOST_BIG_ENDIAN)
return net;
#elif defined(BOOST_LITTLE_ENDIAN)
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse(net);
2014-03-26 02:57:33 +00:00
#elif defined(BOOST_PDP_ENDIAN)
2014-03-26 03:20:13 +00:00
return sprout::net::detail::reverse_words(host);
2014-03-26 02:57:33 +00:00
#endif
}
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// uint32_t htonl(uint32_t host32)
SPROUT_CONSTEXPR std::uint32_t htonl(std::uint32_t host32) {
2014-03-26 03:20:13 +00:00
return sprout::net::hton(host32);
2014-03-26 02:57:33 +00:00
}
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// uint16_t htons(uint16_t host16)
SPROUT_CONSTEXPR std::uint16_t htons(std::uint16_t host16) {
2014-03-26 03:20:13 +00:00
return sprout::net::hton(host16);
2014-03-26 02:57:33 +00:00
}
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// uint32_t ntohl(uint32_t net32)
SPROUT_CONSTEXPR std::uint32_t ntohl(std::uint32_t net32) {
2014-03-26 03:20:13 +00:00
return sprout::net::ntoh(net32);
2014-03-26 02:57:33 +00:00
}
// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3620.pdf
// uint16_t ntohs(uint32_t net16)
SPROUT_CONSTEXPR std::uint16_t ntohs(std::uint16_t net16) {
2014-03-26 03:20:13 +00:00
return sprout::net::ntoh(net16);
2014-03-26 02:57:33 +00:00
}
} //namespace net
} // namespace sprout
#endif // #ifndef SPROUT_NET_ENDIAN_HPP