A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
GhostFile.cc
1#include "GhostFile.hh"
2
3#include <egg/core/Decomp.hh>
4
5#include <cstring>
6
7namespace System {
8
10GhostFile::GhostFile(const RawGhostFile &raw) {
11 u8 *streamPtr = const_cast<u8 *>(raw.buffer());
12 EGG::RamStream stream(streamPtr, RKG_HEADER_SIZE);
13 read(stream);
14 m_inputs = raw.buffer() + RKG_HEADER_SIZE;
15}
16
18GhostFile::~GhostFile() = default;
19
22 constexpr size_t RKG_MII_DATA_SIZE = 0x4A;
23 constexpr size_t RKG_USER_DATA_SIZE = 0x14;
24
25 stream.skip(0x4); // RKGD
26
27 // 0x04 - 0x07
28 u32 data = stream.read_u32();
29 m_raceTime = Timer(data);
30 m_course = static_cast<Course>(data >> 0x2 & 0x3F);
31
32 // 0x08 - 0x0B
33 data = stream.read_u32();
34 m_vehicle = static_cast<Vehicle>(data >> 0x1A);
35 m_character = static_cast<Character>(data >> 0x14 & 0x3F);
36 m_year = (data >> 0xD) & 0x7F;
37 m_month = (data >> 0x9) & 0xF;
38 m_day = (data >> 0x4) & 0x1F;
39
40 // 0x0C - 0x0F
41 data = stream.read_u32();
42 m_type = data >> 0xA & 0x7F;
43 m_driftIsAuto = data >> 0x11 & 0x1;
44 m_inputSize = data & 0xFFFF;
45
46 // 0x10
47 m_lapCount = stream.read_u8();
48
49 // 0x11 - 0x1F
50 for (size_t i = 0; i < 5; ++i) {
51 m_lapTimes[i] = Timer(stream.read_u32());
52 stream.jump(stream.index() - 1);
53 }
54
55 stream.read(m_userData.data(), RKG_USER_DATA_SIZE);
56 m_userData[10] = L'\0';
57
58 // 0x34
59 m_location = stream.read_u8();
60 stream.skip(0x7);
61 stream.read(m_miiData.data(), RKG_MII_DATA_SIZE);
62}
63
64RawGhostFile::RawGhostFile() {
65 memset(m_buffer, 0, sizeof(m_buffer));
66}
67
68RawGhostFile::RawGhostFile(const u8 *rkg) {
69 init(rkg);
70}
71
72RawGhostFile::~RawGhostFile() = default;
73
74RawGhostFile &RawGhostFile::operator=(const u8 *rkg) {
75 init(rkg);
76
77 return *this;
78}
79
80void RawGhostFile::init(const u8 *rkg) {
81 if (!isValid(rkg)) {
82 PANIC("Invalid RKG header");
83 }
84
85 if (compressed(rkg)) {
86 if (!decompress(rkg)) {
87 PANIC("Failed to decompress RKG!");
88 }
89 } else {
90 memcpy(m_buffer, rkg, sizeof(m_buffer));
91 }
92}
93
95bool RawGhostFile::decompress(const u8 *rkg) {
96 memcpy(m_buffer, rkg, RKG_HEADER_SIZE);
97
98 // Unset compressed flag
99 *(m_buffer + 0xC) &= 0xF7;
100
101 // Get uncompressed size. Skip past 0x4 bytes which represents the size of the compressed data
102 s32 uncompressedSize = EGG::Decomp::GetExpandSize(rkg + RKG_HEADER_SIZE + 0x4);
103
104 if (uncompressedSize <= 0 ||
105 static_cast<u32>(uncompressedSize) > RKG_UNCOMPRESSED_INPUT_DATA_SECTION_SIZE) {
106 return false;
107 }
108
109 EGG::Decomp::DecodeSZS(rkg + RKG_HEADER_SIZE + 0x4, m_buffer + RKG_HEADER_SIZE);
110
111 // Set input data section length
112 *reinterpret_cast<u16 *>(m_buffer + 0xE) = static_cast<u16>(uncompressedSize);
113
114 return true;
115}
116
120bool RawGhostFile::isValid(const u8 *rkg) const {
121 if (strncmp(reinterpret_cast<const char *>(rkg), "RKGD", 4) != 0) {
122 PANIC("RKG header malformed");
123 return false;
124 }
125
126 u32 ids = parse<u32>(*reinterpret_cast<const u32 *>(rkg + 0x8));
127 Vehicle vehicle = static_cast<Vehicle>(ids >> 0x1a);
128 Character character = static_cast<Character>((ids >> 0x14) & 0x3f);
129 u8 year = (ids >> 0xd) & 0x7f;
130 u8 day = (ids >> 0x4) & 0x1f;
131 u8 month = (ids >> 0x9) & 0xf;
132
133 if (vehicle >= Vehicle::Max || character >= Character::Max) {
134 return false;
135 }
136
137 if (year >= 100 || day >= 32 || month > 12) {
138 return false;
139 }
140
141 // Validate weight class match
142 WeightClass charWeight = CharacterToWeight(character);
143 WeightClass vehicleWeight = VehicleToWeight(vehicle);
144
145 if (charWeight == WeightClass::Invalid) {
146 PANIC("Invalid character weight class!");
147 }
148 if (vehicleWeight == WeightClass::Invalid) {
149 PANIC("Invalid vehicle weight class!");
150 }
151 if (charWeight != vehicleWeight) {
152 PANIC("Character/Bike weight class mismatch!");
153 }
154
155 return true;
156}
157
158} // namespace System
A stream of data stored in memory.
Definition Stream.hh:64
u32 m_type
The type of ghost.
Definition GhostFile.hh:128
u16 m_inputSize
The size of the decompressed input data section.
Definition GhostFile.hh:131
u32 m_location
0xFFFF if sharing disabled
Definition GhostFile.hh:130
bool m_driftIsAuto
True for automatic, false for manual.
Definition GhostFile.hh:129
void read(EGG::RamStream &stream)
Organizes binary data into members. See RawGhostFile.
Definition GhostFile.cc:21
u8 m_year
The year, relative to 2000.
Definition GhostFile.hh:125
bool isValid(const u8 *rkg) const
Definition GhostFile.cc:120
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5
A simple struct to represent a lap or race finish time.