#pragma once #include "DNACommon.hpp" #include "boo/ThreadLocalPtr.hpp" #include #include "zeus/CMatrix4f.hpp" namespace DataSpec { /** PAK entry stream reader */ class PAKEntryReadStream : public athena::io::IStreamReader { std::unique_ptr m_buf; atUint64 m_sz; atUint64 m_pos; public: PAKEntryReadStream() {} operator bool() const { return m_buf.operator bool(); } PAKEntryReadStream(const PAKEntryReadStream& other) = delete; PAKEntryReadStream(PAKEntryReadStream&& other) = default; PAKEntryReadStream& operator=(const PAKEntryReadStream& other) = delete; PAKEntryReadStream& operator=(PAKEntryReadStream&& other) = default; PAKEntryReadStream(std::unique_ptr&& buf, atUint64 sz, atUint64 pos) : m_buf(std::move(buf)), m_sz(sz), m_pos(pos) { if (m_pos >= m_sz) LogDNACommon.report(logvisor::Fatal, fmt("PAK stream cursor overrun")); } void seek(atInt64 pos, athena::SeekOrigin origin) override { if (origin == athena::Begin) m_pos = pos; else if (origin == athena::Current) m_pos += pos; else if (origin == athena::End) m_pos = m_sz + pos; if (m_pos > m_sz) LogDNACommon.report(logvisor::Fatal, fmt("PAK stream cursor overrun")); } atUint64 position() const override { return m_pos; } atUint64 length() const override { return m_sz; } const atUint8* data() const { return m_buf.get(); } atUint64 readUBytesToBuf(void* buf, atUint64 len) override { atUint64 bufEnd = m_pos + len; if (bufEnd > m_sz) len -= bufEnd - m_sz; memmove(buf, m_buf.get() + m_pos, len); m_pos += len; return len; } }; struct UniqueResult { enum class Type { NotFound, Pak, Level, Area, Layer } m_type = Type::NotFound; const hecl::SystemString* m_levelName = nullptr; const hecl::SystemString* m_areaName = nullptr; const hecl::SystemString* m_layerName = nullptr; UniqueResult() = default; UniqueResult(Type tp) : m_type(tp) {} template void checkEntry(const PAKBRIDGE& pakBridge, const typename PAKBRIDGE::PAKType::Entry& entry); hecl::ProjectPath uniquePath(const hecl::ProjectPath& pakPath) const; }; template class PAKRouter; /** Resource extractor type */ template struct ResExtractor { std::function func_a; std::function&, const typename PAKBRIDGE::PAKType::Entry&, bool, hecl::blender::Token&, std::function)> func_b; std::array fileExts = {}; unsigned weight = 0; std::function&, typename PAKBRIDGE::PAKType::Entry&)> func_name; ResExtractor() = default; ResExtractor(std::function func, std::array&& fileExtsIn, unsigned weightin = 0, std::function&, typename PAKBRIDGE::PAKType::Entry&)> nfunc = {}) : func_a(std::move(func)), fileExts(std::move(fileExtsIn)), weight(weightin), func_name(std::move(nfunc)) {} ResExtractor(std::function&, const typename PAKBRIDGE::PAKType::Entry&, bool, hecl::blender::Token&, std::function)> func, std::array&& fileExtsIn, unsigned weightin = 0, std::function&, typename PAKBRIDGE::PAKType::Entry&)> nfunc = {}) : func_b(std::move(func)), fileExts(std::move(fileExtsIn)), weight(weightin), func_name(std::move(nfunc)) {} bool IsFullyExtracted(const hecl::ProjectPath& path) const { hecl::ProjectPath::Type tp = path.getPathType(); if (tp == hecl::ProjectPath::Type::None) return false; else if (tp == hecl::ProjectPath::Type::Glob) { for (int i = 0; i < 6; ++i) { if (!fileExts[i]) break; hecl::ProjectPath withExt = path.getWithExtension(fileExts[i], true); if (withExt.isNone()) return false; } } return true; } }; /** Level hierarchy representation */ template struct Level { hecl::SystemString name; struct Area { hecl::SystemString name; struct Layer { hecl::SystemString name; bool active; std::unordered_set resources; }; std::vector layers; std::unordered_set resources; }; std::unordered_map areas; }; /** PAKRouter (for detecting shared entry locations) */ template class PAKRouter : public PAKRouterBase { public: using PAKType = typename BRIDGETYPE::PAKType; using IDType = typename PAKType::IDType; using EntryType = typename PAKType::Entry; private: const std::vector* m_bridges = nullptr; std::vector> m_bridgePaths; ThreadLocalPtr m_curBridgeIdx; const hecl::ProjectPath& m_gameWorking; const hecl::ProjectPath& m_gameCooked; hecl::ProjectPath m_sharedWorking; hecl::ProjectPath m_sharedCooked; ThreadLocalPtr m_pak; ThreadLocalPtr m_node; std::unordered_map> m_uniqueEntries; std::unordered_map> m_sharedEntries; std::unordered_map m_overrideEntries; CharacterAssociations m_charAssoc; std::unordered_map m_mapaTransforms; hecl::ProjectPath getCharacterWorking(const EntryType* entry) const; public: PAKRouter(const SpecBase& dataSpec, const hecl::ProjectPath& working, const hecl::ProjectPath& cooked) : PAKRouterBase(dataSpec) , m_gameWorking(working) , m_gameCooked(cooked) , m_sharedWorking(working, "Shared") , m_sharedCooked(cooked, "Shared") {} void build(std::vector& bridges, std::function progress); void enterPAKBridge(const BRIDGETYPE& pakBridge); using PAKRouterBase::getWorking; hecl::ProjectPath getWorking(const EntryType* entry, const ResExtractor& extractor) const; hecl::ProjectPath getWorking(const EntryType* entry) const; hecl::ProjectPath getWorking(const IDType& id, bool silenceWarnings = false) const override; hecl::ProjectPath getCooked(const EntryType* entry) const; hecl::ProjectPath getCooked(const IDType& id, bool silenceWarnings = false) const; hecl::SystemString getResourceRelativePath(const EntryType& a, const IDType& b) const; std::string getBestEntryName(const EntryType& entry, bool stdOverride = true) const; std::string getBestEntryName(const IDType& entry, bool stdOverride = true) const; bool extractResources(const BRIDGETYPE& pakBridge, bool force, hecl::blender::Token& btok, std::function progress); const typename BRIDGETYPE::PAKType::Entry* lookupEntry(const IDType& entry, const nod::Node** nodeOut = nullptr, bool silenceWarnings = false, bool currentPAK = false) const; template bool lookupAndReadDNA(const IDType& id, DNA& out, bool silenceWarnings = false) { const nod::Node* node; const EntryType* entry = lookupEntry(id, &node, silenceWarnings); if (!entry) return false; PAKEntryReadStream rs = entry->beginReadStream(*node); out.read(rs); return true; } PAKEntryReadStream beginReadStreamForId(const IDType& id, bool silenceWarnings = false) { const nod::Node* node; const EntryType* entry = lookupEntry(id, &node, silenceWarnings); return entry->beginReadStream(*node); } const typename CharacterAssociations::RigPair* lookupCMDLRigPair(const IDType& id) const; const typename CharacterAssociations::MultimapIteratorPair lookupCharacterAttachmentRigs(const IDType& id) const; const zeus::CMatrix4f* lookupMAPATransform(const IDType& mapaId) const; hecl::ProjectPath getAreaLayerWorking(const IDType& areaId, int layerIdx) const; hecl::ProjectPath getAreaLayerWorking(const IDType& areaId, int layerIdx, bool& activeOut) const; hecl::ProjectPath getAreaLayerCooked(const IDType& areaId, int layerIdx) const; hecl::ProjectPath getAreaLayerCooked(const IDType& areaId, int layerIdx, bool& activeOut) const; void enumerateResources(const std::function& func); bool mreaHasDupeResources(const IDType& id) const; }; } // namespace DataSpec