A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
MultiDvdArchive.cc
1#include "MultiDvdArchive.hh"
2
3#include <cstring>
4
5namespace System {
6
7static size_t SUFFIX_SIZE = 8;
8
10MultiDvdArchive::MultiDvdArchive(u16 archiveCount) : m_archiveCount(archiveCount) {
11 m_archives = new DvdArchive[archiveCount];
12 m_fileStarts = new void *[archiveCount];
13 m_fileSizes = new size_t[archiveCount];
14 m_suffixes = new char *[archiveCount];
15 m_formats = new Format[archiveCount];
16
17 for (u16 i = 0; i < m_archiveCount; i++) {
18 m_fileStarts[i] = nullptr;
19 m_fileSizes[i] = 0;
20 m_suffixes[i] = new char[SUFFIX_SIZE];
21 strncpy(m_suffixes[i], SZS_EXTENSION, SUFFIX_SIZE);
22 m_formats[i] = Format::Double;
23 }
24}
25
27MultiDvdArchive::~MultiDvdArchive() {
28 delete[] m_archives;
29 // WARN: Could lead to a memory leak if this is the only reference to the file!
30 delete[] m_fileStarts;
31 delete[] m_fileSizes;
32 // WARN: Could lead to a memory leak if the pointer is not static!
33 delete[] m_suffixes;
34 delete[] m_formats;
35}
36
38void *MultiDvdArchive::getFile(const char *filename, size_t *size) const {
39 void *file = nullptr;
40
41 for (u16 i = m_archiveCount; i-- > 0;) {
42 const DvdArchive &archive = m_archives[i];
43
44 if (!archive.isLoaded()) {
45 continue;
46 }
47
48 file = archive.getFile(filename, size);
49 if (file) {
50 break;
51 }
52 }
53
54 return file;
55}
56
58void MultiDvdArchive::load(const char *filename) {
59 char buffer[256];
60
61 for (u16 i = 0; i < m_archiveCount; i++) {
62 switch (m_formats[i]) {
63 case Format::Double:
64 snprintf(buffer, sizeof(buffer), "%s%s", filename, m_suffixes[i]);
65 break;
66 case Format::Single:
67 snprintf(buffer, sizeof(buffer), "%s", filename);
68 break;
69 case Format::None:
70 break;
71 default:
72 continue;
73 }
74
75 if (m_formats[i] == Format::None) {
76 m_archives[i].load(m_fileStarts[i], m_fileSizes[i], true);
77 } else {
78 m_archives[i].load(buffer, true);
79 }
80 }
81}
82
84void MultiDvdArchive::load(const MultiDvdArchive *other) {
85 for (u16 i = 0; i < m_archiveCount; i++) {
86 m_archives[i].load(&other->m_archives[i]);
87 }
88}
89
91void MultiDvdArchive::rip(const char *filename) {
92 char buffer[256];
93
94 for (u16 i = 0; i < m_archiveCount; i++) {
95 switch (m_formats[i]) {
96 case Format::Double:
97 snprintf(buffer, sizeof(buffer), "%s%s", filename, m_suffixes[i]);
98 break;
99 case Format::Single:
100 snprintf(buffer, sizeof(buffer), "%s", filename);
101 break;
102 case Format::None:
103 break;
104 default:
105 continue;
106 }
107 m_archives[i].rip(buffer);
108 }
109}
110
112void MultiDvdArchive::clear() {
113 for (u16 i = 0; i < m_archiveCount; i++) {
114 m_archives[i].clear();
115 }
116}
117
119void MultiDvdArchive::unmount() {
120 for (u16 i = 0; i < m_archiveCount; i++) {
121 m_archives[i].unmount();
122 }
123}
124
126bool MultiDvdArchive::isLoaded() const {
127 for (u16 i = 0; i < m_archiveCount; i++) {
128 if (m_archives[i].isLoaded()) {
129 return true;
130 }
131 }
132
133 return false;
134}
135
137u16 MultiDvdArchive::rippedArchiveCount() const {
138 u16 count = 0;
139 for (u16 i = 0; i < m_archiveCount; i++) {
140 if (m_archives[i].isRipped()) {
141 count++;
142 }
143 }
144
145 return count;
146}
147
148} // namespace System
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5