491 lines
18 KiB
C#
491 lines
18 KiB
C#
/*
|
|
Copyright (C) 2011-2013 de4dot@gmail.com
|
|
|
|
This file is part of de4dot.
|
|
|
|
de4dot is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
de4dot is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with de4dot. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Reflection;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.ExceptionServices;
|
|
using System.Runtime.InteropServices;
|
|
using System.Security;
|
|
using dnlib.DotNet;
|
|
using de4dot.blocks;
|
|
|
|
namespace de4dot.code.deobfuscators.ILProtector {
|
|
sealed class DynamicMethodsDecrypter : IDisposable {
|
|
ModuleDefMD module;
|
|
Module reflectionModule;
|
|
Module reflectionProtectModule;
|
|
TypeDef protectMainType;
|
|
Type reflectionProtectMainType;
|
|
FieldInfo invokerFieldInfo;
|
|
ModuleDefMD moduleProtect;
|
|
IDecrypter decrypter;
|
|
|
|
interface IDecrypter {
|
|
byte[] Decrypt(int methodId, uint rid);
|
|
}
|
|
|
|
abstract class DecrypterBase : IDecrypter {
|
|
protected readonly DynamicMethodsDecrypter dmd;
|
|
protected readonly int appDomainId;
|
|
protected readonly int asmHashCode;
|
|
|
|
class PatchData {
|
|
public int RVA { get; set; }
|
|
public byte[] Data { get; set; }
|
|
|
|
public PatchData() {
|
|
}
|
|
|
|
public PatchData(int rva, byte[] data) {
|
|
this.RVA = rva;
|
|
this.Data = data;
|
|
}
|
|
}
|
|
|
|
class PatchInfo {
|
|
public int RvaDecryptMethod { get; set; }
|
|
public List<PatchData> PatchData { get; set; }
|
|
|
|
public PatchInfo() {
|
|
}
|
|
|
|
public PatchInfo(int rvaDecryptMethod, PatchData patchData) {
|
|
this.RvaDecryptMethod = rvaDecryptMethod;
|
|
this.PatchData = new List<PatchData> { patchData };
|
|
}
|
|
}
|
|
|
|
static readonly byte[] nops2 = new byte[] { 0x90, 0x90 };
|
|
static readonly byte[] nops6 = new byte[] { 0x90, 0x90, 0x90, 0x90, 0x90, 0x90 };
|
|
static readonly Dictionary<Version, PatchInfo> patchInfos32 = new Dictionary<Version, PatchInfo> {
|
|
{ new Version(2, 0, 8, 0), new PatchInfo(0x00020B20, new PatchData(0x00005733, nops2)) },
|
|
{ new Version(2, 0, 8, 5), new PatchInfo(0x000221A0, new PatchData(0x00005742, nops2)) },
|
|
{ new Version(2, 0, 9, 0), new PatchInfo(0x00023360, new PatchData(0x000056F2, nops6)) },
|
|
{ new Version(2, 0, 10, 0), new PatchInfo(0x00023B30, new PatchData(0x00005B12, nops6)) },
|
|
{ new Version(2, 0, 11, 0), new PatchInfo(0x000207C0, new PatchData(0x00018432, nops6)) },
|
|
};
|
|
static readonly Dictionary<Version, PatchInfo> patchInfos64 = new Dictionary<Version, PatchInfo> {
|
|
{ new Version(2, 0, 8, 0), new PatchInfo(0x00026090, new PatchData(0x00005E0C, nops6)) },
|
|
{ new Version(2, 0, 8, 5), new PatchInfo(0x000273D0, new PatchData(0x000060CA, nops6)) },
|
|
{ new Version(2, 0, 9, 0), new PatchInfo(0x00028B00, new PatchData(0x00005F70, nops6)) },
|
|
{ new Version(2, 0, 10, 0), new PatchInfo(0x00029630, new PatchData(0x00006510, nops6)) },
|
|
{ new Version(2, 0, 11, 0), new PatchInfo(0x000257C0, new PatchData(0x0001C9A0, nops6)) },
|
|
};
|
|
|
|
[DllImport("kernel32")]
|
|
static extern bool VirtualProtect(IntPtr addr, int size, uint newProtect, out uint oldProtect);
|
|
const uint PAGE_EXECUTE_READWRITE = 0x40;
|
|
|
|
public DecrypterBase(DynamicMethodsDecrypter dmd) {
|
|
this.dmd = dmd;
|
|
this.appDomainId = AppDomain.CurrentDomain.Id;
|
|
this.asmHashCode = dmd.reflectionModule.Assembly.GetHashCode();
|
|
}
|
|
|
|
protected IntPtr GetDelegateAddress(FieldDef delegateField) {
|
|
FieldInfo delegateFieldInfo = dmd.reflectionProtectModule.ResolveField(0x04000000 + (int)delegateField.Rid);
|
|
object mainTypeInst = ((Delegate)dmd.invokerFieldInfo.GetValue(null)).Target;
|
|
return GetNativeAddressOfDelegate((Delegate)delegateFieldInfo.GetValue(mainTypeInst));
|
|
}
|
|
|
|
static IntPtr GetNativeAddressOfDelegate(Delegate del) {
|
|
var field = typeof(Delegate).GetField("_methodPtrAux", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
|
|
if (field == null)
|
|
return IntPtr.Zero;
|
|
|
|
return (IntPtr)field.GetValue(del);
|
|
}
|
|
|
|
protected void PatchRuntime(IntPtr decryptAddr) {
|
|
if (!PatchRuntimeInternal(decryptAddr))
|
|
throw new ApplicationException("Probably a new version. Could not patch runtime.");
|
|
}
|
|
|
|
bool PatchRuntimeInternal(IntPtr decryptAddr) {
|
|
var patchInfos = IntPtr.Size == 4 ? patchInfos32 : patchInfos64;
|
|
var protectVersion = dmd.reflectionProtectModule.Assembly.GetName().Version;
|
|
PatchInfo info;
|
|
if (!patchInfos.TryGetValue(protectVersion, out info))
|
|
return false;
|
|
return PatchRuntime(decryptAddr, info);
|
|
}
|
|
|
|
[HandleProcessCorruptedStateExceptions, SecurityCritical] // Req'd on .NET 4.0
|
|
static bool PatchRuntime(IntPtr decryptAddr, PatchInfo info) {
|
|
try {
|
|
IntPtr baseAddr = new IntPtr(decryptAddr.ToInt64() - info.RvaDecryptMethod);
|
|
if ((baseAddr.ToInt64() & 0xFFFF) != 0)
|
|
return false;
|
|
|
|
if (Marshal.ReadInt16(baseAddr) != 0x5A4D)
|
|
return false;
|
|
|
|
foreach (var patchData in info.PatchData) {
|
|
var patchAddr = new IntPtr(baseAddr.ToInt64() + patchData.RVA);
|
|
uint oldProtect;
|
|
if (!VirtualProtect(patchAddr, patchData.Data.Length, PAGE_EXECUTE_READWRITE, out oldProtect))
|
|
return false;
|
|
Marshal.Copy(patchData.Data, 0, patchAddr, patchData.Data.Length);
|
|
VirtualProtect(patchAddr, patchData.Data.Length, oldProtect, out oldProtect);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
catch {
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public abstract byte[] Decrypt(int methodId, uint rid);
|
|
}
|
|
|
|
// 1.0.7.0 - 1.0.8.0
|
|
class DecrypterV1_0_7_0 : DecrypterBase {
|
|
DecryptMethod decryptMethod;
|
|
|
|
unsafe delegate bool DecryptMethod(int appDomainId, int asmHashCode, int methodId, out byte* pMethodCode, out int methodSize);
|
|
|
|
public DecrypterV1_0_7_0(DynamicMethodsDecrypter dmd, FieldDef delegateField)
|
|
: base(dmd) {
|
|
IntPtr addr = GetDelegateAddress(delegateField);
|
|
decryptMethod = (DecryptMethod)Marshal.GetDelegateForFunctionPointer(addr, typeof(DecryptMethod));
|
|
}
|
|
|
|
public unsafe override byte[] Decrypt(int methodId, uint rid) {
|
|
byte* pMethodCode;
|
|
int methodSize;
|
|
if (!decryptMethod(appDomainId, asmHashCode, methodId, out pMethodCode, out methodSize))
|
|
return null;
|
|
byte[] methodData = new byte[methodSize];
|
|
Marshal.Copy(new IntPtr(pMethodCode), methodData, 0, methodData.Length);
|
|
return methodData;
|
|
}
|
|
}
|
|
|
|
// 2.0.0.0 - 2.0.7.6
|
|
class DecrypterV2_0_0_0 : DecrypterBase {
|
|
DecryptMethod decryptMethod;
|
|
|
|
unsafe delegate bool DecryptMethod(int clrMajorVersion, int appDomainId, int asmHashCode, int methodId, out byte* pMethodCode, out int methodSize);
|
|
|
|
public DecrypterV2_0_0_0(DynamicMethodsDecrypter dmd, FieldDef delegateField)
|
|
: base(dmd) {
|
|
IntPtr addr = GetDelegateAddress(delegateField);
|
|
decryptMethod = (DecryptMethod)Marshal.GetDelegateForFunctionPointer(addr, typeof(DecryptMethod));
|
|
}
|
|
|
|
public unsafe override byte[] Decrypt(int methodId, uint rid) {
|
|
byte* pMethodCode;
|
|
int methodSize;
|
|
if (!decryptMethod(Environment.Version.Major, appDomainId, asmHashCode, methodId, out pMethodCode, out methodSize))
|
|
return null;
|
|
byte[] methodData = new byte[methodSize];
|
|
Marshal.Copy(new IntPtr(pMethodCode), methodData, 0, methodData.Length);
|
|
return methodData;
|
|
}
|
|
}
|
|
|
|
// 2.0.8.0
|
|
class DecrypterV2_0_8_0 : DecrypterBase {
|
|
DecryptMethod decryptMethod;
|
|
byte[] decryptedData;
|
|
|
|
delegate bool DecryptMethod(int clrMajorVersion, int appDomainId, int asmHashCode, int methodId, [MarshalAs(UnmanagedType.FunctionPtr)] DecryptCallback decryptCallback, [MarshalAs(UnmanagedType.Interface)] out Delegate createdDelegate);
|
|
unsafe delegate bool DecryptCallback(byte* pMethodCode, int methodSize, [MarshalAs(UnmanagedType.Interface)] ref Delegate createdDelegate);
|
|
|
|
public DecrypterV2_0_8_0(DynamicMethodsDecrypter dmd, FieldDef delegateField)
|
|
: base(dmd) {
|
|
IntPtr addr = GetDelegateAddress(delegateField);
|
|
decryptMethod = (DecryptMethod)Marshal.GetDelegateForFunctionPointer(addr, typeof(DecryptMethod));
|
|
PatchRuntime(addr);
|
|
}
|
|
|
|
public unsafe override byte[] Decrypt(int methodId, uint rid) {
|
|
Delegate createdDelegate;
|
|
if (!decryptMethod(Environment.Version.Major, appDomainId, asmHashCode, methodId, MyDecryptCallback, out createdDelegate))
|
|
return null;
|
|
return decryptedData;
|
|
}
|
|
|
|
unsafe bool MyDecryptCallback(byte* pMethodCode, int methodSize, ref Delegate createdDelegate) {
|
|
decryptedData = new byte[methodSize];
|
|
Marshal.Copy(new IntPtr(pMethodCode), decryptedData, 0, decryptedData.Length);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// 2.0.8.5
|
|
class DecrypterV2_0_8_5 : DecrypterBase {
|
|
DecryptMethod decryptMethod;
|
|
byte[] decryptedData;
|
|
bool decryptReturnValue;
|
|
|
|
delegate bool DecryptMethod(int clrMajorVersion, int appDomainId, int asmHashCode, int methodId, [MarshalAs(UnmanagedType.Interface)] StackTrace stackTrace, [MarshalAs(UnmanagedType.FunctionPtr)] DecryptCallback decryptCallback, [MarshalAs(UnmanagedType.Interface)] out Delegate createdDelegate);
|
|
unsafe delegate bool DecryptCallback(byte* pMethodCode, int methodSize, [MarshalAs(UnmanagedType.Interface)] ref Delegate createdDelegate);
|
|
|
|
public DecrypterV2_0_8_5(DynamicMethodsDecrypter dmd, FieldDef delegateField)
|
|
: base(dmd) {
|
|
IntPtr addr = GetDelegateAddress(delegateField);
|
|
decryptMethod = (DecryptMethod)Marshal.GetDelegateForFunctionPointer(addr, typeof(DecryptMethod));
|
|
PatchRuntime(addr);
|
|
}
|
|
|
|
public unsafe override byte[] Decrypt(int methodId, uint rid) {
|
|
Delegate createdDelegate;
|
|
decryptReturnValue = false;
|
|
if (!decryptMethod(Environment.Version.Major, appDomainId, asmHashCode, methodId, new StackTrace(), MyDecryptCallback, out createdDelegate) &&
|
|
!decryptReturnValue)
|
|
return null;
|
|
return decryptedData;
|
|
}
|
|
|
|
unsafe bool MyDecryptCallback(byte* pMethodCode, int methodSize, ref Delegate createdDelegate) {
|
|
createdDelegate = new DecryptCallback(MyDecryptCallback);
|
|
decryptedData = new byte[methodSize];
|
|
Marshal.Copy(new IntPtr(pMethodCode), decryptedData, 0, decryptedData.Length);
|
|
return decryptReturnValue = true;
|
|
}
|
|
}
|
|
|
|
// 2.0.9.0 - 2.0.11.0
|
|
class DecrypterV2_0_9_0 : DecrypterBase {
|
|
DecryptMethod decryptMethod;
|
|
byte[] decryptedData;
|
|
|
|
delegate bool DecryptMethod(int clrMajorVersion, int appDomainId, int asmHashCode, int methodId, [MarshalAs(UnmanagedType.Interface)] StackTrace stackTrace, [MarshalAs(UnmanagedType.FunctionPtr)] DecryptCallback decryptCallback);
|
|
unsafe delegate bool DecryptCallback(byte* pMethodCode, int methodSize, int methodId);
|
|
|
|
public DecrypterV2_0_9_0(DynamicMethodsDecrypter dmd, FieldDef delegateField)
|
|
: base(dmd) {
|
|
IntPtr addr = GetDelegateAddress(delegateField);
|
|
decryptMethod = (DecryptMethod)Marshal.GetDelegateForFunctionPointer(addr, typeof(DecryptMethod));
|
|
PatchRuntime(addr);
|
|
}
|
|
|
|
public unsafe override byte[] Decrypt(int methodId, uint rid) {
|
|
var encMethod = this.dmd.reflectionModule.ResolveMethod(0x06000000 + (int)rid);
|
|
var stackTrace = StackTracePatcher.WriteStackFrame(new StackTrace(), 1, encMethod);
|
|
if (!decryptMethod(Environment.Version.Major, appDomainId, asmHashCode, methodId, stackTrace, MyDecryptCallback))
|
|
return null;
|
|
return decryptedData;
|
|
}
|
|
|
|
unsafe bool MyDecryptCallback(byte* pMethodCode, int methodSize, int methodId) {
|
|
decryptedData = new byte[methodSize];
|
|
Marshal.Copy(new IntPtr(pMethodCode), decryptedData, 0, decryptedData.Length);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
public DynamicMethodsDecrypter(ModuleDefMD module, Module reflectionModule) {
|
|
this.module = module;
|
|
this.reflectionModule = reflectionModule;
|
|
}
|
|
|
|
public void Initialize() {
|
|
RuntimeHelpers.RunModuleConstructor(reflectionModule.ModuleHandle);
|
|
var reflectionProtectAssembly = GetProtectAssembly();
|
|
if (reflectionProtectAssembly == null)
|
|
throw new ApplicationException("Could not find 'Protect' assembly");
|
|
reflectionProtectModule = reflectionProtectAssembly.ManifestModule;
|
|
moduleProtect = ModuleDefMD.Load(reflectionProtectModule);
|
|
protectMainType = FindMainType(moduleProtect);
|
|
if (protectMainType == null)
|
|
throw new ApplicationException("Could not find Protect.MainType");
|
|
var invokerField = FindInvokerField(module);
|
|
|
|
reflectionProtectMainType = reflectionProtectModule.ResolveType(0x02000000 + (int)protectMainType.Rid);
|
|
invokerFieldInfo = reflectionModule.ResolveField(0x04000000 + (int)invokerField.Rid);
|
|
|
|
decrypter = CreateDecrypter();
|
|
if (decrypter == null)
|
|
throw new ApplicationException("Probably a new version. Could not create a decrypter.");
|
|
}
|
|
|
|
public DecryptedMethodInfo Decrypt(int methodId, uint rid) {
|
|
byte[] methodData = decrypter.Decrypt(methodId, rid);
|
|
if (methodData == null)
|
|
throw new ApplicationException(string.Format("Probably a new version. Could not decrypt method. ID:{0}, RID:{1:X4}", methodId, rid));
|
|
return new DecryptedMethodInfo(methodId, methodData);
|
|
}
|
|
|
|
public void Dispose() {
|
|
if (moduleProtect != null)
|
|
moduleProtect.Dispose();
|
|
moduleProtect = null;
|
|
}
|
|
|
|
IDecrypter CreateDecrypter() {
|
|
return CreateDecrypterV1_0_7_1() ??
|
|
CreateDecrypterV2_0_0_0() ??
|
|
CreateDecrypterV2_0_8_0() ??
|
|
CreateDecrypterV2_0_8_5() ??
|
|
CreateDecrypterV2_0_9_0();
|
|
}
|
|
|
|
IDecrypter CreateDecrypterV1_0_7_1() {
|
|
var delegateField = FindDelegateFieldV1_0_7_1(protectMainType);
|
|
if (delegateField == null)
|
|
return null;
|
|
|
|
return new DecrypterV1_0_7_0(this, delegateField);
|
|
}
|
|
|
|
IDecrypter CreateDecrypterV2_0_0_0() {
|
|
var delegateField = FindDelegateFieldV2_0_0_0(protectMainType);
|
|
if (delegateField == null)
|
|
return null;
|
|
|
|
return new DecrypterV2_0_0_0(this, delegateField);
|
|
}
|
|
|
|
IDecrypter CreateDecrypterV2_0_8_0() {
|
|
var delegateField = FindDelegateFieldV2_0_8_0(protectMainType, FindDecryptCallbackV2_0_8_0(protectMainType));
|
|
if (delegateField == null)
|
|
return null;
|
|
|
|
return new DecrypterV2_0_8_0(this, delegateField);
|
|
}
|
|
|
|
IDecrypter CreateDecrypterV2_0_8_5() {
|
|
var delegateField = FindDelegateFieldV2_0_8_5(protectMainType, FindDecryptCallbackV2_0_8_0(protectMainType));
|
|
if (delegateField == null)
|
|
return null;
|
|
|
|
return new DecrypterV2_0_8_5(this, delegateField);
|
|
}
|
|
|
|
IDecrypter CreateDecrypterV2_0_9_0() {
|
|
var delegateField = FindDelegateFieldV2_0_9_0(protectMainType, FindDecryptCallbackV2_0_9_0(protectMainType));
|
|
if (delegateField == null)
|
|
return null;
|
|
|
|
return new DecrypterV2_0_9_0(this, delegateField);
|
|
}
|
|
|
|
static readonly byte[] ilpPublicKeyToken = new byte[8] { 0x20, 0x12, 0xD3, 0xC0, 0x55, 0x1F, 0xE0, 0x3D };
|
|
static Assembly GetProtectAssembly() {
|
|
foreach (var asm in AppDomain.CurrentDomain.GetAssemblies()) {
|
|
if (!string.IsNullOrEmpty(asm.Location))
|
|
continue;
|
|
var asmName = asm.GetName();
|
|
if (asmName.Name != "Protect")
|
|
continue;
|
|
if (!Compare(asmName.GetPublicKeyToken(), ilpPublicKeyToken))
|
|
continue;
|
|
|
|
return asm;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
static bool Compare(byte[] a, byte[] b) {
|
|
if (a == null && b == null)
|
|
return true;
|
|
if (a == null || b == null)
|
|
return false;
|
|
if (a.Length != b.Length)
|
|
return false;
|
|
for (int i = 0; i < a.Length; i++) {
|
|
if (a[i] != b[i])
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static TypeDef FindMainType(ModuleDef module) {
|
|
foreach (var type in module.Types) {
|
|
if (type.FindMethod("Finalize") != null)
|
|
return type;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
static FieldDef FindInvokerField(ModuleDef module) {
|
|
return FindDelegateField(module.GlobalType, "System.Delegate", "(System.Int32)");
|
|
}
|
|
|
|
static FieldDef FindDelegateFieldV1_0_7_1(TypeDef mainType) {
|
|
return FindDelegateField(mainType, "System.Boolean", "(System.Int32,System.Int32,System.Int32,System.Byte*&,System.Int32&)");
|
|
}
|
|
|
|
static FieldDef FindDelegateFieldV2_0_0_0(TypeDef mainType) {
|
|
return FindDelegateField(mainType, "System.Boolean", "(System.Int32,System.Int32,System.Int32,System.Int32,System.Byte*&,System.Int32&)");
|
|
}
|
|
|
|
static FieldDef FindDecryptCallbackV2_0_8_0(TypeDef mainType) {
|
|
return FindDelegateField(mainType, "System.Boolean", "(System.Byte*,System.Int32,System.Delegate&)");
|
|
}
|
|
|
|
static FieldDef FindDecryptCallbackV2_0_9_0(TypeDef mainType) {
|
|
return FindDelegateField(mainType, "System.Boolean", "(System.Byte*,System.Int32,System.Int32)");
|
|
}
|
|
|
|
static FieldDef FindDelegateFieldV2_0_8_0(TypeDef mainType, FieldDef decryptCallbackField) {
|
|
if (decryptCallbackField == null)
|
|
return null;
|
|
var type = decryptCallbackField.FieldSig.GetFieldType().ToTypeDefOrRef() as TypeDef;
|
|
if (type == null)
|
|
return null;
|
|
return FindDelegateField(mainType, "System.Boolean", string.Format("(System.Int32,System.Int32,System.Int32,System.Int32,{0},System.Delegate&)", type.FullName));
|
|
}
|
|
|
|
static FieldDef FindDelegateFieldV2_0_8_5(TypeDef mainType, FieldDef decryptCallbackField) {
|
|
if (decryptCallbackField == null)
|
|
return null;
|
|
var type = decryptCallbackField.FieldSig.GetFieldType().ToTypeDefOrRef() as TypeDef;
|
|
if (type == null)
|
|
return null;
|
|
return FindDelegateField(mainType, "System.Boolean", string.Format("(System.Int32,System.Int32,System.Int32,System.Int32,System.Diagnostics.StackTrace,{0},System.Delegate&)", type.FullName));
|
|
}
|
|
|
|
static FieldDef FindDelegateFieldV2_0_9_0(TypeDef mainType, FieldDef decryptCallbackField) {
|
|
if (decryptCallbackField == null)
|
|
return null;
|
|
var type = decryptCallbackField.FieldSig.GetFieldType().ToTypeDefOrRef() as TypeDef;
|
|
if (type == null)
|
|
return null;
|
|
return FindDelegateField(mainType, "System.Boolean", string.Format("(System.Int32,System.Int32,System.Int32,System.Int32,System.Diagnostics.StackTrace,{0})", type.FullName));
|
|
}
|
|
|
|
static FieldDef FindDelegateField(TypeDef mainType, string returnType, string parameters) {
|
|
foreach (var field in mainType.Fields) {
|
|
var fieldSig = field.FieldSig;
|
|
if (fieldSig == null)
|
|
continue;
|
|
var fieldType = fieldSig.Type.ToTypeDefOrRef() as TypeDef;
|
|
if (fieldType == null)
|
|
continue;
|
|
if (fieldType.BaseType != null && fieldType.BaseType.FullName != "System.MulticastDelegate")
|
|
continue;
|
|
var invokeMethod = fieldType.FindMethod("Invoke");
|
|
if (!DotNetUtils.IsMethod(invokeMethod, returnType, parameters))
|
|
continue;
|
|
|
|
return field;
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
}
|