// C compatibility header -*- C++ -*- // Copyright The GNU Toolchain Authors. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // . /** @file include/stdbit.h * This is a Standard C++ Library header. */ #ifndef _GLIBCXX_STDBIT_H #define _GLIBCXX_STDBIT_H #if __cplusplus > 202302L #include #define __STDC_VERSION_STDBIT_H__ 202311L #define __STDC_ENDIAN_BIG__ __ORDER_BIG_ENDIAN__ #define __STDC_ENDIAN_LITTLE__ __ORDER_LITTLE_ENDIAN__ #define __STDC_ENDIAN_NATIVE__ __BYTE_ORDER__ #ifndef _GLIBCXX_DOXYGEN // We define these in our own namespace, but let Doxygen think otherwise. namespace __gnu_cxx _GLIBCXX_VISIBILITY(default) { #endif /** Count the number of leading zero bits * * @param __value An unsigned integer. * @since C++26 * @{ */ template inline unsigned int stdc_leading_zeros(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::countl_zero(__value); } inline unsigned int stdc_leading_zeros_uc(unsigned char __value) { return stdc_leading_zeros(__value); } inline unsigned int stdc_leading_zeros_us(unsigned short __value) { return stdc_leading_zeros(__value); } inline unsigned int stdc_leading_zeros_ui(unsigned int __value) { return stdc_leading_zeros(__value); } inline unsigned int stdc_leading_zeros_ul(unsigned long int __value) { return stdc_leading_zeros(__value); } inline unsigned int stdc_leading_zeros_ull(unsigned long long int __value) { return stdc_leading_zeros(__value); } /// @} /** Count the number of leading one bits * * @param __value An unsigned integer. * @since C++26 * @{ */ template inline unsigned int stdc_leading_ones(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::countl_one(__value); } inline unsigned int stdc_leading_ones_uc(unsigned char __value) { return stdc_leading_ones(__value); } inline unsigned int stdc_leading_ones_us(unsigned short __value) { return stdc_leading_ones(__value); } inline unsigned int stdc_leading_ones_ui(unsigned int __value) { return stdc_leading_ones(__value); } inline unsigned int stdc_leading_ones_ul(unsigned long int __value) { return stdc_leading_ones(__value); } inline unsigned int stdc_leading_ones_ull(unsigned long long int __value) { return stdc_leading_ones(__value); } /// @} /** Count the number of trailing zero bits * * @param __value An unsigned integer. * @since C++26 * @{ */ template inline unsigned int stdc_trailing_zeros(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::countr_zero(__value); } inline unsigned int stdc_trailing_zeros_uc(unsigned char __value) { return stdc_trailing_zeros(__value); } inline unsigned int stdc_trailing_zeros_us(unsigned short __value) { return stdc_trailing_zeros(__value); } inline unsigned int stdc_trailing_zeros_ui(unsigned int __value) { return stdc_trailing_zeros(__value); } inline unsigned int stdc_trailing_zeros_ul(unsigned long int __value) { return stdc_trailing_zeros(__value); } inline unsigned int stdc_trailing_zeros_ull(unsigned long long int __value) { return stdc_trailing_zeros(__value); } /// @} /** Count the number of trailing one bits * * @param __value An unsigned integer. * @since C++26 * @{ */ template inline unsigned int stdc_trailing_ones(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::countr_one(__value); } inline unsigned int stdc_trailing_ones_uc(unsigned char __value) { return stdc_trailing_ones(__value); } inline unsigned int stdc_trailing_ones_us(unsigned short __value) { return stdc_trailing_ones(__value); } inline unsigned int stdc_trailing_ones_ui(unsigned int __value) { return stdc_trailing_ones(__value); } inline unsigned int stdc_trailing_ones_ul(unsigned long int __value) { return stdc_trailing_ones(__value); } inline unsigned int stdc_trailing_ones_ull(unsigned long long int __value) { return stdc_trailing_ones(__value); } /// @} /** Find the leftmost (i.e. most significant) zero bit * * @param __value An unsigned integer. * @return The one-based index of the first zero bit counting from the left, * or zero if there are no zero bits. * @since C++26 * @{ */ template inline unsigned int stdc_first_leading_zero(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return __value == _Tp(-1) ? 0 : 1 + std::countl_one(__value); } inline unsigned int stdc_first_leading_zero_uc(unsigned char __value) { return stdc_first_leading_zero(__value); } inline unsigned int stdc_first_leading_zero_us(unsigned short __value) { return stdc_first_leading_zero(__value); } inline unsigned int stdc_first_leading_zero_ui(unsigned int __value) { return stdc_first_leading_zero(__value); } inline unsigned int stdc_first_leading_zero_ul(unsigned long int __value) { return stdc_first_leading_zero(__value); } inline unsigned int stdc_first_leading_zero_ull(unsigned long long int __value) { return stdc_first_leading_zero(__value); } /// @} /** Find the leftmost (i.e. most significant) one bit * * @param __value An unsigned integer. * @return The one-based index of the first one bit counting from the left, * or zero if there are no one bits. * @since C++26 * @{ */ template inline unsigned int stdc_first_leading_one(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return __value == 0 ? 0 : 1 + std::countl_zero(__value); } inline unsigned int stdc_first_leading_one_uc(unsigned char __value) { return stdc_first_leading_one(__value); } inline unsigned int stdc_first_leading_one_us(unsigned short __value) { return stdc_first_leading_one(__value); } inline unsigned int stdc_first_leading_one_ui(unsigned int __value) { return stdc_first_leading_one(__value); } inline unsigned int stdc_first_leading_one_ul(unsigned long int __value) { return stdc_first_leading_one(__value); } inline unsigned int stdc_first_leading_one_ull(unsigned long long int __value) { return stdc_first_leading_one(__value); } /// @} /** Find the rightmost (i.e. least significant) zero bit * * @param __value An unsigned integer. * @return The one-based index of the first zero bit counting from the right, * or zero if there are no zero bits. * @since C++26 * @{ */ template inline unsigned int stdc_first_trailing_zero(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return __value == _Tp(-1) ? 0 : 1 + std::countr_one(__value); } inline unsigned int stdc_first_trailing_zero_uc(unsigned char __value) { return stdc_first_trailing_zero(__value); } inline unsigned int stdc_first_trailing_zero_us(unsigned short __value) { return stdc_first_trailing_zero(__value); } inline unsigned int stdc_first_trailing_zero_ui(unsigned int __value) { return stdc_first_trailing_zero(__value); } inline unsigned int stdc_first_trailing_zero_ul(unsigned long int __value) { return stdc_first_trailing_zero(__value); } inline unsigned int stdc_first_trailing_zero_ull(unsigned long long int __value) { return stdc_first_trailing_zero(__value); } /// @} /** Find the rightmost (i.e. least significant) one bit * * @param __value An unsigned integer. * @return The one-based index of the first one bit counting from the right, * or zero if there are no one bits. * @since C++26 * @{ */ template inline unsigned int stdc_first_trailing_one(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return __value == 0 ? 0 : 1 + std::countr_zero(__value); } inline unsigned int stdc_first_trailing_one_uc(unsigned char __value) { return stdc_first_trailing_one(__value); } inline unsigned int stdc_first_trailing_one_us(unsigned short __value) { return stdc_first_trailing_one(__value); } inline unsigned int stdc_first_trailing_one_ui(unsigned int __value) { return stdc_first_trailing_one(__value); } inline unsigned int stdc_first_trailing_one_ul(unsigned long int __value) { return stdc_first_trailing_one(__value); } inline unsigned int stdc_first_trailing_one_ull(unsigned long long int __value) { return stdc_first_trailing_one(__value); } /// @} /** Count zeros * * @param __value An unsigned integer. * @return The total number of zero bits in `__value`. * @since C++26 * @{ */ template inline unsigned int stdc_count_zeros(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::popcount(_Tp(~__value)); } inline unsigned int stdc_count_zeros_uc(unsigned char __value) { return stdc_count_zeros(__value); } inline unsigned int stdc_count_zeros_us(unsigned short __value) { return stdc_count_zeros(__value); } inline unsigned int stdc_count_zeros_ui(unsigned int __value) { return stdc_count_zeros(__value); } inline unsigned int stdc_count_zeros_ul(unsigned long int __value) { return stdc_count_zeros(__value); } inline unsigned int stdc_count_zeros_ull(unsigned long long int __value) { return stdc_count_zeros(__value); } /// @} /** Count ones * * @param __value An unsigned integer. * @return The total number of one bits in `__value`. * @since C++26 * @{ */ template inline unsigned int stdc_count_ones(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::popcount(__value); } inline unsigned int stdc_count_ones_uc(unsigned char __value) { return stdc_count_ones(__value); } inline unsigned int stdc_count_ones_us(unsigned short __value) { return stdc_count_ones(__value); } inline unsigned int stdc_count_ones_ui(unsigned int __value) { return stdc_count_ones(__value); } inline unsigned int stdc_count_ones_ul(unsigned long int __value) { return stdc_count_ones(__value); } inline unsigned int stdc_count_ones_ull(unsigned long long int __value) { return stdc_count_ones(__value); } /// @} /** Power of two check * * @param __value An unsigned integer. * @return True if the value has a single bit set, false otherwise. * @since C++26 * @{ */ template inline bool stdc_has_single_bit(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::has_single_bit(__value); } inline bool stdc_has_single_bit_uc(unsigned char __value) { return stdc_has_single_bit(__value); } inline bool stdc_has_single_bit_us(unsigned short __value) { return stdc_has_single_bit(__value); } inline bool stdc_has_single_bit_ui(unsigned int __value) { return stdc_has_single_bit(__value); } inline bool stdc_has_single_bit_ul(unsigned long int __value) { return stdc_has_single_bit(__value); } inline bool stdc_has_single_bit_ull(unsigned long long int __value) { return stdc_has_single_bit(__value); } /// @} /** Bit width * * @param __value An unsigned integer. * @return The minimum number of bits needed to represent `__value`. * @since C++26 * @{ */ template inline unsigned int stdc_bit_width(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::bit_width(__value); } inline unsigned int stdc_bit_width_uc(unsigned char __value) { return stdc_bit_width(__value); } inline unsigned int stdc_bit_width_us(unsigned short __value) { return stdc_bit_width(__value); } inline unsigned int stdc_bit_width_ui(unsigned int __value) { return stdc_bit_width(__value); } inline unsigned int stdc_bit_width_ul(unsigned long int __value) { return stdc_bit_width(__value); } inline unsigned int stdc_bit_width_ull(unsigned long long int __value) { return stdc_bit_width(__value); } /// @} /** Bit floor * * @param __value An unsigned integer. * @return The largest power of two that is not greater than `__value`. * @since C++26 * @{ */ template inline _Tp stdc_bit_floor(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); return std::bit_floor(__value); } inline unsigned char stdc_bit_floor_uc(unsigned char __value) { return stdc_bit_floor(__value); } inline unsigned short stdc_bit_floor_us(unsigned short __value) { return stdc_bit_floor(__value); } inline unsigned int stdc_bit_floor_ui(unsigned int __value) { return stdc_bit_floor(__value); } inline unsigned long int stdc_bit_floor_ul(unsigned long int __value) { return stdc_bit_floor(__value); } inline unsigned long long int stdc_bit_floor_ull(unsigned long long int __value) { return stdc_bit_floor(__value); } /// @} /** Bit ceiling * * Unlike `std::bit_ceil`, this is defined to return zero for values which * are not representable in the return type. * * @param __value An unsigned integer. * @return The smallest power of two that is not less than `__value`. * @since C++26 * @{ */ template inline _Tp stdc_bit_ceil(_Tp __value) { static_assert(std::__unsigned_integer<_Tp>); constexpr _Tp __msb = _Tp(1) << (__gnu_cxx::__int_traits<_Tp>::__digits - 1); return (__value & __msb) ? 0 : std::bit_ceil(__value); } inline unsigned char stdc_bit_ceil_uc(unsigned char __value) { return stdc_bit_ceil(__value); } inline unsigned short stdc_bit_ceil_us(unsigned short __value) { return stdc_bit_ceil(__value); } inline unsigned int stdc_bit_ceil_ui(unsigned int __value) { return stdc_bit_ceil(__value); } inline unsigned long int stdc_bit_ceil_ul(unsigned long int __value) { return stdc_bit_ceil(__value); } inline unsigned long long int stdc_bit_ceil_ull(unsigned long long int __value) { return stdc_bit_ceil(__value); } /// @} #ifndef _GLIBCXX_DOXYGEN } // namespace __gnu_cxx #define _GLIBCXX_STDBIT_FUNC(F) \ using __gnu_cxx::F ## _uc; \ using __gnu_cxx::F ## _us; \ using __gnu_cxx::F ## _ui; \ using __gnu_cxx::F ## _ul; \ using __gnu_cxx::F ## _ull; \ using __gnu_cxx::F _GLIBCXX_STDBIT_FUNC(stdc_leading_zeros); _GLIBCXX_STDBIT_FUNC(stdc_leading_ones); _GLIBCXX_STDBIT_FUNC(stdc_trailing_zeros); _GLIBCXX_STDBIT_FUNC(stdc_trailing_ones); _GLIBCXX_STDBIT_FUNC(stdc_first_leading_zero); _GLIBCXX_STDBIT_FUNC(stdc_first_leading_one); _GLIBCXX_STDBIT_FUNC(stdc_first_trailing_zero); _GLIBCXX_STDBIT_FUNC(stdc_first_trailing_one); _GLIBCXX_STDBIT_FUNC(stdc_count_zeros); _GLIBCXX_STDBIT_FUNC(stdc_count_ones); _GLIBCXX_STDBIT_FUNC(stdc_has_single_bit); _GLIBCXX_STDBIT_FUNC(stdc_bit_width); _GLIBCXX_STDBIT_FUNC(stdc_bit_floor); _GLIBCXX_STDBIT_FUNC(stdc_bit_ceil); #undef _GLIBCXX_STDBIT_FUNC #endif // !DOXYGEN #endif // C++26 #endif // _GLIBCXX_STDBIT_H