A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
Math.hh
1#pragma once
2
3#include <Common.hh>
4
5#include <cmath>
6
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;
14
16namespace EGG::Mathf {
17
18[[nodiscard]] f32 frsqrt(f32 x);
19
21[[nodiscard]] static inline f32 sqrt(f32 x) {
22 return x > 0.0f ? frsqrt(x) * x : 0.0f;
23}
24
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);
29
32[[nodiscard]] static inline f32 sin(f32 x) {
33 return SinFIdx(x * RAD2FIDX);
34}
35
38[[nodiscard]] static inline f32 cos(f32 x) {
39 return CosFIdx(x * RAD2FIDX);
40}
41
43[[nodiscard]] static inline f32 acos(f32 x) {
44 return ::acosl(x);
45}
46
48[[nodiscard]] static inline f32 atan2(f32 y, f32 x) {
49 return Atan2FIdx(y, x) * FIDX2RAD;
50}
51
52[[nodiscard]] static inline f32 abs(f32 x) {
53 return std::abs(x);
54}
55
59[[nodiscard]] static inline f64 force25Bit(f64 x) {
60 u64 bits = std::bit_cast<u64>(x);
61 bits = (bits & 0xfffffffff8000000ULL) + (bits & 0x8000000);
62 return std::bit_cast<f64>(bits);
63}
64
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));
70}
71
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));
77}
78
79// frsqrte matching
80struct BaseAndDec {
81 int base;
82 int dec;
83};
84
85union c64 {
86 c64(const f64 p) {
87 f = p;
88 }
89
90 [[nodiscard]] u64 _hex() const {
91 return u;
92 }
93
94 u64 u;
95 f64 f;
96};
97
98static const std::array<BaseAndDec, 32> frsqrte_expected = {{
99 {0x3ffa000, 0x7a4},
100 {0x3c29000, 0x700},
101 {0x38aa000, 0x670},
102 {0x3572000, 0x5f2},
103 {0x3279000, 0x584},
104 {0x2fb7000, 0x524},
105 {0x2d26000, 0x4cc},
106 {0x2ac0000, 0x47e},
107 {0x2881000, 0x43a},
108 {0x2665000, 0x3fa},
109 {0x2468000, 0x3c2},
110 {0x2287000, 0x38e},
111 {0x20c1000, 0x35e},
112 {0x1f12000, 0x332},
113 {0x1d79000, 0x30a},
114 {0x1bf4000, 0x2e6},
115 {0x1a7e800, 0x568},
116 {0x17cb800, 0x4f3},
117 {0x1552800, 0x48d},
118 {0x130c000, 0x435},
119 {0x10f2000, 0x3e7},
120 {0x0eff000, 0x3a2},
121 {0x0d2e000, 0x365},
122 {0x0b7c000, 0x32e},
123 {0x09e5000, 0x2fc},
124 {0x0867000, 0x2d0},
125 {0x06ff000, 0x2a8},
126 {0x05ab800, 0x283},
127 {0x046a000, 0x261},
128 {0x0339800, 0x243},
129 {0x0218800, 0x226},
130 {0x0105800, 0x20b},
131}};
132
133[[nodiscard]] static inline f64 frsqrte(const f64 val) {
134 c64 input(val);
135
136 u64 mantissa = input._hex() & ((1LL << 52) - 1);
137 const u64 sign = input._hex() & (1ULL << 63);
138 u64 exponent = input._hex() & (0x7FFLL << 52);
139
140 // Special case 0
141 if (mantissa == 0 && exponent == 0) {
142 return sign ? -std::numeric_limits<f64>::infinity() : std::numeric_limits<f64>::infinity();
143 }
144
145 // Special case NaN-ish numbers
146 if (exponent == (0x7FFLL << 52)) {
147 if (mantissa == 0) {
148 if (sign) {
149 return std::numeric_limits<f64>::quiet_NaN();
150 }
151
152 return 0.0;
153 }
154
155 return 0.0 + val;
156 }
157
158 // Negative numbers return NaN
159 if (sign) {
160 return std::numeric_limits<f64>::quiet_NaN();
161 }
162
163 if (!exponent) {
164 // "Normalize" denormal values
165 do {
166 exponent -= 1LL << 52;
167 mantissa <<= 1;
168 } while (!(mantissa & (1LL << 52)));
169 mantissa &= (1LL << 52) - 1;
170 exponent += 1LL << 52;
171 }
172
173 const bool odd_exponent = !(exponent & (1LL << 52));
174 exponent = ((0x3FFLL << 52) - ((exponent - (0x3FELL << 52)) / 2)) & (0x7FFLL << 52);
175 input.u = sign | exponent;
176
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;
181
182 return input.f;
183}
184
185} // namespace EGG::Mathf
This header houses common data types such as our integral types and enums.
Math functions and constants used in the base game.
Definition Math.cc:3
static f32 fma(f32 x, f32 y, f32 z)
Fused multiply-add operation.
Definition Math.hh:67
f32 frsqrt(f32 x)
Definition Math.cc:315
static f32 sin(f32 x)
Definition Math.hh:32
static f64 force25Bit(f64 x)
This is used to mimic the Wii's floating-point unit.
Definition Math.hh:59
static f32 cos(f32 x)
Definition Math.hh:38
static f32 fms(f32 x, f32 y, f32 z)
Fused multiply-subtract operation.
Definition Math.hh:74