mirror of
				https://github.com/Laupetin/OpenAssetTools.git
				synced 2025-10-20 13:35:20 +00:00 
			
		
		
		
	Dump StructuredDataDef paddings
This commit is contained in:
		| @@ -4,9 +4,11 @@ | ||||
| #include <cassert> | ||||
| #include <sstream> | ||||
|  | ||||
| StructuredDataDefDumper::StructEntry::StructEntry(std::string stringValue, const size_t offset) | ||||
| StructuredDataDefDumper::StructEntry::StructEntry(std::string stringValue, const size_t offset, const size_t sizeInBits, const size_t alignment) | ||||
|     : m_string_value(std::move(stringValue)), | ||||
|       m_offset(offset) | ||||
|       m_offset(offset), | ||||
|       m_size_in_bits(sizeInBits), | ||||
|       m_alignment(alignment) | ||||
| { | ||||
| } | ||||
|  | ||||
| @@ -16,7 +18,9 @@ StructuredDataDefDumper::StructuredDataDefDumper(std::ostream& stream) | ||||
|       m_flags{}, | ||||
|       m_enum_entry_count(0u), | ||||
|       m_struct_property_count(0u), | ||||
|       m_current_property_offset(0u) | ||||
|       m_current_property_offset(0u), | ||||
|       m_current_property_size_in_bits(0u), | ||||
|       m_current_property_alignment(0u) | ||||
| { | ||||
| } | ||||
|  | ||||
| @@ -130,7 +134,7 @@ void StructuredDataDefDumper::WriteEnumEntry(const std::string& entryName, const | ||||
|     m_enum_entries[entryValue] = entryName; | ||||
| } | ||||
|  | ||||
| void StructuredDataDefDumper::BeginStruct(const std::string& structName, const size_t structPropertyCount) | ||||
| void StructuredDataDefDumper::BeginStruct(const std::string& structName, const size_t structPropertyCount, const size_t structSizeInBits, const size_t structInitialOffset) | ||||
| { | ||||
|     assert(m_flags.m_in_version); | ||||
|     assert(m_block == Block::BLOCK_NONE); | ||||
| @@ -139,6 +143,8 @@ void StructuredDataDefDumper::BeginStruct(const std::string& structName, const s | ||||
|         return; | ||||
|  | ||||
|     m_struct_property_count = structPropertyCount; | ||||
|     m_struct_size_in_bits = structSizeInBits; | ||||
|     m_struct_initial_offset = structInitialOffset; | ||||
|     m_struct_properties.reserve(structPropertyCount); | ||||
|  | ||||
|     if (m_flags.m_empty_line_before_block) | ||||
| @@ -166,10 +172,34 @@ void StructuredDataDefDumper::EndStruct() | ||||
|         return e1.m_offset < e2.m_offset; | ||||
|     }); | ||||
|  | ||||
|     auto expectedOffset = m_struct_initial_offset; | ||||
|     for (auto& structProperty : m_struct_properties) | ||||
|     { | ||||
|         if (structProperty.m_alignment > 0) | ||||
|             expectedOffset = (expectedOffset + structProperty.m_alignment - 1) / structProperty.m_alignment * structProperty.m_alignment; | ||||
|  | ||||
|         if(expectedOffset != structProperty.m_offset) | ||||
|         { | ||||
|             assert(structProperty.m_offset > expectedOffset); | ||||
|             assert((structProperty.m_offset - expectedOffset) % 8 == 0); | ||||
|             Indent(); | ||||
|             m_stream << "pad(" << (structProperty.m_offset - expectedOffset) << ");\n"; | ||||
|             expectedOffset = structProperty.m_offset; | ||||
|         } | ||||
|  | ||||
|         Indent(); | ||||
|         m_stream << structProperty.m_string_value << ";\n"; | ||||
|  | ||||
|         expectedOffset += structProperty.m_size_in_bits; | ||||
|     } | ||||
|  | ||||
|     expectedOffset = (expectedOffset + 7) / 8 * 8; | ||||
|     if (m_struct_size_in_bits > 0 && expectedOffset != m_struct_size_in_bits) | ||||
|     { | ||||
|         assert(m_struct_size_in_bits > expectedOffset); | ||||
|         assert((m_struct_size_in_bits - expectedOffset) % 8 == 0); | ||||
|         Indent(); | ||||
|         m_stream << "pad(" << (m_struct_size_in_bits - expectedOffset) / 8 << ");\n"; | ||||
|     } | ||||
|  | ||||
|     DecIndent(); | ||||
| @@ -179,9 +209,11 @@ void StructuredDataDefDumper::EndStruct() | ||||
|     m_flags.m_empty_line_before_block = true; | ||||
|     m_struct_properties.clear(); | ||||
|     m_struct_property_count = 0u; | ||||
|     m_struct_initial_offset = 0u; | ||||
|     m_struct_size_in_bits = 0u; | ||||
| } | ||||
|  | ||||
| void StructuredDataDefDumper::BeginProperty(const std::string& propertyName, const size_t propertyOffset) | ||||
| void StructuredDataDefDumper::BeginProperty(const std::string& propertyName, const size_t propertyOffset, const size_t propertySizeInBits, const size_t alignment) | ||||
| { | ||||
|     assert(m_flags.m_in_version); | ||||
|     assert(m_block == Block::BLOCK_STRUCT); | ||||
| @@ -191,6 +223,8 @@ void StructuredDataDefDumper::BeginProperty(const std::string& propertyName, con | ||||
|  | ||||
|     m_current_property_name = propertyName; | ||||
|     m_current_property_offset = propertyOffset; | ||||
|     m_current_property_size_in_bits = propertySizeInBits; | ||||
|     m_current_property_alignment = alignment; | ||||
|  | ||||
|     m_block = Block::BLOCK_PROPERTY; | ||||
| } | ||||
| @@ -226,7 +260,7 @@ void StructuredDataDefDumper::EndProperty() | ||||
|         ss << " + " << m_current_property_offset % 8 << "bit"; | ||||
|  | ||||
|     ss << " */ "; | ||||
|     m_struct_properties.emplace_back(ss.str(), m_current_property_offset); | ||||
|     m_struct_properties.emplace_back(ss.str(), m_current_property_offset, m_current_property_size_in_bits, m_current_property_alignment); | ||||
|  | ||||
|     m_block = Block::BLOCK_STRUCT; | ||||
|     m_current_property_array_specifiers.clear(); | ||||
|   | ||||
| @@ -10,8 +10,10 @@ class StructuredDataDefDumper : AbstractTextDumper | ||||
|     { | ||||
|         std::string m_string_value; | ||||
|         size_t m_offset; | ||||
|         size_t m_size_in_bits; | ||||
|         size_t m_alignment; | ||||
|  | ||||
|         StructEntry(std::string stringValue, size_t offset); | ||||
|         StructEntry(std::string stringValue, size_t offset, size_t sizeInBits, size_t alignment); | ||||
|     }; | ||||
|  | ||||
|     enum class Block | ||||
| @@ -34,8 +36,12 @@ class StructuredDataDefDumper : AbstractTextDumper | ||||
|  | ||||
|     std::vector<StructEntry> m_struct_properties; | ||||
|     size_t m_struct_property_count; | ||||
|     size_t m_struct_initial_offset; | ||||
|     size_t m_struct_size_in_bits; | ||||
|     std::string m_current_property_name; | ||||
|     size_t m_current_property_offset; | ||||
|     size_t m_current_property_size_in_bits; | ||||
|     size_t m_current_property_alignment; | ||||
|     std::string m_current_property_type_name; | ||||
|     std::vector<std::string> m_current_property_array_specifiers; | ||||
|  | ||||
| @@ -51,9 +57,9 @@ public: | ||||
|     void EndEnum(); | ||||
|     void WriteEnumEntry(const std::string& entryName, size_t entryValue); | ||||
|  | ||||
|     void BeginStruct(const std::string& structName, size_t structPropertyCount); | ||||
|     void BeginStruct(const std::string& structName, size_t structPropertyCount, size_t structSizeInBits, size_t structInitialOffset); | ||||
|     void EndStruct(); | ||||
|     void BeginProperty(const std::string& propertyName, size_t propertyOffset); | ||||
|     void BeginProperty(const std::string& propertyName, size_t propertyOffset, size_t propertySizeInBits, size_t alignment); | ||||
|     void AddPropertyArraySpecifier(const std::string& specifierName); | ||||
|     void SetPropertyTypeName(const std::string& typeName); | ||||
|     void EndProperty(); | ||||
|   | ||||
| @@ -133,9 +133,72 @@ void AssetDumperStructuredDataDefSet::DumpEnum(StructuredDataDefDumper& dumper, | ||||
|     dumper.EndEnum(); | ||||
| } | ||||
|  | ||||
| size_t AssetDumperStructuredDataDefSet::GetPropertySizeInBits(const StructuredDataStructProperty& property, const StructuredDataDef* def) | ||||
| { | ||||
|     switch (property.type.type) | ||||
|     { | ||||
|     case DATA_FLOAT: | ||||
|     case DATA_INT: | ||||
|         return 32; | ||||
|     case DATA_BYTE: | ||||
|         return 8; | ||||
|     case DATA_BOOL: | ||||
|         return 1; | ||||
|     case DATA_SHORT: | ||||
|     case DATA_ENUM: | ||||
|         return 16; | ||||
|     case DATA_STRING: | ||||
|         return property.type.u.stringDataLength * 8; | ||||
|     case DATA_STRUCT: | ||||
|         if (property.type.u.structIndex < 0 || property.type.u.structIndex >= def->structCount || def->structs == nullptr) | ||||
|         { | ||||
|             assert(false); | ||||
|             return 0; | ||||
|         } | ||||
|         return static_cast<size_t>(def->structs[property.type.u.structIndex].size) * 8; | ||||
|     case DATA_INDEXED_ARRAY: | ||||
|         { | ||||
|             if (property.type.u.indexedArrayIndex < 0 || property.type.u.indexedArrayIndex >= def->indexedArrayCount || def->indexedArrays == nullptr) | ||||
|             { | ||||
|                 assert(false); | ||||
|                 return 0; | ||||
|             } | ||||
|             const auto& indexedArray = def->indexedArrays[property.type.u.indexedArrayIndex]; | ||||
|             const auto elementSize = indexedArray.elementType.type == DATA_BOOL ? 1 : indexedArray.elementSize * 8; | ||||
|             return elementSize * static_cast<size_t>(def->indexedArrays[property.type.u.indexedArrayIndex].arraySize); | ||||
|         } | ||||
|     case DATA_ENUM_ARRAY: | ||||
|         { | ||||
|             if (property.type.u.enumedArrayIndex < 0 || property.type.u.enumedArrayIndex >= def->enumedArrayCount || def->enumedArrays == nullptr) | ||||
|             { | ||||
|                 assert(false); | ||||
|                 return 0; | ||||
|             } | ||||
|  | ||||
|             const auto& enumedArray = def->enumedArrays[property.type.u.enumedArrayIndex]; | ||||
|             if (enumedArray.enumIndex < 0 || enumedArray.enumIndex >= def->enumCount || def->enums == nullptr) | ||||
|             { | ||||
|                 assert(false); | ||||
|                 return 0; | ||||
|             } | ||||
|  | ||||
|             const auto elementSize = enumedArray.elementType.type == DATA_BOOL ? 1 : enumedArray.elementSize * 8; | ||||
|             return elementSize * static_cast<size_t>(def->enums[enumedArray.enumIndex].reservedEntryCount); | ||||
|         } | ||||
|     default: | ||||
|         assert(false); | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
|  | ||||
| size_t AssetDumperStructuredDataDefSet::GetPropertyAlign(const StructuredDataStructProperty& property) | ||||
| { | ||||
|     return property.type.type == DATA_BOOL ? 0 : 8; | ||||
| } | ||||
|  | ||||
| void AssetDumperStructuredDataDefSet::DumpProperty(StructuredDataDefDumper& dumper, const StructuredDataStructProperty& property, const StructuredDataDef* def, const int rootStructIndex) | ||||
| { | ||||
|     dumper.BeginProperty(property.name, property.type.type == DATA_BOOL ? property.offset : property.offset * 8); | ||||
|     dumper.BeginProperty(property.name, property.type.type == DATA_BOOL ? property.offset : property.offset * 8, GetPropertySizeInBits(property, def), GetPropertyAlign(property)); | ||||
|  | ||||
|     auto currentType = property.type; | ||||
|     auto stopTypeIteration = false; | ||||
| @@ -242,23 +305,30 @@ void AssetDumperStructuredDataDefSet::DumpProperty(StructuredDataDefDumper& dump | ||||
|     dumper.EndProperty(); | ||||
| } | ||||
|  | ||||
| void AssetDumperStructuredDataDefSet::DumpStruct(StructuredDataDefDumper& dumper, const size_t structIndex, const StructuredDataStruct* _struct, const StructuredDataDef* def, const int rootStructIndex) | ||||
| void AssetDumperStructuredDataDefSet::DumpStruct(StructuredDataDefDumper& dumper, const size_t structIndex, const StructuredDataStruct* _struct, const StructuredDataDef* def, | ||||
|                                                  const int rootStructIndex) | ||||
| { | ||||
|     std::string structName; | ||||
|     size_t actualStructSize; | ||||
|     size_t initialOffset; | ||||
|     if (static_cast<int>(structIndex) == rootStructIndex) | ||||
|     { | ||||
|         structName = "root"; | ||||
|         actualStructSize = def->size; | ||||
|         initialOffset = 64; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         std::ostringstream ss; | ||||
|         ss << "STRUCT_" << structIndex; | ||||
|         structName = ss.str(); | ||||
|         actualStructSize = static_cast<size_t>(_struct->size); | ||||
|         initialOffset = 0; | ||||
|     } | ||||
|  | ||||
|     dumper.WriteLineComment("BitOffset: "s + std::to_string(_struct->bitOffset)); | ||||
|     dumper.WriteLineComment("Size: "s + std::to_string(_struct->size)); | ||||
|     dumper.BeginStruct(structName, static_cast<size_t>(_struct->propertyCount)); | ||||
|     dumper.BeginStruct(structName, static_cast<size_t>(_struct->propertyCount), actualStructSize * 8, initialOffset); | ||||
|  | ||||
|     if (_struct->properties && _struct->propertyCount > 0) | ||||
|     { | ||||
|   | ||||
| @@ -15,6 +15,8 @@ namespace IW4 | ||||
|         static StructuredDataType GetBaseType(const StructuredDataDef* def, StructuredDataType type); | ||||
|         static std::vector<size_t> CalculateStructDumpingOrder(const StructuredDataDef* def); | ||||
|         static void DumpEnum(StructuredDataDefDumper& dumper, int enumIndex, const StructuredDataEnum* _enum); | ||||
|         static size_t GetPropertySizeInBits(const StructuredDataStructProperty& property, const StructuredDataDef* def); | ||||
|         static size_t GetPropertyAlign(const StructuredDataStructProperty& property); | ||||
|         static void DumpProperty(StructuredDataDefDumper& dumper, const StructuredDataStructProperty& property, const StructuredDataDef* def, int rootStructIndex); | ||||
|         static void DumpStruct(StructuredDataDefDumper& dumper, size_t structIndex, const StructuredDataStruct* _struct, const StructuredDataDef* def, int rootStructIndex); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user