| // Copyright 2015 gRPC authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #ifndef GRPC_SRC_CORE_UTIL_USEFUL_H |
| #define GRPC_SRC_CORE_UTIL_USEFUL_H |
| |
| #include <grpc/support/port_platform.h> |
| |
| #include <cstddef> |
| #include <limits> |
| #include <type_traits> |
| #include <variant> |
| |
| #include "absl/log/check.h" |
| #include "absl/numeric/bits.h" |
| #include "absl/strings/string_view.h" |
| |
| /// useful utilities that don't belong anywhere else |
| |
| namespace grpc_core { |
| |
| template <typename T> |
| T Clamp(T val, T min, T max) { |
| if (val < min) return min; |
| if (max < val) return max; |
| return val; |
| } |
| |
| // Set the n-th bit of i |
| template <typename T> |
| T SetBit(T* i, size_t n) { |
| return *i |= (T(1) << n); |
| } |
| |
| // Clear the n-th bit of i |
| template <typename T> |
| T ClearBit(T* i, size_t n) { |
| return *i &= ~(T(1) << n); |
| } |
| |
| // Get the n-th bit of i |
| template <typename T> |
| bool GetBit(T i, size_t n) { |
| return (i & (T(1) << n)) != 0; |
| } |
| |
| // This function uses operator< to implement a qsort-style comparison, whereby: |
| // if a is smaller than b, a number smaller than 0 is returned. |
| // if a is bigger than b, a number greater than 0 is returned. |
| // if a is neither smaller nor bigger than b, 0 is returned. |
| template <typename T> |
| int QsortCompare(const T& a, const T& b) { |
| if (a < b) return -1; |
| if (b < a) return 1; |
| return 0; |
| } |
| |
| template <typename... X> |
| int QsortCompare(const std::variant<X...>& a, const std::variant<X...>& b) { |
| const int index = QsortCompare(a.index(), b.index()); |
| if (index != 0) return index; |
| return std::visit( |
| [&](const auto& x) { |
| return QsortCompare(x, std::get<absl::remove_cvref_t<decltype(x)>>(b)); |
| }, |
| a); |
| } |
| |
| inline int QsortCompare(absl::string_view a, absl::string_view b) { |
| return a.compare(b); |
| } |
| |
| inline int QsortCompare(const std::string& a, const std::string& b) { |
| return a.compare(b); |
| } |
| |
| template <typename A, typename B> |
| int QsortCompare(const std::pair<A, B>& a, const std::pair<A, B>& b) { |
| const int first = QsortCompare(a.first, b.first); |
| if (first != 0) return first; |
| return QsortCompare(a.second, b.second); |
| } |
| |
| template <typename T> |
| constexpr size_t HashPointer(T* p, size_t range) { |
| return (((reinterpret_cast<size_t>(p)) >> 4) ^ |
| ((reinterpret_cast<size_t>(p)) >> 9) ^ |
| ((reinterpret_cast<size_t>(p)) >> 14)) % |
| range; |
| } |
| |
| // Compute a+b. |
| // If the result is greater than MAX, return MAX. |
| // If the result is less than MIN, return MIN. |
| template <typename T> |
| inline T SaturatingAdd(T a, T b) { |
| if (a > 0) { |
| if (b > std::numeric_limits<T>::max() - a) { |
| return std::numeric_limits<T>::max(); |
| } |
| } else if (b < std::numeric_limits<T>::min() - a) { |
| return std::numeric_limits<T>::min(); |
| } |
| return a + b; |
| } |
| |
| template < |
| typename T, |
| std::enable_if_t<std::is_integral_v<T> && std::is_unsigned_v<T>, int> = 0> |
| inline T SaturatingMul(T a, T b) { |
| if (a == 0 || b == 0) return 0; |
| if (b > std::numeric_limits<T>::max() / a) { |
| return std::numeric_limits<T>::max(); |
| } |
| return a * b; |
| } |
| |
| template < |
| typename T, |
| std::enable_if_t<std::is_integral_v<T> && std::is_signed_v<T>, int> = 0> |
| inline T SaturatingMul(T a, T b) { |
| if (a == 0 || b == 0) return 0; |
| if (a == std::numeric_limits<T>::min()) { |
| // negation is ub |
| if (b == -1) return std::numeric_limits<T>::max(); |
| if (b == 1) return std::numeric_limits<T>::min(); |
| if (b > 1) return std::numeric_limits<T>::min(); |
| return std::numeric_limits<T>::max(); |
| } |
| if (b == std::numeric_limits<T>::min()) { |
| if (a == -1) return std::numeric_limits<T>::max(); |
| if (a == 1) return std::numeric_limits<T>::min(); |
| if (a > 1) return std::numeric_limits<T>::min(); |
| return std::numeric_limits<T>::max(); |
| } |
| if (a > 0 && b > 0) { |
| // both positive |
| if (a > std::numeric_limits<T>::max() / b) { |
| return std::numeric_limits<T>::max(); |
| } |
| } else if (a < 0 && b < 0) { |
| // both negative |
| if (a < std::numeric_limits<T>::max() / b) { |
| return std::numeric_limits<T>::max(); |
| } |
| } else { |
| // one positive, one negative |
| if (a > 0) { |
| if (b < std::numeric_limits<T>::min() / a) { |
| return std::numeric_limits<T>::min(); |
| } |
| } else { |
| if (a < std::numeric_limits<T>::min() / b) { |
| return std::numeric_limits<T>::min(); |
| } |
| } |
| } |
| return a * b; |
| } |
| |
| inline uint32_t RoundUpToPowerOf2(uint32_t v) { |
| v--; |
| v |= v >> 1; |
| v |= v >> 2; |
| v |= v >> 4; |
| v |= v >> 8; |
| v |= v >> 16; |
| v++; |
| return v; |
| } |
| |
| // Return a value with only the lowest bit left on. |
| GPR_ATTRIBUTE_ALWAYS_INLINE_FUNCTION inline uint8_t LowestOneBit(uint8_t x) { |
| return x & -x; |
| } |
| |
| GPR_ATTRIBUTE_ALWAYS_INLINE_FUNCTION inline uint16_t LowestOneBit(uint16_t x) { |
| return x & -x; |
| } |
| |
| GPR_ATTRIBUTE_ALWAYS_INLINE_FUNCTION inline uint32_t LowestOneBit(uint32_t x) { |
| return x & -x; |
| } |
| |
| GPR_ATTRIBUTE_ALWAYS_INLINE_FUNCTION inline uint64_t LowestOneBit(uint64_t x) { |
| return x & -x; |
| } |
| |
| } // namespace grpc_core |
| |
| #define GPR_ARRAY_SIZE(array) (sizeof(array) / sizeof(*(array))) |
| |
| #endif // GRPC_SRC_CORE_UTIL_USEFUL_H |