Пример #1
0
        /// <summary>Deserialize instance</summary>
        public void Deserialize(teStructuredData assetFile, STUField_Info field = null)
        {
            BinaryReader data = assetFile.Data;

            uint         instanceHash = teStructuredData.Manager.InstancesInverted[GetType()];
            STUAttribute stuAttribute = teStructuredData.Manager.InstanceAttributes[instanceHash];
            Dictionary <uint, KeyValuePair <FieldInfo, STUFieldAttribute> > fields = teStructuredData.Manager.FieldAttributes[instanceHash];

            if (assetFile.Format == teStructuredDataFormat.V2)
            {
                int fieldBagIdx = data.ReadInt32();
                if (fieldBagIdx == -1)
                {
                    return;
                }
                if (fieldBagIdx >= assetFile.FieldInfoBags.Count)
                {
                    throw new ArgumentOutOfRangeException($"FieldBag index is out of range. Id: {fieldBagIdx}, Type: '{GetType().Name}', Data offset: {data.BaseStream.Position - 4}");
                }
                foreach (STUField_Info stuField in assetFile.FieldInfoBags[fieldBagIdx])
                {
                    int fieldSize = stuField.Size;
                    if (fieldSize == 0)
                    {
                        fieldSize = data.ReadInt32();
                    }
                    long startPosition = data.BaseStream.Position;

                    DeserializeField(assetFile, stuField, fields, stuAttribute);

                    data.BaseStream.Position = startPosition + fieldSize;

                    //long endPosition = data.BaseStream.Position;
                    //if (endPosition - startPosition != fieldSize)
                    //    throw new InvalidDataException($"read len != field size. Type: '{GetType().Name}', Field: {stuField.Hash:X8}, Data offset: {startPosition}");
                }
            }
            else if (assetFile.Format == teStructuredDataFormat.V1)
            {
                //if (instanceHash == 0xEA30C5E9 || instanceHash == 0x298FC27F || instanceHash == 0x75526BC2 ||
                //    instanceHash == 0x5C7059E || instanceHash == 0x646FF8C3 || instanceHash == 0xAD5967B3 ||
                //    instanceHash == 0x74F13350) {
                //    Debugger.Log(0, "STUInstnace", "HACK: SKIPPING DESERIALIZE\r\n");
                //    return;
                //    //Debugger.Break();
                //}
                uint[] fieldOrder = teStructuredData.Manager.InstanceFields[instanceHash];

                foreach (uint fieldHash in fieldOrder)
                {
                    //long fieldStart = data.BaseStream.Position;

                    STUField_Info stuField = new STUField_Info {
                        Hash = fieldHash, Size = -1
                    };

                    DeserializeField(assetFile, stuField, fields, stuAttribute);
                }
            }
        }
Пример #2
0
        public void AddInstance(Type type)
        {
            STUAttribute attribute = type.GetCustomAttribute <STUAttribute>();

            if (attribute == null)
            {
                return;
            }
            if (attribute.Hash == 0)
            {
                return;
            }

            InstanceAttributes[attribute.Hash] = attribute;
            Instances[attribute.Hash]          = type;
            InstancesInverted[type]            = attribute.Hash;
            FieldAttributes[attribute.Hash]    = new Dictionary <uint, KeyValuePair <FieldInfo, STUFieldAttribute> >();
            //List<uint> fieldOrderTemp = new List<uint>();

            foreach (FieldInfo field in type.GetFields())
            {
                STUFieldAttribute fieldAttribute = field.GetCustomAttribute <STUFieldAttribute>();
                if (fieldAttribute == null)
                {
                    continue;
                }
                if (fieldAttribute.Hash == 0)
                {
                    continue;
                }
                //fieldOrderTemp.Add(fieldAttribute.Hash);

                FieldAttributes[attribute.Hash][fieldAttribute.Hash] = new KeyValuePair <FieldInfo, STUFieldAttribute>(field, fieldAttribute);
            }

            //InstanceFields[attribute.Hash] = fieldOrderTemp.ToArray();
            InstanceFields[attribute.Hash] = GetFieldOrder(type).ToArray();  // shrug
        }
Пример #3
0
        /// <summary>Read a specified STU field</summary>
        protected void DeserializeField(teStructuredData assetFile, STUField_Info fieldInfo,
                                        Dictionary <uint, KeyValuePair <FieldInfo, STUFieldAttribute> > fields, STUAttribute stuAttribute)
        {
            if (!fields.TryGetValue(fieldInfo.Hash, out var field))
            {
            #if DEBUG
                string name = stuAttribute.Name ?? $"STU_{stuAttribute.Hash:X8}";
                Debugger.Log(0, "STUInstance", $"Unhandled field: {name}:{fieldInfo.Hash:X8} (size: {fieldInfo.Size})\r\n");
            #endif

                return;
            }

            if (assetFile.Format == teStructuredDataFormat.V1)
            {
                assetFile.DynData = assetFile.Data;
            }

            IStructuredDataFieldReader reader = teStructuredData.Manager.FieldReaders[field.Value.ReaderType];

            if (field.Key.FieldType.IsArray)
            {
                Type elementType = field.Key.FieldType.GetElementType();
                if (elementType == null)
                {
                    return;
                }
                Array array;

                BinaryReader data    = assetFile.Data;
                BinaryReader dynData = assetFile.DynData;

                if (assetFile.Format == teStructuredDataFormat.V2)
                {
                    if (fieldInfo.Size == 0 || reader is InlineInstanceFieldReader)    // inline
                    {
                        int size = data.ReadInt32();
                        if (size == 0)
                        {
                            return;
                        }
                        array = Array.CreateInstance(elementType, size);

                        for (int i = 0; i != size; ++i)
                        {
                            reader.Deserialize_Array(teStructuredData.Manager, assetFile, fieldInfo, array, i);
                        }
                    }
                    else
                    {
                        int offset = data.ReadInt32();
                        if (offset == -1)
                        {
                            return;
                        }
                        dynData.Seek(offset);
                        int size = dynData.ReadInt32();
                        if (size == 0)
                        {
                            return;
                        }
                        array = Array.CreateInstance(elementType, size);
                        uint unknown = dynData.ReadUInt32();
                        dynData.Seek(dynData.ReadInt64()); // Seek to dataoffset

                        for (int i = 0; i != size; ++i)
                        {
                            reader.Deserialize_Array(teStructuredData.Manager, assetFile, fieldInfo, array, i);
                        }
                    }
                }
                else
                {
                    long offset = data.ReadInt32();
                    data.ReadInt32(); // :kyaah:

                    long position = data.Position();
                    if (offset <= 0)
                    {
                        array = null;
                    }
                    else
                    {
                        data.BaseStream.Position = offset + assetFile.StartPos;

                        long count      = data.ReadInt64();
                        long dataOffset = data.ReadInt64();

                        if (count != -1 && dataOffset > 0)
                        {
                            array = Array.CreateInstance(elementType, count);

                            data.BaseStream.Position = dataOffset + assetFile.StartPos;

                            for (int i = 0; i != count; ++i)
                            {
                                reader.Deserialize_Array(teStructuredData.Manager, assetFile, fieldInfo, array, i);
                            }
                        }
                        else
                        {
                            array = null;
                        }
                    }
                    data.BaseStream.Position = position + 8;
                }
                field.Key.SetValue(this, array);
            }
            else
            {
                reader.Deserialize(teStructuredData.Manager, assetFile, fieldInfo, this, field.Key);
            }
        }
Пример #4
0
        /// <summary>Deserialize instance</summary>
        public void Deserialize(teStructuredData assetFile, STUField_Info field = null)
        {
            BinaryReader data = assetFile.Data;

            uint         instanceHash = teStructuredData.Manager.InstancesInverted[GetType()];
            STUAttribute stuAttribute = teStructuredData.Manager.InstanceAttributes[instanceHash];
            Dictionary <uint, KeyValuePair <FieldInfo, STUFieldAttribute> > fields = teStructuredData.Manager.FieldAttributes[instanceHash];

            if (assetFile.Format == teStructuredDataFormat.V2)
            {
                int fieldBagIdx = data.ReadInt32();
                if (fieldBagIdx == -1)
                {
                    return;
                }
                if (fieldBagIdx >= assetFile.FieldInfoBags.Count)
                {
                    throw new ArgumentOutOfRangeException($"FieldBag index is out of range. Id: {fieldBagIdx}, Type: '{GetType().Name}', Data offset: {data.BaseStream.Position - 4}");
                }
                foreach (STUField_Info stuField in assetFile.FieldInfoBags[fieldBagIdx])
                {
                    int fieldSize = stuField.Size;
                    if (fieldSize == 0)
                    {
                        fieldSize = data.ReadInt32();
                    }
                    long startPosition = data.BaseStream.Position;

                    DeserializeField(assetFile, stuField, fields, stuAttribute);

                    data.BaseStream.Position = startPosition + fieldSize;

                    //long endPosition = data.BaseStream.Position;
                    //if (endPosition - startPosition != fieldSize)
                    //    throw new InvalidDataException($"read len != field size. Type: '{GetType().Name}', Field: {stuField.Hash:X8}, Data offset: {startPosition}");
                }
            }
            else if (assetFile.Format == teStructuredDataFormat.V1)
            {
                //if (instanceHash == 0xEA30C5E9 || instanceHash == 0x298FC27F || instanceHash == 0x75526BC2 ||
                //    instanceHash == 0x5C7059E || instanceHash == 0x646FF8C3 || instanceHash == 0xAD5967B3 ||
                //    instanceHash == 0x74F13350) {
                //    Debugger.Log(0, "STUInstnace", "HACK: SKIPPING DESERIALIZE\r\n");
                //    return;
                //    //Debugger.Break();
                //}

                long instanceStartPos = assetFile.Data.BaseStream.Position - m_posInDataAtDeserializeCall;

                uint[] fieldOrder = teStructuredData.Manager.InstanceFields[instanceHash];
                foreach (uint fieldHash in fieldOrder)
                {
                    var fieldDefinition = fields[fieldHash];
                    if (fieldDefinition.Value.m_offset < m_posInDataAtDeserializeCall)
                    {
                        throw new InvalidDataException(
                                  $"Internal STU parser error. Field offset ({fieldDefinition.Value.m_offset}) is located " +
                                  "in data that was read before the instance deserialize method was called. (e.g field is " +
                                  "at 0, called at 8 due to crc+offset thing)");
                    }

                    // go to correct field pos. e.g vecs are aligned to 16 byte so not always exactly sequential
                    assetFile.Data.BaseStream.Position = instanceStartPos + fieldDefinition.Value.m_offset;

                    // todo: how does alignment work on arrays
                    // e.g will an array of vec3a be aligned by the compiler or do we need to process that

                    STUField_Info stuField = new STUField_Info {
                        Hash = fieldHash, Size = -1
                    };
                    DeserializeField(assetFile, stuField, fields, stuAttribute);
                }

                // make sure we are at the correct end of data including alignment padding etc
                assetFile.Data.BaseStream.Position = instanceStartPos + stuAttribute.m_size;
            }
        }