de4dot-cex/de4dot.code/deobfuscators/SmartAssembly/TamperProtectionRemover.cs

357 lines
9.4 KiB
C#
Raw Permalink Normal View History

2011-09-22 10:55:30 +08:00
/*
2015-10-30 05:45:26 +08:00
Copyright (C) 2011-2015 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;
using System.Collections.Generic;
using dnlib.DotNet;
using dnlib.DotNet.Emit;
2011-09-22 10:55:30 +08:00
using de4dot.blocks;
namespace de4dot.code.deobfuscators.SmartAssembly {
2011-09-22 10:55:30 +08:00
class TamperProtectionRemover {
2012-11-19 00:07:02 +08:00
ModuleDefMD module;
List<MethodDef> pinvokeMethods = new List<MethodDef>();
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
enum Type {
V1,
V2,
}
public IList<MethodDef> PinvokeMethods {
2011-09-22 10:55:30 +08:00
get { return pinvokeMethods; }
}
2012-11-19 00:07:02 +08:00
public TamperProtectionRemover(ModuleDefMD module) {
2011-09-22 10:55:30 +08:00
this.module = module;
}
2013-01-19 20:03:57 +08:00
public bool Remove(Blocks blocks) {
2011-09-22 10:55:30 +08:00
if (blocks.Method.Name != ".cctor")
return false;
2013-01-19 20:03:57 +08:00
return RemoveTamperProtection(blocks);
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
bool IsTamperProtected(IEnumerable<Block> allBlocks) {
2011-09-22 10:55:30 +08:00
foreach (var block in allBlocks) {
foreach (var instr in block.Instructions) {
if (instr.OpCode != OpCodes.Ldstr)
continue;
var s = instr.Operand as string;
if (s == "Assembly has been tampered")
return true;
}
}
return false;
}
class BlockInfo {
public Block Block { get; set; }
public int Start { get; set; }
public int End { get; set; }
}
2011-11-24 12:25:34 +08:00
class TamperBlocks {
public Type type;
public MethodDef pinvokeMethod;
2011-11-24 12:25:34 +08:00
public BlockInfo first;
public BlockInfo second;
public BlockInfo bad;
}
2013-01-19 20:03:57 +08:00
TamperBlocks FindTamperBlocks(Blocks blocks, IList<Block> allBlocks) {
2011-11-24 12:25:34 +08:00
var tamperBlocks = new TamperBlocks();
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
if (!FindFirstBlocks(tamperBlocks, allBlocks, blocks.Locals))
2011-09-22 10:55:30 +08:00
return null;
2011-11-24 12:25:34 +08:00
var second = tamperBlocks.second;
2013-01-19 20:03:57 +08:00
var badBlock = second.Block.LastInstr.IsBrfalse() ? second.Block.Targets[0] : second.Block.FallThrough;
tamperBlocks.bad = FindBadBlock(badBlock);
2011-11-24 12:25:34 +08:00
if (tamperBlocks.bad == null)
2011-09-22 10:55:30 +08:00
return null;
2011-11-24 12:25:34 +08:00
return tamperBlocks;
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
bool FindFirstBlocks(TamperBlocks tamperBlocks, IList<Block> allBlocks, IList<Local> locals) {
2011-09-22 10:55:30 +08:00
foreach (var b in allBlocks) {
2011-11-24 12:25:34 +08:00
try {
2013-01-19 20:03:57 +08:00
if (FindFirstBlocks(b, tamperBlocks, allBlocks, locals))
2011-11-24 12:25:34 +08:00
return true;
}
catch (ArgumentOutOfRangeException) {
2011-09-22 10:55:30 +08:00
continue;
2011-11-24 12:25:34 +08:00
}
}
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
return false;
}
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
static int FindCallMethod(Block block, int index, bool keepLooking, Func<IMethod, bool> func) {
2011-11-24 12:25:34 +08:00
var instrs = block.Instructions;
for (int i = index; i < instrs.Count; i++) {
var instr = instrs[i];
if (instr.OpCode.Code != Code.Call && instr.OpCode.Code != Code.Callvirt)
continue;
2011-09-22 10:55:30 +08:00
2012-11-19 00:07:02 +08:00
var calledMethod = instr.Operand as IMethod;
2011-11-24 12:25:34 +08:00
if (calledMethod != null && func(calledMethod))
return i;
if (!keepLooking)
return -1;
}
return -1;
}
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
bool FindFirstBlocks(Block block, TamperBlocks tamperBlocks, IList<Block> allBlocks, IList<Local> locals) {
if (!block.LastInstr.IsBrfalse())
2011-11-24 12:25:34 +08:00
return false;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
/*
* ldc.i4.0
* stloc X
* call GetExecutingAssembly()
* stloc Y
* ldloc Y
* callvirt Location
* ldc.i4.1
* ldloca X
* call StrongNameSignatureVerificationEx
* pop / brfalse bad_code
* ldloc X
* brfalse bad_code
* ldloc Y
* callvirt FullName()
* ldstr "......"
* callvirt EndsWith(string)
* brfalse bad_code / brtrue good_code
*/
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
var instrs = block.Instructions;
int end = instrs.Count - 1;
Instr instr;
2012-11-19 00:07:02 +08:00
IMethod method;
2011-11-24 12:25:34 +08:00
tamperBlocks.type = Type.V1;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
int index = 0;
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
int start = FindCallMethod(block, index, true, (calledMethod) => calledMethod.ToString() == "System.Reflection.Assembly System.Reflection.Assembly::GetExecutingAssembly()");
2011-11-24 12:25:34 +08:00
if (start < 0)
return false;
index = start + 1;
instr = instrs[--start];
2013-01-19 20:03:57 +08:00
if (!instr.IsStloc())
2011-11-24 12:25:34 +08:00
return false;
2013-01-19 20:03:57 +08:00
var loc0 = Instr.GetLocalVar(locals, instr);
2011-11-24 12:25:34 +08:00
instr = instrs[--start];
2013-01-19 20:03:57 +08:00
if (!instr.IsLdcI4())
2011-11-24 12:25:34 +08:00
return false;
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
index = FindCallMethod(block, index, false, (calledMethod) => calledMethod.ToString() == "System.String System.Reflection.Assembly::get_Location()");
2011-11-24 12:25:34 +08:00
if (index < 0)
return false;
index++;
2011-09-22 10:55:30 +08:00
2013-01-19 20:03:57 +08:00
index = FindCallMethod(block, index, false, (calledMethod) => {
tamperBlocks.pinvokeMethod = DotNetUtils.GetMethod(module, calledMethod);
return DotNetUtils.IsPinvokeMethod(tamperBlocks.pinvokeMethod, "mscorwks", "StrongNameSignatureVerificationEx");
2011-11-24 12:25:34 +08:00
});
if (index < 0)
return false;
index++;
2013-01-19 20:03:57 +08:00
if (!instrs[index].IsBrfalse()) {
2011-11-24 12:25:34 +08:00
if (instrs[index].OpCode.Code != Code.Pop)
return false;
instr = instrs[index + 1];
2013-01-19 20:03:57 +08:00
if (!instr.IsLdloc() || Instr.GetLocalVar(locals, instr) != loc0)
2011-11-24 12:25:34 +08:00
return false;
2013-01-19 20:03:57 +08:00
if (!instrs[index + 2].IsBrfalse())
2011-11-24 12:25:34 +08:00
return false;
tamperBlocks.type = Type.V1;
tamperBlocks.first = new BlockInfo {
Block = block,
Start = start,
End = end,
};
}
else {
tamperBlocks.type = Type.V2;
tamperBlocks.first = new BlockInfo {
Block = block,
Start = start,
End = end,
};
block = block.FallThrough;
if (block == null)
return false;
instrs = block.Instructions;
index = 0;
instr = instrs[index];
2013-01-19 20:03:57 +08:00
if (!instr.IsLdloc() || Instr.GetLocalVar(locals, instr) != loc0)
2011-11-24 12:25:34 +08:00
return false;
2013-01-19 20:03:57 +08:00
if (!instrs[index + 1].IsBrfalse())
2011-11-24 12:25:34 +08:00
return false;
}
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
block = block.FallThrough;
instrs = block.Instructions;
start = end = 0;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
instr = instrs[end++];
2013-01-19 20:03:57 +08:00
if (!instr.IsLdloc())
2011-11-24 12:25:34 +08:00
return false;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
instr = instrs[end++];
if (instr.OpCode != OpCodes.Callvirt)
return false;
2012-11-19 00:07:02 +08:00
method = instr.Operand as IMethod;
if (method == null || method.ToString() != "System.String System.Reflection.Assembly::get_FullName()")
2011-11-24 12:25:34 +08:00
return false;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
instr = instrs[end++];
if (instr.OpCode != OpCodes.Ldstr)
return false;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
instr = instrs[end++];
if (instr.OpCode != OpCodes.Callvirt)
return false;
2012-11-19 00:07:02 +08:00
method = instr.Operand as IMethod;
if (method == null || method.ToString() != "System.Boolean System.String::EndsWith(System.String)")
2011-11-24 12:25:34 +08:00
return false;
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
instr = instrs[end++];
2013-01-19 20:03:57 +08:00
if (!instr.IsBrfalse() && !instr.IsBrtrue())
2011-11-24 12:25:34 +08:00
return false;
end--;
tamperBlocks.second = new BlockInfo {
Block = block,
Start = start,
End = end,
};
return true;
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
BlockInfo FindBadBlock(Block last) {
2011-09-22 10:55:30 +08:00
/*
* ldstr "........."
* newobj System.Security.SecurityException(string)
* throw
*/
var instrs = last.Instructions;
if (instrs.Count != 3)
return null;
Instr instr;
int start = 0;
int end = 0;
instr = instrs[end++];
if (instr.OpCode != OpCodes.Ldstr)
return null;
instr = instrs[end++];
if (instr.OpCode != OpCodes.Newobj)
return null;
2012-11-19 00:07:02 +08:00
var method = instr.Operand as IMethod;
if (method == null || method.ToString() != "System.Void System.Security.SecurityException::.ctor(System.String)")
2011-09-22 10:55:30 +08:00
return null;
instr = instrs[end++];
if (instr.OpCode != OpCodes.Throw)
return null;
end--;
return new BlockInfo {
Block = last,
Start = start,
End = end,
};
}
2013-01-19 20:03:57 +08:00
bool RemoveTamperProtection(Blocks blocks) {
var allBlocks = blocks.MethodBlocks.GetAllBlocks();
var tamperBlocks = FindTamperBlocks(blocks, allBlocks);
2011-09-22 10:55:30 +08:00
2011-11-24 12:25:34 +08:00
if (tamperBlocks == null) {
2013-01-19 20:03:57 +08:00
if (IsTamperProtected(allBlocks))
Logger.w("Could not remove tamper protection code: {0} ({1:X8})", Utils.RemoveNewlines(blocks.Method), blocks.Method.MDToken.ToUInt32());
2011-09-22 10:55:30 +08:00
return false;
}
2011-11-24 12:25:34 +08:00
switch (tamperBlocks.type) {
case Type.V1:
2013-01-19 20:03:57 +08:00
RemoveTamperV1(tamperBlocks);
2011-11-24 12:25:34 +08:00
break;
case Type.V2:
2013-01-19 20:03:57 +08:00
RemoveTamperV2(tamperBlocks);
2011-11-24 12:25:34 +08:00
break;
default:
throw new ApplicationException("Unknown type");
}
pinvokeMethods.Add(tamperBlocks.pinvokeMethod);
return true;
}
2013-01-19 20:03:57 +08:00
void RemoveTamperV1(TamperBlocks tamperBlocks) {
2011-11-24 12:25:34 +08:00
var first = tamperBlocks.first;
var second = tamperBlocks.second;
var bad = tamperBlocks.bad;
2013-01-19 20:03:57 +08:00
var goodBlock = second.Block.LastInstr.IsBrtrue() ? second.Block.Targets[0] : second.Block.FallThrough;
2011-11-24 12:25:34 +08:00
2011-09-22 10:55:30 +08:00
if (first.Block.Targets.Count != 1 || first.Block.Targets[0] != bad.Block)
throw new ApplicationException("Invalid state");
2011-11-24 12:25:34 +08:00
2013-01-19 20:03:57 +08:00
first.Block.Remove(first.Start, first.End - first.Start + 1);
first.Block.ReplaceLastInstrsWithBranch(0, goodBlock);
RemoveDeadBlock(second.Block);
RemoveDeadBlock(bad.Block);
2011-11-24 12:25:34 +08:00
}
2013-01-19 20:03:57 +08:00
void RemoveTamperV2(TamperBlocks tamperBlocks) {
2011-11-24 12:25:34 +08:00
var first = tamperBlocks.first;
var second = tamperBlocks.second.Block;
var bad = tamperBlocks.bad.Block;
var firstFallthrough = first.Block.FallThrough;
2013-01-19 20:03:57 +08:00
var goodBlock = second.LastInstr.IsBrtrue() ? second.Targets[0] : second.FallThrough;
2011-11-24 12:25:34 +08:00
if (first.Block.Targets.Count != 1 || first.Block.Targets[0] != bad)
2011-09-22 10:55:30 +08:00
throw new ApplicationException("Invalid state");
2013-01-19 20:03:57 +08:00
first.Block.Remove(first.Start, first.End - first.Start + 1);
first.Block.ReplaceLastInstrsWithBranch(0, goodBlock);
RemoveDeadBlock(firstFallthrough);
RemoveDeadBlock(second);
RemoveDeadBlock(bad);
2011-09-22 10:55:30 +08:00
}
2013-01-19 20:03:57 +08:00
void RemoveDeadBlock(Block block) {
2011-12-15 23:17:04 +08:00
var parent = block.Parent;
2011-09-22 10:55:30 +08:00
if (parent != null) // null if already dead
2013-01-19 20:03:57 +08:00
parent.RemoveDeadBlock(block);
2011-09-22 10:55:30 +08:00
}
}
}