#include "Runtime/CResLoader.hpp" #include "Runtime/CPakFile.hpp" namespace metaforce { static logvisor::Module Log("CResLoader"); CResLoader::CResLoader() { x48_curPak = x18_pakLoadedList.end(); } const std::vector<CAssetId>* CResLoader::GetTagListForFile(std::string_view name) const { const std::string namePak = std::string(name).append(".pak"); for (const std::unique_ptr<CPakFile>& pak : x18_pakLoadedList) { if (CStringExtras::CompareCaseInsensitive(namePak, pak->x18_path)) { return &pak->GetDepList(); } } return nullptr; } void CResLoader::AddPakFileAsync(std::string_view name, bool buildDepList, bool worldPak, bool override) { const std::string namePak = std::string(name).append(".pak"); if (CDvdFile::FileExists(namePak)) { x30_pakLoadingList.emplace_back(std::make_unique<CPakFile>(namePak, buildDepList, worldPak, override)); ++x44_pakLoadingCount; } } void CResLoader::AddPakFile(std::string_view name, bool samusPak, bool worldPak, bool override) { AddPakFileAsync(name, samusPak, worldPak, override); WaitForPakFileLoadingComplete(); } void CResLoader::WaitForPakFileLoadingComplete() { while (x44_pakLoadingCount) AsyncIdlePakLoading(); } std::unique_ptr<CInputStream> CResLoader::LoadNewResourcePartSync(const SObjectTag& tag, u32 length, u32 offset, void* extBuf) { void* buf = extBuf; if (buf == nullptr) { buf = new u8[length]; } CPakFile* const file = FindResourceForLoad(tag); file->SyncSeekRead(buf, length, ESeekOrigin::Begin, x50_cachedResInfo->GetOffset() + offset); return std::make_unique<athena::io::MemoryReader>(buf, length, !extBuf); } void CResLoader::LoadMemResourceSync(const SObjectTag& tag, std::unique_ptr<u8[]>& bufOut, int* sizeOut) { if (CPakFile* file = FindResourceForLoad(tag)) { bufOut = std::unique_ptr<u8[]>(new u8[x50_cachedResInfo->GetSize()]); file->SyncSeekRead(bufOut.get(), x50_cachedResInfo->GetSize(), ESeekOrigin::Begin, x50_cachedResInfo->GetOffset()); *sizeOut = x50_cachedResInfo->GetSize(); } } std::unique_ptr<CInputStream> CResLoader::LoadResourceFromMemorySync(const SObjectTag& tag, const void* buf) { FindResourceForLoad(tag); std::unique_ptr<CInputStream> newStrm = std::make_unique<athena::io::MemoryReader>(buf, x50_cachedResInfo->GetSize()); if (x50_cachedResInfo->IsCompressed()) { newStrm->readUint32Big(); newStrm = std::make_unique<CZipInputStream>(std::move(newStrm)); } return newStrm; } std::unique_ptr<CInputStream> CResLoader::LoadNewResourceSync(const SObjectTag& tag, void* extBuf) { if (CPakFile* const file = FindResourceForLoad(tag)) { const size_t resSz = ROUND_UP_32(x50_cachedResInfo->GetSize()); void* buf = extBuf; if (buf == nullptr) { buf = new u8[resSz]; } file->SyncSeekRead(buf, resSz, ESeekOrigin::Begin, x50_cachedResInfo->GetOffset()); const bool takeOwnership = extBuf == nullptr; std::unique_ptr<CInputStream> newStrm = std::make_unique<athena::io::MemoryReader>(buf, resSz, takeOwnership); if (x50_cachedResInfo->IsCompressed()) { newStrm->readUint32Big(); newStrm = std::make_unique<CZipInputStream>(std::move(newStrm)); } return newStrm; } return nullptr; } std::shared_ptr<IDvdRequest> CResLoader::LoadResourcePartAsync(const SObjectTag& tag, u32 off, u32 size, void* buf) { CPakFile* file = FindResourceForLoad(tag.id); return file->AsyncSeekRead(buf, size, ESeekOrigin::Begin, x50_cachedResInfo->GetOffset() + off); } std::shared_ptr<IDvdRequest> CResLoader::LoadResourceAsync(const SObjectTag& tag, void* buf) { CPakFile* file = FindResourceForLoad(tag.id); return file->AsyncSeekRead(buf, ROUND_UP_32(x50_cachedResInfo->GetSize()), ESeekOrigin::Begin, x50_cachedResInfo->GetOffset()); } std::unique_ptr<u8[]> CResLoader::LoadResourceSync(const metaforce::SObjectTag& tag) { CPakFile* file = FindResourceForLoad(tag.id); u32 size = ROUND_UP_32(x50_cachedResInfo->GetSize()); std::unique_ptr<u8[]> ret(new u8[size]); file->SyncSeekRead(ret.get(), size, ESeekOrigin::Begin, x50_cachedResInfo->GetOffset()); return ret; } std::unique_ptr<u8[]> CResLoader::LoadNewResourcePartSync(const metaforce::SObjectTag& tag, u32 off, u32 size) { CPakFile* file = FindResourceForLoad(tag.id); std::unique_ptr<u8[]> ret(new u8[size]); file->SyncSeekRead(ret.get(), size, ESeekOrigin::Begin, x50_cachedResInfo->GetOffset() + off); return ret; } void CResLoader::GetTagListForFile(const char* pakName, std::vector<SObjectTag>& out) const { std::string path = std::string(pakName) + ".pak"; for (const std::unique_ptr<CPakFile>& file : m_overridePakList) { if (_GetTagListForFile(out, path, file)) return; } for (const std::unique_ptr<CPakFile>& file : x18_pakLoadedList) { if (_GetTagListForFile(out, path, file)) return; } } bool CResLoader::_GetTagListForFile(std::vector<SObjectTag>& out, const std::string& path, const std::unique_ptr<CPakFile>& file) const { if (CStringExtras::CompareCaseInsensitive(file->GetPath(), path)) { const auto& depList = file->GetDepList(); out.reserve(depList.size()); for (const auto& dep : depList) { const auto* const resInfo = file->GetResInfo(dep); out.emplace_back(resInfo->GetType(), dep); } return true; } return false; } bool CResLoader::GetResourceCompression(const SObjectTag& tag) const { if (FindResource(tag.id)) return x50_cachedResInfo->IsCompressed(); return false; } u32 CResLoader::ResourceSize(const SObjectTag& tag) const { if (FindResource(tag.id)) return x50_cachedResInfo->GetSize(); return 0; } bool CResLoader::ResourceExists(const SObjectTag& tag) const { return FindResource(tag.id); } FourCC CResLoader::GetResourceTypeById(CAssetId id) const { if (id.IsValid() && FindResource(id)) return x50_cachedResInfo->GetType(); return {}; } const SObjectTag* CResLoader::GetResourceIdByName(std::string_view name) const { for (const std::unique_ptr<CPakFile>& file : m_overridePakList) if (const SObjectTag* id = file->GetResIdByName(name)) return id; for (const std::unique_ptr<CPakFile>& file : x18_pakLoadedList) if (const SObjectTag* id = file->GetResIdByName(name)) return id; return nullptr; } bool CResLoader::AreAllPaksLoaded() const { return x44_pakLoadingCount == 0; } void CResLoader::AsyncIdlePakLoading() { for (auto it = x30_pakLoadingList.begin(); it != x30_pakLoadingList.end();) { (*it)->AsyncIdle(); if ((*it)->x2c_asyncLoadPhase == CPakFile::EAsyncPhase::Loaded) { MoveToCorrectLoadedList(std::move(*it)); it = x30_pakLoadingList.erase(it); --x44_pakLoadingCount; continue; } ++it; } } bool CResLoader::FindResource(CAssetId id) const { if (x4c_cachedResId == id) return true; for (auto it = m_overridePakList.begin(); it != m_overridePakList.end(); ++it) { if (CacheFromPak(**it, id)) { return true; } } if (x48_curPak != x18_pakLoadedList.end()) if (CacheFromPak(**x48_curPak, id)) return true; for (auto it = x18_pakLoadedList.begin(); it != x18_pakLoadedList.end(); ++it) { if (it == x48_curPak) continue; if (CacheFromPak(**it, id)) return true; } Log.report(logvisor::Warning, FMT_STRING("Unable to find asset {}"), id); return false; } CPakFile* CResLoader::FindResourceForLoad(CAssetId id) { for (auto it = m_overridePakList.begin(); it != m_overridePakList.end(); ++it) { if (CacheFromPakForLoad(**it, id)) { return &**it; } } if (x48_curPak != x18_pakLoadedList.end()) if (CacheFromPakForLoad(**x48_curPak, id)) return &**x48_curPak; for (auto it = x18_pakLoadedList.begin(); it != x18_pakLoadedList.end(); ++it) { if (it == x48_curPak) continue; if (CacheFromPakForLoad(**it, id)) { x48_curPak = it; return &**it; } } Log.report(logvisor::Error, FMT_STRING("Unable to find asset {}"), id); return nullptr; } CPakFile* CResLoader::FindResourceForLoad(const SObjectTag& tag) { return FindResourceForLoad(tag.id); } bool CResLoader::CacheFromPakForLoad(CPakFile& file, CAssetId id) { const CPakFile::SResInfo* info; if (x54_forwardSeek) { info = file.GetResInfoForLoadPreferForward(id); x54_forwardSeek = false; } else { info = file.GetResInfoForLoadDirectionless(id); } if (info) { x4c_cachedResId = id; x50_cachedResInfo = info; return true; } return false; } bool CResLoader::CacheFromPak(const CPakFile& file, CAssetId id) const { const CPakFile::SResInfo* info = file.GetResInfo(id); if (info) { x4c_cachedResId = id; x50_cachedResInfo = info; return true; } return false; } void CResLoader::MoveToCorrectLoadedList(std::unique_ptr<CPakFile>&& file) { if (file->IsOverridePak()) m_overridePakList.push_back(std::move(file)); else x18_pakLoadedList.push_back(std::move(file)); } std::vector<std::pair<std::string, SObjectTag>> CResLoader::GetResourceIdToNameList() const { std::vector<std::pair<std::string, SObjectTag>> ret; for (auto it = x18_pakLoadedList.begin(); it != x18_pakLoadedList.end(); ++it) for (const auto& name : (*it)->GetNameList()) ret.push_back(name); return ret; } void CResLoader::EnumerateResources(const std::function<bool(const SObjectTag&)>& lambda) const { for (auto it = m_overridePakList.begin(); it != m_overridePakList.end(); ++it) { for (const CAssetId& id : (*it)->GetDepList()) { SObjectTag fcc(GetResourceTypeById(id), id); if (!lambda(fcc)) return; } } for (auto it = x18_pakLoadedList.begin(); it != x18_pakLoadedList.end(); ++it) { for (const CAssetId& id : (*it)->GetDepList()) { SObjectTag fcc(GetResourceTypeById(id), id); if (!lambda(fcc)) return; } } } void CResLoader::EnumerateNamedResources(const std::function<bool(std::string_view, const SObjectTag&)>& lambda) const { for (auto it = x18_pakLoadedList.begin(); it != x18_pakLoadedList.end(); ++it) for (const auto& name : (*it)->GetNameList()) if (!lambda(name.first, name.second)) return; } } // namespace metaforce