mirror of https://github.com/AxioDL/metaforce.git
203 lines
5.7 KiB
C++
203 lines
5.7 KiB
C++
#include "CResLoader.hpp"
|
|
#include "CPakFile.hpp"
|
|
|
|
namespace urde
|
|
{
|
|
|
|
const std::vector<CAssetId>* CResLoader::GetTagListForFile(const std::string& name) const
|
|
{
|
|
std::string namePak = name + ".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(const std::string& name, bool samusPak, bool worldPak)
|
|
{
|
|
std::string namePak = name + ".pak";
|
|
if (CDvdFile::FileExists(namePak.c_str()))
|
|
{
|
|
x30_pakLoadingList.emplace_back(new CPakFile(namePak, samusPak, worldPak));
|
|
++x44_pakLoadingCount;
|
|
}
|
|
}
|
|
|
|
void CResLoader::AddPakFile(const std::string& name, bool samusPak, bool worldPak)
|
|
{
|
|
AddPakFileAsync(name, samusPak, worldPak);
|
|
while (x44_pakLoadingCount)
|
|
AsyncIdlePakLoading();
|
|
}
|
|
|
|
CInputStream* CResLoader::LoadNewResourcePartSync(const SObjectTag& tag, int offset, int length, void* extBuf)
|
|
{
|
|
void* buf = extBuf;
|
|
CPakFile* file = FindResourceForLoad(tag);
|
|
if (!buf)
|
|
buf = new u8[length];
|
|
file->SyncSeekRead(buf, length, ESeekOrigin::Begin, x50_cachedResInfo->x4_offset + offset);
|
|
return new CMemoryInStream((atUint8*)buf, length, !extBuf);
|
|
}
|
|
|
|
void CResLoader::LoadMemResourceSync(const SObjectTag& tag, void** bufOut, int* sizeOut)
|
|
{
|
|
CPakFile* file = FindResourceForLoad(tag);
|
|
void* buf = new u8[x50_cachedResInfo->x8_size];
|
|
file->SyncSeekRead(buf, x50_cachedResInfo->x8_size, ESeekOrigin::Begin,
|
|
x50_cachedResInfo->x4_offset);
|
|
*bufOut = buf;
|
|
*sizeOut = x50_cachedResInfo->x8_size;
|
|
}
|
|
|
|
CInputStream* CResLoader::LoadResourceFromMemorySync(const SObjectTag& tag, const void* buf)
|
|
{
|
|
FindResourceForLoad(tag);
|
|
CInputStream* newStrm = new CMemoryInStream((atUint8*)buf, x50_cachedResInfo->x8_size);
|
|
if (x50_cachedResInfo->xb_compressed)
|
|
{
|
|
newStrm->readUint32Big();
|
|
newStrm = new CZipInputStream(std::unique_ptr<CInputStream>(newStrm));
|
|
}
|
|
return newStrm;
|
|
}
|
|
|
|
CInputStream* CResLoader::LoadNewResourceSync(const SObjectTag& tag, void* extBuf)
|
|
{
|
|
void* buf = extBuf;
|
|
CPakFile* file = FindResourceForLoad(tag);
|
|
size_t resSz = ROUND_UP_32(x50_cachedResInfo->x8_size);
|
|
if (!buf)
|
|
buf = new u8[resSz];
|
|
file->SyncSeekRead(buf, resSz, ESeekOrigin::Begin, x50_cachedResInfo->x4_offset);
|
|
CInputStream* newStrm = new CMemoryInStream((atUint8*)buf, resSz, !extBuf);
|
|
if (x50_cachedResInfo->xb_compressed)
|
|
{
|
|
newStrm->readUint32Big();
|
|
newStrm = new CZipInputStream(std::unique_ptr<CInputStream>(newStrm));
|
|
}
|
|
return newStrm;
|
|
}
|
|
|
|
std::shared_ptr<IDvdRequest> CResLoader::LoadResourcePartAsync(const SObjectTag& tag, int offset, int length, void* buf)
|
|
{
|
|
return FindResourceForLoad(tag.id)->AsyncSeekRead(buf, length,
|
|
ESeekOrigin::Begin, x50_cachedResInfo->x4_offset + offset);
|
|
}
|
|
|
|
std::shared_ptr<IDvdRequest> CResLoader::LoadResourceAsync(const SObjectTag& tag, void* buf)
|
|
{
|
|
return FindResourceForLoad(tag.id)->AsyncSeekRead(buf, ROUND_UP_32(x50_cachedResInfo->x8_size),
|
|
ESeekOrigin::Begin, x50_cachedResInfo->x4_offset);
|
|
}
|
|
|
|
bool CResLoader::GetResourceCompression(const SObjectTag& tag)
|
|
{
|
|
if (FindResource(tag.id))
|
|
return x50_cachedResInfo->xb_compressed;
|
|
return false;
|
|
}
|
|
|
|
u32 CResLoader::ResourceSize(const SObjectTag& tag)
|
|
{
|
|
if (FindResource(tag.id))
|
|
return x50_cachedResInfo->x8_size;
|
|
return false;
|
|
}
|
|
|
|
bool CResLoader::ResourceExists(const SObjectTag& tag)
|
|
{
|
|
return FindResource(tag.id);
|
|
}
|
|
|
|
FourCC CResLoader::GetResourceTypeById(CAssetId id) const
|
|
{
|
|
if (FindResource(id))
|
|
return x50_cachedResInfo->x0_type;
|
|
return FourCC();
|
|
}
|
|
|
|
const SObjectTag* CResLoader::GetResourceIdByName(const char* name) const
|
|
{
|
|
for (const std::unique_ptr<CPakFile>& file : x18_pakLoadedList)
|
|
{
|
|
const SObjectTag* id = file->GetResIdByName(name);
|
|
if (id)
|
|
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)
|
|
{
|
|
(*it)->AsyncIdle();
|
|
if ((*it)->x2c_asyncLoadPhase == CPakFile::EAsyncPhase::Loaded)
|
|
{
|
|
MoveToCorrectLoadedList(std::move(*it));
|
|
it = x30_pakLoadingList.erase(it);
|
|
--x44_pakLoadingCount;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CResLoader::FindResource(CAssetId id) const
|
|
{
|
|
for (const std::unique_ptr<CPakFile>& file : x18_pakLoadedList)
|
|
if (const_cast<CResLoader*>(this)->CacheFromPak(*file, id))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
CPakFile* CResLoader::FindResourceForLoad(CAssetId id)
|
|
{
|
|
for (std::unique_ptr<CPakFile>& file : x18_pakLoadedList)
|
|
if (CacheFromPakForLoad(*file, id))
|
|
return file.get();
|
|
return nullptr;
|
|
}
|
|
|
|
CPakFile* CResLoader::FindResourceForLoad(const SObjectTag& tag)
|
|
{
|
|
return FindResourceForLoad(tag.id);
|
|
}
|
|
|
|
bool CResLoader::CacheFromPakForLoad(CPakFile& file, CAssetId id)
|
|
{
|
|
const CPakFile::SResInfo* info = file.GetResInfoForLoad(id);
|
|
if (info)
|
|
{
|
|
x4c_cachedResId = id;
|
|
x50_cachedResInfo = info;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CResLoader::CacheFromPak(const CPakFile& file, CAssetId id)
|
|
{
|
|
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)
|
|
{
|
|
x18_pakLoadedList.push_back(std::move(file));
|
|
}
|
|
|
|
}
|