2022-12-04 11:03:45 +00:00
|
|
|
// 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
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
#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;
|
|
|
|
}
|
2021-06-24 14:19:14 +00:00
|
|
|
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);
|
|
|
|
value |= (value >> 8);
|
|
|
|
value |= (value >> 16);
|
|
|
|
return value - (value >> 1);
|
|
|
|
}
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
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;
|
2023-08-13 03:42:02 +00:00
|
|
|
#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
|