#include "ObjLoaderT6.h" #include "Asset/GlobalAssetPoolsLoader.h" #include "CustomMap/LoaderCustomMapT6.h" #include "FontIcon/CsvLoaderFontIconT6.h" #include "FontIcon/JsonLoaderFontIconT6.h" #include "Game/T6/AssetMarkerT6.h" #include "Game/T6/CommonT6.h" #include "Game/T6/GameT6.h" #include "Game/T6/Image/ImageLoaderEmbeddedT6.h" #include "Game/T6/Image/ImageLoaderExternalT6.h" #include "Game/T6/T6.h" #include "Game/T6/Techset/PixelShaderLoaderT6.h" #include "Game/T6/Techset/VertexShaderLoaderT6.h" #include "Game/T6/XModel/LoaderXModelT6.h" #include "Image/Dx12TextureLoader.h" #include "Image/IwiLoader.h" #include "Image/IwiTypes.h" #include "Image/LoaderImageT6.h" #include "Image/Texture.h" #include "Leaderboard/JsonLoaderLeaderboardT6.h" #include "Localize/LocalizeLoaderT6.h" #include "Material/LoaderMaterialT6.h" #include "ObjContainer/IPak/IPak.h" #include "ObjLoading.h" #include "PhysConstraints/GdtLoaderPhysConstraintsT6.h" #include "PhysConstraints/RawLoaderPhysConstraintsT6.h" #include "PhysPreset/GdtLoaderPhysPresetT6.h" #include "PhysPreset/RawLoaderPhysPresetT6.h" #include "Qdb/LoaderQdbT6.h" #include "RawFile/LoaderRawFileT6.h" #include "Script/LoaderScriptT6.h" #include "Slug/LoaderSlugT6.h" #include "Sound/LoaderSoundBankT6.h" #include "StringTable/LoaderStringTableT6.h" #include "TechniqueSet/LoaderTechniqueSetT6.h" #include "Tracer/GdtLoaderTracerT6.h" #include "Tracer/RawLoaderTracerT6.h" #include "Utils/Logging/Log.h" #include "Vehicle/GdtLoaderVehicleT6.h" #include "Vehicle/RawLoaderVehicleT6.h" #include "Weapon/AttachmentGdtLoaderT6.h" #include "Weapon/AttachmentRawLoaderT6.h" #include "Weapon/AttachmentUniqueGdtLoaderT6.h" #include "Weapon/AttachmentUniqueRawLoaderT6.h" #include "Weapon/CamoJsonLoaderT6.h" #include "Weapon/WeaponGdtLoaderT6.h" #include "Weapon/WeaponRawLoaderT6.h" #include "ZBarrier/GdtLoaderZBarrierT6.h" #include "ZBarrier/RawLoaderZBarrierT6.h" #include #include namespace T6 { constexpr auto IPAK_READ_HASH = Common::Com_HashKey("ipak_read", 64); constexpr auto GLOBAL_HASH = Common::Com_HashKey("GLOBAL", 64); bool ObjLoader::VerifySoundBankChecksum(const SoundBank& soundBank, const SndRuntimeAssetBank& sndRuntimeAssetBank) { SoundAssetBankChecksum checksum{}; static_assert(sizeof(SoundAssetBankChecksum::checksumBytes) == sizeof(SndRuntimeAssetBank::linkTimeChecksum)); for (auto i = 0u; i < sizeof(SoundAssetBankChecksum::checksumBytes); i++) checksum.checksumBytes[i] = sndRuntimeAssetBank.linkTimeChecksum[i]; return soundBank.VerifyChecksum(checksum); } SoundBank* ObjLoader::LoadSoundBankForZone(ISearchPath& searchPath, const std::string& soundBankFileName, Zone& zone) { con::debug("Trying to load sound bank '{}' for zone '{}'", soundBankFileName, zone.m_name); auto* existingSoundBank = SoundBank::Repository.GetContainerByName(soundBankFileName); if (existingSoundBank != nullptr) { con::debug("Referencing loaded sound bank '{}'.", soundBankFileName); SoundBank::Repository.AddContainerReference(existingSoundBank, &zone); return existingSoundBank; } auto file = searchPath.Open(soundBankFileName); if (file.IsOpen()) { auto sndBank = std::make_unique(soundBankFileName, std::move(file.m_stream), file.m_length); auto* sndBankPtr = sndBank.get(); if (!sndBank->Initialize()) { con::error("Failed to load sound bank '{}'", soundBankFileName); return nullptr; } SoundBank::Repository.AddContainer(std::move(sndBank), &zone); con::debug("Found and loaded sound bank '{}'", soundBankFileName); return sndBankPtr; } con::warn("Could not find sound bank '{}'", soundBankFileName); return nullptr; } void ObjLoader::LoadSoundBankFromLinkedInfo(ISearchPath& searchPath, const std::string& soundBankFileName, const SndRuntimeAssetBank& sndBankLinkedInfo, Zone& zone, std::set& loadedBanksForZone, std::stack& dependenciesToLoad) { if (loadedBanksForZone.find(soundBankFileName) == loadedBanksForZone.end()) { auto* soundBank = LoadSoundBankForZone(searchPath, soundBankFileName, zone); if (soundBank) { if (!VerifySoundBankChecksum(*soundBank, sndBankLinkedInfo)) con::warn("Checksum of sound bank does not match link time checksum for '{}'", soundBankFileName); loadedBanksForZone.emplace(soundBankFileName); for (const auto& dependency : soundBank->GetDependencies()) { dependenciesToLoad.emplace(dependency); } } } } void ObjLoader::LoadSoundBanksFromAsset(ISearchPath& searchPath, const SndBank& sndBank, Zone& zone, std::set& loadedBanksForZone) { std::stack dependenciesToLoad; if (sndBank.streamAssetBank.zone) { const auto soundBankFileName = SoundBank::GetFileNameForDefinition(true, sndBank.streamAssetBank.zone, sndBank.streamAssetBank.language); LoadSoundBankFromLinkedInfo(searchPath, soundBankFileName, sndBank.streamAssetBank, zone, loadedBanksForZone, dependenciesToLoad); } if (sndBank.runtimeAssetLoad && sndBank.loadAssetBank.zone) { const auto soundBankFileName = SoundBank::GetFileNameForDefinition(false, sndBank.loadAssetBank.zone, sndBank.loadAssetBank.language); LoadSoundBankFromLinkedInfo(searchPath, soundBankFileName, sndBank.loadAssetBank, zone, loadedBanksForZone, dependenciesToLoad); } while (!dependenciesToLoad.empty()) { auto dependencyFileName = dependenciesToLoad.top(); dependenciesToLoad.pop(); if (loadedBanksForZone.find(dependencyFileName) == loadedBanksForZone.end()) { auto* soundBank = LoadSoundBankForZone(searchPath, dependencyFileName, zone); if (soundBank) { loadedBanksForZone.emplace(dependencyFileName); for (const auto& dependency : soundBank->GetDependencies()) dependenciesToLoad.emplace(dependency); } } } } void ObjLoader::LoadIPakForZone(ISearchPath& searchPath, const std::string& ipakName, Zone& zone) { con::debug("Trying to load ipak '{}' for zone '{}'", ipakName, zone.m_name); auto* existingIPak = IIPak::Repository.GetContainerByName(ipakName); if (existingIPak != nullptr) { con::debug("Referencing loaded ipak '{}'.", ipakName); IIPak::Repository.AddContainerReference(existingIPak, &zone); return; } const auto ipakFilename = std::format("{}.ipak", ipakName); auto file = searchPath.Open(ipakFilename); if (file.IsOpen()) { auto ipak = IIPak::Create(ipakFilename, std::move(file.m_stream)); if (ipak->Initialize()) { IIPak::Repository.AddContainer(std::move(ipak), &zone); con::debug("Found and loaded ipak '{}'.", ipakFilename); } else { con::error("Failed to load ipak '{}'!", ipakFilename); } } } bool ObjLoader::IsMpZone(const Zone& zone) { return zone.m_name.compare(0, 3, "mp_") == 0 || zone.m_name.compare(zone.m_name.length() - 3, 3, "_mp") == 0; } bool ObjLoader::IsZmZone(const Zone& zone) { return zone.m_name.compare(0, 3, "zm_") == 0 || zone.m_name.compare(zone.m_name.length() - 3, 3, "_zm") == 0; } void ObjLoader::LoadCommonIPaks(ISearchPath& searchPath, Zone& zone) { con::debug("Loading common ipaks for zone \"{}\"", zone.m_name); LoadIPakForZone(searchPath, "base", zone); const auto& languagePrefixes = IGame::GetGameById(GameId::T6)->GetLanguagePrefixes(); for (const auto& languagePrefix : languagePrefixes) LoadIPakForZone(searchPath, std::format("{}base", languagePrefix.m_prefix), zone); if (IsMpZone(zone)) { con::debug("Loading multiplayer ipaks for zone \"{}\"", zone.m_name); LoadIPakForZone(searchPath, "mp", zone); LoadIPakForZone(searchPath, "so", zone); } else if (IsZmZone(zone)) { con::debug("Loading zombie ipak for zone \"{}\"", zone.m_name); LoadIPakForZone(searchPath, "zm", zone); } else { con::debug("Loading singleplayer ipak for zone \"{}\"", zone.m_name); LoadIPakForZone(searchPath, "sp", zone); } } void ObjLoader::LoadReferencedContainersForZone(ISearchPath& searchPath, Zone& zone) const { const auto zoneNameHash = Common::Com_HashKey(zone.m_name.c_str(), 64); LoadCommonIPaks(searchPath, zone); for (auto* keyValuePairsEntry : zone.m_pools.PoolAssets()) { const auto* keyValuePairs = keyValuePairsEntry->Asset(); for (auto variableIndex = 0u; variableIndex < keyValuePairs->numVariables; variableIndex++) { auto* variable = &keyValuePairs->keyValuePairs[variableIndex]; if (variable->namespaceHash == zoneNameHash && variable->keyHash == IPAK_READ_HASH) { LoadIPakForZone(searchPath, variable->value, zone); } } } std::set loadedSoundBanksForZone; for (auto* sndBankAssetInfo : zone.m_pools.PoolAssets()) { LoadSoundBanksFromAsset(searchPath, *sndBankAssetInfo->Asset(), zone, loadedSoundBanksForZone); } } void ObjLoader::UnloadContainersOfZone(Zone& zone) const { IIPak::Repository.RemoveContainerReferences(&zone); } namespace { void ConfigureDefaultCreators(AssetCreatorCollection& collection, Zone& zone) { auto& memory = zone.Memory(); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); // AssetImpactFx has no name and cannot be default constructed collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); // AssetEmblemSet has no name and cannot be default constructed collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); collection.AddDefaultAssetCreator(std::make_unique>(memory)); // custom maps have no default } void ConfigureGlobalAssetPoolsLoaders(AssetCreatorCollection& collection, Zone& zone) { collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); // collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); collection.AddAssetCreator(std::make_unique>(zone)); // collection.AddAssetCreator(std::make_unique>(zone)); } void ConfigureLoaders(AssetCreatorCollection& collection, Zone& zone, ISearchPath& searchPath, IGdtQueryable& gdt) { auto& memory = zone.Memory(); collection.AddAssetCreator(phys_preset::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(phys_preset::CreateGdtLoaderT6(memory, gdt, zone)); collection.AddAssetCreator(phys_constraints::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(phys_constraints::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(xmodel::CreateLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(material::CreateLoaderT6(memory, searchPath)); collection.AddAssetCreator(CreateTechniqueSetLoader(memory, searchPath)); collection.AddAssetCreator(image::CreateLoaderEmbeddedT6(memory, searchPath)); collection.AddAssetCreator(image::CreateLoaderExternalT6(memory, searchPath)); collection.AddAssetCreator(image::CreateLoadDefLoaderT6(memory, searchPath)); collection.AddAssetCreator(sound::CreateSoundBankLoaderT6(memory, searchPath)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(font_icon::CreateCsvLoaderT6(memory, searchPath)); collection.AddAssetCreator(font_icon::CreateJsonLoaderT6(memory, searchPath)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(localize::CreateLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(weapon::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(weapon::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); collection.AddAssetCreator(attachment::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(attachment::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); collection.AddAssetCreator(attachment_unique::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(attachment_unique::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); collection.AddAssetCreator(camo::CreateJsonLoaderT6(memory, searchPath)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(raw_file::CreateLoaderT6(memory, searchPath)); collection.AddAssetCreator(string_table::CreateLoaderT6(memory, searchPath)); collection.AddAssetCreator(leaderboard::CreateLoaderT6(memory, searchPath)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(script::CreateLoaderT6(memory, searchPath)); collection.AddAssetCreator(vehicle::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(vehicle::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(qdb::CreateLoaderT6(memory, searchPath)); collection.AddAssetCreator(slug::CreateLoaderT6(memory, searchPath)); // collection.AddAssetCreator(std::make_unique(memory)); // collection.AddAssetCreator(std::make_unique(memory)); collection.AddAssetCreator(z_barrier::CreateRawLoaderT6(memory, searchPath, zone)); collection.AddAssetCreator(z_barrier::CreateGdtLoaderT6(memory, searchPath, gdt, zone)); collection.AddSubAssetCreator(techset::CreateVertexShaderLoaderT6(memory, searchPath)); collection.AddSubAssetCreator(techset::CreatePixelShaderLoaderT6(memory, searchPath)); collection.AddAssetCreator(CreateCustomMapLoader(memory, searchPath, zone)); } } // namespace void ObjLoader::ConfigureCreatorCollection(AssetCreatorCollection& collection, Zone& zone, ISearchPath& searchPath, IGdtQueryable& gdt) const { ConfigureDefaultCreators(collection, zone); ConfigureLoaders(collection, zone, searchPath, gdt); ConfigureGlobalAssetPoolsLoaders(collection, zone); } } // namespace T6