7static constexpr f32 F_PI = 3.1415927f;
8static constexpr f32 DEG2RAD = 0.017453292f;
9static constexpr f32 DEG2RAD360 = 0.034906585f;
10static constexpr f32 RAD2DEG = 57.2957795f;
11static constexpr f32 DEG2FIDX = 256.0f / 360.0f;
12static constexpr f32 RAD2FIDX = 128.0f / F_PI;
13static constexpr f32 FIDX2RAD = F_PI / 128.0f;
18[[nodiscard]] f32
frsqrt(f32 x);
21[[nodiscard]]
static inline f32 sqrt(f32 x) {
22 return x > 0.0f ?
frsqrt(x) * x : 0.0f;
25[[nodiscard]] f32 SinFIdx(f32 fidx);
26[[nodiscard]] f32 CosFIdx(f32 fidx);
27[[nodiscard]] f32 AtanFIdx_(f32 fidx);
28[[nodiscard]] f32 Atan2FIdx(f32 x, f32 y);
32[[nodiscard]]
static inline f32
sin(f32 x) {
33 return SinFIdx(x * RAD2FIDX);
38[[nodiscard]]
static inline f32
cos(f32 x) {
39 return CosFIdx(x * RAD2FIDX);
43[[nodiscard]]
static inline f32 acos(f32 x) {
48[[nodiscard]]
static inline f32 atan2(f32 y, f32 x) {
49 return Atan2FIdx(y, x) * FIDX2RAD;
52[[nodiscard]]
static inline f32 abs(f32 x) {
60 u64 bits = std::bit_cast<u64>(x);
61 bits = (bits & 0xfffffffff8000000ULL) + (bits & 0x8000000);
62 return std::bit_cast<f64>(bits);
67[[nodiscard]]
static inline f32
fma(f32 x, f32 y, f32 z) {
68 return static_cast<f32
>(
69 static_cast<f64
>(x) *
force25Bit(
static_cast<f64
>(y)) +
static_cast<f64
>(z));
74[[nodiscard]]
static inline f32
fms(f32 x, f32 y, f32 z) {
75 return static_cast<f32
>(
76 static_cast<f64
>(x) *
force25Bit(
static_cast<f64
>(y)) -
static_cast<f64
>(z));
90 [[nodiscard]] u64 _hex()
const {
98static const std::array<BaseAndDec, 32> frsqrte_expected = {{
133[[nodiscard]]
static inline f64 frsqrte(
const f64 val) {
136 u64 mantissa = input._hex() & ((1LL << 52) - 1);
137 const u64 sign = input._hex() & (1ULL << 63);
138 u64 exponent = input._hex() & (0x7FFLL << 52);
141 if (mantissa == 0 && exponent == 0) {
142 return sign ? -std::numeric_limits<f64>::infinity() : std::numeric_limits<f64>::infinity();
146 if (exponent == (0x7FFLL << 52)) {
149 return std::numeric_limits<f64>::quiet_NaN();
160 return std::numeric_limits<f64>::quiet_NaN();
166 exponent -= 1LL << 52;
168 }
while (!(mantissa & (1LL << 52)));
169 mantissa &= (1LL << 52) - 1;
170 exponent += 1LL << 52;
173 const bool odd_exponent = !(exponent & (1LL << 52));
174 exponent = ((0x3FFLL << 52) - ((exponent - (0x3FELL << 52)) / 2)) & (0x7FFLL << 52);
175 input.u = sign | exponent;
177 const int i =
static_cast<int>(mantissa >> 37);
178 const int index = i / 2048 + (odd_exponent ? 16 : 0);
179 const auto &entry = frsqrte_expected[index];
180 input.u |=
static_cast<uint64_t
>(entry.base - entry.dec * (i % 2048)) << 26;
This header houses common data types such as our integral types and enums.
Math functions and constants used in the base game.
static f32 fma(f32 x, f32 y, f32 z)
Fused multiply-add operation.
static f64 force25Bit(f64 x)
This is used to mimic the Wii's floating-point unit.
static f32 fms(f32 x, f32 y, f32 z)
Fused multiply-subtract operation.