blob: c59fb574912aa18d4488e4d4a45203f33fb88e55 [file]
// 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