#pragma once #include #include "logvisor/logvisor.hpp" #include "athena/DNAYaml.hpp" #include "hecl/Database.hpp" #include "../SpecBase.hpp" #include "boo/ThreadLocalPtr.hpp" #include "zeus/CColor.hpp" namespace DataSpec { struct SpecBase; extern logvisor::Module LogDNACommon; extern ThreadLocalPtr g_curSpec; extern ThreadLocalPtr g_PakRouter; extern ThreadLocalPtr g_ThreadBlenderToken; /* This comes up a great deal */ using BigDNA = athena::io::DNA; using BigDNAV = athena::io::DNAV; using BigDNAVYaml = athena::io::DNAVYaml; /** FourCC with DNA read/write */ using DNAFourCC = hecl::DNAFourCC; class DNAColor final : public BigDNA, public zeus::CColor { public: DNAColor() = default; DNAColor(const zeus::CColor& color) : zeus::CColor(color) {} AT_DECL_EXPLICIT_DNA_YAML }; template <> inline void DNAColor::Enumerate(typename Read::StreamT& _r) { zeus::CColor::readRGBABig(_r); } template <> inline void DNAColor::Enumerate(typename Write::StreamT& _w) { zeus::CColor::writeRGBABig(_w); } template <> inline void DNAColor::Enumerate(typename ReadYaml::StreamT& _r) { size_t count; if (auto v = _r.enterSubVector(count)) { zeus::simd_floats f; f[0] = (count >= 1) ? _r.readFloat() : 0.f; f[1] = (count >= 2) ? _r.readFloat() : 0.f; f[2] = (count >= 3) ? _r.readFloat() : 0.f; f[3] = (count >= 4) ? _r.readFloat() : 0.f; mSimd.copy_from(f); } } template <> inline void DNAColor::Enumerate(typename WriteYaml::StreamT& _w) { if (auto v = _w.enterSubVector()) { zeus::simd_floats f(mSimd); _w.writeFloat(f[0]); _w.writeFloat(f[1]); _w.writeFloat(f[2]); _w.writeFloat(f[3]); } } template <> inline void DNAColor::Enumerate(typename BinarySize::StreamT& _s) { _s += 16; } using FourCC = hecl::FourCC; class UniqueID32; class UniqueID64; class UniqueID128; /** Common virtual interface for runtime ambiguity resolution */ class PAKRouterBase { protected: const SpecBase& m_dataSpec; public: PAKRouterBase(const SpecBase& dataSpec) : m_dataSpec(dataSpec) {} hecl::Database::Project& getProject() const { return m_dataSpec.getProject(); } virtual hecl::ProjectPath getWorking(const UniqueID32&, bool silenceWarnings = false) const { LogDNACommon.report(logvisor::Fatal, FMT_STRING("PAKRouter IDType mismatch; expected UniqueID32 specialization")); return hecl::ProjectPath(); } virtual hecl::ProjectPath getWorking(const UniqueID64&, bool silenceWarnings = false) const { LogDNACommon.report(logvisor::Fatal, FMT_STRING("PAKRouter IDType mismatch; expected UniqueID64 specialization")); return hecl::ProjectPath(); } virtual hecl::ProjectPath getWorking(const UniqueID128&, bool silenceWarnings = false) const { LogDNACommon.report(logvisor::Fatal, FMT_STRING("PAKRouter IDType mismatch; expected UniqueID128 specialization")); return hecl::ProjectPath(); } }; /** Globally-accessed manager allowing UniqueID* classes to directly * lookup destination paths of resources */ class UniqueIDBridge { friend class UniqueID32; friend class UniqueID64; static ThreadLocalPtr s_Project; public: template static hecl::ProjectPath TranslatePakIdToPath(const IDType& id, bool silenceWarnings = false); template static hecl::ProjectPath MakePathFromString(std::string_view str); static void SetThreadProject(hecl::Database::Project& project); }; /** PAK 32-bit Unique ID */ class UniqueID32 : public BigDNA { protected: uint32_t m_id = 0xffffffff; public: using value_type = uint32_t; static UniqueID32 kInvalidId; AT_DECL_EXPLICIT_DNA_YAML bool isValid() const noexcept { return m_id != 0xffffffff && m_id != 0; } void assign(uint32_t id) noexcept { m_id = id ? id : 0xffffffff; } UniqueID32& operator=(const hecl::ProjectPath& path) noexcept { assign(path.parsedHash32()); return *this; } bool operator!=(const UniqueID32& other) const noexcept { return m_id != other.m_id; } bool operator==(const UniqueID32& other) const noexcept { return m_id == other.m_id; } bool operator<(const UniqueID32& other) const noexcept { return m_id < other.m_id; } uint32_t toUint32() const noexcept { return m_id; } uint64_t toUint64() const noexcept { return m_id; } std::string toString() const; void clear() noexcept { m_id = 0xffffffff; } UniqueID32() noexcept = default; UniqueID32(uint32_t idin) noexcept { assign(idin); } UniqueID32(athena::io::IStreamReader& reader) { read(reader); } UniqueID32(const hecl::ProjectPath& path) noexcept { *this = path; } UniqueID32(const char* hexStr) noexcept { char copy[9]; strncpy(copy, hexStr, 8); copy[8] = '\0'; assign(strtoul(copy, nullptr, 16)); } UniqueID32(const wchar_t* hexStr) noexcept { wchar_t copy[9]; wcsncpy(copy, hexStr, 8); copy[8] = L'\0'; assign(wcstoul(copy, nullptr, 16)); } static constexpr size_t BinarySize() noexcept { return 4; } }; /** PAK 32-bit Unique ID - writes zero when invalid */ class UniqueID32Zero : public UniqueID32 { public: AT_DECL_DNA_YAML Delete __d2; using UniqueID32::UniqueID32; }; /** PAK 64-bit Unique ID */ class UniqueID64 : public BigDNA { uint64_t m_id = 0xffffffffffffffff; public: using value_type = uint64_t; AT_DECL_EXPLICIT_DNA_YAML bool isValid() const noexcept { return m_id != 0xffffffffffffffff && m_id != 0; } void assign(uint64_t id) noexcept { m_id = id ? id : 0xffffffffffffffff; } UniqueID64& operator=(const hecl::ProjectPath& path) noexcept { assign(path.hash().val64()); return *this; } bool operator!=(const UniqueID64& other) const noexcept { return m_id != other.m_id; } bool operator==(const UniqueID64& other) const noexcept { return m_id == other.m_id; } bool operator<(const UniqueID64& other) const noexcept { return m_id < other.m_id; } uint64_t toUint64() const noexcept { return m_id; } std::string toString() const; void clear() noexcept { m_id = 0xffffffffffffffff; } UniqueID64() noexcept = default; UniqueID64(uint64_t idin) noexcept { assign(idin); } UniqueID64(athena::io::IStreamReader& reader) { read(reader); } UniqueID64(const hecl::ProjectPath& path) noexcept { *this = path; } UniqueID64(const char* hexStr) noexcept { char copy[17]; std::strncpy(copy, hexStr, 16); copy[16] = '\0'; assign(std::strtoull(copy, nullptr, 16)); } UniqueID64(const wchar_t* hexStr) noexcept { wchar_t copy[17]; std::wcsncpy(copy, hexStr, 16); copy[16] = L'\0'; assign(std::wcstoull(copy, nullptr, 16)); } static constexpr size_t BinarySize() noexcept { return 8; } }; /** PAK 128-bit Unique ID */ class UniqueID128 : public BigDNA { public: union Value { uint64_t id[2]; #if __SSE__ __m128i id128; #endif }; private: Value m_id; public: using value_type = uint64_t; AT_DECL_EXPLICIT_DNA_YAML UniqueID128() noexcept { m_id.id[0] = 0xffffffffffffffff; m_id.id[1] = 0xffffffffffffffff; } UniqueID128(uint64_t idin) noexcept { m_id.id[0] = idin; m_id.id[1] = 0; } bool isValid() const noexcept { return m_id.id[0] != 0xffffffffffffffff && m_id.id[0] != 0 && m_id.id[1] != 0xffffffffffffffff && m_id.id[1] != 0; } UniqueID128& operator=(const hecl::ProjectPath& path) noexcept { m_id.id[0] = path.hash().val64(); m_id.id[1] = 0; return *this; } UniqueID128(const hecl::ProjectPath& path) noexcept { *this = path; } bool operator!=(const UniqueID128& other) const noexcept { #if __SSE__ __m128i vcmp = _mm_cmpeq_epi32(m_id.id128, other.m_id.id128); int vmask = _mm_movemask_epi8(vcmp); return vmask != 0xffff; #else return (m_id.id[0] != other.m_id.id[0]) || (m_id.id[1] != other.m_id.id[1]); #endif } bool operator==(const UniqueID128& other) const noexcept { #if __SSE__ __m128i vcmp = _mm_cmpeq_epi32(m_id.id128, other.m_id.id128); int vmask = _mm_movemask_epi8(vcmp); return vmask == 0xffff; #else return (m_id.id[0] == other.m_id.id[0]) && (m_id.id[1] == other.m_id.id[1]); #endif } bool operator<(const UniqueID128& other) const noexcept { return m_id.id[0] < other.m_id.id[0] || (m_id.id[0] == other.m_id.id[0] && m_id.id[1] < other.m_id.id[1]); } void clear() noexcept { m_id.id[0] = 0xffffffffffffffff; m_id.id[1] = 0xffffffffffffffff; } uint64_t toUint64() const noexcept { return m_id.id[0]; } uint64_t toHighUint64() const noexcept { return m_id.id[0]; } uint64_t toLowUint64() const noexcept { return m_id.id[1]; } std::string toString() const; static constexpr size_t BinarySize() noexcept { return 16; } }; /** Casts ID type to its null-zero equivalent */ template using CastIDToZero = typename std::conditional_t, UniqueID32Zero, T>; /** Word Bitmap reader/writer */ class WordBitmap { std::vector m_words; size_t m_bitCount = 0; public: void read(athena::io::IStreamReader& reader, size_t bitCount); void write(athena::io::IStreamWriter& writer) const; void reserve(size_t bitCount) { m_words.reserve((bitCount + 31) / 32); } void binarySize(size_t& __isz) const; size_t getBitCount() const { return m_bitCount; } bool getBit(size_t idx) const { size_t wordIdx = idx / 32; if (wordIdx >= m_words.size()) return false; size_t wordCur = idx % 32; return (m_words[wordIdx] >> wordCur) & 0x1; } void setBit(size_t idx) { size_t wordIdx = idx / 32; while (wordIdx >= m_words.size()) m_words.push_back(0); size_t wordCur = idx % 32; m_words[wordIdx] |= (1 << wordCur); m_bitCount = std::max(m_bitCount, idx + 1); } void unsetBit(size_t idx) { size_t wordIdx = idx / 32; while (wordIdx >= m_words.size()) m_words.push_back(0); size_t wordCur = idx % 32; m_words[wordIdx] &= ~(1 << wordCur); m_bitCount = std::max(m_bitCount, idx + 1); } void clear() { m_words.clear(); m_bitCount = 0; } class Iterator { friend class WordBitmap; const WordBitmap& m_bmp; size_t m_idx = 0; Iterator(const WordBitmap& bmp, size_t idx) : m_bmp(bmp), m_idx(idx) {} public: using iterator_category = std::forward_iterator_tag; using value_type = bool; using difference_type = std::ptrdiff_t; using pointer = bool*; using reference = bool&; Iterator& operator++() { ++m_idx; return *this; } bool operator*() const { return m_bmp.getBit(m_idx); } bool operator!=(const Iterator& other) const { return m_idx != other.m_idx; } }; Iterator begin() const { return Iterator(*this, 0); } Iterator end() const { return Iterator(*this, m_bitCount); } }; /** Resource cooker function */ using ResCooker = std::function; /** Mappings of resources involved in extracting characters */ template struct CharacterAssociations { struct RigPair { IDType cskr, cinf; }; struct ModelRigPair { IDType cinf, cmdl; }; /* CMDL -> (CSKR, CINF) */ std::unordered_map m_cmdlRigs; /* CSKR -> ANCS */ std::unordered_map> m_cskrToCharacter; /* ANCS -> (CINF, CMDL) */ std::unordered_multimap> m_characterToAttachmentRigs; using MultimapIteratorPair = std::pair>::const_iterator, typename std::unordered_multimap>::const_iterator>; void addAttachmentRig(IDType character, IDType cinf, IDType cmdl, const char* name) { auto range = m_characterToAttachmentRigs.equal_range(character); for (auto it = range.first; it != range.second; ++it) if (it->second.second == name) return; m_characterToAttachmentRigs.insert(std::make_pair(character, std::make_pair(ModelRigPair{cinf, cmdl}, name))); } }; hecl::ProjectPath GetPathBeginsWith(const hecl::DirectoryEnumerator& dEnum, const hecl::ProjectPath& parentPath, hecl::SystemStringView test); inline hecl::ProjectPath GetPathBeginsWith(const hecl::ProjectPath& parentPath, hecl::SystemStringView test) { return GetPathBeginsWith(hecl::DirectoryEnumerator(parentPath.getAbsolutePath()), parentPath, test); } } // namespace DataSpec /* Hash template-specializations for UniqueID types */ namespace std { template <> struct hash { size_t operator()(const DataSpec::DNAFourCC& fcc) const noexcept { return fcc.toUint32(); } }; template <> struct hash { size_t operator()(const DataSpec::UniqueID32& id) const noexcept { return id.toUint32(); } }; template <> struct hash { size_t operator()(const DataSpec::UniqueID64& id) const noexcept { return id.toUint64(); } }; template <> struct hash { size_t operator()(const DataSpec::UniqueID128& id) const noexcept { return id.toHighUint64() ^ id.toLowUint64(); } }; } // namespace std FMT_CUSTOM_FORMATTER(DataSpec::UniqueID32, "{:08X}", obj.toUint32()) FMT_CUSTOM_FORMATTER(DataSpec::UniqueID32Zero, "{:08X}", obj.toUint32()) FMT_CUSTOM_FORMATTER(DataSpec::UniqueID64, "{:016X}", obj.toUint64()) FMT_CUSTOM_FORMATTER(DataSpec::UniqueID128, "{:016X}{:016X}", obj.toHighUint64(), obj.toLowUint64())