// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #pragma once #include #include "compiler_builtins.hpp" #include "helpers.hpp" #ifdef _WIN32 #include #endif namespace Common { template constexpr bool isAligned(T value, unsigned int alignment) { return (value % static_cast(alignment)) == 0; } template constexpr T alignUp(T value, unsigned int alignment) { return (value + static_cast(alignment - 1)) / static_cast(alignment) * static_cast(alignment); } template constexpr T alignDown(T value, unsigned int alignment) { return value / static_cast(alignment) * static_cast(alignment); } template constexpr bool isAlignedPow2(T value, unsigned int alignment) { return (value & static_cast(alignment - 1)) == 0; } template constexpr T alignUpPow2(T value, unsigned int alignment) { return (value + static_cast(alignment - 1)) & static_cast(~static_cast(alignment - 1)); } template constexpr T alignDownPow2(T value, unsigned int alignment) { return value & static_cast(~static_cast(alignment - 1)); } template constexpr bool isPow2(T value) { return (value & (value - 1)) == 0; } template constexpr T previousPow2(T value) { if (value == static_cast(0)) return 0; value |= (value >> 1); value |= (value >> 2); value |= (value >> 4); if constexpr (sizeof(T) >= 16) value |= (value >> 8); if constexpr (sizeof(T) >= 32) value |= (value >> 16); if constexpr (sizeof(T) >= 64) value |= (value >> 32); return value - (value >> 1); } template constexpr T nextPow2(T value) { // https://graphics.stanford.edu/~seander/bithacks.html#RoundUpPowerOf2 if (value == static_cast(0)) return 0; value--; value |= (value >> 1); value |= (value >> 2); value |= (value >> 4); if constexpr (sizeof(T) >= 16) value |= (value >> 8); if constexpr (sizeof(T) >= 32) value |= (value >> 16); if constexpr (sizeof(T) >= 64) value |= (value >> 32); value++; return value; } ALWAYS_INLINE static void* alignedMalloc(size_t size, size_t alignment) { #ifdef _WIN32 return _aligned_malloc(size, alignment); #else // Unaligned sizes are slow on macOS. #ifdef __APPLE__ if (isPow2(alignment)) size = (size + alignment - 1) & ~(alignment - 1); #endif void* ret = nullptr; return (posix_memalign(&ret, alignment, size) == 0) ? ret : nullptr; #endif } ALWAYS_INLINE static void alignedFree(void* ptr) { #ifdef _MSC_VER _aligned_free(ptr); #else free(ptr); #endif } } // namespace Common