/* * Copyright (c) 2022, Leon Albrecht * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include namespace AK { template constexpr T exp2(T exponent) { return static_cast(1) << exponent; } template constexpr T log2(T x) { return x ? (8 * sizeof(T) - 1) - count_leading_zeroes(static_cast>(x)) : 0; } template constexpr T ceil_log2(T x) { if (!x) return 0; T log = AK::log2(x); log += (x & ((((T)1) << (log - 1)) - 1)) != 0; return log; } template constexpr I pow(I base, I exponent) { // https://en.wikipedia.org/wiki/Exponentiation_by_squaring if (exponent < 0) return 0; if (exponent == 0) return 1; I res = 1; while (exponent > 0) { if (exponent & 1) res *= base; base *= base; exponent /= 2u; } return res; } template constexpr bool is_power_of(U x) { if constexpr (base == 1) return x == 1; else if constexpr (base == 2) return is_power_of_two(x); if (base == 0 && x == 0) return true; if (base == 0 || x == 0) return false; while (x != 1) { if (x % base != 0) return false; x /= base; } return true; } }