de4dot-cex/AssemblyData/AssemblyService.cs

126 lines
3.7 KiB
C#
Raw Normal View History

2011-09-22 10:55:30 +08:00
/*
2012-01-10 06:02:47 +08:00
Copyright (C) 2011-2012 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;
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();
2011-09-22 10:55:30 +08:00
public void doNothing() {
}
void checkStringDecrypter() {
if (stringDecrypter == null)
throw new ApplicationException("setStringDecrypterType() hasn't been called yet.");
}
void checkAssembly() {
if (assembly == null)
throw new ApplicationException("loadAssembly() hasn't been called yet.");
}
public void loadAssembly(string filename) {
if (assembly != null)
throw new ApplicationException("Only one assembly can be explicitly loaded");
try {
2011-09-28 10:30:00 +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));
}
}
public void setStringDecrypterType(StringDecrypterType type) {
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));
}
}
public int defineStringDecrypter(int methodToken) {
2011-09-22 10:55:30 +08:00
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));
2011-09-22 10:55:30 +08:00
return stringDecrypter.defineStringDecrypter(methodInfo);
}
public object[] decryptStrings(int stringDecrypterMethod, object[] args, int callerToken) {
2011-09-22 10:55:30 +08:00
checkStringDecrypter();
var caller = getCaller(callerToken);
2011-09-22 10:55:30 +08:00
foreach (var arg in args)
SimpleData.unpack((object[])arg);
return SimpleData.pack(stringDecrypter.decryptStrings(stringDecrypterMethod, args, caller));
2011-09-22 10:55:30 +08:00
}
MethodBase getCaller(int callerToken) {
try {
return assembly.GetModules()[0].ResolveMethod(callerToken);
}
catch {
return null;
}
}
2011-09-22 10:55:30 +08:00
public void exit() {
exitEvent.Set();
}
public void waitExit() {
exitEvent.WaitOne();
}
public override object InitializeLifetimeService() {
return null;
}
MethodInfo findMethod(int methodToken) {
2011-09-22 10:55:30 +08:00
checkAssembly();
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;
}
}
}