3#include <egg/core/Decomp.hh>
10GhostFile::GhostFile(
const RawGhostFile &raw) {
11 u8 *streamPtr =
const_cast<u8 *
>(raw.buffer());
14 m_inputs = raw.buffer() + RKG_HEADER_SIZE;
18GhostFile::~GhostFile() =
default;
22 constexpr size_t RKG_MII_DATA_SIZE = 0x4A;
23 constexpr size_t RKG_USER_DATA_SIZE = 0x14;
28 u32 data = stream.read_u32();
29 m_raceTime =
Timer(data);
30 m_course =
static_cast<Course
>(data >> 0x2 & 0x3F);
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;
41 data = stream.read_u32();
42 m_type = data >> 0xA & 0x7F;
47 m_lapCount = stream.read_u8();
50 for (
size_t i = 0; i < 5; ++i) {
51 m_lapTimes[i] =
Timer(stream.read_u32());
52 stream.jump(stream.index() - 1);
55 stream.read(m_userData.data(), RKG_USER_DATA_SIZE);
56 m_userData[10] = L
'\0';
61 stream.read(m_miiData.data(), RKG_MII_DATA_SIZE);
64RawGhostFile::RawGhostFile() {
65 memset(m_buffer, 0,
sizeof(m_buffer));
68RawGhostFile::RawGhostFile(
const u8 *rkg) {
72RawGhostFile::~RawGhostFile() =
default;
74RawGhostFile &RawGhostFile::operator=(
const u8 *rkg) {
80void RawGhostFile::init(
const u8 *rkg) {
82 PANIC(
"Invalid RKG header");
85 if (compressed(rkg)) {
86 if (!decompress(rkg)) {
87 PANIC(
"Failed to decompress RKG!");
90 memcpy(m_buffer, rkg,
sizeof(m_buffer));
95bool RawGhostFile::decompress(
const u8 *rkg) {
96 memcpy(m_buffer, rkg, RKG_HEADER_SIZE);
99 *(m_buffer + 0xC) &= 0xF7;
102 s32 uncompressedSize = EGG::Decomp::GetExpandSize(rkg + RKG_HEADER_SIZE + 0x4);
104 if (uncompressedSize <= 0 ||
105 static_cast<u32
>(uncompressedSize) > RKG_UNCOMPRESSED_INPUT_DATA_SECTION_SIZE) {
109 EGG::Decomp::DecodeSZS(rkg + RKG_HEADER_SIZE + 0x4, m_buffer + RKG_HEADER_SIZE);
112 *
reinterpret_cast<u16 *
>(m_buffer + 0xE) =
static_cast<u16>(uncompressedSize);
121 if (strncmp(
reinterpret_cast<const char *
>(rkg),
"RKGD", 4) != 0) {
122 PANIC(
"RKG header malformed");
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;
133 if (vehicle >= Vehicle::Max || character >= Character::Max) {
137 if (year >= 100 || day >= 32 || month > 12) {
142 WeightClass charWeight = CharacterToWeight(character);
143 WeightClass vehicleWeight = VehicleToWeight(vehicle);
145 if (charWeight == WeightClass::Invalid) {
146 PANIC(
"Invalid character weight class!");
148 if (vehicleWeight == WeightClass::Invalid) {
149 PANIC(
"Invalid vehicle weight class!");
151 if (charWeight != vehicleWeight) {
152 PANIC(
"Character/Bike weight class mismatch!");
A stream of data stored in memory.
u32 m_type
The type of ghost.
u16 m_inputSize
The size of the decompressed input data section.
u32 m_location
0xFFFF if sharing disabled
bool m_driftIsAuto
True for automatic, false for manual.
void read(EGG::RamStream &stream)
Organizes binary data into members. See RawGhostFile.
u8 m_year
The year, relative to 2000.
bool isValid(const u8 *rkg) const
High-level handling for generic system operations, such as input reading, race configuration,...
A simple struct to represent a lap or race finish time.