Files
ss/include/nw4r/ut/ut_algorithm.h
T
Elijah Thomas 732a119127 Nw4r splits (#28)
* snd / math / g3d
2024-09-15 14:55:25 -04:00

104 lines
2.5 KiB
C++

#ifndef NW4R_UT_ALGORITHM_H
#define NW4R_UT_ALGORITHM_H
#include <nw4r/types_nw4r.h>
namespace nw4r {
namespace ut {
namespace {
/**
* Value operations
*/
template <typename T> inline T Max(T t1, T t2) {
return (t1 < t2) ? t2 : t1;
}
template <typename T> inline T Min(T t1, T t2) {
return (t1 > t2) ? t2 : t1;
}
template <typename T> inline T Clamp(T value, T min, T max) {
return value > max ? max : (value < min ? min : value);
}
template <typename T> inline T Abs(T x) {
// Static cast needed to break abs optimization during instruction selection
return x < 0 ? static_cast<T>(-x) : static_cast<T>(x);
}
template <> f32 inline Abs(register f32 x) {
register f32 ax;
// clang-format off
asm {
fabs ax, x
}
// clang-format on
return ax;
}
/**
* Bit operations
*/
template <typename T> inline T BitExtract(T bits, int pos, int len) {
T mask = (1 << len) - 1;
return (bits >> pos) & mask;
}
template <typename T> inline bool TestBit(T t, int pos) {
return BitExtract<T>(t, sizeof(T), pos);
}
/**
* Pointer operations
*/
inline u32 GetIntPtr(const void* pPtr) {
return reinterpret_cast<u32>(pPtr);
}
template <typename T>
inline const void* AddOffsetToPtr(const void* pBase, T offset) {
return reinterpret_cast<const void*>(GetIntPtr(pBase) + offset);
}
template <typename T> inline void* AddOffsetToPtr(void* pBase, T offset) {
return reinterpret_cast<void*>(GetIntPtr(pBase) + offset);
}
inline s32 GetOffsetFromPtr(const void* pStart, const void* pEnd) {
return static_cast<s32>(GetIntPtr(pEnd) - GetIntPtr(pStart));
}
inline int ComparePtr(const void* pPtr1, const void* pPtr2) {
return static_cast<int>(GetIntPtr(pPtr1) - GetIntPtr(pPtr2));
}
/**
* Rounding
*/
template <typename T> inline T RoundUp(T t, unsigned int alignment) {
return (alignment + t - 1) & ~(alignment - 1);
}
template <typename T> inline void* RoundUp(T* pPtr, unsigned int alignment) {
u32 value = reinterpret_cast<u32>(pPtr);
u32 rounded = (alignment + value - 1) & ~(alignment - 1);
return reinterpret_cast<void*>(rounded);
}
template <typename T> inline T RoundDown(T t, unsigned int alignment) {
return t & ~(alignment - 1);
}
template <typename T> inline void* RoundDown(T* pPtr, unsigned int alignment) {
u32 value = reinterpret_cast<u32>(pPtr);
u32 rounded = value & ~(alignment - 1);
return reinterpret_cast<void*>(rounded);
}
} // namespace
} // namespace ut
} // namespace nw4r
#endif