| //===-- A self contained equivalent of std::bitset --------------*- C++ -*-===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef LLVM_LIBC_SRC___SUPPORT_CPP_BITSET_H |
| #define LLVM_LIBC_SRC___SUPPORT_CPP_BITSET_H |
| |
| #include "src/__support/macros/attributes.h" |
| #include <stddef.h> // For size_t. |
| |
| namespace LIBC_NAMESPACE::cpp { |
| |
| template <size_t NumberOfBits> struct bitset { |
| static_assert(NumberOfBits != 0, |
| "Cannot create a LIBC_NAMESPACE::cpp::bitset of size 0."); |
| |
| LIBC_INLINE constexpr void set(size_t Index) { |
| Data[Index / BITS_PER_UNIT] |= mask(Index); |
| } |
| |
| LIBC_INLINE constexpr void reset() { |
| for (size_t i = 0; i < NUMBER_OF_UNITS; ++i) |
| Data[i] = 0; |
| } |
| |
| LIBC_INLINE constexpr bool test(size_t Index) const { |
| return Data[Index / BITS_PER_UNIT] & mask(Index); |
| } |
| |
| LIBC_INLINE constexpr void flip() { |
| for (size_t i = 0; i < NUMBER_OF_UNITS; ++i) |
| Data[i] = ~Data[i]; |
| } |
| |
| // This function sets all bits in the range from Start to End (inclusive) to |
| // true. It assumes that Start <= End. |
| LIBC_INLINE constexpr void set_range(size_t Start, size_t End) { |
| size_t start_index = Start / BITS_PER_UNIT; |
| size_t end_index = End / BITS_PER_UNIT; |
| |
| if (start_index == end_index) { |
| // The reason the left shift is split into two parts (instead of just left |
| // shifting by End - Start + 1) is because when a number is shifted left |
| // by 64 then it wraps around to doing nothing, but shifting by 63 and the |
| // shifting by 1 correctly shifts away all of the bits. |
| size_t bit_mask = (((size_t(1) << (End - Start)) << 1) - 1) |
| << (Start - (start_index * BITS_PER_UNIT)); |
| Data[start_index] |= bit_mask; |
| } else { |
| size_t low_bit_mask = |
| ~((size_t(1) << (Start - (start_index * BITS_PER_UNIT))) - 1); |
| Data[start_index] |= low_bit_mask; |
| |
| for (size_t i = start_index + 1; i < end_index; ++i) |
| Data[i] = ~size_t(0); |
| |
| // Same as above, by splitting the shift the behavior is more consistent. |
| size_t high_bit_mask = |
| ((size_t(1) << (End - (end_index * BITS_PER_UNIT))) << 1) - 1; |
| Data[end_index] |= high_bit_mask; |
| } |
| } |
| |
| LIBC_INLINE constexpr bool |
| operator==(const bitset<NumberOfBits> &other) const { |
| for (size_t i = 0; i < NUMBER_OF_UNITS; ++i) { |
| if (Data[i] != other.Data[i]) |
| return false; |
| } |
| return true; |
| } |
| |
| private: |
| static constexpr size_t BITS_PER_BYTE = 8; |
| static constexpr size_t BITS_PER_UNIT = BITS_PER_BYTE * sizeof(size_t); |
| static constexpr size_t NUMBER_OF_UNITS = |
| (NumberOfBits + BITS_PER_UNIT - 1) / BITS_PER_UNIT; |
| |
| LIBC_INLINE static constexpr size_t mask(size_t Index) { |
| return size_t{1} << (Index % BITS_PER_UNIT); |
| } |
| size_t Data[NUMBER_OF_UNITS] = {0}; |
| }; |
| |
| } // namespace LIBC_NAMESPACE::cpp |
| |
| #endif // LLVM_LIBC_SRC___SUPPORT_CPP_BITSET_H |