Better StructuredData patching

This commit is contained in:
momo5502 2016-11-06 11:50:55 +01:00
parent 000e2177e3
commit 708a739650

View File

@ -27,112 +27,62 @@ namespace Components
Game::StructuredDataEnum* dataEnum = &data->enums[type]; Game::StructuredDataEnum* dataEnum = &data->enums[type];
// Find last index so we can add our offset to it. // Build index-sorted data vector
// This whole procedure is potentially unsafe. std::vector<const char*> dataVector;
// If any index changes, everything gets shifted and the stats are fucked.
int lastIndex = 0;
for (int i = 0; i < dataEnum->numIndices; ++i) for (int i = 0; i < dataEnum->numIndices; ++i)
{ {
if (dataEnum->indices[i].index > lastIndex) int index = 0;
for (; index < dataEnum->numIndices; ++index)
{ {
lastIndex = dataEnum->indices[i].index; if (dataEnum->indices[index].index == i)
}
}
std::vector<std::string> rebaseEntries;
for (unsigned int i = 0; i < entries.size(); ++i)
{
for (int pos = 0; pos < dataEnum->numIndices; ++pos)
{
if (dataEnum->indices[pos].key == entries[i])
{
rebaseEntries.push_back(entries[i]);
}
}
}
// Calculate new count
unsigned int indexCount = dataEnum->numIndices + entries.size() - rebaseEntries.size();
// Allocate new entries
Game::StructuredDataEnumEntry* indices = StructuredData::MemAllocator.AllocateArray<Game::StructuredDataEnumEntry>(indexCount);
std::memcpy(indices, dataEnum->indices, sizeof(Game::StructuredDataEnumEntry) * dataEnum->numIndices);
int skipped = 0;
for (unsigned int i = 0; i < entries.size(); ++i)
{
unsigned int pos = 0;
for (; pos < (dataEnum->numIndices + i - rebaseEntries.size()); ++pos)
{
if (indices[pos].key == entries[i])
{
Logger::Print("Playerdatadef entry %s will be rebased!\n", entries[i].data());
pos = 0xFFFFFFFF;
break;
//Logger::Error("Duplicate playerdatadef entry found: %s", entries[i].data());
}
// We found our position
if (entries[i] < indices[pos].key)
{ {
break; break;
} }
} }
// Rebasing needed dataVector.push_back(dataEnum->indices[index].key);
if (pos == 0xFFFFFFFF)
{
++skipped;
continue;
}
// TODO directly shift the data using memmove
for (unsigned int j = dataEnum->numIndices + i - skipped; j > pos && j < indexCount; --j)
{
std::memcpy(&indices[j], &indices[j - 1], sizeof(Game::StructuredDataEnumEntry));
}
indices[pos].index = i + lastIndex - skipped;
indices[pos].key = StructuredData::MemAllocator.DuplicateString(entries[i]);
} }
// Meh, but best way for now // Rebase or add new entries
for (unsigned int i = 0; i < rebaseEntries.size(); ++i) for (auto entry : entries)
{ {
int index = -1; bool rebased = false;
unsigned int pos = 0; for (auto i = dataVector.begin(); i != dataVector.end(); ++i)
for (; pos < indexCount; ++pos)
{ {
if (indices[pos].key == rebaseEntries[i]) if (*i == entry)
{ {
index = indices[pos].index; const char* value = *i;
indices[pos].index = entries.size() + lastIndex - rebaseEntries.size() + i; dataVector.erase(i);
dataVector.push_back(value);
Logger::Print("Playerdatadef entry '%s' was rebased!\n", value);
rebased = true;
break; break;
} }
} }
if (index < 0) if(rebased) continue;
{ dataVector.push_back(StructuredData::MemAllocator.DuplicateString(entry));
Logger::Error("Playerdatadef entry %s cannot be rebased!", rebaseEntries[i].data());
}
for (unsigned int j = 0; j < indexCount; ++j)
{
if (pos != j && indices[j].index > index)
{
indices[j].index--;
}
}
} }
for (unsigned int j = 0; j < indexCount && type == StructuredData::PlayerDataType::ENUM_WEAPONS; ++j) // Map data back to the game structure
Game::StructuredDataEnumEntry* indices = StructuredData::MemAllocator.AllocateArray<Game::StructuredDataEnumEntry>(dataVector.size());
for (unsigned int i = 0; i < dataVector.size(); ++i)
{ {
OutputDebugStringA(Utils::String::VA("%s: %d\n", indices[j].key, indices[j].index)); indices[i].index = i;
indices[i].key = dataVector[i];
} }
// Sort alphabetically
qsort(indices, dataVector.size(), sizeof(Game::StructuredDataEnumEntry), [] (const void* first, const void* second)
{
const Game::StructuredDataEnumEntry* entry1 = reinterpret_cast<const Game::StructuredDataEnumEntry*>(first);
const Game::StructuredDataEnumEntry* entry2 = reinterpret_cast<const Game::StructuredDataEnumEntry*>(second);
return std::string(entry1->key).compare(entry2->key);
});
// Apply our patches // Apply our patches
dataEnum->numIndices = indexCount; dataEnum->numIndices = dataVector.size();
dataEnum->indices = indices; dataEnum->indices = indices;
} }