#pragma once #include "hecl/ClientProcess.hpp" #include "hecl/Database.hpp" #include "Runtime/IFactory.hpp" #include "Runtime/CFactoryMgr.hpp" #include "Runtime/CResFactory.hpp" #include "DataSpec/SpecBase.hpp" #include #include #include namespace urde { class ProjectResourceFactoryBase : public IFactory { friend class ProjectResourcePool; hecl::ClientProcess& m_clientProc; public: struct AsyncTask : urde::IDvdRequest { ProjectResourceFactoryBase& m_parent; SObjectTag x0_tag; // IDvdRequest* x8_dvdReq = nullptr; std::unique_ptr* xc_targetDataPtr = nullptr; u8* xc_targetDataRawPtr = nullptr; std::unique_ptr* xc_targetObjPtr = nullptr; std::unique_ptr x10_loadBuffer; u32 x14_resSize = UINT32_MAX; u32 x14_resOffset = 0; CVParamTransfer x18_cvXfer; CObjectReference* m_selfRef = nullptr; hecl::ProjectPath m_workingPath; hecl::ProjectPath m_cookedPath; std::shared_ptr m_cookTransaction; std::shared_ptr m_bufTransaction; bool m_failed = false; bool m_complete = false; AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag, std::unique_ptr& ptr) : m_parent(parent), x0_tag(tag), xc_targetDataPtr(&ptr) {} AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag, std::unique_ptr& ptr, u32 size, u32 off) : m_parent(parent), x0_tag(tag), xc_targetDataPtr(&ptr), x14_resSize(size), x14_resOffset(off) {} AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag, u8* ptr) : m_parent(parent), x0_tag(tag), xc_targetDataRawPtr(ptr) {} AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag, u8* ptr, u32 size, u32 off) : m_parent(parent), x0_tag(tag), xc_targetDataRawPtr(ptr), x14_resSize(size), x14_resOffset(off) {} AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag, std::unique_ptr* ptr, const CVParamTransfer& xfer, CObjectReference* selfRef) : m_parent(parent), x0_tag(tag), xc_targetObjPtr(ptr), x18_cvXfer(xfer), m_selfRef(selfRef) {} /* Cook only */ AsyncTask(ProjectResourceFactoryBase& parent, const SObjectTag& tag) : m_parent(parent), x0_tag(tag) {} void EnsurePath(const urde::SObjectTag& tag, const hecl::ProjectPath& path); void CookComplete(); bool AsyncPump(); void WaitUntilComplete(); bool IsComplete() { return m_complete; } void PostCancelRequest() {} EMediaType GetMediaType() const { return EMediaType::Real; } }; protected: const hecl::Database::Project* m_proj = nullptr; const hecl::Database::DataSpecEntry* m_origSpec = nullptr; const hecl::Database::DataSpecEntry* m_pcSpec = nullptr; /* Used to resolve cooked paths */ std::unique_ptr m_cookSpec; urde::CFactoryMgr m_factoryMgr; std::list> m_asyncLoadList; std::unordered_map>::iterator> m_asyncLoadMap; std::shared_ptr _AddTask(const std::shared_ptr& ptr); std::list>::iterator _RemoveTask(std::list>::iterator it); std::unordered_map>::iterator>::iterator _RemoveTask(std::unordered_map>::iterator>::iterator it); std::unordered_map>::iterator>::iterator _RemoveTask(const SObjectTag& tag); static AsyncTask& _GetAsyncTask(std::list>::iterator it) { return **it; } static AsyncTask& _GetAsyncTask(std::unordered_map>::iterator>::iterator it) { return **it->second; } bool PrepForReadSync(const SObjectTag& tag, const hecl::ProjectPath& path, std::optional& fr); bool WaitForTagReady(const urde::SObjectTag& tag, const hecl::ProjectPath*& pathOut) { return static_cast(*m_cookSpec).waitForTagReady(tag, pathOut); } SObjectTag TagFromPath(const hecl::ProjectPath& path) const { return static_cast(*m_cookSpec).tagFromPath(path); } SObjectTag BuildTagFromPath(const hecl::ProjectPath& path) const { return static_cast(*m_cookSpec).buildTagFromPath(path); } void GetTagListForFile(const char* pakName, std::vector& out) const { return static_cast(*m_cookSpec).getTagListForFile(pakName, out); } void CancelBackgroundIndex() { if (m_cookSpec) return static_cast(*m_cookSpec).cancelBackgroundIndex(); } void BeginBackgroundIndex(hecl::Database::Project& proj, const hecl::Database::DataSpecEntry& origSpec, const hecl::Database::DataSpecEntry& pcSpec); bool SyncCook(const hecl::ProjectPath& working); CFactoryFnReturn BuildSync(const SObjectTag& tag, const hecl::ProjectPath& path, const CVParamTransfer& paramXfer, CObjectReference* selfRef); std::shared_ptr BuildAsyncInternal(const urde::SObjectTag&, const urde::CVParamTransfer&, std::unique_ptr*, CObjectReference* selfRef); public: ProjectResourceFactoryBase(hecl::ClientProcess& clientProc) : m_clientProc(clientProc) {} std::unique_ptr Build(const urde::SObjectTag&, const urde::CVParamTransfer&, CObjectReference* selfRef); void BuildAsync(const urde::SObjectTag&, const urde::CVParamTransfer&, std::unique_ptr*, CObjectReference* selfRef); void CancelBuild(const urde::SObjectTag&); bool CanBuild(const urde::SObjectTag&); const urde::SObjectTag* GetResourceIdByName(std::string_view) const; FourCC GetResourceTypeById(CAssetId id) const; hecl::ProjectPath GetCookedPath(const hecl::ProjectPath& working, bool pcTarget) const { return static_cast(*m_cookSpec).getCookedPath(working, pcTarget); } void EnumerateResources(const std::function& lambda) const; void EnumerateNamedResources(const std::function& lambda) const; u32 ResourceSize(const SObjectTag& tag); std::shared_ptr LoadResourceAsync(const urde::SObjectTag& tag, void* target); std::shared_ptr LoadResourcePartAsync(const urde::SObjectTag& tag, u32 off, u32 size, void* target); std::unique_ptr LoadResourceSync(const urde::SObjectTag& tag); std::unique_ptr LoadNewResourcePartSync(const urde::SObjectTag& tag, u32 off, u32 size); std::shared_ptr CookResourceAsync(const urde::SObjectTag& tag); template bool AsyncPumpTask(ItType& it); void AsyncIdle(); void Shutdown() { CancelBackgroundIndex(); } bool IsBusy() const { return m_asyncLoadMap.size() != 0; } SObjectTag TagFromPath(hecl::SystemStringView path) const { return TagFromPath(hecl::ProjectPath(*(hecl::Database::Project*)m_proj, path)); } ~ProjectResourceFactoryBase() { Shutdown(); } }; } // namespace urde