blob: 1fb691e36c184b2f747cd5e8fe43458bc54d326c [file] [log] [blame]
// C compatibility header <stdbit.h> -*- 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
// <http://www.gnu.org/licenses/>.
/** @file include/stdbit.h
* This is a Standard C++ Library header.
*/
#ifndef _GLIBCXX_STDBIT_H
#define _GLIBCXX_STDBIT_H
#if __cplusplus > 202302L
#include <bit>
#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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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<typename _Tp>
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