Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improvements to struct interop for DOTS #70

Open
wants to merge 36 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
5b5cefe
Basic support of struct injection
limoka Oct 28, 2022
3be8a27
Merge branch 'BepInEx:master' into master
limoka Nov 2, 2022
04ebe53
Fix methods with pointers generated wrong in interop assemblies
limoka Nov 2, 2022
a73c475
Don't generate default constructor for structs
limoka Nov 5, 2022
0e785af
Merge branch 'BepInEx:master' into master
limoka Nov 11, 2022
bdb3538
Merge branch 'BepInEx:master' into master
limoka Dec 26, 2022
010800e
Allow generic structs to be blittable
limoka Jan 21, 2023
82f7e23
Prevent duplicate IsUnmanagedAttribute classes from being generated
limoka Jan 21, 2023
d795fd9
Fix formatting
limoka Jan 22, 2023
715fbd8
Cleanup
limoka Jan 22, 2023
277efc6
Determine type usage. Add GenericBlittable struct type
limoka Jan 23, 2023
c767a78
Generate Boxed variations for generic blittable structs
limoka Feb 2, 2023
2c4a31f
Append _Unboxed for blittable types instead of non blittable
limoka Feb 14, 2023
fb8ac21
Fix incorrect generic parameter field generation
limoka May 22, 2023
9c4692e
Address comments
limoka May 22, 2023
c00299e
Merge remote-tracking branch 'origin/master' into dots-improvements
limoka May 23, 2023
3e44ef8
Fix method generic parameter constraints
limoka May 23, 2023
636648e
Relax generic constraints which are not used in any fields
limoka May 23, 2023
4232b6a
Force Nullable struct to be non blittable
limoka May 23, 2023
8b03a8d
Remove default constructor constraints
limoka May 23, 2023
682cd0b
Check nested types for generic parameter relaxation
limoka May 23, 2023
3321c8d
Fix generic constraint violations in nested types
limoka May 25, 2023
02c1637
Perform generic constraint checks to ensure no type load exceptions
limoka May 30, 2023
8963b77
Check for field generic parameter usage
limoka May 31, 2023
debe584
Split generic parameter code into pass
limoka May 31, 2023
fa78b1c
Adjust generic parameter constraint logic
limoka May 31, 2023
4ebb59c
Set Strict generic parameter constraint when found no non blittable u…
limoka Jun 1, 2023
1f16eb8
Fix formatting
limoka Jun 1, 2023
8587912
Ensure structs are not generated as blittable only when non blittable…
limoka Jun 3, 2023
ae8a1bf
Generate nested types for boxed type variants
limoka Jun 3, 2023
802eba3
Split compute specifics pass in two to prevent use cycles
limoka Jun 3, 2023
48a7cc5
Prevent relaxed generic parameters from affecting blittablity.
limoka Jun 4, 2023
de2c125
Revert forcing Nullable struct to be non blittable
limoka Jun 5, 2023
35d2507
Fix invalid constructor being generated in unboxed struct variants
limoka Jun 9, 2023
9963313
Generate boolean fields in generic structs as a custom struct.
limoka Jun 9, 2023
f38c606
Allow fields to be ignored
limoka Jun 10, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
97 changes: 81 additions & 16 deletions Il2CppInterop.Generator/Contexts/AssemblyRewriteContext.cs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
using System.Collections.Generic;
using Il2CppInterop.Generator.Extensions;
using Il2CppInterop.Generator.Utils;
using Mono.Cecil;
using Mono.Cecil.Cil;

namespace Il2CppInterop.Generator.Contexts;

Expand All @@ -15,6 +15,7 @@ public class AssemblyRewriteContext
public readonly RuntimeAssemblyReferences Imports;
private readonly Dictionary<string, TypeRewriteContext> myNameTypeMap = new();
private readonly Dictionary<TypeDefinition, TypeRewriteContext> myNewTypeMap = new();
private TypeDefinition isUnmanagedAttributeType;

private readonly Dictionary<TypeDefinition, TypeRewriteContext> myOldTypeMap = new();
public readonly AssemblyDefinition NewAssembly;
Expand All @@ -32,7 +33,8 @@ public AssemblyRewriteContext(RewriteGlobalContext globalContext, AssemblyDefini
mod => new RuntimeAssemblyReferences(mod, globalContext));
}

public IEnumerable<TypeRewriteContext> Types => myOldTypeMap.Values;
public IEnumerable<TypeRewriteContext> Types => myNewTypeMap.Values;
public IEnumerable<TypeRewriteContext> OriginalTypes => myOldTypeMap.Values;

public TypeRewriteContext GetContextForOriginalType(TypeDefinition type)
{
Expand All @@ -49,6 +51,11 @@ public TypeRewriteContext GetContextForNewType(TypeDefinition type)
return myNewTypeMap[type];
}

public void RegisterTypeContext(TypeRewriteContext context)
{
myNewTypeMap.Add(context.NewType, context);
}

public void RegisterTypeRewrite(TypeRewriteContext context)
{
if (context.OriginalType != null)
Expand All @@ -63,7 +70,7 @@ public MethodReference RewriteMethodRef(MethodReference methodRef)
return newType.GetMethodByOldMethod(methodRef.Resolve()).NewMethod;
}

public TypeReference RewriteTypeRef(TypeReference? typeRef)
public TypeReference RewriteTypeRef(TypeReference? typeRef, bool typeIsBoxed)
{
if (typeRef == null) return Imports.Il2CppObjectBase;

Expand All @@ -78,7 +85,7 @@ public TypeReference RewriteTypeRef(TypeReference? typeRef)
if (elementType.FullName == "System.String")
return Imports.Il2CppStringArray;

var convertedElementType = RewriteTypeRef(elementType);
var convertedElementType = RewriteTypeRef(elementType, typeIsBoxed);
if (elementType.IsGenericParameter)
return new GenericInstanceType(Imports.Il2CppArrayBase) { GenericArguments = { convertedElementType } };

Expand All @@ -92,24 +99,36 @@ public TypeReference RewriteTypeRef(TypeReference? typeRef)
{
var genericParameterDeclaringType = genericParameter.DeclaringType;
if (genericParameterDeclaringType != null)
return RewriteTypeRef(genericParameterDeclaringType).GenericParameters[genericParameter.Position];
return RewriteTypeRef(genericParameterDeclaringType, typeIsBoxed).GenericParameters[genericParameter.Position];

return RewriteMethodRef(genericParameter.DeclaringMethod).GenericParameters[genericParameter.Position];
}

if (typeRef is ByReferenceType byRef)
return new ByReferenceType(RewriteTypeRef(byRef.ElementType));
return new ByReferenceType(RewriteTypeRef(byRef.ElementType, typeIsBoxed));

if (typeRef is PointerType pointerType)
return new PointerType(RewriteTypeRef(pointerType.ElementType));
return new PointerType(RewriteTypeRef(pointerType.ElementType, typeIsBoxed));

if (typeRef is GenericInstanceType genericInstance)
{
var newRef = new GenericInstanceType(RewriteTypeRef(genericInstance.ElementType));
foreach (var originalParameter in genericInstance.GenericArguments)
newRef.GenericArguments.Add(RewriteTypeRef(originalParameter));

return newRef;
var genericTypeContext = GetTypeContext(genericInstance);
if (genericTypeContext.ComputedTypeSpecifics == TypeRewriteContext.TypeSpecifics.GenericBlittableStruct && !IsUnmanaged(typeRef, typeIsBoxed))
{
var newRef = new GenericInstanceType(sourceModule.ImportReference(genericTypeContext.BoxedTypeContext.NewType));
foreach (var originalParameter in genericInstance.GenericArguments)
newRef.GenericArguments.Add(RewriteTypeRef(originalParameter, typeIsBoxed));

return newRef;
}
else
{
var newRef = new GenericInstanceType(RewriteTypeRef(genericInstance.ElementType, typeIsBoxed));
foreach (var originalParameter in genericInstance.GenericArguments)
newRef.GenericArguments.Add(RewriteTypeRef(originalParameter, typeIsBoxed));

return newRef;
}
}

if (typeRef.IsPrimitive || typeRef.FullName == "System.TypedReference")
Expand All @@ -129,11 +148,57 @@ public TypeReference RewriteTypeRef(TypeReference? typeRef)
return sourceModule.ImportReference(GlobalContext.GetAssemblyByName("mscorlib")
.GetTypeByName("System.Attribute").NewType);

var originalTypeDef = typeRef.Resolve();
var targetAssembly = GlobalContext.GetNewAssemblyForOriginal(originalTypeDef.Module.Assembly);
var target = targetAssembly.GetContextForOriginalType(originalTypeDef).NewType;
var target = GetTypeContext(typeRef);

if (typeIsBoxed && target.BoxedTypeContext != null)
{
target = target.BoxedTypeContext;
}

return sourceModule.ImportReference(target.NewType);
}

private TypeRewriteContext GetTypeContext(TypeReference typeRef)
{
return GlobalContext.GetNewTypeForOriginal(typeRef.Resolve());
}

private bool IsUnmanaged(TypeReference originalType, bool typeIsBoxed)
{
if (originalType is GenericParameter genericParameter)
{
GenericParameter newGenericParameter = (GenericParameter)RewriteTypeRef(genericParameter, typeIsBoxed);
return newGenericParameter.CustomAttributes.Any(attribute => attribute.AttributeType.Name.Equals("IsUnmanagedAttribute"));
}

return sourceModule.ImportReference(target);
if (originalType is GenericInstanceType genericInstanceType)
{
foreach (TypeReference genericArgument in genericInstanceType.GenericArguments)
{
if (!IsUnmanaged(genericArgument, typeIsBoxed))
return false;
}
}

var paramTypeContext = GetTypeContext(originalType);
return paramTypeContext.ComputedTypeSpecifics.IsBlittable();
}

public TypeDefinition GetOrInjectIsUnmanagedAttribute()
{
if (isUnmanagedAttributeType != null)
return isUnmanagedAttributeType;

isUnmanagedAttributeType = new TypeDefinition("System.Runtime.CompilerServices", "IsUnmanagedAttribute", TypeAttributes.AnsiClass | TypeAttributes.BeforeFieldInit | TypeAttributes.Public | TypeAttributes.Sealed, NewAssembly.MainModule.ImportReference(typeof(Attribute)));
NewAssembly.MainModule.Types.Add(isUnmanagedAttributeType);

var attributeCctr = new MethodDefinition(".ctor", MethodAttributes.Public | MethodAttributes.HideBySig | MethodAttributes.RTSpecialName | MethodAttributes.SpecialName, NewAssembly.MainModule.TypeSystem.Void);
isUnmanagedAttributeType.Methods.Add(attributeCctr);
var ilProcessor = attributeCctr.Body.GetILProcessor();
ilProcessor.Emit(OpCodes.Ldarg_0);
ilProcessor.Emit(OpCodes.Call, NewAssembly.MainModule.ImportReference(isUnmanagedAttributeType.BaseType.DefaultCtorFor()));
ilProcessor.Emit(OpCodes.Ret);
return isUnmanagedAttributeType;
}

public TypeRewriteContext GetTypeByName(string name)
Expand Down
2 changes: 1 addition & 1 deletion Il2CppInterop.Generator/Contexts/FieldRewriteContext.cs
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ private string UnmangleFieldNameBase(FieldDefinition field, GeneratorOptions opt
var accessModString = MethodAccessTypeLabels[(int)(field.Attributes & FieldAttributes.FieldAccessMask)];
var staticString = field.IsStatic ? "_Static" : "";
return "field_" + accessModString + staticString + "_" +
DeclaringType.AssemblyContext.RewriteTypeRef(field.FieldType).GetUnmangledName();
DeclaringType.AssemblyContext.RewriteTypeRef(field.FieldType, false).GetUnmangledName();
}

private string UnmangleFieldName(FieldDefinition field, GeneratorOptions options,
Expand Down
67 changes: 53 additions & 14 deletions Il2CppInterop.Generator/Contexts/MethodRewriteContext.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
using System.Collections.Generic;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Text;
using Il2CppInterop.Common.XrefScans;
Expand Down Expand Up @@ -69,14 +67,23 @@ public MethodRewriteContext(TypeRewriteContext declaringType, MethodDefinition o
foreach (var oldParameter in genericParams)
{
var genericParameter = new GenericParameter(oldParameter.Name, newMethod);
genericParameter.Attributes = oldParameter.Attributes.StripValueTypeConstraint();
if (ShouldParameterBeBlittable(originalMethod, oldParameter))
{
genericParameter.Attributes = oldParameter.Attributes;
genericParameter.MakeUnmanaged(DeclaringType.AssemblyContext);
}
else
{
genericParameter.Attributes = oldParameter.Attributes.StripValueTypeConstraint();
}

newMethod.GenericParameters.Add(genericParameter);
}
}

if (!Pass15GenerateMemberContexts.HasObfuscatedMethods && !passthroughNames &&
if (!Pass16GenerateMemberContexts.HasObfuscatedMethods && !passthroughNames &&
originalMethod.Name.IsObfuscated(declaringType.AssemblyContext.GlobalContext.Options))
Pass15GenerateMemberContexts.HasObfuscatedMethods = true;
Pass16GenerateMemberContexts.HasObfuscatedMethods = true;

FileOffset = originalMethod.ExtractOffset();
// Workaround for garbage file offsets passed by Cpp2IL
Expand All @@ -86,6 +93,38 @@ public MethodRewriteContext(TypeRewriteContext declaringType, MethodDefinition o
declaringType.AssemblyContext.GlobalContext.MethodStartAddresses.Add(FileOffset);
}

private bool ShouldParameterBeBlittable(MethodDefinition method, GenericParameter genericParameter)
{
if (HasGenericParameter(method.ReturnType, genericParameter, out GenericParameter parameter))
{
return parameter.IsUnmanaged();
}

foreach (ParameterDefinition methodParameter in method.Parameters)
{
if (HasGenericParameter(methodParameter.ParameterType, genericParameter, out parameter))
{
return parameter.IsUnmanaged();
}
}

return false;
}

private bool HasGenericParameter(TypeReference typeReference, GenericParameter inputGenericParameter, out GenericParameter typeGenericParameter)
{
typeGenericParameter = null;
if (typeReference is not GenericInstanceType genericInstance) return false;

var index = genericInstance.GenericArguments.IndexOf(inputGenericParameter);
if (index < 0) return false;

var globalContext = DeclaringType.AssemblyContext.GlobalContext;
var returnTypeContext = globalContext.GetNewTypeForOriginal(typeReference.Resolve());
typeGenericParameter = returnTypeContext.NewType.GenericParameters[index];
return true;
}

public string UnmangledName { get; private set; }
public string UnmangledNameWithSignature { get; private set; }

Expand All @@ -102,7 +141,7 @@ public void CtorPhase2()
UnmangledNameWithSignature = UnmangleMethodNameWithSignature();

NewMethod.Name = UnmangledName;
NewMethod.ReturnType = DeclaringType.AssemblyContext.RewriteTypeRef(OriginalMethod.ReturnType);
NewMethod.ReturnType = DeclaringType.AssemblyContext.RewriteTypeRef(OriginalMethod.ReturnType, DeclaringType.isBoxedTypeVariant);

var nonGenericMethodInfoPointerField = new FieldDefinition(
"NativeMethodInfoPtr_" + UnmangledNameWithSignature,
Expand Down Expand Up @@ -142,7 +181,7 @@ public void CtorPhase2()
oldConstraint.ConstraintType.Resolve()?.IsInterface == true) continue;

newParameter.Constraints.Add(new GenericParameterConstraint(
DeclaringType.AssemblyContext.RewriteTypeRef(oldConstraint.ConstraintType)));
DeclaringType.AssemblyContext.RewriteTypeRef(oldConstraint.ConstraintType, DeclaringType.isBoxedTypeVariant)));
}
}

Expand Down Expand Up @@ -222,17 +261,17 @@ private string ProduceMethodSignatureBase()
builder.Append(str);

builder.Append('_');
builder.Append(DeclaringType.AssemblyContext.RewriteTypeRef(method.ReturnType).GetUnmangledName());
builder.Append(DeclaringType.AssemblyContext.RewriteTypeRef(method.ReturnType, DeclaringType.isBoxedTypeVariant).GetUnmangledName());

foreach (var param in method.Parameters)
{
builder.Append('_');
builder.Append(DeclaringType.AssemblyContext.RewriteTypeRef(param.ParameterType).GetUnmangledName());
builder.Append(DeclaringType.AssemblyContext.RewriteTypeRef(param.ParameterType, DeclaringType.isBoxedTypeVariant).GetUnmangledName());
}

var address = Rva;
if (address != 0 && Pass15GenerateMemberContexts.HasObfuscatedMethods &&
!Pass16ScanMethodRefs.NonDeadMethods.Contains(address)) builder.Append("_PDM");
if (address != 0 && Pass16GenerateMemberContexts.HasObfuscatedMethods &&
!Pass17ScanMethodRefs.NonDeadMethods.Contains(address)) builder.Append("_PDM");

return builder.ToString();
}
Expand Down Expand Up @@ -278,13 +317,13 @@ private bool ParameterSignatureMatchesThis(MethodRewriteContext otherRewriteCont
if (a[i].ParameterType.FullName != b[i].ParameterType.FullName)
return false;

if (Pass15GenerateMemberContexts.HasObfuscatedMethods)
if (Pass16GenerateMemberContexts.HasObfuscatedMethods)
{
var addressA = otherRewriteContext.Rva;
var addressB = Rva;
if (addressA != 0 && addressB != 0)
if (Pass16ScanMethodRefs.NonDeadMethods.Contains(addressA) !=
Pass16ScanMethodRefs.NonDeadMethods.Contains(addressB))
if (Pass17ScanMethodRefs.NonDeadMethods.Contains(addressA) !=
Pass17ScanMethodRefs.NonDeadMethods.Contains(addressB))
return false;
}

Expand Down
Loading