2019-05-24 06:11:21 -07:00
|
|
|
#ifndef AL_BYTE_H
|
|
|
|
#define AL_BYTE_H
|
|
|
|
|
2019-06-08 23:33:59 -07:00
|
|
|
#include <cstddef>
|
|
|
|
#include <limits>
|
2019-05-24 06:11:21 -07:00
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
namespace al {
|
|
|
|
|
|
|
|
/* The "canonical" way to store raw byte data. Like C++17's std::byte, it's not
|
|
|
|
* treated as a character type and does not work with arithmatic ops. Only
|
|
|
|
* bitwise ops are allowed.
|
|
|
|
*/
|
|
|
|
enum class byte : unsigned char { };
|
|
|
|
|
2020-03-22 21:15:12 -07:00
|
|
|
#define REQUIRES(...) std::enable_if_t<(__VA_ARGS__),bool> = true
|
2019-05-28 08:49:53 -07:00
|
|
|
|
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)>
|
2019-05-24 06:11:21 -07:00
|
|
|
inline constexpr T to_integer(al::byte b) noexcept { return T(b); }
|
|
|
|
|
|
|
|
|
2019-05-28 08:49:53 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)>
|
2019-05-24 06:11:21 -07:00
|
|
|
inline constexpr al::byte operator<<(al::byte lhs, T rhs) noexcept
|
|
|
|
{ return al::byte(to_integer<unsigned int>(lhs) << rhs); }
|
|
|
|
|
2019-05-28 08:49:53 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)>
|
2019-05-24 06:11:21 -07:00
|
|
|
inline constexpr al::byte operator>>(al::byte lhs, T rhs) noexcept
|
|
|
|
{ return al::byte(to_integer<unsigned int>(lhs) >> rhs); }
|
|
|
|
|
2019-05-28 08:49:53 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)>
|
2019-05-24 06:11:21 -07:00
|
|
|
inline al::byte& operator<<=(al::byte &lhs, T rhs) noexcept
|
|
|
|
{ lhs = lhs << rhs; return lhs; }
|
|
|
|
|
2019-05-28 08:49:53 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)>
|
2019-05-24 06:11:21 -07:00
|
|
|
inline al::byte& operator>>=(al::byte &lhs, T rhs) noexcept
|
|
|
|
{ lhs = lhs >> rhs; return lhs; }
|
|
|
|
|
|
|
|
#define AL_DECL_OP(op) \
|
2019-05-28 08:49:53 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)> \
|
2019-06-11 22:29:39 -07:00
|
|
|
inline constexpr al::byte operator op (al::byte lhs, T rhs) noexcept \
|
2019-09-11 05:53:10 -07:00
|
|
|
{ return al::byte(to_integer<unsigned int>(lhs) op static_cast<unsigned int>(rhs)); } \
|
2019-06-11 22:29:39 -07:00
|
|
|
template<typename T, REQUIRES(std::is_integral<T>::value)> \
|
2019-05-24 13:30:40 -07:00
|
|
|
inline al::byte& operator op##= (al::byte &lhs, T rhs) noexcept \
|
|
|
|
{ lhs = lhs op rhs; return lhs; } \
|
2019-06-11 22:29:39 -07:00
|
|
|
inline constexpr al::byte operator op (al::byte lhs, al::byte rhs) noexcept \
|
|
|
|
{ return al::byte(lhs op to_integer<unsigned int>(rhs)); } \
|
2019-05-24 06:11:21 -07:00
|
|
|
inline al::byte& operator op##= (al::byte &lhs, al::byte rhs) noexcept \
|
|
|
|
{ lhs = lhs op rhs; return lhs; }
|
|
|
|
|
|
|
|
AL_DECL_OP(|)
|
|
|
|
AL_DECL_OP(&)
|
|
|
|
AL_DECL_OP(^)
|
|
|
|
|
|
|
|
#undef AL_DECL_OP
|
|
|
|
|
2019-06-11 22:29:39 -07:00
|
|
|
inline constexpr al::byte operator~(al::byte b) noexcept
|
|
|
|
{ return al::byte(~to_integer<unsigned int>(b)); }
|
|
|
|
|
|
|
|
|
2019-06-08 23:33:59 -07:00
|
|
|
template<size_t N>
|
|
|
|
class bitfield {
|
2020-03-22 21:34:50 -07:00
|
|
|
template<size_t> struct ElemT { };
|
|
|
|
template<> struct ElemT<1> { using type = uint8_t; };
|
|
|
|
template<> struct ElemT<2> { using type = uint16_t; };
|
|
|
|
template<> struct ElemT<3> { using type = uint32_t; };
|
|
|
|
template<> struct ElemT<4> { using type = uint32_t; };
|
|
|
|
|
2019-06-08 23:33:59 -07:00
|
|
|
static constexpr size_t bits_per_byte{std::numeric_limits<unsigned char>::digits};
|
|
|
|
static constexpr size_t NumElems{(N+bits_per_byte-1) / bits_per_byte};
|
|
|
|
|
2020-03-22 21:34:50 -07:00
|
|
|
typename ElemT<NumElems>::type vals{};
|
2019-06-08 23:33:59 -07:00
|
|
|
|
|
|
|
public:
|
2019-06-09 02:20:30 -07:00
|
|
|
template<size_t b>
|
|
|
|
inline void set() noexcept
|
|
|
|
{
|
|
|
|
static_assert(b < N, "Bit index out of range");
|
2020-03-22 21:34:50 -07:00
|
|
|
vals |= 1 << b;
|
2019-06-09 02:20:30 -07:00
|
|
|
}
|
|
|
|
template<size_t b>
|
|
|
|
inline void unset() noexcept
|
|
|
|
{
|
|
|
|
static_assert(b < N, "Bit index out of range");
|
2020-03-22 21:34:50 -07:00
|
|
|
vals &= ~(1 << b);
|
2019-06-09 02:20:30 -07:00
|
|
|
}
|
|
|
|
template<size_t b>
|
|
|
|
inline bool get() const noexcept
|
|
|
|
{
|
|
|
|
static_assert(b < N, "Bit index out of range");
|
2020-03-22 21:34:50 -07:00
|
|
|
return (vals & (1 << b)) != 0;
|
2019-06-09 02:20:30 -07:00
|
|
|
}
|
2019-06-08 23:33:59 -07:00
|
|
|
|
2019-06-09 02:20:30 -07:00
|
|
|
template<size_t b, size_t ...args, REQUIRES(sizeof...(args) > 0)>
|
|
|
|
void set() noexcept
|
2019-06-08 23:33:59 -07:00
|
|
|
{
|
2019-06-09 02:20:30 -07:00
|
|
|
set<b>();
|
|
|
|
set<args...>();
|
2019-06-08 23:33:59 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-05-28 08:49:53 -07:00
|
|
|
#undef REQUIRES
|
|
|
|
|
2019-05-24 06:11:21 -07:00
|
|
|
} // namespace al
|
|
|
|
|
|
|
|
#endif /* AL_BYTE_H */
|