mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 22:25:41 +12:00
Add 3DSX RomFS reads via NCCH archive
This commit is contained in:
parent
53917841bc
commit
a380aa83f0
5 changed files with 15 additions and 8 deletions
|
@ -18,7 +18,8 @@ public:
|
||||||
// Returns whether the cart has a RomFS
|
// Returns whether the cart has a RomFS
|
||||||
bool hasRomFS() {
|
bool hasRomFS() {
|
||||||
auto cxi = mem.getCXI();
|
auto cxi = mem.getCXI();
|
||||||
return (cxi != nullptr && cxi->hasRomFS());
|
auto hb3dsx = mem.get3DSX();
|
||||||
|
return (cxi != nullptr && cxi->hasRomFS()) || (hb3dsx != nullptr && hb3dsx->hasRomFs());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns whether the cart has an ExeFS (All executable carts should have an ExeFS. This is just here to be safe)
|
// Returns whether the cart has an ExeFS (All executable carts should have an ExeFS. This is just here to be safe)
|
||||||
|
|
|
@ -19,7 +19,7 @@ public:
|
||||||
bool hasRomFS() {
|
bool hasRomFS() {
|
||||||
auto cxi = mem.getCXI();
|
auto cxi = mem.getCXI();
|
||||||
auto hb3dsx = mem.get3DSX();
|
auto hb3dsx = mem.get3DSX();
|
||||||
return (cxi != nullptr && cxi->hasRomFS()) | (hb3dsx != nullptr && hb3dsx->hasRomFs());
|
return (cxi != nullptr && cxi->hasRomFS()) || (hb3dsx != nullptr && hb3dsx->hasRomFs());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns whether the cart has an ExeFS (All executable carts should have an ExeFS. This is just here to be safe)
|
// Returns whether the cart has an ExeFS (All executable carts should have an ExeFS. This is just here to be safe)
|
||||||
|
|
|
@ -131,20 +131,20 @@ std::optional<u32> NCCHArchive::readFile(FileSession* file, u64 offset, u32 size
|
||||||
}
|
}
|
||||||
|
|
||||||
auto cxi = mem.getCXI();
|
auto cxi = mem.getCXI();
|
||||||
IOFile& ioFile = mem.CXIFile;
|
auto hb3dsx = mem.get3DSX();
|
||||||
|
|
||||||
NCCH::FSInfo fsInfo;
|
// If isCXI is true then we've loaded a CXI/3DS file, else it's 3DSX
|
||||||
|
bool isCXI = cxi != nullptr;
|
||||||
|
|
||||||
// Seek to file offset depending on if we're reading from RomFS, ExeFS, etc
|
// Seek to file offset depending on if we're reading from RomFS, ExeFS, etc
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case PathType::RomFS: {
|
case PathType::RomFS: {
|
||||||
const u64 romFSSize = cxi->romFS.size;
|
const u64 romFSSize = isCXI ? cxi->romFS.size : hb3dsx->romFSSize;
|
||||||
const u64 romFSOffset = cxi->romFS.offset;
|
const u64 romFSOffset = isCXI ? cxi->romFS.offset : hb3dsx->romFSOffset;
|
||||||
if ((offset >> 32) || (offset >= romFSSize) || (offset + size >= romFSSize)) {
|
if ((offset >> 32) || (offset >= romFSSize) || (offset + size >= romFSSize)) {
|
||||||
Helpers::panic("Tried to read from NCCH with too big of an offset");
|
Helpers::panic("Tried to read from NCCH with too big of an offset");
|
||||||
}
|
}
|
||||||
|
|
||||||
fsInfo = cxi->romFS;
|
|
||||||
offset += 0x1000;
|
offset += 0x1000;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -154,7 +154,8 @@ std::optional<u32> NCCHArchive::readFile(FileSession* file, u64 offset, u32 size
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<u8[]> data(new u8[size]);
|
std::unique_ptr<u8[]> data(new u8[size]);
|
||||||
auto [success, bytesRead] = cxi->readFromFile(ioFile, fsInfo, &data[0], offset, size);
|
auto [success, bytesRead] =
|
||||||
|
isCXI ? cxi->readFromFile(mem.CXIFile, cxi->romFS, &data[0], offset, size) : hb3dsx->readRomFSBytes(&data[0], offset, size);
|
||||||
|
|
||||||
if (!success) {
|
if (!success) {
|
||||||
Helpers::panic("Failed to read from NCCH archive");
|
Helpers::panic("Failed to read from NCCH archive");
|
||||||
|
|
|
@ -125,6 +125,7 @@ std::optional<u32> SelfNCCHArchive::readFile(FileSession* file, u64 offset, u32
|
||||||
data.resize(size);
|
data.resize(size);
|
||||||
std::tie(success, bytesRead) = cxi->readFromFile(ioFile, fsInfo, &data[0], offset, size);
|
std::tie(success, bytesRead) = cxi->readFromFile(ioFile, fsInfo, &data[0], offset, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if (auto hb3dsx = mem.get3DSX(); hb3dsx != nullptr) {
|
else if (auto hb3dsx = mem.get3DSX(); hb3dsx != nullptr) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case PathType::RomFS: {
|
case PathType::RomFS: {
|
||||||
|
|
|
@ -426,6 +426,10 @@ bool Emulator::loadROM(const std::filesystem::path& path) {
|
||||||
reset(ReloadOption::NoReload);
|
reset(ReloadOption::NoReload);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Reset whatever state needs to be reset before loading a new ROM
|
||||||
|
memory.loadedCXI = std::nullopt;
|
||||||
|
memory.loaded3DSX = std::nullopt;
|
||||||
|
|
||||||
// Get path for saving files (AppData on Windows, /home/user/.local/share/ApplcationName on Linux, etc)
|
// Get path for saving files (AppData on Windows, /home/user/.local/share/ApplcationName on Linux, etc)
|
||||||
// Inside that path, we be use a game-specific folder as well. Eg if we were loading a ROM called PenguinDemo.3ds, the savedata would be in
|
// Inside that path, we be use a game-specific folder as well. Eg if we were loading a ROM called PenguinDemo.3ds, the savedata would be in
|
||||||
// %APPDATA%/Alber/PenguinDemo/SaveData on Windows, and so on. We do this because games save data in their own filesystem on the cart
|
// %APPDATA%/Alber/PenguinDemo/SaveData on Windows, and so on. We do this because games save data in their own filesystem on the cart
|
||||||
|
|
Loading…
Add table
Reference in a new issue