A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
MapdataCheckPoint.cc
1#include "MapdataCheckPoint.hh"
2
3#include "game/system/CourseMap.hh"
4#include "game/system/map/MapdataCheckPath.hh"
5
6#include <ranges>
7
8namespace System {
9
11MapdataCheckPoint::MapdataCheckPoint(const SData *data)
12 : m_rawData(data), m_nextCount(0), m_prevCount(0) {
13 EGG::RamStream stream = EGG::RamStream(data, sizeof(SData));
14 read(stream);
15 m_midpoint = 0.5f * (m_left + m_right);
16 m_dir = EGG::Vector2f(m_right.y - m_left.y, m_left.x - m_right.x);
17 m_dir.normalise();
18}
19
20void MapdataCheckPoint::read(EGG::Stream &stream) {
21 m_left.read(stream);
22 m_right.read(stream);
23 m_jugemIndex = stream.read_s8();
24 m_checkArea = stream.read_s8();
25 m_prevPt = stream.read_u8();
26 m_nextPt = stream.read_u8();
27}
28
33void MapdataCheckPoint::initCheckpointLinks(MapdataCheckPointAccessor &accessor, int id) {
34 m_id = id;
35 const auto *checkPathAccessor = CourseMap::Instance()->checkPath();
36
37 // Calculate the quadrilateral's `m_prevPoints`. If the check point is the first in its group,
38 // it has multiple previous checkpoints defined by its preceding checkpaths
39 if (m_prevPt == 0xFF) {
40 MapdataCheckPath *checkpath = checkPathAccessor->findCheckpathForCheckpoint(id);
41 if (checkpath) {
42 m_prevCount = 0;
43
44 for (auto [i, prevID] : std::views::enumerate(checkpath->prev())) {
45 if (prevID == 0xFF) {
46 continue;
47 }
48
49 m_prevPoints[i] = accessor.get(checkPathAccessor->get(prevID)->end());
50 ++m_prevCount;
51 }
52 }
53 } else {
54 m_prevPoints[0] = accessor.get(m_prevPt);
55 ++m_prevCount;
56 }
57
58 // Calculate the quadrilateral's `m_nextPoints`. If the checkpoint is the last in its group, it
59 // can have multiple quadrilaterals (and nextCheckpoint) which are determined by its next
60 // path(s)
61 if (m_nextPt == 0xFF) {
62 MapdataCheckPath *checkpath = checkPathAccessor->findCheckpathForCheckpoint(id);
63 if (checkpath) {
64 m_nextCount = 0;
65
66 for (auto [i, nextID] : std::views::enumerate(checkpath->next())) {
67 if (nextID == 0xFF) {
68 continue;
69 }
70
71 m_nextPoints[i].checkpoint = accessor.get(checkPathAccessor->get(nextID)->start());
72 ++m_nextCount;
73 }
74 }
75 } else {
76 m_nextPoints[0].checkpoint = accessor.get(m_nextPt);
77 ++m_nextCount;
78 }
79
80 // Form the checkpoint's quadrilateral(s)
81 for (auto [i, next] : std::views::enumerate(m_nextPoints)) {
82 if (i < m_nextCount) {
83 auto &nextLinked = m_nextPoints[i];
84 auto *nextPoint = nextLinked.checkpoint;
85
86 next.distance = (nextPoint->m_midpoint - m_midpoint).normalise();
87 next.p0diff = nextPoint->m_left - m_left;
88 next.p1diff = nextPoint->m_right - m_right;
89 } else {
90 next.distance = 0.0f;
91 next.p0diff = EGG::Vector2f::zero;
92 next.p1diff = EGG::Vector2f::zero;
93 }
94 }
95}
96
98MapdataCheckPoint::SectorOccupancy MapdataCheckPoint::checkSectorAndDistanceRatio(
99 const EGG::Vector3f &pos, f32 &distanceRatio) const {
100 bool betweenSides = false;
101 EGG::Vector2f p1 = m_right;
102 p1.y = pos.z - p1.y;
103 p1.x = pos.x - p1.x;
104
105 for (size_t i = 0; i < m_nextCount; ++i) {
106 const LinkedCheckpoint &next = m_nextPoints[i];
107 EGG::Vector2f p0 = next.checkpoint->m_left;
108 p0.y = pos.z - p0.y;
109 p0.x = pos.x - p0.x;
111 checkSectorAndDistanceRatio(next, p0, p1, distanceRatio);
112
113 if (result == SectorOccupancy::InsideSector) {
114 return SectorOccupancy::InsideSector;
115 } else if (result == SectorOccupancy::BetweenSides) {
116 betweenSides = true;
117 }
118 }
119
120 return betweenSides ? SectorOccupancy::BetweenSides : SectorOccupancy::OutsideSector;
121}
122
129u16 MapdataCheckPoint::getEntryOffsetMs(const EGG::Vector2f &prevPos,
130 const EGG::Vector2f &pos) const {
131 constexpr f32 REFRESH_PERIOD = 1000.0f / 59.94f;
132
133 EGG::Vector2f velocity = pos - prevPos;
134 velocity *= 1.0f / REFRESH_PERIOD;
135
136 // d_k = p_0 - m + kv
137 // We walk along the line via k increments, until we pass it
138 // The goal of this function is to approximate k as an integer
139 EGG::Vector2f displacement = prevPos - m_midpoint + velocity;
140
141 u16 k = 1;
142 for (; static_cast<f32>(k) < REFRESH_PERIOD && displacement.dot(m_dir) < 0.0f; ++k) {
143 displacement += velocity;
144 }
145
146 return k;
147}
148
155f32 MapdataCheckPoint::getEntryOffsetExact(const EGG::Vector2f &prevPos,
156 const EGG::Vector2f &pos) const {
157 constexpr f32 REFRESH_PERIOD = 1000.0f / 59.94f;
158
159 EGG::Vector2f velocity = pos - prevPos;
160 velocity *= 1.0f / REFRESH_PERIOD;
161
162 // d_k = p_0 - m + kv
163 // d_k dot r = 0 => k is the exact offset to the finish line
164 // Therefore, k = ((m - p_0) dot r) / (v dot r)
165
166 f32 x = (m_midpoint - prevPos).dot(m_dir);
167 f32 y = velocity.dot(m_dir);
168
169 // y = 0 => v is parallel to the checkpoint line
170 return y != 0.0f ? x / y : 0.0f;
171}
172
174MapdataCheckPoint::SectorOccupancy MapdataCheckPoint::checkSectorAndDistanceRatio(
175 const LinkedCheckpoint &next, const EGG::Vector2f &p0, const EGG::Vector2f &p1,
176 f32 &distanceRatio) const {
177 if (!checkSector(next, p0, p1)) {
178 return SectorOccupancy::OutsideSector;
179 }
180
181 return checkDistanceRatio(next, p0, p1, distanceRatio) ? SectorOccupancy::InsideSector :
182 SectorOccupancy::BetweenSides;
183}
184
187bool MapdataCheckPoint::checkSector(const LinkedCheckpoint &next, const EGG::Vector2f &p0,
188 const EGG::Vector2f &p1) const {
189 if (-(next.p0diff.y) * p0.x + next.p0diff.x * p0.y < 0.0f) {
190 return false;
191 }
192
193 if (next.p1diff.y * p1.x - next.p1diff.x * p1.y < 0.0f) {
194 return false;
195 }
196
197 return true;
198}
199
204bool MapdataCheckPoint::checkDistanceRatio(const LinkedCheckpoint &next, const EGG::Vector2f &p0,
205 const EGG::Vector2f &p1, f32 &distanceRatio) const {
206 f32 d1 = m_dir.dot(p1);
207 f32 d2 = -(next.checkpoint->m_dir.dot(p0));
208 distanceRatio = d1 / (d1 + d2);
209 return distanceRatio >= 0.0f && distanceRatio <= 1.0f;
210}
211
212MapdataCheckPointAccessor::MapdataCheckPointAccessor(const MapSectionHeader *header)
214 MapdataAccessorBase::init(
215 reinterpret_cast<const MapdataCheckPoint::SData *>(m_sectionHeader + 1),
216 parse<u16>(m_sectionHeader->count));
217 init();
218}
219
220MapdataCheckPointAccessor::~MapdataCheckPointAccessor() = default;
221
225 s8 lastKcpType = -1;
226 s16 finishLineCheckpointId = -1;
227
228 for (size_t ckptId = 0; ckptId < size(); ckptId++) {
229 MapdataCheckPoint *checkpoint = get(ckptId);
230 checkpoint->initCheckpointLinks(*this, ckptId);
231
232 if (checkpoint->isFinishLine()) {
233 finishLineCheckpointId = ckptId;
234 }
235
236 lastKcpType = std::max(lastKcpType, checkpoint->checkArea());
237 }
238
239 m_lastKcpType = lastKcpType;
240 m_finishLineCheckpointId = finishLineCheckpointId;
241}
242
243} // namespace System
A stream of data stored in memory.
Definition Stream.hh:64
A stream of data, abstracted to allow for continuous seeking.
Definition Stream.hh:10
void init()
Initializes all checkpoint links, and finds the finish line and last key checkpoint.
void initCheckpointLinks(MapdataCheckPointAccessor &accessor, int id)
Calculates m_nextPoints and m_prevPoints from m_nextPt and m_prevPt.
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5
A 2D float vector.
Definition Vector.hh:12
A 3D float vector.
Definition Vector.hh:83