mirror of
https://github.com/jakcron/nstool
synced 2024-11-15 02:06:40 +00:00
134 lines
3.6 KiB
C++
134 lines
3.6 KiB
C++
|
#include "RomfsProcess.h"
|
||
|
#include "util.h"
|
||
|
|
||
|
#include <tc/io/VirtualFileSystem.h>
|
||
|
#include <nn/hac/RomFsMetaGenerator.h>
|
||
|
|
||
|
|
||
|
nstool::RomfsProcess::RomfsProcess() :
|
||
|
mModuleName("nstool::RomfsProcess"),
|
||
|
mFile(),
|
||
|
mCliOutputMode(true, false, false, false),
|
||
|
mVerify(false),
|
||
|
mDirNum(0),
|
||
|
mFileNum(0),
|
||
|
mFileSystem(),
|
||
|
mFsProcess()
|
||
|
{
|
||
|
mFsProcess.setFsLabel("RomFS");
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::process()
|
||
|
{
|
||
|
importHeader();
|
||
|
|
||
|
if (mCliOutputMode.show_basic_info)
|
||
|
{
|
||
|
displayHeader();
|
||
|
}
|
||
|
|
||
|
mFsProcess.process();
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setInputFile(const std::shared_ptr<tc::io::IStream>& file)
|
||
|
{
|
||
|
mFile = file;
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setCliOutputMode(CliOutputMode type)
|
||
|
{
|
||
|
mCliOutputMode = type;
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setVerifyMode(bool verify)
|
||
|
{
|
||
|
mVerify = verify;
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setMountPointName(const std::string& mount_name)
|
||
|
{
|
||
|
mFsProcess.setFsLabel(mount_name);
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setExtractPath(const tc::io::Path& path)
|
||
|
{
|
||
|
mFsProcess.setExtractPath(path);
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::setListFs(bool list_fs)
|
||
|
{
|
||
|
mFsProcess.setCliOutputMode(list_fs);
|
||
|
}
|
||
|
|
||
|
const std::shared_ptr<tc::io::IStorage>& nstool::RomfsProcess::getFileSystem() const
|
||
|
{
|
||
|
return mFileSystem;
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::importHeader()
|
||
|
{
|
||
|
if (mFile == nullptr)
|
||
|
{
|
||
|
throw tc::Exception(mModuleName, "No file reader set.");
|
||
|
}
|
||
|
|
||
|
tc::ByteData scratch;
|
||
|
|
||
|
// read base header to determine complete header size
|
||
|
if (mFile->length() < tc::io::IOUtil::castSizeToInt64(sizeof(nn::hac::sRomfsHeader)))
|
||
|
{
|
||
|
throw tc::Exception(mModuleName, "Corrupt RomFs: File too small");
|
||
|
}
|
||
|
|
||
|
mFile->seek(0, tc::io::SeekOrigin::Begin);
|
||
|
mFile->read((byte_t*)&mRomfsHeader, sizeof(mRomfsHeader));
|
||
|
if (mRomfsHeader.header_size.unwrap() != sizeof(nn::hac::sRomfsHeader) ||
|
||
|
mRomfsHeader.dir_hash_bucket.offset.unwrap() != sizeof(nn::hac::sRomfsHeader) ||
|
||
|
mRomfsHeader.data_offset.unwrap() != align<int64_t>(mRomfsHeader.header_size.unwrap(), nn::hac::romfs::kRomfsHeaderAlign))
|
||
|
{
|
||
|
throw tc::ArgumentOutOfRangeException("nn::hac::RomFsMetaGenerator", "Corrupt RomFs: RomFsHeader is corrupted.");
|
||
|
}
|
||
|
|
||
|
// get dir entry ptr
|
||
|
tc::ByteData dir_entry_table = tc::ByteData(tc::io::IOUtil::castInt64ToSize(mRomfsHeader.dir_entry.size.unwrap()));
|
||
|
mFile->seek(mRomfsHeader.dir_entry.offset.unwrap(), tc::io::SeekOrigin::Begin);
|
||
|
mFile->read(dir_entry_table.data(), dir_entry_table.size());
|
||
|
|
||
|
// get file entry ptr
|
||
|
tc::ByteData file_entry_table = tc::ByteData(tc::io::IOUtil::castInt64ToSize(mRomfsHeader.file_entry.size.unwrap()));
|
||
|
mFile->seek(mRomfsHeader.file_entry.offset.unwrap(), tc::io::SeekOrigin::Begin);
|
||
|
mFile->read(file_entry_table.data(), file_entry_table.size());
|
||
|
|
||
|
// count dir num
|
||
|
mDirNum = 0;
|
||
|
for (uint32_t v_addr = 0; size_t(v_addr) < dir_entry_table.size();)
|
||
|
{
|
||
|
uint32_t total_size = sizeof(nn::hac::sRomfsDirEntry) + align<uint32_t>(((nn::hac::sRomfsDirEntry*)(dir_entry_table.data() + v_addr))->name_size.unwrap(), 4);
|
||
|
|
||
|
mDirNum += 1;
|
||
|
|
||
|
v_addr += total_size;
|
||
|
}
|
||
|
|
||
|
// count file num
|
||
|
mFileNum = 0;
|
||
|
for (uint32_t v_addr = 0; size_t(v_addr) < file_entry_table.size();)
|
||
|
{
|
||
|
uint32_t total_size = sizeof(nn::hac::sRomfsFileEntry) + align<uint32_t>(((nn::hac::sRomfsFileEntry*)(file_entry_table.data() + v_addr))->name_size.unwrap(), 4);
|
||
|
|
||
|
mFileNum += 1;
|
||
|
|
||
|
v_addr += total_size;
|
||
|
}
|
||
|
|
||
|
// create virtual filesystem
|
||
|
mFileSystem = std::make_shared<tc::io::VirtualFileSystem>(tc::io::VirtualFileSystem(nn::hac::RomFsMetaGenerator(mFile)));
|
||
|
mFsProcess.setInputFileSystem(mFileSystem);
|
||
|
}
|
||
|
|
||
|
void nstool::RomfsProcess::displayHeader()
|
||
|
{
|
||
|
fmt::print("[RomFS]\n");
|
||
|
fmt::print(" DirNum: {:d}\n", mDirNum);
|
||
|
fmt::print(" FileNum: {:d}\n", mFileNum);
|
||
|
}
|