252 lines
7.4 KiB
C#
252 lines
7.4 KiB
C#
/*
|
|
Copyright (C) 2011-2014 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.IO;
|
|
using dnlib.IO;
|
|
using dnlib.DotNet;
|
|
using dnlib.DotNet.Emit;
|
|
using de4dot.blocks;
|
|
|
|
namespace de4dot.code.deobfuscators.Babel_NET {
|
|
class MethodsDecrypter {
|
|
ModuleDefMD module;
|
|
ResourceDecrypter resourceDecrypter;
|
|
IDeobfuscatorContext deobfuscatorContext;
|
|
Dictionary<string, ImageReader> imageReaders = new Dictionary<string, ImageReader>(StringComparer.Ordinal);
|
|
TypeDef methodsDecrypterCreator;
|
|
TypeDef methodsDecrypter;
|
|
MethodDef decryptExecuteMethod;
|
|
EmbeddedResource encryptedResource;
|
|
|
|
public bool Detected {
|
|
get { return methodsDecrypterCreator != null; }
|
|
}
|
|
|
|
public MethodsDecrypter(ModuleDefMD module, ResourceDecrypter resourceDecrypter, IDeobfuscatorContext deobfuscatorContext) {
|
|
this.module = module;
|
|
this.resourceDecrypter = resourceDecrypter;
|
|
this.deobfuscatorContext = deobfuscatorContext;
|
|
}
|
|
|
|
public void Find() {
|
|
var requiredFields = new string[] {
|
|
"System.Threading.ReaderWriterLock",
|
|
"System.Collections.Hashtable",
|
|
};
|
|
foreach (var type in module.GetTypes()) {
|
|
var fieldTypes = new FieldTypes(type);
|
|
if (!fieldTypes.All(requiredFields))
|
|
continue;
|
|
if (type.FindMethod("Finalize") == null)
|
|
continue;
|
|
var executeMethod = DotNetUtils.GetMethod(type, "System.Object", "(System.String,System.Object[])");
|
|
if (executeMethod == null || !executeMethod.IsStatic || executeMethod.Body == null)
|
|
continue;
|
|
|
|
var decrypterType = FindMethodsDecrypterType(type);
|
|
if (decrypterType == null)
|
|
continue;
|
|
|
|
resourceDecrypter.DecryptMethod = FindDecryptMethod(decrypterType);
|
|
|
|
methodsDecrypterCreator = type;
|
|
methodsDecrypter = decrypterType;
|
|
decryptExecuteMethod = executeMethod;
|
|
return;
|
|
}
|
|
}
|
|
|
|
static MethodDef FindDecryptMethod(TypeDef type) {
|
|
foreach (var method in type.Methods) {
|
|
var decryptMethod = ResourceDecrypter.FindDecrypterMethod(method);
|
|
if (decryptMethod != null)
|
|
return decryptMethod;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
TypeDef FindMethodsDecrypterType(TypeDef type) {
|
|
foreach (var field in type.Fields) {
|
|
var fieldType = DotNetUtils.GetType(module, field.FieldSig.GetFieldType());
|
|
if (fieldType == null)
|
|
continue;
|
|
if (fieldType.FindMethod("Finalize") == null)
|
|
continue;
|
|
if (!new FieldTypes(fieldType).Exists("System.Collections.Hashtable"))
|
|
continue;
|
|
if (DotNetUtils.GetMethod(fieldType, "System.String", "()") == null)
|
|
continue;
|
|
|
|
return fieldType;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public void Initialize(ISimpleDeobfuscator simpleDeobfuscator, IDeobfuscator deob) {
|
|
if (methodsDecrypter == null)
|
|
return;
|
|
|
|
encryptedResource = BabelUtils.FindEmbeddedResource(module, methodsDecrypter, simpleDeobfuscator, deob);
|
|
if (encryptedResource != null)
|
|
AddImageReader("", resourceDecrypter.Decrypt(encryptedResource.Data.ReadAllBytes()));
|
|
}
|
|
|
|
ImageReader AddImageReader(string name, byte[] data) {
|
|
var imageReader = new ImageReader(deobfuscatorContext, module, data);
|
|
if (!imageReader.Initialize()) {
|
|
Logger.w("Could not read encrypted methods");
|
|
return null;
|
|
}
|
|
if (imageReaders.ContainsKey(name))
|
|
throw new ApplicationException(string.Format("ImageReader for name '{0}' already exists", name));
|
|
imageReaders[name] = imageReader;
|
|
return imageReader;
|
|
}
|
|
|
|
class EncryptInfo {
|
|
public string encryptedMethodName;
|
|
public string feature;
|
|
public MethodDef method;
|
|
|
|
public string FullName {
|
|
get {
|
|
if (string.IsNullOrEmpty(feature))
|
|
return encryptedMethodName;
|
|
return string.Format("{0}:{1}", feature, encryptedMethodName);
|
|
}
|
|
}
|
|
|
|
public EncryptInfo(string encryptedMethodName, string feature, MethodDef method) {
|
|
this.encryptedMethodName = encryptedMethodName;
|
|
this.feature = feature;
|
|
this.method = method;
|
|
}
|
|
|
|
public override string ToString() {
|
|
if (feature != "")
|
|
return string.Format("{0}:{1} {2:X8}", feature, encryptedMethodName, method.MDToken.ToInt32());
|
|
else
|
|
return string.Format("{0} {1:X8}", encryptedMethodName, method.MDToken.ToInt32());
|
|
}
|
|
}
|
|
|
|
public void decrypt() {
|
|
int numNonDecryptedMethods = 0;
|
|
int totalEncryptedMethods = 0;
|
|
foreach (var info in GetEncryptedMethods()) {
|
|
totalEncryptedMethods++;
|
|
var imageReader = GetImageReader(info.feature);
|
|
if (imageReader == null) {
|
|
numNonDecryptedMethods++;
|
|
continue;
|
|
}
|
|
Logger.v("Decrypting method {0:X8}", info.method.MDToken.ToInt32());
|
|
imageReader.Restore(info.FullName, info.method);
|
|
}
|
|
if (numNonDecryptedMethods > 0)
|
|
Logger.w("{0}/{1} methods not decrypted", numNonDecryptedMethods, totalEncryptedMethods);
|
|
}
|
|
|
|
ImageReader GetImageReader(string feature) {
|
|
ImageReader imageReader;
|
|
if (imageReaders.TryGetValue(feature, out imageReader))
|
|
return imageReader;
|
|
|
|
return CreateImageReader(feature);
|
|
}
|
|
|
|
ImageReader CreateImageReader(string feature) {
|
|
if (string.IsNullOrEmpty(feature))
|
|
return null;
|
|
|
|
try {
|
|
var encrypted = File.ReadAllBytes(GetFile(Path.GetDirectoryName(module.Location), feature));
|
|
var decrypted = resourceDecrypter.Decrypt(encrypted);
|
|
return AddImageReader(feature, decrypted);
|
|
}
|
|
catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
static string GetFile(string dir, string name) {
|
|
try {
|
|
var di = new DirectoryInfo(dir);
|
|
foreach (var file in di.GetFiles()) {
|
|
if (Utils.StartsWith(file.Name, name, StringComparison.OrdinalIgnoreCase))
|
|
return file.FullName;
|
|
}
|
|
}
|
|
catch {
|
|
}
|
|
return null;
|
|
}
|
|
|
|
List<EncryptInfo> GetEncryptedMethods() {
|
|
var infos = new List<EncryptInfo>();
|
|
|
|
foreach (var type in module.GetTypes()) {
|
|
foreach (var method in type.Methods) {
|
|
EncryptInfo info;
|
|
if (CheckEncryptedMethod(method, out info))
|
|
infos.Add(info);
|
|
}
|
|
}
|
|
|
|
return infos;
|
|
}
|
|
|
|
bool CheckEncryptedMethod(MethodDef method, out EncryptInfo info) {
|
|
info = null;
|
|
if (method.Body == null)
|
|
return false;
|
|
if (!CallsExecuteMethod(method))
|
|
return false;
|
|
|
|
var strings = DotNetUtils.GetCodeStrings(method);
|
|
if (strings.Count != 1)
|
|
throw new ApplicationException(string.Format("Could not find name of encrypted method"));
|
|
|
|
string feature = "";
|
|
string name = strings[0];
|
|
int index = name.IndexOf(':');
|
|
if (index >= 0) {
|
|
feature = name.Substring(0, index);
|
|
name = name.Substring(index + 1);
|
|
}
|
|
|
|
info = new EncryptInfo(name, feature, method);
|
|
return true;
|
|
}
|
|
|
|
bool CallsExecuteMethod(MethodDef method) {
|
|
foreach (var instr in method.Body.Instructions) {
|
|
if (instr.OpCode.Code != Code.Call && instr.OpCode.Code != Code.Callvirt)
|
|
continue;
|
|
if (MethodEqualityComparer.CompareDeclaringTypes.Equals(decryptExecuteMethod, instr.Operand as IMethod))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|