#ifndef SPROUT_BIT_CLZ_HPP #define SPROUT_BIT_CLZ_HPP #include #include #include namespace sprout { namespace detail { # if SPROUT_USE_BUILTIN_BIT_OPERATION inline SPROUT_CONSTEXPR int clz(unsigned n) { return __builtin_clz(n); } inline SPROUT_CONSTEXPR int clz(unsigned long n) { return __builtin_clzl(n); } inline SPROUT_CONSTEXPR int clz(unsigned long long n) { return __builtin_clzll(n); } # endif template inline SPROUT_CONSTEXPR int clz_impl(T n, T m = T(1) << (CHAR_BIT * sizeof(T) - 1)) { return m == 0 || n & m ? 0 : 1 + sprout::detail::clz_impl(n, static_cast(m >> 1)) ; } template inline SPROUT_CONSTEXPR typename std::enable_if< std::is_unsigned::value, int >::type clz(T n) { return sprout::detail::clz_impl(static_cast(n)); } template inline SPROUT_CONSTEXPR typename std::enable_if< std::is_signed::value, int >::type clz(T n) { return sprout::detail::clz(static_cast::type>(n)); } } // namespace detail // // clz // template inline SPROUT_CONSTEXPR typename std::enable_if< std::is_integral::value, int >::type clz(T n) { return sprout::detail::clz(n); } } // namespace sprout #endif // #ifndef SPROUT_BIT_CLZ_HPP