mirror of
https://github.com/sp-tarkov/assembly-tool.git
synced 2025-02-13 07:50:47 -05:00
Adds the source code used for this modification, this de4dot source code has been cleaned of any things not needed for deobfuscating the tarkov assembly Co-authored-by: 静穏靄 <170472707+seionmoya@users.noreply.github.com>
215 lines
6.0 KiB
C#
215 lines
6.0 KiB
C#
/*
|
|
Copyright (C) 2011-2015 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.Reflection;
|
|
using dnlib.DotNet;
|
|
|
|
namespace AssemblyData.methodsrewriter {
|
|
static class Resolver {
|
|
static Dictionary<string, AssemblyResolver> assemblyResolvers = new Dictionary<string, AssemblyResolver>(StringComparer.Ordinal);
|
|
static Dictionary<Module, MModule> modules = new Dictionary<Module, MModule>();
|
|
|
|
public static MModule LoadAssembly(Module module) {
|
|
if (modules.TryGetValue(module, out var info))
|
|
return info;
|
|
|
|
info = new MModule(module, ModuleDefMD.Load(module.FullyQualifiedName));
|
|
modules[module] = info;
|
|
return info;
|
|
}
|
|
|
|
static MModule GetModule(ModuleDef moduleDef) {
|
|
foreach (var mm in modules.Values) {
|
|
if (mm.moduleDef == moduleDef)
|
|
return mm;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
static MModule GetModule(AssemblyRef asmRef) {
|
|
foreach (var mm in modules.Values) {
|
|
var asm = mm.moduleDef.Assembly;
|
|
if (asm != null && asm.FullName == asmRef.FullName)
|
|
return mm;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public static MModule GetModule(IScope scope) {
|
|
if (scope.ScopeType == ScopeType.ModuleDef)
|
|
return GetModule((ModuleDef)scope);
|
|
else if (scope.ScopeType == ScopeType.AssemblyRef)
|
|
return GetModule((AssemblyRef)scope);
|
|
|
|
return null;
|
|
}
|
|
|
|
public static MType GetType(IType typeRef) {
|
|
if (typeRef == null)
|
|
return null;
|
|
var module = GetModule(typeRef.Scope);
|
|
if (module != null)
|
|
return module.GetType(typeRef);
|
|
return null;
|
|
}
|
|
|
|
public static MMethod GetMethod(IMethod methodRef) {
|
|
if (methodRef == null)
|
|
return null;
|
|
var module = GetModule(methodRef.DeclaringType.Scope);
|
|
if (module != null)
|
|
return module.GetMethod(methodRef);
|
|
return null;
|
|
}
|
|
|
|
public static MField GetField(IField fieldRef) {
|
|
if (fieldRef == null)
|
|
return null;
|
|
var module = GetModule(fieldRef.DeclaringType.Scope);
|
|
if (module != null)
|
|
return module.GetField(fieldRef);
|
|
return null;
|
|
}
|
|
|
|
public static object GetRtObject(ITokenOperand memberRef) {
|
|
if (memberRef == null)
|
|
return null;
|
|
if (memberRef is ITypeDefOrRef tdr)
|
|
return GetRtType(tdr);
|
|
if (memberRef is IField field && field.FieldSig != null)
|
|
return GetRtField(field);
|
|
if (memberRef is IMethod method && method.MethodSig != null)
|
|
return GetRtMethod(method);
|
|
|
|
throw new ApplicationException($"Unknown MemberRef: {memberRef}");
|
|
}
|
|
|
|
public static Type GetRtType(IType typeRef) {
|
|
var mtype = GetType(typeRef);
|
|
if (mtype != null)
|
|
return mtype.type;
|
|
|
|
return Resolver.Resolve(typeRef);
|
|
}
|
|
|
|
public static FieldInfo GetRtField(IField fieldRef) {
|
|
var mfield = GetField(fieldRef);
|
|
if (mfield != null)
|
|
return mfield.fieldInfo;
|
|
|
|
return Resolver.Resolve(fieldRef);
|
|
}
|
|
|
|
public static MethodBase GetRtMethod(IMethod methodRef) {
|
|
var mmethod = GetMethod(methodRef);
|
|
if (mmethod != null)
|
|
return mmethod.methodBase;
|
|
|
|
return Resolver.Resolve(methodRef);
|
|
}
|
|
|
|
static AssemblyResolver GetAssemblyResolver(ITypeDefOrRef type) {
|
|
var asmName = type.DefinitionAssembly.FullName;
|
|
if (!assemblyResolvers.TryGetValue(asmName, out var resolver))
|
|
assemblyResolvers[asmName] = resolver = new AssemblyResolver(asmName);
|
|
return resolver;
|
|
}
|
|
|
|
static Type Resolve(IType typeRef) {
|
|
if (typeRef == null)
|
|
return null;
|
|
var scopeType = typeRef.ScopeType;
|
|
var resolver = GetAssemblyResolver(scopeType);
|
|
var resolvedType = resolver.Resolve(scopeType);
|
|
if (resolvedType != null)
|
|
return FixType(typeRef, resolvedType);
|
|
throw new ApplicationException($"Could not resolve type {typeRef} ({typeRef.MDToken.Raw:X8}) in assembly {resolver}");
|
|
}
|
|
|
|
static FieldInfo Resolve(IField fieldRef) {
|
|
if (fieldRef == null)
|
|
return null;
|
|
var resolver = GetAssemblyResolver(fieldRef.DeclaringType);
|
|
var fieldInfo = resolver.Resolve(fieldRef);
|
|
if (fieldInfo != null)
|
|
return fieldInfo;
|
|
throw new ApplicationException($"Could not resolve field {fieldRef} ({fieldRef.MDToken.Raw:X8}) in assembly {resolver}");
|
|
}
|
|
|
|
static MethodBase Resolve(IMethod methodRef) {
|
|
if (methodRef == null)
|
|
return null;
|
|
var resolver = GetAssemblyResolver(methodRef.DeclaringType);
|
|
var methodBase = resolver.Resolve(methodRef);
|
|
if (methodBase != null)
|
|
return methodBase;
|
|
throw new ApplicationException($"Could not resolve method {methodRef} ({methodRef.MDToken.Raw:X8}) in assembly {resolver}");
|
|
}
|
|
|
|
static Type FixType(IType typeRef, Type type) {
|
|
var sig = typeRef as TypeSig;
|
|
if (sig == null) {
|
|
if (typeRef is TypeSpec ts)
|
|
sig = ts.TypeSig;
|
|
}
|
|
while (sig != null) {
|
|
switch (sig.ElementType) {
|
|
case ElementType.SZArray:
|
|
type = type.MakeArrayType();
|
|
break;
|
|
|
|
case ElementType.Array:
|
|
type = type.MakeArrayType((int)((ArraySig)sig).Rank);
|
|
break;
|
|
|
|
case ElementType.ByRef:
|
|
type = type.MakeByRefType();
|
|
break;
|
|
|
|
case ElementType.Ptr:
|
|
type = type.MakePointerType();
|
|
break;
|
|
|
|
case ElementType.GenericInst:
|
|
var git = (GenericInstSig)sig;
|
|
var args = new Type[git.GenericArguments.Count];
|
|
bool isGenericTypeDef = true;
|
|
for (int i = 0; i < args.Length; i++) {
|
|
var arg = git.GenericArguments[i];
|
|
if (!(arg is GenericSig))
|
|
isGenericTypeDef = false;
|
|
args[i] = Resolver.Resolve(arg);
|
|
}
|
|
if (!isGenericTypeDef)
|
|
type = type.MakeGenericType(args);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
sig = sig.Next;
|
|
}
|
|
return type;
|
|
}
|
|
}
|
|
}
|