#include #include "StructuredData.hpp" namespace Components { Utils::Memory::Allocator StructuredData::MemAllocator; const char* StructuredData::EnumTranslation[COUNT] = { "features", "weapons", "attachements", "challenges", "camos", "perks", "killstreaks", "accolades", "cardicons", "cardtitles", "cardnameplates", "teams", "gametypes" }; void StructuredData::PatchPlayerDataEnum(Game::StructuredDataDef* data, StructuredData::PlayerDataType type, std::vector& entries) { if (!data || type >= StructuredData::PlayerDataType::COUNT) return; Game::StructuredDataEnum* dataEnum = &data->enums[type]; // Build index-sorted data vector std::vector dataVector; for (int i = 0; i < dataEnum->entryCount; ++i) { int index = 0; for (; index < dataEnum->entryCount; ++index) { if (dataEnum->entries[index].index == i) { break; } } dataVector.push_back(dataEnum->entries[index].string); } // Rebase or add new entries for (auto entry : entries) { const char* value = nullptr; for (auto i = dataVector.begin(); i != dataVector.end(); ++i) { if (*i == entry) { value = *i; dataVector.erase(i); Logger::Print("Playerdatadef entry '{}' will be rebased!\n", value); break; } } if (!value) value = StructuredData::MemAllocator.duplicateString(entry); dataVector.push_back(value); } // Map data back to the game structure Game::StructuredDataEnumEntry* indices = StructuredData::MemAllocator.allocateArray(dataVector.size()); for (unsigned short i = 0; i < dataVector.size(); ++i) { indices[i].index = i; indices[i].string = dataVector[i]; } // Sort alphabetically qsort(indices, dataVector.size(), sizeof(Game::StructuredDataEnumEntry), [](void const* first, void const* second) { const Game::StructuredDataEnumEntry* entry1 = reinterpret_cast(first); const Game::StructuredDataEnumEntry* entry2 = reinterpret_cast(second); return std::string(entry1->string).compare(entry2->string); }); // Apply our patches dataEnum->entryCount = dataVector.size(); dataEnum->entries = indices; } void StructuredData::PatchCustomClassLimit(Game::StructuredDataDef* data, int count) { const int customClassSize = 64; for (int i = 0; i < data->structs[0].propertyCount; ++i) { // 3003 is the offset of the customClasses structure if (data->structs[0].properties[i].offset >= 3643) { // -10 because 10 is the default amount of custom classes. data->structs[0].properties[i].offset += ((count - 10) * customClassSize); } } // update structure size data->size += ((count - 10) * customClassSize); // Update amount of custom classes data->indexedArrays[5].arraySize = count; } void StructuredData::PatchAdditionalData(Game::StructuredDataDef* data, std::unordered_map& patches) { for (auto& item : patches) { if (item.first == "classes") { StructuredData::PatchCustomClassLimit(data, atoi(item.second.data())); } } } bool StructuredData::UpdateVersionOffsets(Game::StructuredDataDefSet *set, Game::StructuredDataBuffer *buffer, Game::StructuredDataDef *whatever) { Game::StructuredDataDef* newDef = &set->defs[0]; Game::StructuredDataDef* oldDef = &set->defs[0]; for (unsigned int i = 0; i < set->defCount; ++i) { if (newDef->version < set->defs[i].version) { newDef = &set->defs[i]; } if (set->defs[i].version == *reinterpret_cast(buffer->data)) { oldDef = &set->defs[i]; } } if (newDef->version >= 159 && oldDef->version <= 158) { // this should move the data 320 bytes infront std::memmove(&buffer->data[3963], &buffer->data[3643], oldDef->size - 3643); } // StructuredData_UpdateVersion return Utils::Hook::Call(0x456830)(set, buffer, whatever); } StructuredData::StructuredData() { if (Dedicated::IsEnabled()) return; // Do not execute this when building zones if (!ZoneBuilder::IsEnabled()) { // Correctly upgrade stats Utils::Hook(0x42F088, StructuredData::UpdateVersionOffsets, HOOK_CALL).install()->quick(); // 15 or more custom classes Utils::Hook::Set(0x60A2FE, NUM_CUSTOM_CLASSES); #ifdef _DEBUG // Reset empty names Command::Add("checkClasses", [](Command::Params*) { for (int i = 0; i < NUM_CUSTOM_CLASSES; ++i) { // TODO: Correctly lookup using structured data char* className = (reinterpret_cast(0x1AD3694) - 4 + 3003 + (64 * i) + 0x29); if (!*className) strcpy_s(className, 24, Game::SEH_StringEd_GetString(Utils::String::VA("CLASS_SLOT%i", i + 1))); } }); #endif return; } AssetHandler::OnLoad([](Game::XAssetType type, Game::XAssetHeader asset, const std::string& filename, bool* /*restrict*/) { // Only intercept playerdatadef loading if (type != Game::XAssetType::ASSET_TYPE_STRUCTURED_DATA_DEF || filename != "mp/playerdata.def") return; // Store asset Game::StructuredDataDefSet* data = asset.structuredDataDefSet; if (!data) return; if (data->defCount != 1) { Logger::Error(Game::ERR_FATAL, "PlayerDataDefSet contains more than 1 definition!"); return; } if (data->defs[0].version != 155) { Logger::Error(Game::ERR_FATAL, "Initial PlayerDataDef is not version 155, patching not possible!"); return; } std::unordered_map>> patchDefinitions; std::unordered_map> otherPatchDefinitions; // First check if all versions are present for (int i = 156;; ++i) { // We're on DB thread (OnLoad) so use DB thread for FS FileSystem::File definition(std::format("{}/{}.json", filename, i), Game::FsThread::FS_THREAD_DATABASE); if (!definition.exists()) break; std::vector> enumContainer; std::unordered_map otherPatches; nlohmann::json defData; try { defData = nlohmann::json::parse(definition.getBuffer()); } catch (const nlohmann::json::parse_error& ex) { Logger::PrintError(Game::CON_CHANNEL_ERROR, "Json Parse Error: {}\n", ex.what()); return; } if (!defData.is_object()) { Logger::Error(Game::ERR_FATAL, "PlayerDataDef patch for version {} is invalid!", i); return; } for (auto pType = 0; pType < StructuredData::PlayerDataType::COUNT; ++pType) { auto enumData = defData[StructuredData::EnumTranslation[pType]]; std::vector entryData; if (enumData.is_array()) { for (const auto& rawEntry : enumData) { if (rawEntry.is_string()) { entryData.push_back(rawEntry.get()); } } } enumContainer.push_back(entryData); } auto other = defData["other"]; if (other.is_object()) { for (auto& item : other.items()) { if (item.value().is_string()) { otherPatches[item.key()] = item.value().get(); } } } patchDefinitions[i] = enumContainer; otherPatchDefinitions[i] = otherPatches; } // Nothing to patch if (patchDefinitions.empty()) return; // Reallocate the definition auto* newData = StructuredData::MemAllocator.allocateArray(data->defCount + patchDefinitions.size()); std::memcpy(&newData[patchDefinitions.size()], data->defs, sizeof Game::StructuredDataDef * data->defCount); // Prepare the buffers for (unsigned int i = 0; i < patchDefinitions.size(); ++i) { std::memcpy(&newData[i], data->defs, sizeof Game::StructuredDataDef); newData[i].version = (patchDefinitions.size() - i) + 155; // Reallocate the enum array auto* newEnums = StructuredData::MemAllocator.allocateArray(data->defs->enumCount); std::memcpy(newEnums, data->defs->enums, sizeof Game::StructuredDataEnum * data->defs->enumCount); newData[i].enums = newEnums; } // Apply new data data->defs = newData; data->defCount += patchDefinitions.size(); // Patch the definition for (unsigned int i = 0; i < data->defCount; ++i) { // No need to patch version 155 if (newData[i].version == 155) continue; if (patchDefinitions.contains(newData[i].version)) { auto patchData = patchDefinitions[newData[i].version]; auto otherData = otherPatchDefinitions[newData[i].version]; // Invalid patch data if (patchData.size() != StructuredData::PlayerDataType::COUNT) { Logger::Error(Game::ERR_FATAL, "PlayerDataDef patch for version {} wasn't parsed correctly!", newData[i].version); continue; } // Apply the patch data for (auto pType = 0; pType < StructuredData::PlayerDataType::COUNT; ++pType) { if (!patchData[pType].empty()) { StructuredData::PatchPlayerDataEnum(&newData[i], static_cast(pType), patchData[pType]); } } StructuredData::PatchAdditionalData(&newData[i], otherData); } } }); } }