123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144 |
- #ifndef AL_BIT_H
- #define AL_BIT_H
- #include <limits>
- #include <type_traits>
- #if !defined(__GNUC__) && (defined(_WIN32) || defined(_WIN64))
- #include <intrin.h>
- #include "opthelpers.h"
- #endif
- namespace al {
- #ifdef __BYTE_ORDER__
- enum class endian {
- little = __ORDER_LITTLE_ENDIAN__,
- big = __ORDER_BIG_ENDIAN__,
- native = __BYTE_ORDER__
- };
- #else
- /* This doesn't support mixed-endian. */
- namespace detail_ {
- constexpr inline bool EndianTest() noexcept
- {
- static_assert(sizeof(char) < sizeof(int), "char is too big");
- constexpr int test_val{1};
- return static_cast<const char&>(test_val);
- }
- } // namespace detail_
- enum class endian {
- little = 0,
- big = 1,
- native = detail_::EndianTest() ? little : big
- };
- #endif
- /* Define popcount (population count/count 1 bits) and countr_zero (count
- * trailing zero bits, starting from the lsb) methods, for various integer
- * types.
- */
- #ifdef __GNUC__
- namespace detail_ {
- inline int popcount(unsigned long long val) noexcept { return __builtin_popcountll(val); }
- inline int popcount(unsigned long val) noexcept { return __builtin_popcountl(val); }
- inline int popcount(unsigned int val) noexcept { return __builtin_popcount(val); }
- inline int countr_zero(unsigned long long val) noexcept { return __builtin_ctzll(val); }
- inline int countr_zero(unsigned long val) noexcept { return __builtin_ctzl(val); }
- inline int countr_zero(unsigned int val) noexcept { return __builtin_ctz(val); }
- } // namespace detail_
- template<typename T>
- inline std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> popcount(T v) noexcept { return detail_::popcount(v); }
- template<typename T>
- inline std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> countr_zero(T val) noexcept
- { return val ? detail_::countr_zero(val) : std::numeric_limits<T>::digits; }
- #else
- /* There be black magics here. The popcount method is derived from
- * https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetParallel
- * while the ctz-utilizing-popcount algorithm is shown here
- * http://www.hackersdelight.org/hdcodetxt/ntz.c.txt
- * as the ntz2 variant. These likely aren't the most efficient methods, but
- * they're good enough if the GCC built-ins aren't available.
- */
- namespace detail_ {
- template<typename T>
- constexpr T repbits(unsigned char bits) noexcept
- {
- T ret{bits};
- for(size_t i{1};i < sizeof(T);++i)
- ret = (ret<<8) | bits;
- return ret;
- }
- } // namespace detail_
- template<typename T>
- constexpr std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> popcount(T v) noexcept
- {
- constexpr T m55{detail_::repbits<T>(0x55)};
- constexpr T m33{detail_::repbits<T>(0x33)};
- constexpr T m0f{detail_::repbits<T>(0x0f)};
- constexpr T m01{detail_::repbits<T>(0x01)};
- v = v - ((v >> 1) & m55);
- v = (v & m33) + ((v >> 2) & m33);
- v = (v + (v >> 4)) & m0f;
- return static_cast<int>((v * m01) >> ((sizeof(T)-1)*8));
- }
- #if defined(_WIN64)
- template<typename T>
- inline std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> countr_zero(T v)
- {
- unsigned long idx{std::numeric_limits<T>::digits};
- if_constexpr(std::numeric_limits<T>::digits <= 32)
- _BitScanForward(&idx, static_cast<uint32_t>(v));
- else // std::numeric_limits<T>::digits > 32
- _BitScanForward64(&idx, v);
- return static_cast<int>(idx);
- }
- #elif defined(_WIN32)
- template<typename T>
- inline std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> countr_zero(T v)
- {
- unsigned long idx{std::numeric_limits<T>::digits};
- if_constexpr(std::numeric_limits<T>::digits <= 32)
- _BitScanForward(&idx, static_cast<uint32_t>(v));
- else if(!_BitScanForward(&idx, static_cast<uint32_t>(v)))
- {
- if(_BitScanForward(&idx, static_cast<uint32_t>(v>>32)))
- idx += 32;
- }
- return static_cast<int>(idx);
- }
- #else
- template<typename T>
- constexpr std::enable_if_t<std::is_integral<T>::value && std::is_unsigned<T>::value,
- int> countr_zero(T value)
- { return popcount(static_cast<T>(~value & (value - 1))); }
- #endif
- #endif
- } // namespace al
- #endif /* AL_BIT_H */
|