de4dot-cex/AssemblyData/AssemblyService.cs

154 lines
4.6 KiB
C#
Raw Normal View History

2011-09-22 10:55:30 +08:00
/*
2013-01-02 00:03:16 +08:00
Copyright (C) 2011-2013 de4dot@gmail.com
2011-09-22 10:55:30 +08:00
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;
2011-09-28 10:30:00 +08:00
using System.Collections.Generic;
2011-09-22 10:55:30 +08:00
using System.Reflection;
using System.Threading;
using dnlib.DotNet;
using de4dot.blocks;
using de4dot.mdecrypt;
2011-09-22 10:55:30 +08:00
namespace AssemblyData {
public class AssemblyService : MarshalByRefObject, IAssemblyService {
IStringDecrypter stringDecrypter = null;
ManualResetEvent exitEvent = new ManualResetEvent(false);
Assembly assembly = null;
2011-09-28 10:30:00 +08:00
AssemblyResolver assemblyResolver = new AssemblyResolver();
bool installCompileMethodCalled = false;
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
public void DoNothing() {
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
public void Exit() {
exitEvent.Set();
}
2013-01-19 20:03:57 +08:00
public void WaitExit() {
exitEvent.WaitOne();
}
public override object InitializeLifetimeService() {
return null;
}
2013-01-19 20:03:57 +08:00
void CheckStringDecrypter() {
2011-09-22 10:55:30 +08:00
if (stringDecrypter == null)
throw new ApplicationException("setStringDecrypterType() hasn't been called yet.");
}
2013-01-19 20:03:57 +08:00
void CheckAssembly() {
2011-09-22 10:55:30 +08:00
if (assembly == null)
throw new ApplicationException("loadAssembly() hasn't been called yet.");
}
2013-01-19 20:03:57 +08:00
public void LoadAssembly(string filename) {
2011-09-22 10:55:30 +08:00
if (assembly != null)
throw new ApplicationException("Only one assembly can be explicitly loaded");
try {
2013-01-19 20:03:57 +08:00
assembly = assemblyResolver.Load(filename);
2011-09-22 10:55:30 +08:00
}
catch (BadImageFormatException) {
throw new ApplicationException(string.Format("Could not load assembly {0}. Maybe it's 32-bit or 64-bit only?", filename));
}
}
2013-01-19 20:03:57 +08:00
public void SetStringDecrypterType(StringDecrypterType type) {
2011-09-22 10:55:30 +08:00
if (stringDecrypter != null)
throw new ApplicationException("StringDecrypterType already set");
switch (type) {
case StringDecrypterType.Delegate:
stringDecrypter = new DelegateStringDecrypter();
break;
2011-09-24 16:26:29 +08:00
case StringDecrypterType.Emulate:
stringDecrypter = new EmuStringDecrypter();
break;
2011-09-22 10:55:30 +08:00
default:
throw new ApplicationException(string.Format("Unknown StringDecrypterType {0}", type));
}
}
2013-01-19 20:03:57 +08:00
public int DefineStringDecrypter(int methodToken) {
CheckStringDecrypter();
var methodInfo = FindMethod(methodToken);
2011-09-22 10:55:30 +08:00
if (methodInfo == null)
throw new ApplicationException(string.Format("Could not find method {0:X8}", methodToken));
if (methodInfo.ReturnType != typeof(string) && methodInfo.ReturnType != typeof(object))
throw new ApplicationException(string.Format("Method return type must be string or object: {0}", methodInfo));
2013-01-19 20:03:57 +08:00
return stringDecrypter.DefineStringDecrypter(methodInfo);
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
public object[] DecryptStrings(int stringDecrypterMethod, object[] args, int callerToken) {
CheckStringDecrypter();
var caller = GetCaller(callerToken);
2011-09-22 10:55:30 +08:00
foreach (var arg in args)
2013-01-19 20:03:57 +08:00
SimpleData.Unpack((object[])arg);
return SimpleData.Pack(stringDecrypter.DecryptStrings(stringDecrypterMethod, args, caller));
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
MethodBase GetCaller(int callerToken) {
try {
return assembly.GetModules()[0].ResolveMethod(callerToken);
}
catch {
return null;
}
}
2013-01-19 20:03:57 +08:00
MethodInfo FindMethod(int methodToken) {
CheckAssembly();
2011-09-22 10:55:30 +08:00
foreach (var module in assembly.GetModules()) {
var method = module.ResolveMethod(methodToken) as MethodInfo;
if (method != null)
return method;
2011-09-22 10:55:30 +08:00
}
return null;
}
2013-01-19 20:03:57 +08:00
public void InstallCompileMethod(DecryptMethodsInfo decryptMethodsInfo) {
if (installCompileMethodCalled)
throw new ApplicationException("installCompileMethod() has already been called");
installCompileMethodCalled = true;
DynamicMethodsDecrypter.Instance.DecryptMethodsInfo = decryptMethodsInfo;
2013-01-19 20:03:57 +08:00
DynamicMethodsDecrypter.Instance.InstallCompileMethod();
}
2013-01-19 20:03:57 +08:00
public void LoadObfuscator(string filename) {
LoadAssembly(filename);
DynamicMethodsDecrypter.Instance.Module = assembly.ManifestModule;
2013-01-19 20:03:57 +08:00
DynamicMethodsDecrypter.Instance.LoadObfuscator();
}
2013-01-19 20:03:57 +08:00
public bool CanDecryptMethods() {
CheckAssembly();
return DynamicMethodsDecrypter.Instance.CanDecryptMethods();
}
2013-01-19 20:03:57 +08:00
public DumpedMethods DecryptMethods() {
CheckAssembly();
return DynamicMethodsDecrypter.Instance.DecryptMethods();
}
2011-09-22 10:55:30 +08:00
}
}