Duckstation/src/common/align.h

113 lines
2.6 KiB
C
Raw Normal View History

// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
2020-01-10 03:31:12 +00:00
#pragma once
#include "types.h"
#include <cstdlib>
#ifdef _MSC_VER
#include <malloc.h>
#endif
2020-01-10 03:31:12 +00:00
namespace Common {
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr bool IsAligned(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return (value % static_cast<T>(alignment)) == 0;
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr T AlignUp(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return (value + static_cast<T>(alignment - 1)) / static_cast<T>(alignment) * static_cast<T>(alignment);
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr T AlignDown(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return value / static_cast<T>(alignment) * static_cast<T>(alignment);
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr bool IsAlignedPow2(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return (value & static_cast<T>(alignment - 1)) == 0;
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr T AlignUpPow2(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return (value + static_cast<T>(alignment - 1)) & static_cast<T>(~static_cast<T>(alignment - 1));
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr T AlignDownPow2(T value, unsigned int alignment)
2020-01-10 03:31:12 +00:00
{
return value & static_cast<T>(~static_cast<T>(alignment - 1));
}
template<typename T>
2020-10-11 10:34:56 +00:00
constexpr bool IsPow2(T value)
2020-01-10 03:31:12 +00:00
{
return (value & (value - 1)) == 0;
}
template<typename T>
constexpr T PreviousPow2(T value)
{
if (value == static_cast<T>(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<typename T>
constexpr T NextPow2(T value)
{
// https://graphics.stanford.edu/~seander/bithacks.html#RoundUpPowerOf2
if (value == static_cast<T>(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 _MSC_VER
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;
2023-09-05 12:58:11 +00:00
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
}
2020-01-10 03:31:12 +00:00
} // namespace Common