// File: crn_packed_uint // See Copyright Notice and license at the end of inc/crnlib.h #pragma once namespace crnlib { template<unsigned int N> struct packed_uint { inline packed_uint() { } inline packed_uint(unsigned int val) { *this = val; } inline packed_uint(const packed_uint& other) { *this = other; } inline packed_uint& operator= (const packed_uint& rhs) { if (this != &rhs) memcpy(m_buf, rhs.m_buf, sizeof(m_buf)); return *this; } inline packed_uint& operator= (unsigned int val) { #ifdef CRNLIB_BUILD_DEBUG if (N == 1) { CRNLIB_ASSERT(val <= 0xFFU); } else if (N == 2) { CRNLIB_ASSERT(val <= 0xFFFFU); } else if (N == 3) { CRNLIB_ASSERT(val <= 0xFFFFFFU); } #endif val <<= (8U * (4U - N)); for (unsigned int i = 0; i < N; i++) { m_buf[i] = static_cast<unsigned char>(val >> 24U); val <<= 8U; } return *this; } inline operator unsigned int() const { switch (N) { case 1: return m_buf[0]; case 2: return (m_buf[0] << 8U) | m_buf[1]; case 3: return (m_buf[0] << 16U) | (m_buf[1] << 8U) | (m_buf[2]); default: return (m_buf[0] << 24U) | (m_buf[1] << 16U) | (m_buf[2] << 8U) | (m_buf[3]); } } unsigned char m_buf[N]; }; template<typename T> class packed_value { public: packed_value() { } packed_value(T val) { *this = val; } inline operator T() const { T result = 0; for (int i = sizeof(T) - 1; i >= 0; i--) result = static_cast<T>((result << 8) | m_bytes[i]); return result; } packed_value& operator= (T val) { for (int i = 0; i < sizeof(T); i++) { m_bytes[i] = static_cast<uint8>(val); val >>= 8; } return *this; } private: uint8 m_bytes[sizeof(T)]; }; } // namespace crnlib