// Copyright 2013 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // This file defines some bit utilities. #ifndef PARTITION_ALLOC_PARTITION_ALLOC_BASE_BITS_H_ #define PARTITION_ALLOC_PARTITION_ALLOC_BASE_BITS_H_ #include <cstddef> #include <cstdint> #include <type_traits> #include "partition_alloc/build_config.h" #include "partition_alloc/partition_alloc_base/bits.h" #include "partition_alloc/partition_alloc_base/check.h" #include "partition_alloc/partition_alloc_base/compiler_specific.h" namespace partition_alloc::internal::base::bits { // Backport of C++20 std::has_single_bit in <bit>. // // Returns true iff |value| is a power of 2. template <typename T, typename = std::enable_if_t<std::is_integral_v<T>>> constexpr bool HasSingleBit(T value) { … } // Round down |size| to a multiple of alignment, which must be a power of two. template <typename T> inline constexpr T AlignDown(T size, T alignment) { … } // Move |ptr| back to the previous multiple of alignment, which must be a power // of two. Defined for types where sizeof(T) is one byte. template <typename T> inline T* AlignDown(T* ptr, size_t alignment) { … } // Round up |size| to a multiple of alignment, which must be a power of two. template <typename T> inline constexpr T AlignUp(T size, T alignment) { … } // Advance |ptr| to the next multiple of alignment, which must be a power of // two. Defined for types where sizeof(T) is one byte. template <typename T> inline T* AlignUp(T* ptr, size_t alignment) { … } // Backport of C++20 std::countl_zero in <bit>. // // CountlZero(value) returns the number of zero bits following the // most significant 1 bit in |value| if |value| is non-zero, otherwise it // returns {sizeof(T) * 8}. // Example: 00100010 -> 2 // // CountrZero(value) returns the number of zero bits preceding the // least significant 1 bit in |value| if |value| is non-zero, otherwise it // returns {sizeof(T) * 8}. // Example: 00100010 -> 1 // // C does not have an operator to do this, but fortunately the various // compilers have built-ins that map to fast underlying processor instructions. // __builtin_clz has undefined behaviour for an input of 0, even though there's // clearly a return value that makes sense, and even though some processor clz // instructions have defined behaviour for 0. We could drop to raw __asm__ to // do better, but we'll avoid doing that unless we see proof that we need to. template <typename T, int bits = sizeof(T) * 8> PA_ALWAYS_INLINE constexpr typename std::enable_if<std::is_unsigned_v<T> && sizeof(T) <= 8, int>::type CountlZero(T value) { … } // Backport of C++20 std::countr_zero in <bit>. // // Returns the number of consecutive 0 bits, starting from the least significant // one. template <typename T, int bits = sizeof(T) * 8> PA_ALWAYS_INLINE constexpr typename std::enable_if<std::is_unsigned_v<T> && sizeof(T) <= 8, int>::type CountrZero(T value) { … } // Backport of C++20 std::bit_width in <bit>. // // Returns the smallest i such as n <= 2^i. // This represent the number of bits needed to store values up to n. constexpr int BitWidth(uint32_t n) { … } // Returns the integer i such as 2^(i-1) < n <= 2^i. constexpr int Log2Ceiling(uint32_t n) { … } } // namespace partition_alloc::internal::base::bits #endif // PARTITION_ALLOC_PARTITION_ALLOC_BASE_BITS_H_