A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
DvdArchive.cc
1#include "DvdArchive.hh"
2
3#include <abstract/File.hh>
4
5#include <egg/core/Decomp.hh>
6
7namespace System {
8
10DvdArchive::DvdArchive()
11 : m_archive(nullptr), m_archiveStart(nullptr), m_archiveSize(0), m_fileStart(nullptr),
12 m_fileSize(0), m_state(State::Cleared) {}
13
15DvdArchive::~DvdArchive() {
16 unmount();
17}
18
20void DvdArchive::decompress() {
21 m_archiveSize = EGG::Decomp::GetExpandSize(reinterpret_cast<u8 *>(m_fileStart));
22 m_archiveStart = new u8[m_archiveSize];
23 EGG::Decomp::DecodeSZS(reinterpret_cast<u8 *>(m_fileStart),
24 reinterpret_cast<u8 *>(m_archiveStart));
25 m_state = State::Decompressed;
26}
27
29void *DvdArchive::getFile(const char *filename, size_t *size) const {
30 if (m_state != State::Mounted) {
31 return nullptr;
32 }
33
34 char buffer[256];
35 if (filename[0] == '/') {
36 snprintf(buffer, sizeof(buffer), "%s", filename);
37 } else {
38 snprintf(buffer, sizeof(buffer), "/%s", filename);
39 }
40 buffer[sizeof(buffer) - 1] = '\0';
41
43 s32 entryId = m_archive->convertPathToEntryId(buffer);
44 if (entryId == -1) {
45 return nullptr;
46 }
47
48 void *file = m_archive->getFileFast(entryId, fileInfo);
49 if (file && size) {
50 *size = fileInfo.length;
51 }
52
53 return file;
54}
55
57void DvdArchive::load(const char *path, bool decompress_) {
58 if (m_state == State::Cleared) {
59 rip(path);
60 }
61
62 if (m_state == State::Ripped) {
63 if (decompress_) {
64 decompress();
65 clearFile();
66 } else {
67 move();
68 }
69 mount();
70 }
71}
72
74void DvdArchive::load(const DvdArchive *other) {
75 if (other->m_state == State::Ripped) {
76 m_fileStart = other->m_fileStart;
77 m_state = State::Ripped;
78 decompress();
79 mount();
80 m_fileStart = nullptr;
81 m_fileSize = 0;
82 } else {
83 m_state = State::Cleared;
84 }
85}
86
88void DvdArchive::load(void *fileStart, size_t fileSize, bool decompress_) {
89 m_fileStart = fileStart;
90 m_fileSize = fileSize;
91 if (decompress_) {
92 decompress();
93 m_fileStart = nullptr;
94 m_fileSize = 0;
95 }
96 mount();
97}
98
100void DvdArchive::mount() {
101 m_archive = EGG::Archive::Mount(m_archiveStart);
102 m_state = State::Mounted;
103}
104
106void DvdArchive::move() {
107 m_archiveStart = m_fileStart;
108 m_archiveSize = m_fileSize;
109 m_fileStart = nullptr;
110 m_fileSize = 0;
111 m_state = State::Decompressed;
112}
113
115void DvdArchive::rip(const char *path) {
116 m_fileStart = Abstract::File::Load(path, m_fileSize);
117 if (m_fileSize != 0 && m_fileStart) {
118 m_state = State::Ripped;
119 }
120}
121
123void DvdArchive::clear() {
124 clearArchive();
125 clearFile();
126}
127
129void DvdArchive::clearArchive() {
130 if (!m_archiveStart) {
131 return;
132 }
133
134 delete[] static_cast<u8 *>(m_archiveStart);
135 m_archiveStart = nullptr;
136 m_archiveSize = 0;
137}
138
140void DvdArchive::clearFile() {
141 if (!m_fileStart) {
142 return;
143 }
144
145 delete[] static_cast<u8 *>(m_fileStart);
146 m_fileStart = nullptr;
147 m_fileSize = 0;
148}
149
151void DvdArchive::unmount() {
152 if (m_state == State::Mounted) {
153 m_archive->unmount();
154 }
155 clear();
156 m_state = State::Cleared;
157}
158
159} // namespace System
static Archive * Mount(void *archiveStart)
Creates a new Archive object or increments the ref count for an already existing Archive.
Definition Archive.cc:56
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5