1. Revision history
1.1. Changes since R1
- 
     Mentioned [P3104R2], now that it has been published. 
- 
     Greatly expanded § 4 Design considerations based on feedback from LEWGI. 
1.2. Changes since R0
- 
     Simplified proposed wording by defining rotl rotr 
- 
     Fixed minor editorial mistakes. 
2. Introduction
[P0553R4] added the bit manipulation library to C++20, which introduced many useful utility functions.
Some of these already have a counterpart in 
3. Proposed changes
For each of the functions from the bit manipulation library that are not yet available
in 
| function | Proposed member | 
|---|---|
|  |  | 
|  |  | 
|  | |
|  |  | 
|  | |
|  |  | 
|  | |
|  |  | 
|  | |
|  |  | 
|  |  | 
|  | 
The additional overloads for the counting functions allow counting from a starting position. This can be useful for iterating over all set bits:
bitset < 128 > bits ; for ( size_t i = 0 ; i != 128 ; ++ i ) { i += bits . countr_zero ( i ); if ( i == 128 ) break ; // ... } 
Note: 
4. Design considerations
The names and signatures of the proposed member functions is a mixture of 
This proposal generally prefers local consistency, i.e. it is more important to
be consistent with 
4.1. reverse 
   
This function is added because the method for reversing integers may be tremendously faster than
doing so bit by bit.
ARM has a dedicated 
4.1.1. In-place reversal vs. reverse-copy
I propose 
Compilers also should not be expected to find the optimization from a 
void reverse_in_place ( bitset & x ) { x = x . reverse_copy (); } 
... to the in-place counterpart for a 4096-bit 
I believe that all operations of a 
A perfect optimization would be very difficult due to substantial differences between the 
4.1.2. reverse ( bitset ) 
   Instead of a 
I do not oppose this idea; LEWG needs to vote on this.
4.2. Common interface with < bit > 
   Another considered alternative is the creation of a common interface with the
function templates of 
This proposal does not pursue this design for multiple reasons:
- 
     The use case of this is unclear. I have personally never needed to write generic code which does bit manipulation on either integers or bitsets, and don’t know of any code base where this is done. 
- 
     There is too much friction in the design of bitset < bit > size_t int 
- 
     The functions in < bit > bitset < N > N 
- 
     A common interface does not eliminate the need for new member functions. It would be very surprising to users if some functionality was only available through < bit > countr_zero < bit > popcount . count () 
- 
     A future proposal could still create a common interface for < bit > bitset 
Also note that most of these issues equally apply to sharing a common interface with 
4.3. Counting overloads
This is preferable to a single member function with a defaulted argument because the overloads
have different 
4.3.1. Alternative designs facilitating iteration
Besides the proposed design for 
bitset < 128 > bits ; for ( size_t i = 0 ; i != 128 ; ++ i ) { i += bits . countr_zero ( i ); if ( i == 128 ) break ; // ... } 
LEWG should vote on whether to go ahead with the current design or prefer one of the alternatives below.
4.3.1.1. Infallible countr_zero 
   If 
bitset < 128 > bits ; for ( size_t i = 0 ; i += bits . countr_zero ( i ) != 128 ; ++ i ) { // ... } 
An infallible 
4.3.1.2. find_next_one 
   It would be possible to add a member function which increments and advances to the next bit in a one operation.
bitset < 128 > bits ; for ( size_t i = 0 ; ( i = bits . find_next_one ( i )) != bitset < N >:: npos ; ) { // ... } 
A possible issue is that this operation is extremely specialized towards iteration.
As with 
4.3.1.3. No novel overloads
It is also possible to drop the overloads taking 
However, such functions are well within the scope of this proposal, especially if they share a name with other proposed members, and they do not pose a substantial design/wording hurdle. In short, we may as well do it.
4.3.2. New overloads for < bit > 
   The overloads taking 
int countl_zero ( std :: unsigned_integral auto x , int off ) { return std :: countl_zero ( x >> off ); } 
I consider these too trivial to justify adding such convenience functions to 
5. Impact on existing code
The semantics of existing 
This proposal is purely an extension of the functionality of 
6. Implementation experience
[Bontus1] provides  a 
7. Proposed wording
The proposed changes are relative to the working draft of the standard as of [N4917].
Modify the header synopsis in subclause 17.3.2 [version.syn] as follows:
#define __cpp_lib_bitset 202306L 20XXXXL // also in <bitset> 
Modify the header synopsis in subclause 22.9.2.1 [template.bitset.general] as follows:
namespace std { template < size_t N > class bitset { public : // bit reference [...] // [bitset.members], bitset operations constexpr bitset & operator &= ( const bitset & rhs ) noexcept ; constexpr bitset & operator |= ( const bitset & rhs ) noexcept ; constexpr bitset & operator ^= ( const bitset & rhs ) noexcept ; constexpr bitset & operator <<= ( size_t pos ) noexcept ; constexpr bitset & operator >>= ( size_t pos ) noexcept ; constexpr bitset operator << ( size_t pos ) const noexcept ; constexpr bitset operator >> ( size_t pos ) const noexcept ; + constexpr bitset & rotl ( size_t pos ) noexcept ; + constexpr bitset & rotr ( size_t pos ) noexcept ; + constexpr bitset & reverse () noexcept ; constexpr bitset & set () noexcept ; constexpr bitset & set ( size_t pos , bool val = true); constexpr bitset & reset () noexcept ; constexpr bitset & reset ( size_t pos ); constexpr bitset operator ~ () const noexcept ; constexpr bitset & flip () noexcept ; constexpr bitset & flip ( size_t pos ); // element access [...] // observers constexpr size_t count () const noexcept ; + constexpr size_t countl_zero () const noexcept ; + constexpr size_t countl_zero ( size_t pos ) const ; + constexpr size_t countl_one () const noexcept ; + constexpr size_t countl_one ( size_t pos ) const ; + constexpr size_t countr_zero () const noexcept ; + constexpr size_t countr_zero ( size_t pos ) const ; + constexpr size_t countr_one () const noexcept ; + constexpr size_t countr_one ( size_t pos ) const ; constexpr size_t size () const noexcept ; constexpr bool operator == ( const bitset & rhs ) const noexcept ; constexpr bool test ( size_t pos ) const ; constexpr bool all () const noexcept ; constexpr bool any () const noexcept ; constexpr bool none () const noexcept ; + constexpr bool one () const noexcept ; }; // [bitset.hash], hash support template < class T > struct hash ; template < size_t N > struct hash < bitset < N >> ; } 
Modify subclause 22.9.2.3 [bitset.members] as follows:
constexpr bitset operator >> ( size_t pos ) const noexcept ; Returns:
.bitset ( * this ) >>= pos 
constexpr bitset & rotl ( size_t pos ) noexcept ; Effects: Equivalent to
.rotr ( N - pos % N ) 
constexpr bitset & rotr ( size_t pos ) noexcept ; Effects: Replaces each bit at position
inI with the bit at position* this , where( static_cast < U > ( pos ) + static_cast < U > ( I )) % N is a hypothetical unsigned integer type whose width is greater than the width ofU .size_t Returns:
.* this 
constexpr bitset & reverse () noexcept ; Effects: Replaces each bit at position
inI with the bit at position* this .N - I - 1 Returns:
.* this [...]
constexpr size_t count () noexcept ; Returns: A count of the number of bits set in
.* this 
constexpr size_t countl_zero ( size_t pos ) const ; Returns: The number of consecutive zero-bits in
, starting at position* this , and traversingpos in decreasing position direction.* this Throws:
ifout_of_range does not correspond to a valid bit position.pos 
constexpr size_t countl_zero () const noexcept ; Returns:
.countl_zero ( N - 1 ) 
constexpr size_t countl_one ( size_t pos ) const ; Returns: The number of consecutive one-bits in
, starting at position* this , and traversingpos in decreasing position direction.* this Throws:
ifout_of_range does not correspond to a valid bit position.pos 
constexpr size_t countl_one () const noexcept ; Returns:
.countl_one ( N - 1 ) 
constexpr size_t countr_zero ( size_t pos ) const ; Returns: The number of consecutive zero-bits in
, starting at position* this , and traversingpos in increasing position direction.* this Throws:
ifout_of_range does not correspond to a valid bit position.pos 
constexpr size_t countr_zero () const noexcept ; Returns:
.countr_zero ( 0 ) 
constexpr size_t countr_one ( size_t pos ) const ; Returns: The number of consecutive one-bits in
, starting at position* this , and traversingpos in increasing position direction.* this Throws:
ifout_of_range does not correspond to a valid bit position.pos 
constexpr size_t countr_one () const noexcept ; Returns:
.countr_one ( 0 ) [...]
constexpr bool none () const noexcept ; Returns:
.count () == 0 
constexpr bool one () const noexcept ; Returns:
.count () == 1 
Note: The use of a hypothetical integer type in the specification of