mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 22:25:41 +12:00
Add IVFC parser
This commit is contained in:
parent
72b45377e1
commit
f3ce29bf2d
5 changed files with 137 additions and 1 deletions
|
@ -126,7 +126,8 @@ set(PICA_SOURCE_FILES src/core/PICA/gpu.cpp src/core/PICA/regs.cpp src/core/PICA
|
|||
|
||||
set(LOADER_SOURCE_FILES src/core/loader/elf.cpp src/core/loader/ncsd.cpp src/core/loader/ncch.cpp src/core/loader/lz77.cpp)
|
||||
set(FS_SOURCE_FILES src/core/fs/archive_self_ncch.cpp src/core/fs/archive_save_data.cpp src/core/fs/archive_sdmc.cpp
|
||||
src/core/fs/archive_ext_save_data.cpp src/core/fs/archive_ncch.cpp
|
||||
src/core/fs/archive_ext_save_data.cpp src/core/fs/archive_ncch.cpp src/core/fs/romfs.cpp
|
||||
src/core/fs/ivfc.cpp
|
||||
)
|
||||
|
||||
set(RENDERER_SW_SOURCE_FILES src/core/renderer_sw/renderer_sw.cpp)
|
||||
|
@ -156,6 +157,7 @@ set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/termcolor.hpp
|
|||
include/crypto/aes_engine.hpp include/metaprogramming.hpp include/PICA/pica_vertex.hpp
|
||||
include/config.hpp include/services/ir_user.hpp include/http_server.hpp include/cheats.hpp
|
||||
include/action_replay.hpp include/renderer_sw/renderer_sw.hpp include/compiler_builtins.hpp
|
||||
include/fs/romfs.hpp include/fs/ivfc.hpp
|
||||
)
|
||||
|
||||
set(THIRD_PARTY_SOURCE_FILES third_party/imgui/imgui.cpp
|
||||
|
|
22
include/fs/ivfc.hpp
Normal file
22
include/fs/ivfc.hpp
Normal file
|
@ -0,0 +1,22 @@
|
|||
#pragma once
|
||||
|
||||
#include <cstddef>
|
||||
#include <cstring>
|
||||
#include "helpers.hpp"
|
||||
|
||||
namespace IVFC {
|
||||
|
||||
struct IVFCLevel {
|
||||
u64 logicalOffset;
|
||||
u64 size;
|
||||
u64 blockSize;
|
||||
};
|
||||
|
||||
struct IVFC {
|
||||
u64 masterHashSize;
|
||||
std::vector<IVFCLevel> levels;
|
||||
};
|
||||
|
||||
size_t parseIVFC(uintptr_t ivfcStart, IVFC& ivfc);
|
||||
|
||||
} // namespace IVFC
|
13
include/fs/romfs.hpp
Normal file
13
include/fs/romfs.hpp
Normal file
|
@ -0,0 +1,13 @@
|
|||
#pragma once
|
||||
#include "helpers.hpp"
|
||||
#include <vector>
|
||||
|
||||
namespace RomFS {
|
||||
|
||||
struct RomFSNode {
|
||||
std::vector<RomFSNode> children;
|
||||
};
|
||||
|
||||
RomFSNode parseRomFSTree(uintptr_t romFS, u64 romFSSize);
|
||||
|
||||
} // namespace RomFS
|
78
src/core/fs/ivfc.cpp
Normal file
78
src/core/fs/ivfc.cpp
Normal file
|
@ -0,0 +1,78 @@
|
|||
#include "fs/ivfc.hpp"
|
||||
|
||||
namespace IVFC {
|
||||
|
||||
size_t parseIVFC(uintptr_t ivfcStart, IVFC& ivfc) {
|
||||
uintptr_t ivfcPointer = ivfcStart;
|
||||
std::string magicIVFC((char*)ivfcPointer, 4);
|
||||
ivfcPointer += 4;
|
||||
|
||||
if (magicIVFC != "IVFC") {
|
||||
printf("Invalid IVFC magic: %s\n", magicIVFC.c_str());
|
||||
return 0;
|
||||
}
|
||||
|
||||
uint32_t magicIdentifier = *(u32*)ivfcPointer;
|
||||
ivfcPointer += 4;
|
||||
|
||||
// RomFS IVFC uses 0x10000, DISA/DIFF IVFC uses 0x20000 here
|
||||
if (magicIdentifier != 0x10000 && magicIdentifier != 0x20000) {
|
||||
printf("Invalid IVFC magic identifier: %08X\n", magicIdentifier);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (magicIdentifier == 0x10000) {
|
||||
ivfc.masterHashSize = *(u32*)ivfcPointer;
|
||||
ivfcPointer += 4;
|
||||
// RomFS IVFC uses 3 levels
|
||||
ivfc.levels.resize(3);
|
||||
} else {
|
||||
ivfc.masterHashSize = *(u64*)ivfcPointer;
|
||||
ivfcPointer += 8;
|
||||
// DISA/DIFF IVFC uses 4 levels
|
||||
ivfc.levels.resize(4);
|
||||
}
|
||||
|
||||
for (size_t i = 0; i < ivfc.levels.size(); i++) {
|
||||
IVFCLevel level;
|
||||
|
||||
level.logicalOffset = *(u64*)ivfcPointer;
|
||||
ivfcPointer += 8;
|
||||
|
||||
level.size = *(u64*)ivfcPointer;
|
||||
ivfcPointer += 8;
|
||||
|
||||
// This field is in log2
|
||||
level.blockSize = 1 << *(u32*)ivfcPointer;
|
||||
ivfcPointer += 4;
|
||||
|
||||
// Skip 4 reserved bytes
|
||||
ivfcPointer += 4;
|
||||
|
||||
ivfc.levels[i] = level;
|
||||
}
|
||||
|
||||
u64 ivfcDescriptorSize = *(u64*)ivfcPointer;
|
||||
ivfcPointer += 8;
|
||||
|
||||
uintptr_t ivfcActualSize = ivfcPointer - ivfcStart;
|
||||
|
||||
// According to 3DBrew, this is usually the case but not guaranteed
|
||||
if (ivfcActualSize != ivfcDescriptorSize) {
|
||||
printf("IVFC descriptor size mismatch: %lx != %lx\n", ivfcActualSize, ivfcDescriptorSize);
|
||||
}
|
||||
|
||||
if (magicIdentifier == 0x10000 && ivfcActualSize != 0x5C) {
|
||||
// This is always 0x5C bytes long
|
||||
printf("Invalid IVFC size: %08x\n", (u32)ivfcActualSize);
|
||||
return 0;
|
||||
} else if (magicIdentifier == 0x20000 && ivfcActualSize != 0x78) {
|
||||
// This is always 0x78 bytes long
|
||||
printf("Invalid IVFC size: %08x\n", (u32)ivfcActualSize);
|
||||
return 0;
|
||||
}
|
||||
|
||||
return ivfcActualSize;
|
||||
}
|
||||
|
||||
} // namespace IVFC
|
21
src/core/fs/romfs.cpp
Normal file
21
src/core/fs/romfs.cpp
Normal file
|
@ -0,0 +1,21 @@
|
|||
#include "fs/romfs.hpp"
|
||||
#include "fs/ivfc.hpp"
|
||||
#include <cstdio>
|
||||
|
||||
namespace RomFS {
|
||||
|
||||
RomFSNode parseRomFSTree(uintptr_t romFS, u64 romFSSize) {
|
||||
RomFSNode root;
|
||||
|
||||
IVFC::IVFC ivfc;
|
||||
size_t ivfcSize = IVFC::parseIVFC((uintptr_t)romFS, ivfc);
|
||||
|
||||
if (ivfcSize == 0) {
|
||||
printf("Failed to parse IVFC\n");
|
||||
return {};
|
||||
}
|
||||
|
||||
return root;
|
||||
}
|
||||
|
||||
} // namespace RomFS
|
Loading…
Add table
Reference in a new issue