From 9e56631950724128905a3b0808900e19a38b67f4 Mon Sep 17 00:00:00 2001
From: Jeremy Pritts <49847914+ds5678@users.noreply.github.com>
Date: Sun, 31 Dec 2023 15:58:57 -0500
Subject: [PATCH] Use AssetRipper.CIL for method filling
---
Cpp2IL.Core/Cpp2IL.Core.csproj | 5 +-
.../AsmResolver/AsmResolverMethodFiller.cs | 94 +--------------
.../CilInstructionCollectionExtensions.cs | 114 ------------------
.../AsmResolver/TypeSignatureExtensions.cs | 11 --
4 files changed, 5 insertions(+), 219 deletions(-)
delete mode 100644 Cpp2IL.Core/Utils/AsmResolver/CilInstructionCollectionExtensions.cs
delete mode 100644 Cpp2IL.Core/Utils/AsmResolver/TypeSignatureExtensions.cs
diff --git a/Cpp2IL.Core/Cpp2IL.Core.csproj b/Cpp2IL.Core/Cpp2IL.Core.csproj
index 1f34073f..20b01911 100644
--- a/Cpp2IL.Core/Cpp2IL.Core.csproj
+++ b/Cpp2IL.Core/Cpp2IL.Core.csproj
@@ -34,7 +34,8 @@
-
+
+
@@ -51,7 +52,7 @@
-
+
diff --git a/Cpp2IL.Core/Utils/AsmResolver/AsmResolverMethodFiller.cs b/Cpp2IL.Core/Utils/AsmResolver/AsmResolverMethodFiller.cs
index 208438cf..0f832abc 100644
--- a/Cpp2IL.Core/Utils/AsmResolver/AsmResolverMethodFiller.cs
+++ b/Cpp2IL.Core/Utils/AsmResolver/AsmResolverMethodFiller.cs
@@ -1,9 +1,5 @@
-using System.Diagnostics.CodeAnalysis;
-using System.Linq;
using AsmResolver.DotNet;
-using AsmResolver.DotNet.Collections;
-using AsmResolver.DotNet.Signatures.Types;
-using AsmResolver.PE.DotNet.Cil;
+using AssetRipper.CIL;
using Cpp2IL.Core.Model.Contexts;
namespace Cpp2IL.Core.Utils.AsmResolver;
@@ -25,8 +21,7 @@ public static void FillManagedMethodBodies(AssemblyAnalysisContext asmContext)
{
var managedMethod = methodCtx.GetExtraData("AsmResolverMethod") ?? throw new($"AsmResolver method not found in method analysis context for {typeContext.Definition?.FullName}.{methodCtx.Definition?.Name}");
- if (managedMethod.IsManagedMethodWithBody())
- FillMethodBodyWithStub(managedMethod);
+ managedMethod.FillMethodBodyWithStub();
}
}
#if !DEBUG
@@ -38,89 +33,4 @@ public static void FillManagedMethodBodies(AssemblyAnalysisContext asmContext)
#endif
}
}
-
- private static void FillMethodBodyWithStub(MethodDefinition methodDefinition)
- {
- methodDefinition.CilMethodBody = new(methodDefinition);
- var methodInstructions = methodDefinition.CilMethodBody.Instructions;
-
- if (methodDefinition.IsConstructor && !methodDefinition.IsStatic && !methodDefinition.DeclaringType!.IsValueType)
- {
- var baseConstructor = TryGetBaseConstructor(methodDefinition);
- if (baseConstructor is not null)
- {
- methodInstructions.Add(CilOpCodes.Ldarg_0);
- foreach (var baseParameter in baseConstructor.Parameters)
- {
- var importedBaseParameterType = methodDefinition.DeclaringType.Module!.DefaultImporter.ImportTypeSignature(baseParameter.ParameterType);
- methodInstructions.AddDefaultValueForType(importedBaseParameterType);
- }
- methodInstructions.Add(CilOpCodes.Call, methodDefinition.DeclaringType.Module!.DefaultImporter.ImportMethod(baseConstructor));
- }
- }
-
- foreach (var parameter in methodDefinition.Parameters)
- {
- //Although Roslyn-compiled code will only emit the out flag on ByReferenceTypeSignatures,
- //Some Unity libraries have it on a handful (less than 100) of parameters with incompatible type signatures.
- //One example on 2021.3.6 is int System.IO.CStreamReader.Read([In][Out] char[] dest, int index, int count)
- //All the instances I investigated were clearly not meant to be out parameters.
- //The [In][Out] attributes are not a decompilation issue and compile fine on .NET 7.
- if (parameter.IsOutParameter(out var parameterType))
- {
- if (parameterType.IsValueTypeOrGenericParameter())
- {
- methodInstructions.Add(CilOpCodes.Ldarg, parameter);
- methodInstructions.Add(CilOpCodes.Initobj, parameterType.ToTypeDefOrRef());
- }
- else
- {
- methodInstructions.Add(CilOpCodes.Ldarg, parameter);
- methodInstructions.Add(CilOpCodes.Ldnull);
- methodInstructions.Add(CilOpCodes.Stind_Ref);
- }
- }
- }
- methodInstructions.AddDefaultValueForType(methodDefinition.Signature!.ReturnType);
- methodInstructions.Add(CilOpCodes.Ret);
- methodInstructions.OptimizeMacros();
- }
-
- ///
- /// Is this an out parameter?
- ///
- ///
- /// The base type of the
- ///
- private static bool IsOutParameter(this Parameter parameter, [NotNullWhen(true)] out TypeSignature? parameterType)
- {
- if ((parameter.Definition?.IsOut ?? false) && parameter.ParameterType is ByReferenceTypeSignature byReferenceTypeSignature)
- {
- parameterType = byReferenceTypeSignature.BaseType;
- return true;
- }
- else
- {
- parameterType = default;
- return false;
- }
- }
-
- private static MethodDefinition? TryGetBaseConstructor(MethodDefinition methodDefinition)
- {
- var declaringType = methodDefinition.DeclaringType!;
- var baseType = declaringType.BaseType?.Resolve();
- if (baseType is null)
- {
- return null;
- }
- else if (declaringType.Module == baseType.Module)
- {
- return baseType.Methods.FirstOrDefault(m => m.IsConstructor && !m.IsStatic && !m.IsPrivate);
- }
- else
- {
- return baseType.Methods.FirstOrDefault(m => m.IsConstructor && !m.IsStatic && (m.IsFamily || m.IsPublic));
- }
- }
}
diff --git a/Cpp2IL.Core/Utils/AsmResolver/CilInstructionCollectionExtensions.cs b/Cpp2IL.Core/Utils/AsmResolver/CilInstructionCollectionExtensions.cs
deleted file mode 100644
index d4f0b3df..00000000
--- a/Cpp2IL.Core/Utils/AsmResolver/CilInstructionCollectionExtensions.cs
+++ /dev/null
@@ -1,114 +0,0 @@
-using System;
-using System.Diagnostics;
-using System.Runtime.CompilerServices;
-using AsmResolver.DotNet.Code.Cil;
-using AsmResolver.DotNet.Signatures.Types;
-using AsmResolver.PE.DotNet.Cil;
-using AsmResolver.PE.DotNet.Metadata.Tables.Rows;
-
-namespace Cpp2IL.Core.Utils.AsmResolver;
-
-internal static class CilInstructionCollectionExtensions
-{
- public static CilLocalVariable AddLocalVariable(this CilInstructionCollection instructions, TypeSignature variableType)
- {
- var variable = new CilLocalVariable(variableType);
- instructions.Owner.LocalVariables.Add(variable);
- return variable;
- }
-
- public static void AddDefaultValueForType(this CilInstructionCollection instructions, TypeSignature type)
- {
- if (type is CorLibTypeSignature { IsValueType: true } corLibTypeSignature)
- {
- instructions.AddDefaultPrimitiveValue(corLibTypeSignature);
- }
- else if (type is ByReferenceTypeSignature)
- {
- instructions.AddNullRef();
- }
- else if (type.IsValueTypeOrGenericParameter())
- {
- instructions.AddDefaultValueForUnknownType(type);
- }
- else
- {
- instructions.Add(CilOpCodes.Ldnull);
- }
- }
-
- ///
- /// Load a null reference onto the stack.
- ///
- ///
- /// The specified instructions come from the documentation for .
- /// They decompile as: ref *(T*)null
- /// The decompilation is valid C# and is the most optimized implementation.
- /// However, it does produce a compiler warning.
- ///
- ///
- private static void AddNullRef(this CilInstructionCollection instructions)
- {
- instructions.Add(CilOpCodes.Ldc_I4_0);
- instructions.Add(CilOpCodes.Conv_U);
- }
-
- ///
- /// Load the default value onto the stack for an unknown type.
- ///
- ///
- /// This is a silver bullet. It handles any type except void and by ref.
- ///
- ///
- ///
- private static void AddDefaultValueForUnknownType(this CilInstructionCollection instructions, TypeSignature type)
- {
- Debug.Assert(type is not CorLibTypeSignature { ElementType: ElementType.Void } and not ByReferenceTypeSignature);
- var variable = instructions.AddLocalVariable(type);
- instructions.Add(CilOpCodes.Ldloca, variable);
- instructions.Add(CilOpCodes.Initobj, type.ToTypeDefOrRef());
- instructions.Add(CilOpCodes.Ldloc, variable);
- }
-
- private static void AddDefaultPrimitiveValue(this CilInstructionCollection instructions, CorLibTypeSignature type)
- {
- switch (type.ElementType)
- {
- case ElementType.Void:
- break;
- case ElementType.U1 or ElementType.U2 or ElementType.U4 or ElementType.I1 or ElementType.I2 or ElementType.I4 or ElementType.Boolean or ElementType.Char:
- instructions.Add(CilOpCodes.Ldc_I4_0);
- break;
- case ElementType.I8:
- instructions.Add(CilOpCodes.Ldc_I4_0);
- instructions.Add(CilOpCodes.Conv_I8);
- break;
- case ElementType.U8:
- instructions.Add(CilOpCodes.Ldc_I4_0);
- instructions.Add(CilOpCodes.Conv_U8);
- break;
- case ElementType.I:
- instructions.Add(CilOpCodes.Ldc_I4_0);
- instructions.Add(CilOpCodes.Conv_I);
- break;
- case ElementType.U:
- instructions.Add(CilOpCodes.Ldc_I4_0);
- instructions.Add(CilOpCodes.Conv_U);
- break;
- case ElementType.R4:
- instructions.Add(CilOpCodes.Ldc_R4, 0f);
- break;
- case ElementType.R8:
- instructions.Add(CilOpCodes.Ldc_R8, 0d);
- break;
- case ElementType.Object or ElementType.String:
- instructions.Add(CilOpCodes.Ldnull);
- break;
- case ElementType.TypedByRef:
- instructions.AddDefaultValueForUnknownType(type);
- break;
- default:
- throw new ArgumentOutOfRangeException(null);
- }
- }
-}
diff --git a/Cpp2IL.Core/Utils/AsmResolver/TypeSignatureExtensions.cs b/Cpp2IL.Core/Utils/AsmResolver/TypeSignatureExtensions.cs
deleted file mode 100644
index 40545076..00000000
--- a/Cpp2IL.Core/Utils/AsmResolver/TypeSignatureExtensions.cs
+++ /dev/null
@@ -1,11 +0,0 @@
-using AsmResolver.DotNet.Signatures.Types;
-
-namespace Cpp2IL.Core.Utils.AsmResolver;
-
-internal static class TypeSignatureExtensions
-{
- public static bool IsValueTypeOrGenericParameter(this TypeSignature type)
- {
- return type is { IsValueType: true } or GenericParameterSignature;
- }
-}