/// <summary> /// Scans the specified type's non-static fields. /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan the non-static fields of.</param> /// <param name="FieldTablesBlock">The ASM block for the fields table for the library currently being compiled.</param> private static void ScanFields(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock FieldTablesBlock) { string currentTypeId = TheTypeInfo.ID; string currentTypeName = TheTypeInfo.UnderlyingType.FullName; List <Tuple <string, string, string> > AllFieldInfo = new List <Tuple <string, string, string> >(); if (TheTypeInfo.UnderlyingType.BaseType == null || (TheTypeInfo.UnderlyingType.BaseType.FullName != "System.Array" && TheTypeInfo.UnderlyingType.BaseType.FullName != "System.MulticastDelegate")) { foreach (Types.FieldInfo anOwnField in TheTypeInfo.FieldInfos) { if (!anOwnField.IsStatic) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(anOwnField.FieldType); string fieldOffsetVal = anOwnField.OffsetInBytes.ToString(); string fieldSizeVal = (FieldTypeInfo.IsValueType ? FieldTypeInfo.SizeOnHeapInBytes : FieldTypeInfo.SizeOnStackInBytes).ToString(); string fieldTypeIdVal = FieldTypeInfo.ID; FieldTablesBlock.AddExternalLabel(fieldTypeIdVal); AllFieldInfo.Add(new Tuple <string, string, string>(fieldOffsetVal, fieldSizeVal, fieldTypeIdVal)); } } } string parentTypeFieldTablePtr = "0"; bool parentPtrIsExternal = false; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); parentPtrIsExternal = (ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo); parentTypeFieldTablePtr = baseTypeInfo.ID + "_FieldTable"; } } { string fieldOffsetVal = "0"; string fieldSizeVal = "0"; string fieldTypeIdVal = parentTypeFieldTablePtr; if (parentPtrIsExternal) { FieldTablesBlock.AddExternalLabel(fieldTypeIdVal); } AllFieldInfo.Add(new Tuple <string, string, string>(fieldOffsetVal, fieldSizeVal, fieldTypeIdVal)); } List <Tuple <string, int> > TableEntryFieldInfos = GetSpecialClassFieldInfo(TheLibrary, typeof(Attributes.FieldInfoStructAttribute)); ASM.ASMOp newFieldTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.FieldTable, currentTypeId, currentTypeName, AllFieldInfo, TableEntryFieldInfos); FieldTablesBlock.Append(newFieldTableOp); }
/// <summary> /// Scans the specified type (excludes fields and methods). /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan.</param> /// <param name="TypesTableBlock">The ASM block for the types table for the library currently being compiled.</param> private static void ScanType(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock TypesTableBlock) { string TypeId = TheTypeInfo.ID; string SizeVal = TheTypeInfo.SizeOnHeapInBytes.ToString(); string IdVal = (TypesScanned++).ToString(); string StackSizeVal = TheTypeInfo.SizeOnStackInBytes.ToString(); string IsValueTypeVal = (TheTypeInfo.IsValueType ? "1" : "0"); string MethodTablePointer = TypeId + "_MethodTable"; string IsPointerTypeVal = (TheTypeInfo.IsPointer ? "1" : "0"); string BaseTypeIdVal = "0"; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); BaseTypeIdVal = baseTypeInfo.ID; //Declared external to this library, so won't appear in this library's type tables if ((ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo)) { TypesTableBlock.AddExternalLabel(BaseTypeIdVal); } } } string FieldTablePointer = TypeId + "_FieldTable"; string TypeSignatureLiteralLabel = TheLibrary.AddStringLiteral(TheTypeInfo.UnderlyingType.FullName); // Legacy string TypeIdLiteralLabel = TheLibrary.AddStringLiteral(TheTypeInfo.ID); Types.TypeInfo typeTypeInfo = ILLibrary.SpecialClasses[typeof(Attributes.TypeClassAttribute)].First(); List <Types.FieldInfo> OrderedFields = typeTypeInfo.FieldInfos.Where(x => !x.IsStatic).OrderBy(x => x.OffsetInBytes).ToList(); List <Tuple <string, Types.TypeInfo> > FieldInformation = new List <Tuple <string, Types.TypeInfo> >(); foreach (Types.FieldInfo aTypeField in OrderedFields) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(aTypeField.FieldType); FieldInformation.Add(new Tuple <string, Types.TypeInfo>(aTypeField.Name, FieldTypeInfo)); } ASM.ASMOp newTypeTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.TypeTable, TypeId, SizeVal, IdVal, StackSizeVal, IsValueTypeVal, MethodTablePointer, IsPointerTypeVal, BaseTypeIdVal, FieldTablePointer, TypeSignatureLiteralLabel, TypeIdLiteralLabel, FieldInformation); TypesTableBlock.Append(newTypeTableOp); TypesTableBlock.AddExternalLabel(MethodTablePointer); TypesTableBlock.AddExternalLabel(FieldTablePointer); TypesTableBlock.AddExternalLabel(TypeSignatureLiteralLabel); TypesTableBlock.AddExternalLabel(TypeIdLiteralLabel); }
/// <summary> /// Scans the specified type's methods. /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan the methods of.</param> /// <param name="MethodTablesBlock">The ASM block for the methods table for the library currently being compiled.</param> private static void ScanMethods(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock MethodTablesBlock) { string currentTypeId = TheTypeInfo.ID; string currentTypeName = TheTypeInfo.UnderlyingType.FullName; List <Tuple <string, string> > AllMethodInfo = new List <Tuple <string, string> >(); if (TheTypeInfo.UnderlyingType.BaseType == null || TheTypeInfo.UnderlyingType.BaseType.FullName != "System.Array") { foreach (Types.MethodInfo aMethodInfo in TheTypeInfo.MethodInfos) { if (!aMethodInfo.IsStatic && !aMethodInfo.UnderlyingInfo.IsAbstract) { string methodID = aMethodInfo.ID; string methodIDValue = aMethodInfo.IDValue.ToString(); MethodTablesBlock.AddExternalLabel(methodID); AllMethodInfo.Add(new Tuple <string, string>(methodID, methodIDValue)); } } } string parentTypeMethodTablePtr = "0"; bool parentPtrIsExternal = false; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); parentPtrIsExternal = (ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo); parentTypeMethodTablePtr = baseTypeInfo.ID + "_MethodTable"; } } { string methodID = parentTypeMethodTablePtr; string methodIDValue = "0"; if (parentPtrIsExternal) { MethodTablesBlock.AddExternalLabel(methodID); } AllMethodInfo.Add(new Tuple <string, string>(methodID, methodIDValue)); } List <Tuple <string, int> > TableEntryFieldInfos = GetSpecialClassFieldInfo(TheLibrary, typeof(Attributes.MethodInfoStructAttribute)); ASM.ASMOp newMethodTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.MethodTable, currentTypeId, currentTypeName, AllMethodInfo, TableEntryFieldInfos); MethodTablesBlock.Append(newMethodTableOp); }
private static List <Tuple <string, int> > GetSpecialClassFieldInfo(ILLibrary TheLibrary, Type SpecialClassType) { Types.TypeInfo InformationAboutInfoStruct = ILLibrary.SpecialClasses[SpecialClassType].First(); List <Types.FieldInfo> InfoStruct_OrderedFields = InformationAboutInfoStruct.FieldInfos.Where(x => !x.IsStatic).OrderBy(x => x.OffsetInBytes).ToList(); List <Tuple <string, int> > InfoStruct_OrderedFieldInfo_Subset = new List <Tuple <string, int> >(); foreach (Types.FieldInfo aField in InfoStruct_OrderedFields) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(aField.FieldType); InfoStruct_OrderedFieldInfo_Subset.Add(new Tuple <string, int>(aField.Name, FieldTypeInfo.IsValueType ? FieldTypeInfo.SizeOnHeapInBytes : FieldTypeInfo.SizeOnStackInBytes)); } return(InfoStruct_OrderedFieldInfo_Subset); }
/// <summary> /// Scans the specified type's static fields. /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan the static fields of.</param> /// <param name="StaticFieldsBlock">The ASM block for the static fields for the library currently being compiled.</param> private static void ScanStaticFields(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock StaticFieldsBlock) { foreach (Types.FieldInfo aFieldInfo in TheTypeInfo.FieldInfos) { if (aFieldInfo.IsStatic) { Types.TypeInfo fieldTypeInfo = TheLibrary.GetTypeInfo(aFieldInfo.FieldType); string FieldID = aFieldInfo.ID; string Size = fieldTypeInfo.SizeOnStackInBytes.ToString(); ASM.ASMOp newStaticFieldOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.StaticField, FieldID, Size); StaticFieldsBlock.Append(newStaticFieldOp); } } }
/// <summary> /// Gets the next unique ID for a method of the specified type. /// </summary> /// <remarks> /// Used for generatign IDs to go in the method tables for use in virtual calls (callvirt Il ops). /// </remarks> /// <param name="TheLibrary">The IL library being compiled.</param> /// <param name="aType">The type to get the next method ID from.</param> /// <returns>The next unique method ID.</returns> private static int GetMethodIDGenerator(ILLibrary TheLibrary, Type aType) { Types.TypeInfo aTypeInfo = TheLibrary.GetTypeInfo(aType); return GetMethodIDGenerator(TheLibrary, aTypeInfo); }
/// <summary> /// Injects the garbage collector related IL ops into the specified method. /// </summary> /// <param name="TheLibrary">The library being compiled.</param> /// <param name="theMethodInfo">The method to inject ops into.</param> /// <param name="theILBlock">The IL block for the method to inject ops into.</param> private static void InjectGC(ILLibrary TheLibrary, Types.MethodInfo theMethodInfo, ILBlock theILBlock) { if (theMethodInfo.ApplyGC) { // Find the index of the MethodStart op int MethodStartOpPos = theILBlock.PositionOf(theILBlock.ILOps.Where(x => x is ILOps.MethodStart).First()); // Inject ops for incrementing ref. count of args at start of method int InjectIncArgsRefCountPos = MethodStartOpPos + 1; foreach (Types.VariableInfo argInfo in theMethodInfo.ArgumentInfos) { if (argInfo.TheTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(InjectIncArgsRefCountPos, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldarg, ValueBytes = BitConverter.GetBytes(argInfo.Position) }); theILBlock.ILOps.Insert(InjectIncArgsRefCountPos + 1, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.IncrementRefCountMethodAttribute)].First().UnderlyingInfo }); } } // The following two things can be done within the same loop: // Inject ops for inc./dec. ref. counts of objects when written to: // - Arguments / Locals // - Fields / Static Fields // - Elements of Arrays // Add Cleanup Block and Inject finally-block ops for it // - Also remember the op for storing return value (if any) ILPreprocessState preprocessState = new ILPreprocessState() { TheILLibrary = TheLibrary, Input = theILBlock, CurrentStackFrame = new StackFrame() }; ExceptionHandledBlock CleanupExBlock = new ExceptionHandledBlock(); for (int opIndx = 0; opIndx < theILBlock.ILOps.Count; opIndx++) { ILOp currOp = theILBlock.ILOps[opIndx]; bool IncRefCount = false; int incOpIndexBy = 0; switch ((ILOp.OpCodes)currOp.opCode.Value) { case ILOp.OpCodes.Stsfld: #region Stsfld { int metadataToken = Utilities.ReadInt32(currOp.ValueBytes, 0); System.Reflection.FieldInfo theField = theMethodInfo.UnderlyingInfo.Module.ResolveField(metadataToken); Types.TypeInfo theFieldTypeInfo = TheLibrary.GetTypeInfo(theField.FieldType); if (theFieldTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(opIndx, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldsfld, ValueBytes = currOp.ValueBytes }); theILBlock.ILOps.Insert(opIndx + 1, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); IncRefCount = true; incOpIndexBy = 2; } } #endregion break; case ILOp.OpCodes.Stloc: case ILOp.OpCodes.Stloc_0: case ILOp.OpCodes.Stloc_1: case ILOp.OpCodes.Stloc_2: case ILOp.OpCodes.Stloc_3: case ILOp.OpCodes.Stloc_S: #region Stloc { UInt16 localIndex = 0; switch ((ILOp.OpCodes)currOp.opCode.Value) { case ILOp.OpCodes.Stloc: localIndex = (UInt16)Utilities.ReadInt16(currOp.ValueBytes, 0); break; case ILOp.OpCodes.Stloc_0: localIndex = 0; break; case ILOp.OpCodes.Stloc_1: localIndex = 1; break; case ILOp.OpCodes.Stloc_2: localIndex = 2; break; case ILOp.OpCodes.Stloc_3: localIndex = 3; break; case ILOp.OpCodes.Stloc_S: localIndex = (UInt16)currOp.ValueBytes[0]; break; } Types.TypeInfo LocalTypeInfo = theMethodInfo.LocalInfos[localIndex].TheTypeInfo; if (LocalTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(opIndx, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldloc, ValueBytes = BitConverter.GetBytes(localIndex) }); theILBlock.ILOps.Insert(opIndx + 1, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); IncRefCount = true; incOpIndexBy = 2; } } #endregion break; case ILOp.OpCodes.Stfld: #region Stfld { int metadataToken = Utilities.ReadInt32(currOp.ValueBytes, 0); System.Reflection.FieldInfo theField = theMethodInfo.UnderlyingInfo.Module.ResolveField(metadataToken); Types.TypeInfo theFieldTypeInfo = TheLibrary.GetTypeInfo(theField.FieldType); if (theFieldTypeInfo.IsGCManaged) { // Items on stack: // - Object reference // - (New) Value to store // // We want to load the current value of the field // for which we must duplicate the object ref // But first, we must remove the (new) value to store // off the stack, while also storing it to put back // on the stack after so the store can continue // // So: // 1. Switch value to store and object ref on stack // 3. Duplicate the object ref // 4. Load the field value // 5. Call dec ref count // 6. Switch value to store and object ref back again //USE A SWITCH STACK ITEMS OP!! theILBlock.ILOps.Insert(opIndx, new ILOps.StackSwitch() { StackSwitch_Items = 2 }); theILBlock.ILOps.Insert(opIndx + 1, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Dup }); theILBlock.ILOps.Insert(opIndx + 2, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldfld, ValueBytes = currOp.ValueBytes }); theILBlock.ILOps.Insert(opIndx + 3, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); theILBlock.ILOps.Insert(opIndx + 4, new ILOps.StackSwitch() { StackSwitch_Items = 2 }); IncRefCount = true; incOpIndexBy = 5; } } #endregion break; case ILOp.OpCodes.Stelem: case ILOp.OpCodes.Stelem_Ref: #region Stelem / Stelem_Ref { bool doDecrement = false; bool isRefOp = false; if ((ILOp.OpCodes)currOp.opCode.Value == ILOp.OpCodes.Stelem_Ref) { doDecrement = preprocessState.CurrentStackFrame.Stack.Peek().isGCManaged; isRefOp = true; } else { int metadataToken = Utilities.ReadInt32(currOp.ValueBytes, 0); Type elementType = theMethodInfo.UnderlyingInfo.Module.ResolveType(metadataToken); doDecrement = TheLibrary.GetTypeInfo(elementType).IsGCManaged; } if (doDecrement) { // Items on stack: // - Array reference // - Index // - (New) Value to store // // We want to load the current value of the element at Index in the array // for which we must duplicate the array ref and index // But first, we must remove the (new) value to store // off the stack, while also storing it to put back // on the stack after so the store can continue // // So: // 1. Switch (rotate) 1 times the top 3 values so that index is on top // 2. Duplicate the index // 3. Switch (rotate) 2 times the top 4 values so that array ref is on top // 4. Duplicate the array ref // 5. Switch (rotate) 4 times the top 5 values so that duplicate array ref and index are on top // 6. Do LdElem op to load existing element value // 7. Call GC.DecrementRefCount // 8. Switch (rotate) 1 times the top 3 values so that the stack is in its original state // (9. Continue to increment ref count as normal) // // The following is a diagram of the stack manipulation occurring here: // Key: A=Array ref, I=Index, V=Value to store, E=Loaded element // Top-most stack item appears last // // 1) Rotate x 1 2) Duplicate 3) Rot x 2 4) Dup // A,I,V ---------> V,A,I ---------> V,A,I,I ---------> I,I,V,A ---------> I,I,V,A,A // // // 5) Rot x 4 6) Ldelem 7) Call GC (Dec) // I,I,V,A,A ---------> I,V,A,A,I ---------> I,V,A,E ---------> I,V,A // // // 8) Rot x 1 9) Dup 10) Call GC (Inc) // I,V,A ---------> A,I,V ---------> A,I,V,V ---------> A,I,V #region 1. theILBlock.ILOps.Insert(opIndx, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(3), StackSwitch_Items = 3 }); incOpIndexBy++; #endregion #region 2. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Dup }); incOpIndexBy++; #endregion #region 3. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(4), StackSwitch_Items = 4 }); incOpIndexBy++; theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(4), StackSwitch_Items = 4 }); incOpIndexBy++; #endregion #region 4. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Dup }); incOpIndexBy++; #endregion #region 5. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(5), StackSwitch_Items = 5 }); incOpIndexBy++; theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(5), StackSwitch_Items = 5 }); incOpIndexBy++; theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(5), StackSwitch_Items = 5 }); incOpIndexBy++; theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(5), StackSwitch_Items = 5 }); incOpIndexBy++; #endregion #region 6. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = isRefOp ? System.Reflection.Emit.OpCodes.Ldelem_Ref : System.Reflection.Emit.OpCodes.Ldelem, ValueBytes = currOp.ValueBytes }); incOpIndexBy++; #endregion #region 7. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); incOpIndexBy++; #endregion #region 8. theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOps.StackSwitch() { ValueBytes = BitConverter.GetBytes(3), StackSwitch_Items = 3 }); incOpIndexBy++; #endregion IncRefCount = true; } } #endregion break; case ILOp.OpCodes.Starg: case ILOp.OpCodes.Starg_S: #region Starg { UInt16 index = (ILOp.OpCodes)currOp.opCode.Value == ILOp.OpCodes.Starg_S ? (UInt16)currOp.ValueBytes[0] : (UInt16)Utilities.ReadInt16(currOp.ValueBytes, 0); if (theMethodInfo.ArgumentInfos[index].TheTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(opIndx, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldarg, ValueBytes = BitConverter.GetBytes(index) }); theILBlock.ILOps.Insert(opIndx + 1, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); IncRefCount = true; incOpIndexBy = 2; } } #endregion break; } if (IncRefCount && !preprocessState.CurrentStackFrame.Stack.Peek().isNewGCObject) { theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Dup }); incOpIndexBy++; theILBlock.ILOps.Insert(opIndx + incOpIndexBy, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.IncrementRefCountMethodAttribute)].First().UnderlyingInfo }); incOpIndexBy++; } // If op changed if (theILBlock.ILOps[opIndx] != currOp) { theILBlock.ILOps[opIndx].Offset = currOp.Offset; theILBlock.ILOps[opIndx].BytesSize = currOp.BytesSize; } // <= is correct. E.g. if 1 extra op added, incOpIndex=1 so <= results in currOp processed // + extra op processed bool UseNextOpAsCleanupStart = false; for (int i = 0; i <= incOpIndexBy; i++) { currOp = theILBlock.ILOps[opIndx]; if (UseNextOpAsCleanupStart) { UseNextOpAsCleanupStart = false; CleanupExBlock.Offset = currOp.Offset; } if (currOp is ILOps.MethodStart) { TargetArchitecture.MethodStartOp.PerformStackOperations(preprocessState, theILBlock.ILOps[opIndx]); UseNextOpAsCleanupStart = true; } else if (currOp is ILOps.MethodEnd) { TargetArchitecture.MethodEndOp.PerformStackOperations(preprocessState, theILBlock.ILOps[opIndx]); } else if (currOp is ILOps.StackSwitch) { TargetArchitecture.StackSwitchOp.PerformStackOperations(preprocessState, theILBlock.ILOps[opIndx]); } else { // Leave unsupported ops for the IL Scanner to deal with (or later code e.g. castclass op) if (TargetArchitecture.TargetILOps.ContainsKey((ILOp.OpCodes)currOp.opCode.Value)) { ILOp ConverterOp = TargetArchitecture.TargetILOps[(ILOp.OpCodes)currOp.opCode.Value]; ConverterOp.PerformStackOperations(preprocessState, currOp); } } if (i > 0) { opIndx++; } } } if (theMethodInfo.ArgumentInfos.Count > 0 || theMethodInfo.LocalInfos.Count > 0) { bool AddCleanupBlock = false; foreach (Types.VariableInfo anArgInfo in theMethodInfo.ArgumentInfos) { if (anArgInfo.TheTypeInfo.IsGCManaged) { AddCleanupBlock = true; break; } } if (!AddCleanupBlock) { foreach (Types.VariableInfo aLocInfo in theMethodInfo.LocalInfos) { if (aLocInfo.TheTypeInfo.IsGCManaged) { AddCleanupBlock = true; break; } } } if (AddCleanupBlock) { ILOp lastOp = theILBlock.ILOps.Last(); int lastOpOffset = lastOp.Offset; int lastOpIndex = theILBlock.ILOps.Count - 1; bool MethodHasReturnValue = false; // If there is a return value, we will need to temp store it if (theMethodInfo.UnderlyingInfo is System.Reflection.MethodInfo) { Type returnType = ((System.Reflection.MethodInfo)theMethodInfo.UnderlyingInfo).ReturnType; //Void return type = no return value if (returnType != typeof(void)) { // Add local variable for storing return value int lastLocalOffset = theMethodInfo.LocalInfos.Count > 0 ? theMethodInfo.LocalInfos.Last().Offset : 0; int lastLocalSize = theMethodInfo.LocalInfos.Count > 0 ? theMethodInfo.LocalInfos.Last().TheTypeInfo.SizeOnStackInBytes : 0; theMethodInfo.LocalInfos.Add(new Types.VariableInfo() { UnderlyingType = returnType, TheTypeInfo = TheLibrary.GetTypeInfo(returnType), Position = theMethodInfo.LocalInfos.Count, Offset = lastLocalOffset + lastLocalSize }); // Add op for storing return value, update op offsets theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Stloc, Offset = lastOpOffset, BytesSize = lastOp.BytesSize, ValueBytes = BitConverter.GetBytes(theMethodInfo.LocalInfos.Count - 1) }); lastOpIndex++; MethodHasReturnValue = true; } } ILOp leaveOp; // Add the Leave op of the try-block theILBlock.ILOps.Insert(lastOpIndex, leaveOp = new ILOp() { opCode = System.Reflection.Emit.OpCodes.Leave, Offset = lastOpOffset, BytesSize = lastOp.BytesSize, ValueBytes = BitConverter.GetBytes(0) }); lastOpIndex++; FinallyBlock CleanupFinallyBlock = new FinallyBlock() { Offset = lastOpOffset + lastOp.BytesSize, Length = 0 }; CleanupExBlock.Length = lastOpOffset - CleanupExBlock.Offset; CleanupExBlock.FinallyBlocks.Add(CleanupFinallyBlock); int cleanupOpsOffset = lastOpOffset + 1; // Add cleanup code for local variables (including the return value local) foreach (Types.VariableInfo aLocInfo in theMethodInfo.LocalInfos) { if (aLocInfo.TheTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldloc, Offset = cleanupOpsOffset, BytesSize = 1, ValueBytes = BitConverter.GetBytes(aLocInfo.Position) }); cleanupOpsOffset++; lastOpIndex++; theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, Offset = cleanupOpsOffset, BytesSize = 1, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); cleanupOpsOffset++; lastOpIndex++; CleanupFinallyBlock.Length += 2; } } // Add cleanup code for arguments foreach (Types.VariableInfo anArgInfo in theMethodInfo.ArgumentInfos) { if (anArgInfo.TheTypeInfo.IsGCManaged) { theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldarg, Offset = cleanupOpsOffset, BytesSize = 1, ValueBytes = BitConverter.GetBytes(anArgInfo.Position) }); cleanupOpsOffset++; lastOpIndex++; theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Call, Offset = cleanupOpsOffset, BytesSize = 1, MethodToCall = ILLibrary.SpecialMethods[typeof(Attributes.DecrementRefCountMethodAttribute)].First().UnderlyingInfo }); cleanupOpsOffset++; lastOpIndex++; CleanupFinallyBlock.Length += 2; } } // Add end finally op theILBlock.ILOps.Insert(lastOpIndex, leaveOp.LoadAtILOpAfterOp = new ILOp() { opCode = System.Reflection.Emit.OpCodes.Endfinally, Offset = cleanupOpsOffset, BytesSize = 1 }); cleanupOpsOffset++; lastOpIndex++; CleanupFinallyBlock.Length++; // Add restore return value op if (MethodHasReturnValue) { theILBlock.ILOps.Insert(lastOpIndex, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Ldloc, Offset = cleanupOpsOffset, BytesSize = 1, ValueBytes = BitConverter.GetBytes(theMethodInfo.LocalInfos.Count - 1) }); cleanupOpsOffset++; lastOpIndex++; } // Add ex block to the method theILBlock.ExceptionHandledBlocks.Add(CleanupExBlock); // Replace any Ret ops contained within Cleanup Block with: // - Op to store return value (if any) // - Leave op bool Inside = false; for(int opIndx = 0; opIndx < theILBlock.ILOps.Count; opIndx++) { if (theILBlock.ILOps[opIndx].Offset > CleanupExBlock.Offset + CleanupExBlock.Length) { break; } else if (theILBlock.ILOps[opIndx].Offset >= CleanupExBlock.Offset) { Inside = true; } if (Inside && (ILOp.OpCodes)theILBlock.ILOps[opIndx].opCode.Value == ILOp.OpCodes.Ret) { ILOp ARetOp = theILBlock.ILOps[opIndx]; theILBlock.ILOps.RemoveAt(opIndx); if (MethodHasReturnValue) { theILBlock.ILOps.Insert(opIndx, new ILOp() { opCode = System.Reflection.Emit.OpCodes.Stloc, Offset = ARetOp.Offset, ValueBytes = BitConverter.GetBytes(theMethodInfo.LocalInfos.Count - 1) }); theILBlock.ILOps.Insert(opIndx + 1, new ILOp() { Offset = ARetOp.Offset, opCode = System.Reflection.Emit.OpCodes.Leave, ValueBytes = BitConverter.GetBytes(0), LoadAtILOpAfterOp = leaveOp.LoadAtILOpAfterOp }); } else { theILBlock.ILOps.Insert(opIndx, new ILOp() { Offset = ARetOp.Offset, opCode = System.Reflection.Emit.OpCodes.Leave, ValueBytes = BitConverter.GetBytes(0), LoadAtILOpAfterOp = leaveOp.LoadAtILOpAfterOp }); } } } } } } }
/// <summary> /// Gets the next unique ID for a method of the specified type. /// </summary> /// <remarks> /// Used for generatign IDs to go in the method tables for use in virtual calls (callvirt Il ops). /// </remarks> /// <param name="TheLibrary">The IL library being compiled.</param> /// <param name="aType">The type to get the next method ID from.</param> /// <returns>The next unique method ID.</returns> private static int GetMethodIDGenerator(ILLibrary TheLibrary, Type aType, bool useMethodCount = false) { Types.TypeInfo aTypeInfo = TheLibrary.GetTypeInfo(aType); return GetMethodIDGenerator(TheLibrary, aTypeInfo, useMethodCount); }
/// <summary> /// Preprocesses the specified method. /// </summary> /// <param name="TheLibrary">The library being compiled.</param> /// <param name="theMethodInfo">The method to preprocess.</param> private static void PreprocessMethodInfo(ILLibrary TheLibrary, Types.MethodInfo theMethodInfo) { if (theMethodInfo.Preprocessed) { return; } theMethodInfo.Preprocessed = true; string sig = theMethodInfo.Signature; bool SetMethodID = true; if (!theMethodInfo.IsConstructor) { System.Reflection.MethodInfo methodInf = (System.Reflection.MethodInfo)theMethodInfo.UnderlyingInfo; if (methodInf.GetBaseDefinition() != methodInf) { Types.MethodInfo baseMethodInfo = TheLibrary.GetMethodInfo(methodInf.GetBaseDefinition()); PreprocessMethodInfo(TheLibrary, baseMethodInfo); theMethodInfo.IDValue = baseMethodInfo.IDValue; SetMethodID = false; } } if (SetMethodID) { Types.TypeInfo declarerTypeInfo = TheLibrary.GetTypeInfo(theMethodInfo.UnderlyingInfo.DeclaringType); int ID = GetMethodIDGenerator(TheLibrary, declarerTypeInfo); theMethodInfo.IDValue = ID + 1; declarerTypeInfo.MethodIDGenerator++; } int totalLocalsOffset = 0; foreach (Types.VariableInfo aVarInfo in theMethodInfo.LocalInfos) { //Causes processing of the type - in case it hasn't already been processed Types.TypeInfo aTypeInfo = TheLibrary.GetTypeInfo(aVarInfo.UnderlyingType); aVarInfo.TheTypeInfo = aTypeInfo; aVarInfo.Offset = totalLocalsOffset; totalLocalsOffset += aTypeInfo.SizeOnStackInBytes; } int totalArgsSize = 0; if (!theMethodInfo.IsStatic) { Types.VariableInfo newVarInfo = new Types.VariableInfo() { UnderlyingType = theMethodInfo.UnderlyingInfo.DeclaringType, Position = 0, TheTypeInfo = TheLibrary.GetTypeInfo(theMethodInfo.UnderlyingInfo.DeclaringType) }; theMethodInfo.ArgumentInfos.Add(newVarInfo); totalArgsSize += newVarInfo.TheTypeInfo.SizeOnStackInBytes; } System.Reflection.ParameterInfo[] args = theMethodInfo.UnderlyingInfo.GetParameters(); foreach (System.Reflection.ParameterInfo argItem in args) { Types.VariableInfo newVarInfo = new Types.VariableInfo() { UnderlyingType = argItem.ParameterType, Position = theMethodInfo.ArgumentInfos.Count, TheTypeInfo = TheLibrary.GetTypeInfo(argItem.ParameterType) }; theMethodInfo.ArgumentInfos.Add(newVarInfo); totalArgsSize += newVarInfo.TheTypeInfo.SizeOnStackInBytes; } //System.Reflection.ParameterInfo returnArgItem = (theMethodInfo.IsConstructor ? null : ((System.Reflection.MethodInfo)theMethodInfo.UnderlyingInfo).ReturnParameter); //if (returnArgItem != null) //{ // Types.VariableInfo newVarInfo = new Types.VariableInfo() // { // UnderlyingType = returnArgItem.ParameterType, // Position = theMethodInfo.ArgumentInfos.Count, // TheTypeInfo = TheLibrary.GetTypeInfo(returnArgItem.ParameterType) // }; // theMethodInfo.ArgumentInfos.Add(newVarInfo); // totalArgsSize += newVarInfo.TheTypeInfo.SizeOnStackInBytes; //} int offset = totalArgsSize; for (int i = 0; i < theMethodInfo.ArgumentInfos.Count; i++) { offset -= theMethodInfo.ArgumentInfos[i].TheTypeInfo.SizeOnStackInBytes; theMethodInfo.ArgumentInfos[i].Offset = offset; } }
private static List<Tuple<string, int>> GetSpecialClassFieldInfo(ILLibrary TheLibrary, Type SpecialClassType) { Types.TypeInfo InformationAboutInfoStruct = ILLibrary.SpecialClasses[SpecialClassType].First(); List<Types.FieldInfo> InfoStruct_OrderedFields = InformationAboutInfoStruct.FieldInfos.Where(x => !x.IsStatic).OrderBy(x => x.OffsetInBytes).ToList(); List<Tuple<string, int>> InfoStruct_OrderedFieldInfo_Subset = new List<Tuple<string, int>>(); foreach (Types.FieldInfo aField in InfoStruct_OrderedFields) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(aField.FieldType); InfoStruct_OrderedFieldInfo_Subset.Add(new Tuple<string, int>(aField.Name, FieldTypeInfo.IsValueType ? FieldTypeInfo.SizeOnHeapInBytes : FieldTypeInfo.SizeOnStackInBytes)); } return InfoStruct_OrderedFieldInfo_Subset; }
/// <summary> /// Gets the field info by name for the specified field of the specified type. /// </summary> /// <param name="aType">The type to which the field belongs.</param> /// <param name="FieldName">The name of the field to get.</param> /// <returns>The field information.</returns> public Types.FieldInfo GetFieldInfo(Type aType, string FieldName) { Types.TypeInfo aTypeInfo = TheILLibrary.GetTypeInfo(aType); return(TheILLibrary.GetFieldInfo(aTypeInfo, FieldName)); }
/// <summary> /// Scans the specified type's non-static fields. /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan the non-static fields of.</param> /// <param name="FieldTablesBlock">The ASM block for the fields table for the library currently being compiled.</param> private static void ScanFields(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock FieldTablesBlock) { string currentTypeId = TheTypeInfo.ID; string currentTypeName = TheTypeInfo.UnderlyingType.FullName; List<Tuple<string, string, string>> AllFieldInfo = new List<Tuple<string, string, string>>(); if (TheTypeInfo.UnderlyingType.BaseType == null || (TheTypeInfo.UnderlyingType.BaseType.FullName != "System.Array" && TheTypeInfo.UnderlyingType.BaseType.FullName != "System.MulticastDelegate")) { foreach (Types.FieldInfo anOwnField in TheTypeInfo.FieldInfos) { if (!anOwnField.IsStatic) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(anOwnField.FieldType); string fieldOffsetVal = anOwnField.OffsetInBytes.ToString(); string fieldSizeVal = (FieldTypeInfo.IsValueType ? FieldTypeInfo.SizeOnHeapInBytes : FieldTypeInfo.SizeOnStackInBytes).ToString(); string fieldTypeIdVal = FieldTypeInfo.ID; FieldTablesBlock.AddExternalLabel(fieldTypeIdVal); AllFieldInfo.Add(new Tuple<string, string, string>(fieldOffsetVal, fieldSizeVal, fieldTypeIdVal)); } } } string parentTypeFieldTablePtr = "0"; bool parentPtrIsExternal = false; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); parentPtrIsExternal = (ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo); parentTypeFieldTablePtr = baseTypeInfo.ID + "_FieldTable"; } } { string fieldOffsetVal = "0"; string fieldSizeVal = "0"; string fieldTypeIdVal = parentTypeFieldTablePtr; if (parentPtrIsExternal) { FieldTablesBlock.AddExternalLabel(fieldTypeIdVal); } AllFieldInfo.Add(new Tuple<string, string, string>(fieldOffsetVal, fieldSizeVal, fieldTypeIdVal)); } List<Tuple<string, int>> TableEntryFieldInfos = GetSpecialClassFieldInfo(TheLibrary, typeof(Attributes.FieldInfoStructAttribute)); ASM.ASMOp newFieldTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.FieldTable, currentTypeId, currentTypeName, AllFieldInfo, TableEntryFieldInfos); FieldTablesBlock.Append(newFieldTableOp); }
/// <summary> /// Scans the specified type's methods. /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan the methods of.</param> /// <param name="MethodTablesBlock">The ASM block for the methods table for the library currently being compiled.</param> private static void ScanMethods(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock MethodTablesBlock) { string currentTypeId = TheTypeInfo.ID; string currentTypeName = TheTypeInfo.UnderlyingType.FullName; List<Tuple<string, string>> AllMethodInfo = new List<Tuple<string, string>>(); if (TheTypeInfo.UnderlyingType.BaseType == null || TheTypeInfo.UnderlyingType.BaseType.FullName != "System.Array") { foreach (Types.MethodInfo aMethodInfo in TheTypeInfo.MethodInfos) { if (!aMethodInfo.IsStatic && !aMethodInfo.UnderlyingInfo.IsAbstract) { string methodID = aMethodInfo.ID; string methodIDValue = aMethodInfo.IDValue.ToString(); MethodTablesBlock.AddExternalLabel(methodID); AllMethodInfo.Add(new Tuple<string, string>(methodID, methodIDValue)); } } } string parentTypeMethodTablePtr = "0"; bool parentPtrIsExternal = false; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); parentPtrIsExternal = (ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo); parentTypeMethodTablePtr = baseTypeInfo.ID + "_MethodTable"; } } { string methodID = parentTypeMethodTablePtr; string methodIDValue = "0"; if (parentPtrIsExternal) { MethodTablesBlock.AddExternalLabel(methodID); } AllMethodInfo.Add(new Tuple<string,string>(methodID, methodIDValue)); } List<Tuple<string, int>> TableEntryFieldInfos = GetSpecialClassFieldInfo(TheLibrary, typeof(Attributes.MethodInfoStructAttribute)); ASM.ASMOp newMethodTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.MethodTable, currentTypeId, currentTypeName, AllMethodInfo, TableEntryFieldInfos); MethodTablesBlock.Append(newMethodTableOp); }
/// <summary> /// Scans the specified type (excludes fields and methods). /// </summary> /// <param name="TheLibrary">The library currently being compiled.</param> /// <param name="TheTypeInfo">The type to scan.</param> /// <param name="TypesTableBlock">The ASM block for the types table for the library currently being compiled.</param> private static void ScanType(ILLibrary TheLibrary, Types.TypeInfo TheTypeInfo, ASM.ASMBlock TypesTableBlock) { string TypeId = TheTypeInfo.ID; string SizeVal = TheTypeInfo.SizeOnHeapInBytes.ToString(); string IdVal = (TypesScanned++).ToString(); string StackSizeVal = TheTypeInfo.SizeOnStackInBytes.ToString(); string IsValueTypeVal = (TheTypeInfo.IsValueType ? "1" : "0"); string MethodTablePointer = TypeId + "_MethodTable"; string IsPointerTypeVal = (TheTypeInfo.IsPointer ? "1" : "0"); string BaseTypeIdVal = "0"; if (TheTypeInfo.UnderlyingType.BaseType != null) { if (!TheTypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Types.TypeInfo baseTypeInfo = TheLibrary.GetTypeInfo(TheTypeInfo.UnderlyingType.BaseType); BaseTypeIdVal = baseTypeInfo.ID; //Declared external to this library, so won't appear in this library's type tables if ((ScannedTypes.ContainsKey(baseTypeInfo.ID) && ScannedTypes[baseTypeInfo.ID] != TheLibrary) || !TheLibrary.TypeInfos.Contains(baseTypeInfo)) { TypesTableBlock.AddExternalLabel(BaseTypeIdVal); } } } string FieldTablePointer = TypeId + "_FieldTable"; string TypeSignatureLiteralLabel = TheLibrary.AddStringLiteral(TheTypeInfo.UnderlyingType.FullName); // Legacy string TypeIdLiteralLabel = TheLibrary.AddStringLiteral(TheTypeInfo.ID); Types.TypeInfo typeTypeInfo = ILLibrary.SpecialClasses[typeof(Attributes.TypeClassAttribute)].First(); List<Types.FieldInfo> OrderedFields = typeTypeInfo.FieldInfos.Where(x => !x.IsStatic).OrderBy(x => x.OffsetInBytes).ToList(); List<Tuple<string, Types.TypeInfo>> FieldInformation = new List<Tuple<string, Types.TypeInfo>>(); foreach (Types.FieldInfo aTypeField in OrderedFields) { Types.TypeInfo FieldTypeInfo = TheLibrary.GetTypeInfo(aTypeField.FieldType); FieldInformation.Add(new Tuple<string, Types.TypeInfo>(aTypeField.Name, FieldTypeInfo)); } ASM.ASMOp newTypeTableOp = TargetArchitecture.CreateASMOp(ASM.OpCodes.TypeTable, TypeId, SizeVal, IdVal, StackSizeVal, IsValueTypeVal, MethodTablePointer, IsPointerTypeVal, BaseTypeIdVal, FieldTablePointer, TypeSignatureLiteralLabel, TypeIdLiteralLabel, FieldInformation); TypesTableBlock.Append(newTypeTableOp); TypesTableBlock.AddExternalLabel(MethodTablePointer); TypesTableBlock.AddExternalLabel(FieldTablePointer); TypesTableBlock.AddExternalLabel(TypeSignatureLiteralLabel); TypesTableBlock.AddExternalLabel(TypeIdLiteralLabel); }
public static void SaveLibraryInfo(string FolderPath, IL.ILLibrary TheLibrary) { string RootAssemblyName = Utilities.CleanFileName(TheLibrary.TheAssembly.GetName().Name); using (StreamWriter Str = new StreamWriter(Path.Combine(FolderPath, RootAssemblyName + "_Dependencies.txt"), false)) { foreach (IL.ILLibrary DependencyLibrary in TheLibrary.Dependencies) { Str.WriteLine(Utilities.CleanFileName(DependencyLibrary.TheAssembly.GetName().Name)); } } using (StreamWriter Str = new StreamWriter(Path.Combine(FolderPath, RootAssemblyName + "_Library.txt"), false)) { foreach (Types.TypeInfo ATypeInfo in TheLibrary.TypeInfos) { //TypeID //¬BaseTypeID:[ID] //¬IsGCManaged:[Boolean] //¬IsPointer:[Boolean] //¬IsValueType:[Boolean] //¬SizeOnHeapInBytes:[Integer] //¬SizeOnStackInBytes:[Integer] //|Field:[ID] //~Type:[TypeID] //~IsStatic:[Boolean] //~Name:[String] //~OffsetInBytes:[Integer] //|Method:[ID] //~ApplyDebug:[Boolean] //~ApplyGC:[Boolean] //~IDValue:[Integer] //~IsConstructor:[Boolean] //~IsPlugged:[Boolean] //~IsStatic:[Boolean] //~Signature:[String] //~Argument:Offset|Position|TypeID //~Local:Offset|Position|TypeID Str.WriteLine(ATypeInfo.ID); if (ATypeInfo.UnderlyingType.BaseType != null && !ATypeInfo.UnderlyingType.BaseType.AssemblyQualifiedName.Contains("mscorlib")) { Str.WriteLine("¬BaseTypeID:" + TheLibrary.GetTypeInfo(ATypeInfo.UnderlyingType.BaseType).ID); } Str.WriteLine("¬IsGCManaged:" + ATypeInfo.IsGCManaged.ToString()); Str.WriteLine("¬IsPointer:" + ATypeInfo.IsPointer.ToString()); Str.WriteLine("¬IsValueType:" + ATypeInfo.IsValueType.ToString()); Str.WriteLine("¬SizeOnHeapInBytes:" + ATypeInfo.SizeOnHeapInBytes.ToString()); Str.WriteLine("¬SizeOnStackInBytes:" + ATypeInfo.SizeOnStackInBytes.ToString()); foreach (Types.FieldInfo AFieldInfo in ATypeInfo.FieldInfos) { Str.WriteLine("|Field:" + AFieldInfo.ID); Str.WriteLine("~Type:" + TheLibrary.GetTypeInfo(AFieldInfo.FieldType).ID); Str.WriteLine("~IsStatic:" + AFieldInfo.IsStatic.ToString()); Str.WriteLine("~Name:" + AFieldInfo.Name); Str.WriteLine("~OffsetInBytes:" + AFieldInfo.OffsetInBytes.ToString()); } foreach (Types.MethodInfo AMethodInfo in ATypeInfo.MethodInfos) { Str.WriteLine("|Method:" + AMethodInfo.ID); Str.WriteLine("~ApplyDebug:" + AMethodInfo.ApplyDebug.ToString()); Str.WriteLine("~ApplyGC:" + AMethodInfo.ApplyGC.ToString()); Str.WriteLine("~IDValue:" + AMethodInfo.IDValue.ToString()); Str.WriteLine("~IsConstructor:" + AMethodInfo.IsConstructor.ToString()); Str.WriteLine("~IsPlugged:" + AMethodInfo.IsPlugged.ToString()); Str.WriteLine("~IsStatic:" + AMethodInfo.IsStatic.ToString()); Str.WriteLine("~Signature:" + AMethodInfo.Signature); Type RetType = (AMethodInfo.IsConstructor ? typeof(void) : ((System.Reflection.MethodInfo)AMethodInfo.UnderlyingInfo).ReturnType); Str.WriteLine("~ReturnSize:" + Types.TypeScanner.GetSizeOnStackInBytes(RetType)); foreach (Types.VariableInfo AnArgumentInfo in AMethodInfo.ArgumentInfos) { Str.WriteLine("~Argument:" + AnArgumentInfo.Offset.ToString() + "|" + AnArgumentInfo.Position.ToString() + "|" + AnArgumentInfo.TheTypeInfo.ID); } foreach (Types.VariableInfo ALocalInfo in AMethodInfo.LocalInfos) { Str.WriteLine("~Local:" + ALocalInfo.Offset.ToString() + "|" + ALocalInfo.Position.ToString() + "|" + ALocalInfo.TheTypeInfo.ID); } } } } }