de4dot-cex/de4dot.code/deobfuscators/Babel_NET/MethodsDecrypter.cs
2015-10-29 22:45:26 +01:00

252 lines
7.4 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.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;
}
}
}