using System; using System.Linq; using System.Reflection; using Mono.Cecil; using Mono.Cecil.Cil; namespace StardewModdingAPI.Framework.ModLoading { /// Provides helper methods for field rewriters. internal static class RewriteHelper { /********* ** Properties *********/ /// The comparer which heuristically compares type definitions. private static readonly TypeReferenceComparer TypeDefinitionComparer = new TypeReferenceComparer(); /********* ** Public methods *********/ /// Get the field reference from an instruction if it matches. /// The IL instruction. public static FieldReference AsFieldReference(Instruction instruction) { return instruction.OpCode == OpCodes.Ldfld || instruction.OpCode == OpCodes.Ldsfld || instruction.OpCode == OpCodes.Stfld || instruction.OpCode == OpCodes.Stsfld ? (FieldReference)instruction.Operand : null; } /// Get the method reference from an instruction if it matches. /// The IL instruction. public static MethodReference AsMethodReference(Instruction instruction) { return instruction.OpCode == OpCodes.Call || instruction.OpCode == OpCodes.Callvirt || instruction.OpCode == OpCodes.Newobj ? (MethodReference)instruction.Operand : null; } /// Get whether a type matches a type reference. /// The defined type. /// The type reference. public static bool IsSameType(Type type, TypeReference reference) { // same namespace & name if (type.Namespace != reference.Namespace || type.Name != reference.Name) return false; // same generic parameters if (type.IsGenericType) { if (!reference.IsGenericInstance) return false; Type[] defGenerics = type.GetGenericArguments(); TypeReference[] refGenerics = ((GenericInstanceType)reference).GenericArguments.ToArray(); if (defGenerics.Length != refGenerics.Length) return false; for (int i = 0; i < defGenerics.Length; i++) { if (!RewriteHelper.IsSameType(defGenerics[i], refGenerics[i])) return false; } } return true; } /// Determine whether two type IDs look like the same type, accounting for placeholder values such as !0. /// The type ID to compare. /// The other type ID to compare. /// true if the type IDs look like the same type, false if not. public static bool LooksLikeSameType(TypeReference typeA, TypeReference typeB) { return RewriteHelper.TypeDefinitionComparer.Equals(typeA, typeB); } /// Get whether a method definition matches the signature expected by a method reference. /// The method definition. /// The method reference. public static bool HasMatchingSignature(MethodInfo definition, MethodReference reference) { // same name if (definition.Name != reference.Name) return false; // same arguments ParameterInfo[] definitionParameters = definition.GetParameters(); ParameterDefinition[] referenceParameters = reference.Parameters.ToArray(); if (referenceParameters.Length != definitionParameters.Length) return false; for (int i = 0; i < referenceParameters.Length; i++) { if (!RewriteHelper.IsSameType(definitionParameters[i].ParameterType, referenceParameters[i].ParameterType)) return false; } return true; } /// Get whether a type has a method whose signature matches the one expected by a method reference. /// The type to check. /// The method reference. public static bool HasMatchingSignature(Type type, MethodReference reference) { return type .GetMethods(BindingFlags.Instance | BindingFlags.DeclaredOnly | BindingFlags.Public) .Any(method => RewriteHelper.HasMatchingSignature(method, reference)); } } }