VMProtect/core/il.cc

7841 lines
239 KiB
C++
Raw Normal View History

2023-05-14 16:21:09 +03:00
#include "../runtime/common.h"
#include "../runtime/crypto.h"
#include "objects.h"
#include "osutils.h"
#include "streams.h"
#include "files.h"
#include "processors.h"
#include "lang.h"
#include "core.h"
#include "pefile.h"
#include "packer.h"
#include "dotnet.h"
#include "dotnetfile.h"
#include "il.h"
/**
* ILCommand
*/
ILCommand::ILCommand(IFunction *owner, OperandSize size, uint64_t address)
: BaseCommand(owner), address_(address), size_(size), type_(icUnknown), operand_value_(0), original_dump_size_(0),
section_options_(0), operand_pos_(0), token_reference_(NULL), ext_vm_entry_(NULL), param_(0), fixup_(NULL)
{
if (address_)
include_option(roClearOriginalCode);
}
ILCommand::ILCommand(IFunction *owner, OperandSize size, ILCommandType type, uint64_t operand_value, IFixup *fixup)
: BaseCommand(owner), address_(0), size_(size), type_(type), operand_value_(operand_value), original_dump_size_(0),
section_options_(0), operand_pos_(0), token_reference_(NULL), ext_vm_entry_(NULL), param_(0), fixup_(fixup)
{
}
ILCommand::ILCommand(IFunction *owner, OperandSize size, const std::string &value)
: BaseCommand(owner, value), address_(0), size_(size), type_(icUnknown), operand_value_(0), original_dump_size_(0),
section_options_(0), operand_pos_(0), token_reference_(NULL), ext_vm_entry_(NULL), param_(0), fixup_(NULL)
{
type_ = icData;
}
ILCommand::ILCommand(IFunction *owner, OperandSize size, const Data &value)
: BaseCommand(owner, value), address_(0), size_(size), type_(icUnknown), operand_value_(0), original_dump_size_(0),
section_options_(0), operand_pos_(0), token_reference_(NULL), ext_vm_entry_(NULL), param_(0), fixup_(NULL)
{
type_ = icData;
}
ILCommand::ILCommand(IFunction *owner, const ILCommand &src)
: BaseCommand(owner, src), section_options_(0), ext_vm_entry_(NULL), fixup_(NULL)
{
address_ = src.address_;
size_ = src.size_;
type_ = src.type_;
operand_value_ = src.operand_value_;
original_dump_size_ = src.original_dump_size_;
operand_pos_ = src.operand_pos_;
token_reference_ = src.token_reference_;
param_ = src.param_;
}
void ILCommand::Init(ILCommandType type, uint64_t operand_value, TokenReference *token_reference)
{
type_ = type;
operand_value_ = operand_value;
token_reference_ = token_reference;
}
void ILCommand::clear()
{
type_ = icUnknown;
operand_value_ = 0;
original_dump_size_ = 0;
operand_pos_ = 0;
token_reference_ = NULL;
BaseCommand::clear();
}
void ILCommand::InitUnknown()
{
clear();
type_ = icUnknown;
PushByte(0);
}
void ILCommand::InitComment(const std::string &comment)
{
type_ = icComment;
set_comment(CommentInfo(ttNone, comment));
}
ISEHandler * ILCommand::seh_handler() const
{
throw std::runtime_error("The method or operation is not implemented.");
}
void ILCommand::set_seh_handler(ISEHandler *handler)
{
throw std::runtime_error("The method or operation is not implemented.");
}
std::string ILCommand::text() const
{
std::string res;
const ILOpCode opcode = ILOpCodes[type_];
res = (type_ == icComment) ? comment_text() : opcode.name;
if (type_ == icData) {
for (size_t i = 0; i < dump_size(); i++) {
if (i > 0)
res.append(",");
res.append(string_format(" %.2X", dump(i)));
}
} else {
switch (opcode.operand_type) {
case InlineNone:
case InlinePhi:
case Inline8:
// do nothing
break;
case ShortInlineI:
case ShortInlineVar:
res.append(string_format(" %.2X", static_cast<uint8_t>(operand_value_)));
break;
case InlineVar:
res.append(string_format(" %.4X", static_cast<uint16_t>(operand_value_)));
break;
case InlineI8:
case InlineR:
res.append(string_format(" %.16llX", operand_value_));
break;
case InlineSwitch:
break;
case ShortInlineBrTarget:
case InlineBrTarget:
res.append(string_format(" %.8X", static_cast<uint32_t>(operand_value_)));
break;
default:
res.append(string_format(" %.8X", static_cast<uint32_t>(operand_value_)));
break;
}
}
return res;
}
CommentInfo ILCommand::comment()
{
CommentInfo res = BaseCommand::comment();
if (res.type != ttUnknown)
return res;
res.type = ttNone;
NETArchitecture *file = owner()->owner() ? dynamic_cast<NETArchitecture *>(owner()->owner()->owner()) : NULL;
if (file) {
switch (ILOpCodes[type_].operand_type) {
case InlineBrTarget:
case ShortInlineBrTarget:
res.value = (next_address() > operand_value_) ? char(2) : char(4);
res.type = ttJmp;
break;
case InlineType:
case InlineField:
case InlineTok:
case InlineSig:
case InlineString:
case InlineMethod:
if (ILToken *token = token_reference_ ? token_reference_->owner()->owner() : NULL) {
switch (token->type()) {
case ttTypeRef:
{
ILTypeRef *type = reinterpret_cast<ILTypeRef *>(token);
std::string name = type->full_name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttImport;
}
break;
case ttTypeDef:
{
ILTypeDef *type = reinterpret_cast<ILTypeDef *>(token);
std::string name = type->full_name();
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttFunction;
}
break;
case ttTypeSpec:
{
ILTypeSpec *type = reinterpret_cast<ILTypeSpec *>(token);
std::string name = type->name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttFunction;
}
break;
case ttField:
{
ILField *field = reinterpret_cast<ILField *>(token);
std::string name = field->full_name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttFunction;
}
break;
case ttMemberRef:
{
ILMemberRef *member = reinterpret_cast<ILMemberRef *>(token);
std::string name = member->full_name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = (member->declaring_type() && member->declaring_type()->type() == ttTypeRef) ? ttImport : ttFunction;
}
break;
case ttStandAloneSig:
{
ILStandAloneSig *signature = reinterpret_cast<ILStandAloneSig*>(token);
std::string name = signature->name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttFunction;
}
break;
case ttUserString:
{
ILUserString *str = reinterpret_cast<ILUserString*>(token);
res.value = string_format("%c \"%s\"", 3, str->name().c_str());
res.type = ttString;
}
break;
case ttMethodDef:
{
ILMethodDef *method = reinterpret_cast<ILMethodDef*>(token);
std::string name = method->full_name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = ttFunction;
}
break;
case ttMethodSpec:
{
ILMethodSpec *method_spec = reinterpret_cast<ILMethodSpec*>(token);
std::string name = method_spec->full_name(true);
res.value = string_format("%c %s", 3, name.c_str());
res.type = (method_spec->parent() && method_spec->parent()->type() == ttMethodDef) ? ttFunction : ttImport;
}
break;
}
if (res.type == ttFunction && ILOpCodes[type_].operand_type == InlineField)
res.type = ttVariable;
}
break;
}
set_comment(res);
}
return res;
}
void ILCommand::CompileToNative()
{
if (type_ == icComment || type_ == icData)
return;
BaseCommand::clear();
switch (type_) {
case icByte: break;
case icWord: break;
case icDword: break;
case icQword: break;
case icCase: break;
case icNop: PushByte(0x00); break;
case icBreak: PushByte(0x01); break;
case icLdarg_0: PushByte(0x02); break;
case icLdarg_1: PushByte(0x03); break;
case icLdarg_2: PushByte(0x04); break;
case icLdarg_3: PushByte(0x05); break;
case icLdloc_0: PushByte(0x06); break;
case icLdloc_1: PushByte(0x07); break;
case icLdloc_2: PushByte(0x08); break;
case icLdloc_3: PushByte(0x09); break;
case icStloc_0: PushByte(0x0a); break;
case icStloc_1: PushByte(0x0b); break;
case icStloc_2: PushByte(0x0c); break;
case icStloc_3: PushByte(0x0d); break;
case icLdarg_s: PushByte(0x0e); break;
case icLdarga_s: PushByte(0x0f); break;
case icStarg_s: PushByte(0x10); break;
case icLdloc_s: PushByte(0x11); break;
case icLdloca_s: PushByte(0x12); break;
case icStloc_s: PushByte(0x13); break;
case icLdnull: PushByte(0x14); break;
case icLdc_i4_m1: PushByte(0x15); break;
case icLdc_i4_0: PushByte(0x16); break;
case icLdc_i4_1: PushByte(0x17); break;
case icLdc_i4_2: PushByte(0x18); break;
case icLdc_i4_3: PushByte(0x19); break;
case icLdc_i4_4: PushByte(0x1a); break;
case icLdc_i4_5: PushByte(0x1b); break;
case icLdc_i4_6: PushByte(0x1c); break;
case icLdc_i4_7: PushByte(0x1d); break;
case icLdc_i4_8: PushByte(0x1e); break;
case icLdc_i4_s: PushByte(0x1f); break;
case icLdc_i4: PushByte(0x20); break;
case icLdc_i8: PushByte(0x21); break;
case icLdc_r4: PushByte(0x22); break;
case icLdc_r8: PushByte(0x23); break;
case icDup: PushByte(0x25); break;
case icPop: PushByte(0x26); break;
case icJmp: PushByte(0x27); break;
case icCall: PushByte(0x28); break;
case icCalli: PushByte(0x29); break;
case icRet: PushByte(0x2a); break;
case icBr_s: PushByte(0x2b); break;
case icBrfalse_s: PushByte(0x2c); break;
case icBrtrue_s: PushByte(0x2d); break;
case icBeq_s: PushByte(0x2e); break;
case icBge_s: PushByte(0x2f); break;
case icBgt_s: PushByte(0x30); break;
case icBle_s: PushByte(0x31); break;
case icBlt_s: PushByte(0x32); break;
case icBne_un_s: PushByte(0x33); break;
case icBge_un_s: PushByte(0x34); break;
case icBgt_un_s: PushByte(0x35); break;
case icBle_un_s: PushByte(0x36); break;
case icBlt_un_s: PushByte(0x37); break;
case icBr: PushByte(0x38); break;
case icBrfalse: PushByte(0x39); break;
case icBrtrue: PushByte(0x3a); break;
case icBeq: PushByte(0x3b); break;
case icBge: PushByte(0x3c); break;
case icBgt: PushByte(0x3d); break;
case icBle: PushByte(0x3e); break;
case icBlt: PushByte(0x3f); break;
case icBne_un: PushByte(0x40); break;
case icBge_un: PushByte(0x41); break;
case icBgt_un: PushByte(0x42); break;
case icBle_un: PushByte(0x43); break;
case icBlt_un: PushByte(0x44); break;
case icSwitch: PushByte(0x45); break;
case icLdind_i1: PushByte(0x46); break;
case icLdind_u1: PushByte(0x47); break;
case icLdind_i2: PushByte(0x48); break;
case icLdind_u2: PushByte(0x49); break;
case icLdind_i4: PushByte(0x4a); break;
case icLdind_u4: PushByte(0x4b); break;
case icLdind_i8: PushByte(0x4c); break;
case icLdind_i: PushByte(0x4d); break;
case icLdind_r4: PushByte(0x4e); break;
case icLdind_r8: PushByte(0x4f); break;
case icLdind_ref: PushByte(0x50); break;
case icStind_ref: PushByte(0x51); break;
case icStind_i1: PushByte(0x52); break;
case icStind_i2: PushByte(0x53); break;
case icStind_i4: PushByte(0x54); break;
case icStind_i8: PushByte(0x55); break;
case icStind_r4: PushByte(0x56); break;
case icStind_r8: PushByte(0x57); break;
case icAdd: PushByte(0x58); break;
case icSub: PushByte(0x59); break;
case icMul: PushByte(0x5a); break;
case icDiv: PushByte(0x5b); break;
case icDiv_un: PushByte(0x5c); break;
case icRem: PushByte(0x5d); break;
case icRem_un: PushByte(0x5e); break;
case icAnd: PushByte(0x5f); break;
case icOr: PushByte(0x60); break;
case icXor: PushByte(0x61); break;
case icShl: PushByte(0x62); break;
case icShr: PushByte(0x63); break;
case icShr_un: PushByte(0x64); break;
case icNeg: PushByte(0x65); break;
case icNot: PushByte(0x66); break;
case icConv_i1: PushByte(0x67); break;
case icConv_i2: PushByte(0x68); break;
case icConv_i4: PushByte(0x69); break;
case icConv_i8: PushByte(0x6a); break;
case icConv_r4: PushByte(0x6b); break;
case icConv_r8: PushByte(0x6c); break;
case icConv_u4: PushByte(0x6d); break;
case icConv_u8: PushByte(0x6e); break;
case icCallvirt: PushByte(0x6f); break;
case icCpobj: PushByte(0x70); break;
case icLdobj: PushByte(0x71); break;
case icLdstr: PushByte(0x72); break;
case icNewobj: PushByte(0x73); break;
case icCastclass: PushByte(0x74); break;
case icIsinst: PushByte(0x75); break;
case icConv_r_un: PushByte(0x76); break;
case icUnbox: PushByte(0x79); break;
case icThrow: PushByte(0x7a); break;
case icLdfld: PushByte(0x7b); break;
case icLdflda: PushByte(0x7c); break;
case icStfld: PushByte(0x7d); break;
case icLdsfld: PushByte(0x7e); break;
case icLdsflda: PushByte(0x7f); break;
case icStsfld: PushByte(0x80); break;
case icStobj: PushByte(0x81); break;
case icConv_ovf_i1_un: PushByte(0x82); break;
case icConv_ovf_i2_un: PushByte(0x83); break;
case icConv_ovf_i4_un: PushByte(0x84); break;
case icConv_ovf_i8_un: PushByte(0x85); break;
case icConv_ovf_u1_un: PushByte(0x86); break;
case icConv_ovf_u2_un: PushByte(0x87); break;
case icConv_ovf_u4_un: PushByte(0x88); break;
case icConv_ovf_u8_un: PushByte(0x89); break;
case icConv_ovf_i_un: PushByte(0x8a); break;
case icConv_ovf_u_un: PushByte(0x8b); break;
case icBox: PushByte(0x8c); break;
case icNewarr: PushByte(0x8d); break;
case icLdlen: PushByte(0x8e); break;
case icLdelema: PushByte(0x8f); break;
case icLdelem_i1: PushByte(0x90); break;
case icLdelem_u1: PushByte(0x91); break;
case icLdelem_i2: PushByte(0x92); break;
case icLdelem_u2: PushByte(0x93); break;
case icLdelem_i4: PushByte(0x94); break;
case icLdelem_u4: PushByte(0x95); break;
case icLdelem_i8: PushByte(0x96); break;
case icLdelem_i: PushByte(0x97); break;
case icLdelem_r4: PushByte(0x98); break;
case icLdelem_r8: PushByte(0x99); break;
case icLdelem_ref: PushByte(0x9a); break;
case icStelem_i: PushByte(0x9b); break;
case icStelem_i1: PushByte(0x9c); break;
case icStelem_i2: PushByte(0x9d); break;
case icStelem_i4: PushByte(0x9e); break;
case icStelem_i8: PushByte(0x9f); break;
case icStelem_r4: PushByte(0xa0); break;
case icStelem_r8: PushByte(0xa1); break;
case icStelem_ref: PushByte(0xa2); break;
case icLdelem: PushByte(0xa3); break;
case icStelem: PushByte(0xa4); break;
case icUnbox_any: PushByte(0xa5); break;
case icConv_ovf_i1: PushByte(0xb3); break;
case icConv_ovf_u1: PushByte(0xb4); break;
case icConv_ovf_i2: PushByte(0xb5); break;
case icConv_ovf_u2: PushByte(0xb6); break;
case icConv_ovf_i4: PushByte(0xb7); break;
case icConv_ovf_u4: PushByte(0xb8); break;
case icConv_ovf_i8: PushByte(0xb9); break;
case icConv_ovf_u8: PushByte(0xba); break;
case icLdtoken: PushByte(0xd0); break;
case icConv_u2: PushByte(0xd1); break;
case icConv_u1: PushByte(0xd2); break;
case icConv_i: PushByte(0xd3); break;
case icConv_ovf_i: PushByte(0xd4); break;
case icConv_ovf_u: PushByte(0xd5); break;
case icAdd_ovf: PushByte(0xd6); break;
case icAdd_ovf_un: PushByte(0xd7); break;
case icMul_ovf: PushByte(0xd8); break;
case icMul_ovf_un: PushByte(0xd9); break;
case icSub_ovf: PushByte(0xda); break;
case icSub_ovf_un: PushByte(0xdb); break;
case icEndfinally: PushByte(0xdc); break;
case icLeave: PushByte(0xdd); break;
case icLeave_s: PushByte(0xde); break;
case icStind_i: PushByte(0xdf); break;
case icConv_u: PushByte(0xe0); break;
case icArglist: PushByte(0xfe); PushByte(0x00); break;
case icCeq: PushByte(0xfe); PushByte(0x01); break;
case icCgt: PushByte(0xfe); PushByte(0x02); break;
case icCgt_un: PushByte(0xfe); PushByte(0x03); break;
case icClt: PushByte(0xfe); PushByte(0x04); break;
case icClt_un: PushByte(0xfe); PushByte(0x05); break;
case icLdftn: PushByte(0xfe); PushByte(0x06); break;
case icLdvirtftn: PushByte(0xfe); PushByte(0x07); break;
case icLdarg: PushByte(0xfe); PushByte(0x09); break;
case icLdarga: PushByte(0xfe); PushByte(0x0a); break;
case icStarg: PushByte(0xfe); PushByte(0x0b); break;
case icLdloc: PushByte(0xfe); PushByte(0x0c); break;
case icLdloca: PushByte(0xfe); PushByte(0x0d); break;
case icStloc: PushByte(0xfe); PushByte(0x0e); break;
case icLocalloc: PushByte(0xfe); PushByte(0x0f); break;
case icEndfilter: PushByte(0xfe); PushByte(0x11); break;
case icUnaligned: PushByte(0xfe); PushByte(0x12); break;
case icVolatile: PushByte(0xfe); PushByte(0x13); break;
case icTail: PushByte(0xfe); PushByte(0x14); break;
case icInitobj: PushByte(0xfe); PushByte(0x15); break;
case icConstrained: PushByte(0xfe); PushByte(0x16); break;
case icCpblk: PushByte(0xfe); PushByte(0x17); break;
case icInitblk: PushByte(0xfe); PushByte(0x18); break;
case icNo: PushByte(0xfe); PushByte(0x19); break;
case icRethrow: PushByte(0xfe); PushByte(0x1a); break;
case icSizeof: PushByte(0xfe); PushByte(0x1c); break;
case icRefanytype: PushByte(0xfe); PushByte(0x1d); break;
case icReadonly: PushByte(0xfe); PushByte(0x1e); break;
default:
throw std::runtime_error("Runtime error at CompileToNative: " + text());
}
operand_pos_ = dump_size();
switch (ILOpCodes[type_].operand_type) {
case InlineNone:
case InlinePhi:
case Inline8:
// do nothing
break;
case ShortInlineI:
case ShortInlineVar:
PushByte(static_cast<uint8_t>(operand_value_));
break;
case InlineVar:
PushWord(static_cast<uint16_t>(operand_value_));
break;
case InlineI8:
case InlineR:
PushQWord(operand_value_);
break;
case InlineSwitch:
PushDWord(static_cast<uint32_t>(operand_value_));
break;
case ShortInlineBrTarget:
PushByte(static_cast<uint8_t>(operand_value_ - next_address() - 1));
break;
case InlineBrTarget:
PushDWord(static_cast<uint32_t>(operand_value_ - next_address() - 4));
break;
default:
PushDWord(static_cast<uint32_t>(operand_value_));
break;
}
if (options() & roFillNop) {
for (size_t j = dump_size(); j < original_dump_size_; j++) {
PushByte(0x00);
}
}
}
void ILCommand::CompileLink(const CompileContext &ctx)
{
uint64_t value;
if (block()->type() & mtExecutable) {
// native block
if (!link() || link()->operand_index() == -1)
return;
ICommand *to_command = link()->to_command();
if (to_command) {
value = (to_command->block()->type() & mtExecutable) ? to_command->address() : to_command->ext_vm_address();
} else if (link()->type() == ltDelta) {
value = link()->to_address();
} else {
return;
}
switch (link()->type()) {
case ltDelta:
value -= address();
break;
case ltCase:
value -= link()->parent_command()->next_address() + reinterpret_cast<ILCommand *>(link()->parent_command())->operand_value() * sizeof(uint32_t);
break;
}
set_operand_value(link()->operand_index(), link()->Encrypt(value));
CompileToNative();
} else {
// VM block
for (size_t i = 0; i < internal_links_.count(); i++) {
InternalLink *internal_link = internal_links_.item(i);
ILVMCommand *vm_command = reinterpret_cast<ILVMCommand *>(internal_link->from_command());
switch (internal_link->type()) {
case vlCRCTableAddress:
value = reinterpret_cast<ILFunctionList *>(ctx.file->function_list())->runtime_crc_table()->entry()->address() - ctx.file->image_base();
break;
case vlCRCTableCount:
value = reinterpret_cast<ILFunctionList *>(ctx.file->function_list())->runtime_crc_table()->region_count();
break;
default:
ILCommand *command = reinterpret_cast<ILCommand*>(internal_link->to_command());
if (!command)
continue;
value = command->vm_address() - ctx.file->image_base();
}
vm_command->set_value(value);
vm_command->Compile();
}
if (vm_links_.empty())
return;
ICommand *to_command = link()->to_command();
ICommand *next_command = link()->next_command();
uint64_t image_base = ctx.file->image_base();
switch (link()->type()) {
case ltJmp:
case ltCase:
set_link_value(0, to_command->vm_address() - image_base);
break;
case ltJmpWithFlag:
set_link_value(0, to_command->vm_address() - image_base);
set_link_value(1, next_command->vm_address() - image_base);
break;
case ltOffset:
if (to_command) {
value = (section_options_ & rtLinkedFromOtherType) ? to_command->address() : to_command->vm_address();
set_link_value(0, link()->Encrypt(value));
}
break;
case ltSwitch:
set_link_value(0, vm_links_[4]->address() - image_base);
set_link_value(1, vm_links_[2]->address() - image_base);
set_link_value(3, next_command->vm_address() - image_base);
set_link_value(5, to_command->vm_address() - image_base);
break;
case ltCall:
if (section_options_ & rtLinkedFrom)
set_link_value(0, to_command->vm_address() - image_base);
break;
}
}
}
void ILCommand::PrepareLink(const CompileContext &ctx)
{
bool from_native = block() && (block()->type() & mtExecutable) ? true : owner()->compilation_type() == ctMutation;
ICommand *to_command = link()->to_command();
if (to_command) {
bool to_native = to_command->block() && (to_command->block()->type() & mtExecutable) ? true : to_command->owner()->compilation_type() == ctMutation;
if (from_native == to_native) {
section_options_ |= rtLinkedFrom;
} else {
section_options_ |= rtLinkedFromOtherType;
}
if ((section_options_ & rtLinkedFromOtherType)
|| link()->type() == ltSEHBlock
|| link()->type() == ltFinallyBlock
|| (link()->type() == ltJmp && (type_ == icLeave || type_ == icLeave_s))) {
to_command->include_section_option(rtLinkedToExt);
}
else {
to_command->include_section_option(rtLinkedToInt);
}
}
if (from_native)
return;
bool need_next_command = false;
ICommand *next_command;
switch (link()->type()) {
case ltJmpWithFlag:
case ltSwitch:
need_next_command = true;
break;
}
if (need_next_command) {
size_t j = owner()->IndexOf(this) + 1;
if (type_ == icSwitch)
j += static_cast<uint32_t>(operand_value_);
if (j < owner()->count()) {
next_command = owner()->item(j);
include_section_option(rtLinkedNext);
link()->set_next_command(next_command);
next_command->include_section_option(rtLinkedToInt);
}
}
}
void ILCommand::set_link_value(size_t link_index, uint64_t value)
{
ILVMCommand *vm_command = vm_links_[link_index];
vm_command->set_value(value);
vm_command->Compile();
}
void ILCommand::set_operand_value(size_t operand_index, uint64_t value)
{
operand_value_ = value;
}
void ILCommand::set_operand_fixup(IFixup *fixup)
{
fixup_ = fixup;
}
void ILCommand::set_address(uint64_t address)
{
address_ = address;
switch (ILOpCodes[type_].operand_type) {
case ShortInlineBrTarget:
case InlineBrTarget:
CompileToNative();
break;
}
}
void ILCommand::ReadFromBuffer(Buffer &buffer, IArchitecture &file)
{
size_t i, r;
r = buffer.ReadByte();
address_ = buffer.ReadDWord() + file.image_base();
type_ = static_cast<ILCommandType>(buffer.ReadWord());
BaseCommand::ReadFromBuffer(buffer, file);
original_dump_size_ = (r & 0x10) ? buffer.ReadWord() : buffer.ReadByte();
if (type_ == icComment) {
for (i = 0; i < original_dump_size_; i++) {
PushByte(buffer.ReadByte());
}
} else {
for (i = 0; i < original_dump_size_; i++) {
PushByte(0);
}
}
if (r & 1) {
operand_value_ = buffer.ReadQWord();
}
if (r & 0x08) {
uint32_t value = static_cast<uint32_t>(operand_value_);
ILMetaData *meta = dynamic_cast<NETArchitecture &>(file).command_list();
ILToken *token = meta->token(value);
if (!token && TOKEN_TYPE(value) == ttUserString)
token = meta->us_table()->Add(value);
if (token) {
uint64_t reference_address = address() + original_dump_size_ - sizeof(uint32_t);
if (type_ == icComment) {
token_reference_ = token->reference_list()->GetReferenceByAddress(reference_address);
if (!token_reference_)
throw std::runtime_error("Invalid token reference");
}
else
token_reference_ = token->reference_list()->Add(reference_address);
}
}
}
void ILCommand::WriteToFile(IArchitecture &file)
{
if (fixup_) {
if (fixup_ == NEED_FIXUP) {
ISection *segment = file.segment_list()->GetSectionByAddress(address_);
fixup_ = file.fixup_list()->AddDefault(file.cpu_address_size(), segment && (segment->memory_type() & mtExecutable) != 0);
}
fixup_->set_address(address_ + operand_pos_);
}
BaseCommand::WriteToFile(file);
}
void ILCommand::Rebase(uint64_t delta_base)
{
if (!address_)
return;
switch (ILOpCodes[type_].operand_type) {
case ShortInlineBrTarget:
case InlineBrTarget:
operand_value_ += delta_base;
break;
}
address_ += delta_base;
}
ILCommand * ILCommand::Clone(IFunction *owner) const
{
ILCommand *command = new ILCommand(owner, *this);
return command;
}
bool ILCommand::Merge(ICommand *command)
{
return false;
}
bool ILCommand::is_data() const
{
return (type_ == icByte || type_ == icWord || type_ == icDword || type_ == icData);
}
bool ILCommand::is_end() const
{
ILFlowControl flow_type = ILOpCodes[type_].flow_type;
return (flow_type == Branch || flow_type == Return || flow_type == Throw);
}
bool ILCommand::is_prefix() const
{
return (ILOpCodes[type_].opcode_type == Prefix);
}
std::string ILCommand::dump_str() const
{
std::string res;
if (type_ == icUnknown) {
for (size_t i = 0; i < dump_size(); i++) {
res += "??";
}
return res;
}
res = BaseCommand::dump_str();
size_t value_size;
const ILOpCode opcode = ILOpCodes[type_];
switch (opcode.operand_type) {
case InlineNone:
case InlinePhi:
case Inline8:
value_size = 0;
break;
case ShortInlineI:
case ShortInlineVar:
case ShortInlineBrTarget:
value_size = sizeof(uint8_t);
break;
case InlineVar:
value_size = sizeof(uint16_t);
break;
case InlineI8:
case InlineR:
value_size = sizeof(uint64_t);
break;
default:
value_size = sizeof(uint32_t);
break;
}
if (value_size) {
size_t end = dump_size();
while (end > operand_pos_) {
res.insert((end - value_size) * 2, " ");
end -= value_size;
}
}
return res;
}
std::string ILCommand::display_address() const
{
return DisplayValue(size(), address());
}
size_t ILCommand::ReadFromFile(IArchitecture & file)
{
clear();
switch (ReadByte(file)) {
case 0x00: type_ = icNop; break;
case 0x01: type_ = icBreak; break;
case 0x02: type_ = icLdarg_0; break;
case 0x03: type_ = icLdarg_1; break;
case 0x04: type_ = icLdarg_2; break;
case 0x05: type_ = icLdarg_3; break;
case 0x06: type_ = icLdloc_0; break;
case 0x07: type_ = icLdloc_1; break;
case 0x08: type_ = icLdloc_2; break;
case 0x09: type_ = icLdloc_3; break;
case 0x0a: type_ = icStloc_0; break;
case 0x0b: type_ = icStloc_1; break;
case 0x0c: type_ = icStloc_2; break;
case 0x0d: type_ = icStloc_3; break;
case 0x0e: type_ = icLdarg_s; break;
case 0x0f: type_ = icLdarga_s; break;
case 0x10: type_ = icStarg_s; break;
case 0x11: type_ = icLdloc_s; break;
case 0x12: type_ = icLdloca_s; break;
case 0x13: type_ = icStloc_s; break;
case 0x14: type_ = icLdnull; break;
case 0x15: type_ = icLdc_i4_m1; break;
case 0x16: type_ = icLdc_i4_0; break;
case 0x17: type_ = icLdc_i4_1; break;
case 0x18: type_ = icLdc_i4_2; break;
case 0x19: type_ = icLdc_i4_3; break;
case 0x1a: type_ = icLdc_i4_4; break;
case 0x1b: type_ = icLdc_i4_5; break;
case 0x1c: type_ = icLdc_i4_6; break;
case 0x1d: type_ = icLdc_i4_7; break;
case 0x1e: type_ = icLdc_i4_8; break;
case 0x1f: type_ = icLdc_i4_s; break;
case 0x20: type_ = icLdc_i4; break;
case 0x21: type_ = icLdc_i8; break;
case 0x22: type_ = icLdc_r4; break;
case 0x23: type_ = icLdc_r8; break;
// 0x24 reserved for standardization
case 0x25: type_ = icDup; break;
case 0x26: type_ = icPop; break;
case 0x27: type_ = icJmp; break;
case 0x28: type_ = icCall; break;
case 0x29: type_ = icCalli; break;
case 0x2a: type_ = icRet; break;
case 0x2b: type_ = icBr_s; break;
case 0x2c: type_ = icBrfalse_s; break;
case 0x2d: type_ = icBrtrue_s; break;
case 0x2e: type_ = icBeq_s; break;
case 0x2f: type_ = icBge_s; break;
case 0x30: type_ = icBgt_s; break;
case 0x31: type_ = icBle_s; break;
case 0x32: type_ = icBlt_s; break;
case 0x33: type_ = icBne_un_s; break;
case 0x34: type_ = icBge_un_s; break;
case 0x35: type_ = icBgt_un_s; break;
case 0x36: type_ = icBle_un_s; break;
case 0x37: type_ = icBlt_un_s; break;
case 0x38: type_ = icBr; break;
case 0x39: type_ = icBrfalse; break;
case 0x3a: type_ = icBrtrue; break;
case 0x3b: type_ = icBeq; break;
case 0x3c: type_ = icBge; break;
case 0x3d: type_ = icBgt; break;
case 0x3e: type_ = icBle; break;
case 0x3f: type_ = icBlt; break;
case 0x40: type_ = icBne_un; break;
case 0x41: type_ = icBge_un; break;
case 0x42: type_ = icBgt_un; break;
case 0x43: type_ = icBle_un; break;
case 0x44: type_ = icBlt_un; break;
case 0x45: type_ = icSwitch; break;
case 0x46: type_ = icLdind_i1; break;
case 0x47: type_ = icLdind_u1; break;
case 0x48: type_ = icLdind_i2; break;
case 0x49: type_ = icLdind_u2; break;
case 0x4a: type_ = icLdind_i4; break;
case 0x4b: type_ = icLdind_u4; break;
case 0x4c: type_ = icLdind_i8; break;
case 0x4d: type_ = icLdind_i; break;
case 0x4e: type_ = icLdind_r4; break;
case 0x4f: type_ = icLdind_r8; break;
case 0x50: type_ = icLdind_ref; break;
case 0x51: type_ = icStind_ref; break;
case 0x52: type_ = icStind_i1; break;
case 0x53: type_ = icStind_i2; break;
case 0x54: type_ = icStind_i4; break;
case 0x55: type_ = icStind_i8; break;
case 0x56: type_ = icStind_r4; break;
case 0x57: type_ = icStind_r8; break;
case 0x58: type_ = icAdd; break;
case 0x59: type_ = icSub; break;
case 0x5a: type_ = icMul; break;
case 0x5b: type_ = icDiv; break;
case 0x5c: type_ = icDiv_un; break;
case 0x5d: type_ = icRem; break;
case 0x5e: type_ = icRem_un; break;
case 0x5f: type_ = icAnd; break;
case 0x60: type_ = icOr; break;
case 0x61: type_ = icXor; break;
case 0x62: type_ = icShl; break;
case 0x63: type_ = icShr; break;
case 0x64: type_ = icShr_un; break;
case 0x65: type_ = icNeg; break;
case 0x66: type_ = icNot; break;
case 0x67: type_ = icConv_i1; break;
case 0x68: type_ = icConv_i2; break;
case 0x69: type_ = icConv_i4; break;
case 0x6a: type_ = icConv_i8; break;
case 0x6b: type_ = icConv_r4; break;
case 0x6c: type_ = icConv_r8; break;
case 0x6d: type_ = icConv_u4; break;
case 0x6e: type_ = icConv_u8; break;
case 0x6f: type_ = icCallvirt; break;
case 0x70: type_ = icCpobj; break;
case 0x71: type_ = icLdobj; break;
case 0x72: type_ = icLdstr; break;
case 0x73: type_ = icNewobj; break;
case 0x74: type_ = icCastclass; break;
case 0x75: type_ = icIsinst; break;
case 0x76: type_ = icConv_r_un; break;
// 0x77-0x78 reserved for standardization
case 0x79: type_ = icUnbox; break;
case 0x7a: type_ = icThrow; break;
case 0x7b: type_ = icLdfld; break;
case 0x7c: type_ = icLdflda; break;
case 0x7d: type_ = icStfld; break;
case 0x7e: type_ = icLdsfld; break;
case 0x7f: type_ = icLdsflda; break;
case 0x80: type_ = icStsfld; break;
case 0x81: type_ = icStobj; break;
case 0x82: type_ = icConv_ovf_i1_un; break;
case 0x83: type_ = icConv_ovf_i2_un; break;
case 0x84: type_ = icConv_ovf_i4_un; break;
case 0x85: type_ = icConv_ovf_i8_un; break;
case 0x86: type_ = icConv_ovf_u1_un; break;
case 0x87: type_ = icConv_ovf_u2_un; break;
case 0x88: type_ = icConv_ovf_u4_un; break;
case 0x89: type_ = icConv_ovf_u8_un; break;
case 0x8a: type_ = icConv_ovf_i_un; break;
case 0x8b: type_ = icConv_ovf_u_un; break;
case 0x8c: type_ = icBox; break;
case 0x8d: type_ = icNewarr; break;
case 0x8e: type_ = icLdlen; break;
case 0x8f: type_ = icLdelema; break;
case 0x90: type_ = icLdelem_i1; break;
case 0x91: type_ = icLdelem_u1; break;
case 0x92: type_ = icLdelem_i2; break;
case 0x93: type_ = icLdelem_u2; break;
case 0x94: type_ = icLdelem_i4; break;
case 0x95: type_ = icLdelem_u4; break;
case 0x96: type_ = icLdelem_i8; break;
case 0x97: type_ = icLdelem_i; break;
case 0x98: type_ = icLdelem_r4; break;
case 0x99: type_ = icLdelem_r8; break;
case 0x9a: type_ = icLdelem_ref; break;
case 0x9b: type_ = icStelem_i; break;
case 0x9c: type_ = icStelem_i1; break;
case 0x9d: type_ = icStelem_i2; break;
case 0x9e: type_ = icStelem_i4; break;
case 0x9f: type_ = icStelem_i8; break;
case 0xa0: type_ = icStelem_r4; break;
case 0xa1: type_ = icStelem_r8; break;
case 0xa2: type_ = icStelem_ref; break;
case 0xa3: type_ = icLdelem; break;
case 0xa4: type_ = icStelem; break;
case 0xa5: type_ = icUnbox_any; break;
// 0xA6-0xB2 reserved for standardization
case 0xb3: type_ = icConv_ovf_i1; break;
case 0xb4: type_ = icConv_ovf_u1; break;
case 0xb5: type_ = icConv_ovf_i2; break;
case 0xb6: type_ = icConv_ovf_u2; break;
case 0xb7: type_ = icConv_ovf_i4; break;
case 0xb8: type_ = icConv_ovf_u4; break;
case 0xb9: type_ = icConv_ovf_i8; break;
case 0xba: type_ = icConv_ovf_u8; break;
// 0xBB-0xC1 reserved for standardization
case 0xc2: type_ = icRefanyval; break;
case 0xc3: type_ = icCkfinite; break;
// 0xC4-0xC5 reserved for standardization
case 0xc6: type_ = icMkrefany; break;
// 0xC7-0xCF reserved for standardization
case 0xd0: type_ = icLdtoken; break;
case 0xd1: type_ = icConv_u2; break;
case 0xd2: type_ = icConv_u1; break;
case 0xd3: type_ = icConv_i; break;
case 0xd4: type_ = icConv_ovf_i; break;
case 0xd5: type_ = icConv_ovf_u; break;
case 0xd6: type_ = icAdd_ovf; break;
case 0xd7: type_ = icAdd_ovf_un; break;
case 0xd8: type_ = icMul_ovf; break;
case 0xd9: type_ = icMul_ovf_un; break;
case 0xda: type_ = icSub_ovf; break;
case 0xdb: type_ = icSub_ovf_un; break;
case 0xdc: type_ = icEndfinally; break;
case 0xdd: type_ = icLeave; break;
case 0xde: type_ = icLeave_s; break;
case 0xdf: type_ = icStind_i; break;
case 0xe0: type_ = icConv_u; break;
// 0xE1-0xEF reserved for standardization
// 0xF0-0xFB are ecma_experimental
// 0xFC-0xFD reserved for standardization
case 0xfe:
//two-byte opcodes
switch (ReadByte(file)) {
case 0x00: type_ = icArglist; break;
case 0x01: type_ = icCeq; break;
case 0x02: type_ = icCgt; break;
case 0x03: type_ = icCgt_un; break;
case 0x04: type_ = icClt; break;
case 0x05: type_ = icClt_un; break;
case 0x06: type_ = icLdftn; break;
case 0x07: type_ = icLdvirtftn; break;
// 0xFE08 reserved for standardization
case 0x09: type_ = icLdarg; break;
case 0x0a: type_ = icLdarga; break;
case 0x0b: type_ = icStarg; break;
case 0x0c: type_ = icLdloc; break;
case 0x0d: type_ = icLdloca; break;
case 0x0e: type_ = icStloc; break;
case 0x0f: type_ = icLocalloc; break;
// 0xFE10 reserved for standardization
case 0x11: type_ = icEndfilter; break;
case 0x12: type_ = icUnaligned; break;
case 0x13: type_ = icVolatile; break;
case 0x14: type_ = icTail; break;
case 0x15: type_ = icInitobj; break;
case 0x16: type_ = icConstrained; break;
case 0x17: type_ = icCpblk; break;
case 0x18: type_ = icInitblk; break;
case 0x19: type_ = icNo; break;
case 0x1a: type_ = icRethrow; break;
// 0xFE1B reserved for standardization
case 0x1c: type_ = icSizeof; break;
case 0x1d: type_ = icRefanytype; break;
case 0x1e: type_ = icReadonly; break;
}
break;
// 0xFF reserved for standardization
}
operand_pos_ = dump_size();
switch (ILOpCodes[type_].operand_type)
{
case InlineNone: // No operand
case InlinePhi: // Obsolete. The operand is reserved and should not be used.
case Inline8: // not used
// all read already
break;
case ShortInlineI: // 8-bit integer
operand_value_ = static_cast<int8_t>(ReadByte(file));
break;
case ShortInlineVar: // 8-bit integer containing the ordinal of a local variable or an argument
operand_value_ = ReadByte(file);
break;
case InlineVar: // 16-bit integer containing the ordinal of a local variable or an argument
operand_value_ = ReadWord(file);
break;
case InlineI8: // 64-bit integer
case InlineR: // 64-bit IEEE floating point number
operand_value_ = ReadQWord(file);
break;
case InlineSwitch:
operand_value_ = ReadDWord(file);
break;
case ShortInlineBrTarget: // 8-bit integer branch target
operand_value_ = static_cast<int8_t>(ReadByte(file));
operand_value_ += next_address();
break;
case InlineBrTarget: // 32-bit integer branch target
operand_value_ = static_cast<int32_t>(ReadDWord(file));
operand_value_ += next_address();
break;
case InlineField:
case InlineMethod:
case InlineSig:
case InlineTok:
case InlineType:
case InlineString:
operand_value_ = ReadDWord(file);
{
ILToken *token = dynamic_cast<NETArchitecture &>(file).command_list()->token(static_cast<uint32_t>(operand_value_));
if (token)
token_reference_ = token->reference_list()->GetReferenceByAddress(address() + operand_pos_);
}
break;
default: // 4 byte
operand_value_ = ReadDWord(file);
break;
}
original_dump_size_ = dump_size();
return original_dump_size_;
}
uint64_t ILCommand::ReadValueFromFile(IArchitecture &file, OperandSize value_size, bool is_token)
{
switch (value_size) {
case osByte:
type_ = icByte;
operand_value_ = ReadByte(file);
break;
case osWord:
type_ = icWord;
operand_value_ = ReadWord(file);
break;
case osDWord:
type_ = icDword;
operand_value_ = ReadDWord(file);
break;
default:
throw std::runtime_error("Invalid value size");
}
if (is_token) {
ILToken *token = dynamic_cast<NETArchitecture &>(file).command_list()->token(static_cast<uint32_t>(operand_value_));
if (token)
token_reference_ = token->reference_list()->GetReferenceByAddress(address_);
}
return operand_value_;
}
void ILCommand::ReadString(IArchitecture &file, size_t len)
{
type_ = icData;
Read(file, len);
}
void ILCommand::ReadCaseCommand(IArchitecture &file)
{
type_ = icCase;
operand_value_ = DWordToInt64(ReadDWord(file));
}
int ILCommand::GetStackLevel(size_t *pop_ref) const
{
if (type_ == icRet)
return 0;
size_t push = 0;
size_t pop = 0;
if (ILOpCodes[type_].flow_type == Call) {
ILToken *token = token_reference_->owner()->owner();
if (token->type() == ttMethodSpec)
token = reinterpret_cast<ILMethodSpec *>(token)->parent();
ILSignature *signature;
switch (token->type()) {
case ttMethodDef:
signature = reinterpret_cast<ILMethodDef *>(token)->signature();
break;
case ttMemberRef:
signature = reinterpret_cast<ILMemberRef *>(token)->signature();
break;
case ttStandAloneSig:
signature = reinterpret_cast<ILStandAloneSig *>(token)->signature();
break;
default:
throw std::runtime_error("Invalid token type: " + text());
}
if (type_ != icNewobj && signature->has_this() && !signature->explicit_this())
pop += 1;
if (type_ == icCalli)
pop += 1;
pop += static_cast<int>(signature->count());
if (type_ == icNewobj || signature->ret()->type() != ELEMENT_TYPE_VOID)
push += 1;
}
else {
switch (ILOpCodes[type_].pop) {
case Pop0:
break;
case Pop1:
case Popi:
case Popref:
pop += 1;
break;
case Pop1_pop1:
case Popi_pop1:
case Popi_popi:
case Popi_popi8:
case Popi_popr4:
case Popi_popr8:
case Popref_pop1:
case Popref_popi:
pop += 2;
break;
case Popi_popi_popi:
case Popref_popi_popi:
case Popref_popi_popi8:
case Popref_popi_popr4:
case Popref_popi_popr8:
case Popref_popi_popref:
case Popref_popi_pop1:
pop += 3;
break;
}
switch (ILOpCodes[type_].push) {
case Push0:
break;
case Push1:
case Pushi:
case Pushi8:
case Pushr4:
case Pushr8:
case Pushref:
push += 1;
break;
case Push1_push1:
push += 2;
break;
}
}
if (pop_ref)
*pop_ref = pop;
return (int)(push - pop);
}
ILVMCommand *ILCommand::AddVMCommand(const CompileContext &ctx, ILCommandType command_type, uint64_t value, uint32_t options, TokenReference *token_reference, ILCommand *to_command)
{
ILVMCommand *vm_command = NULL;
#ifdef ULTIMATE
if ((owner()->compilation_options() & coLockToKey) && command_type == icLdc_i4 && (options & voNoCryptValue) == 0) {
vm_command = new ILVMCommand(this, command_type, value, token_reference);
vm_command->set_crypt_command(icAdd, osDWord, ctx.options.licensing_manager->product_code());
}
#endif
if (!vm_command)
vm_command = new ILVMCommand(this, command_type, value, token_reference);
AddObject(vm_command);
if (options & voLinkCommand)
vm_links_.push_back(vm_command);
switch (command_type) {
case icBr:
case icRet:
section_options_ |= rtCloseSection;
break;
}
if (to_command)
internal_links_.Add(vlNone, vm_command, to_command);
uint32_t new_options = (options & voSectionCommand);
if (vm_command->crypt_command() == icAdd) {
new_options |= voNoCryptValue;
size_t i;
ILVMCommand *cur_command = vm_command;
ILToken *type = reinterpret_cast<NETArchitecture *>(ctx.file)->command_list()->ImportType(ELEMENT_TYPE_U4);
// add session key
uint64_t address = ctx.runtime->export_list()->GetAddressByType(atLoaderData);
ILFunction *loader_data_func = reinterpret_cast<ILFunction *>(ctx.runtime->function_list()->GetFunctionByAddress(address));
ILCommand *field_command = loader_data_func->item(1);
AddVMCommand(ctx, icLdnull, 0, new_options);
AddVMCommand(ctx, icLdc_i4, field_command->operand_value(), new_options, field_command->token_reference());
AddVMCommand(ctx, icLdfld, 0, new_options);
AddVMCommand(ctx, icLdc_i4, ctx.runtime_var_index[VAR_SESSION_KEY], new_options);
AddVMCommand(ctx, icLdc_i4, type->id(), new_options, type->reference_list()->Add(0));
AddVMCommand(ctx, icLdelem_ref, 0, new_options);
AddVMCommand(ctx, icAdd, 0, new_options);
for (i = 1; i < 4; i++) {
cur_command->set_link_command(AddVMCommand(ctx, icLdc_i4, 0, new_options));
// add session key
AddVMCommand(ctx, icLdnull, 0, new_options);
AddVMCommand(ctx, icLdc_i4, field_command->operand_value(), new_options, field_command->token_reference());
AddVMCommand(ctx, icLdfld, 0, new_options);
AddVMCommand(ctx, icLdc_i4, ctx.runtime_var_index[VAR_SESSION_KEY], new_options);
AddVMCommand(ctx, icLdc_i4, type->id(), new_options, type->reference_list()->Add(0));
AddVMCommand(ctx, icLdelem_ref, 0, new_options);
AddVMCommand(ctx, icAdd, 0, new_options);
cur_command = cur_command->link_command();
}
address = ctx.runtime->export_list()->GetAddressByType(atDecryptBuffer);
ILVMCommand *from_command = AddVMCommand(ctx, icLdc_i4, 0, new_options);
ICommand *to_command = ctx.file->function_list()->GetCommandByAddress(address, true);
if (to_command)
internal_links_.Add(vlNone, from_command, to_command);
AddVMCommand(ctx, icCallvm, 0, new_options);
// add session key
AddVMCommand(ctx, icLdnull, 0, new_options);
AddVMCommand(ctx, icLdc_i4, field_command->operand_value(), new_options, field_command->token_reference());
AddVMCommand(ctx, icLdfld, 0, new_options);
AddVMCommand(ctx, icLdc_i4, ctx.runtime_var_index[VAR_SESSION_KEY], new_options);
AddVMCommand(ctx, icLdc_i4, type->id(), new_options, type->reference_list()->Add(0));
AddVMCommand(ctx, icLdelem_ref, 0, new_options);
AddVMCommand(ctx, icAdd, 0, new_options);
}
return vm_command;
}
void ILCommand::AddCmpSection(const CompileContext &ctx, ILCommandType cmp_command)
{
ILCommandType cmp_type;
switch (cmp_command) {
case icBne_un:
case icClt_un:
case icBlt_un:
case icBle_un:
case icCgt_un:
case icBgt_un:
case icBge_un:
cmp_type = icCmp_un;
break;
default:
cmp_type = icCmp;
break;
}
switch (cmp_command) {
case icCeq:
case icBeq:
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
case icBne_un:
AddVMCommand(ctx, icLdc_i4, 0);
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
case icClt:
case icClt_un:
case icBlt:
case icBlt_un:
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icShr_un, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
case icBle:
case icBle_un:
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icShr_un, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
case icCgt:
case icCgt_un:
case icBgt:
case icBgt_un:
AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(-1));
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icShr_un, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
case icBge:
case icBge_un:
AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(-1));
AddVMCommand(ctx, cmp_type, 0);
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icShr_un, 0);
AddVMCommand(ctx, icLdc_i4, 1);
AddVMCommand(ctx, icAnd, 0);
break;
}
}
void ILCommand::AddJmpWithFlagSection(const CompileContext &ctx, bool is_true)
{
ILToken *type = reinterpret_cast<NETArchitecture *>(ctx.file)->command_list()->ImportType(ELEMENT_TYPE_BOOLEAN);
AddVMCommand(ctx, icLdc_i4, type->id(), 0, type->reference_list()->Add(0));
AddVMCommand(ctx, icConv, 0);
AddVMCommand(ctx, icNeg, 0);
if (!is_true)
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icAnd, 0);
AddVMCommand(ctx, icDup, 0);
AddVMCommand(ctx, icLdc_i4, type->id(), 0, type->reference_list()->Add(0));
AddVMCommand(ctx, icConv, 0);
AddVMCommand(ctx, icNeg, 0);
AddVMCommand(ctx, icNot, 0);
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icAnd, 0);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icBr, 0);
}
void ILCommand::CompileToVM(const CompileContext &ctx)
{
ILToken *internal_type = NULL;
TokenReference *internal_token_reference = NULL;
switch (type_) {
case icLdelem_i:
case icLdelem_i1:
case icLdelem_i2:
case icLdelem_i4:
case icLdelem_i8:
case icLdelem_u1:
case icLdelem_u2:
case icLdelem_u4:
case icLdelem_r4:
case icLdelem_r8:
case icLdelem_ref:
case icStelem_i:
case icStelem_i1:
case icStelem_i2:
case icStelem_i4:
case icStelem_i8:
case icStelem_r4:
case icStelem_r8:
case icStelem_ref:
case icLdind_i:
case icLdind_i1:
case icLdind_i2:
case icLdind_i4:
case icLdind_i8:
case icLdind_u1:
case icLdind_u2:
case icLdind_u4:
case icLdind_r4:
case icLdind_r8:
case icLdind_ref:
case icStind_i:
case icStind_i1:
case icStind_i2:
case icStind_i4:
case icStind_i8:
case icStind_r4:
case icStind_r8:
case icStind_ref:
case icConv_i:
case icConv_u:
case icConv_i1:
case icConv_u1:
case icConv_i2:
case icConv_u2:
case icConv_i4:
case icConv_u4:
case icConv_i8:
case icConv_u8:
case icConv_r4:
case icConv_r8:
case icConv_r_un:
case icConv_ovf_i:
case icConv_ovf_u:
case icConv_ovf_i1:
case icConv_ovf_u1:
case icConv_ovf_i2:
case icConv_ovf_u2:
case icConv_ovf_i4:
case icConv_ovf_u4:
case icConv_ovf_i8:
case icConv_ovf_u8:
case icConv_ovf_i_un:
case icConv_ovf_u_un:
case icConv_ovf_i1_un:
case icConv_ovf_u1_un:
case icConv_ovf_i2_un:
case icConv_ovf_u2_un:
case icConv_ovf_i4_un:
case icConv_ovf_u4_un:
case icConv_ovf_i8_un:
case icConv_ovf_u8_un:
{
CorElementType element_type;
switch (type_) {
case icLdelem_i1:
case icStelem_i1:
case icLdind_i1:
case icStind_i1:
case icConv_i1:
case icConv_ovf_i1:
case icConv_ovf_i1_un:
element_type = ELEMENT_TYPE_I1;
break;
case icLdelem_i2:
case icStelem_i2:
case icLdind_i2:
case icStind_i2:
case icConv_i2:
case icConv_ovf_i2:
case icConv_ovf_i2_un:
element_type = ELEMENT_TYPE_I2;
break;
case icLdelem_i4:
case icStelem_i4:
case icLdind_i4:
case icStind_i4:
case icConv_i4:
case icConv_ovf_i4:
case icConv_ovf_i4_un:
element_type = ELEMENT_TYPE_I4;
break;
case icLdelem_i8:
case icStelem_i8:
case icLdind_i8:
case icStind_i8:
case icConv_i8:
case icConv_ovf_i8:
case icConv_ovf_i8_un:
element_type = ELEMENT_TYPE_I8;
break;
case icLdelem_u1:
case icLdind_u1:
case icConv_u1:
case icConv_ovf_u1:
case icConv_ovf_u1_un:
element_type = ELEMENT_TYPE_U1;
break;
case icLdelem_u2:
case icLdind_u2:
case icConv_u2:
case icConv_ovf_u2:
case icConv_ovf_u2_un:
element_type = ELEMENT_TYPE_U2;
break;
case icLdelem_u4:
case icLdind_u4:
case icConv_u4:
case icConv_ovf_u4:
case icConv_ovf_u4_un:
element_type = ELEMENT_TYPE_U4;
break;
case icConv_u8:
case icConv_ovf_u8:
case icConv_ovf_u8_un:
element_type = ELEMENT_TYPE_U8;
break;
case icLdelem_r4:
case icStelem_r4:
case icLdind_r4:
case icStind_r4:
case icConv_r4:
element_type = ELEMENT_TYPE_R4;
break;
case icLdelem_r8:
case icStelem_r8:
case icLdind_r8:
case icStind_r8:
case icConv_r8:
case icConv_r_un:
element_type = ELEMENT_TYPE_R8;
break;
case icLdelem_ref:
case icStelem_ref:
case icLdind_ref:
case icStind_ref:
element_type = ELEMENT_TYPE_OBJECT;
break;
case icConv_u:
case icConv_ovf_u:
case icConv_ovf_u_un:
element_type = ELEMENT_TYPE_U;
break;
default:
element_type = ELEMENT_TYPE_I;
break;
}
internal_type = reinterpret_cast<NETArchitecture *>(ctx.file)->command_list()->ImportType(element_type);
internal_token_reference = internal_type->reference_list()->Add(0);
}
break;
}
switch (type_) {
case icNop:
break;
case icLdarg:
case icLdarg_s:
case icLdarg_0:
case icLdarg_1:
case icLdarg_2:
case icLdarg_3:
AddVMCommand(ctx, icLdarg, param_);
break;
case icLdarga:
case icLdarga_s:
AddVMCommand(ctx, icLdarga, param_);
break;
case icStarg:
case icStarg_s:
AddVMCommand(ctx, icStarg, param_);
break;
case icLdloc:
case icLdloc_s:
case icLdloc_0:
case icLdloc_1:
case icLdloc_2:
case icLdloc_3:
AddVMCommand(ctx, icLdarg, param_);
break;
case icLdloca:
case icLdloca_s:
AddVMCommand(ctx, icLdarga, param_);
break;
case icStloc:
case icStloc_s:
case icStloc_0:
case icStloc_1:
case icStloc_2:
case icStloc_3:
AddVMCommand(ctx, icStarg, param_);
break;
case icLdc_i4_0:
AddVMCommand(ctx, icLdc_i4, 0);
break;
case icLdc_i4_1:
AddVMCommand(ctx, icLdc_i4, 1);
break;
case icLdc_i4_2:
AddVMCommand(ctx, icLdc_i4, 2);
break;
case icLdc_i4_3:
AddVMCommand(ctx, icLdc_i4, 3);
break;
case icLdc_i4_4:
AddVMCommand(ctx, icLdc_i4, 4);
break;
case icLdc_i4_5:
AddVMCommand(ctx, icLdc_i4, 5);
break;
case icLdc_i4_6:
AddVMCommand(ctx, icLdc_i4, 6);
break;
case icLdc_i4_7:
AddVMCommand(ctx, icLdc_i4, 7);
break;
case icLdc_i4_8:
AddVMCommand(ctx, icLdc_i4, 8);
break;
case icLdc_i4:
case icLdc_i4_s:
AddVMCommand(ctx, icLdc_i4, operand_value_, (link() && link()->operand_index() == 0) ? voLinkCommand : 0);
break;
case icLdc_i4_m1:
AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(-1));
break;
case icLdc_i8:
AddVMCommand(ctx, icLdc_i8, operand_value_);
break;
case icLdc_r4:
AddVMCommand(ctx, icLdc_r4, operand_value_);
break;
case icLdc_r8:
AddVMCommand(ctx, icLdc_r8, operand_value_);
break;
case icLdnull:
AddVMCommand(ctx, icLdnull, 0);
break;
case icConv_i:
case icConv_u:
case icConv_i1:
case icConv_u1:
case icConv_i2:
case icConv_u2:
case icConv_i4:
case icConv_u4:
case icConv_i8:
case icConv_u8:
case icConv_r4:
case icConv_r8:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icConv, 0);
break;
case icConv_ovf_i:
case icConv_ovf_u:
case icConv_ovf_i1:
case icConv_ovf_u1:
case icConv_ovf_i2:
case icConv_ovf_u2:
case icConv_ovf_i4:
case icConv_ovf_u4:
case icConv_ovf_i8:
case icConv_ovf_u8:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icConv_ovf, 0);
break;
case icConv_ovf_i_un:
case icConv_ovf_u_un:
case icConv_ovf_i1_un:
case icConv_ovf_u1_un:
case icConv_ovf_i2_un:
case icConv_ovf_u2_un:
case icConv_ovf_i4_un:
case icConv_ovf_u4_un:
case icConv_ovf_i8_un:
case icConv_ovf_u8_un:
case icConv_r_un:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icConv_ovf_un, 0);
break;
case icRem:
AddVMCommand(ctx, icRem, 0);
break;
case icRem_un:
AddVMCommand(ctx, icRem_un, 0);
break;
case icAdd:
AddVMCommand(ctx, icAdd, 0);
break;
case icSub:
AddVMCommand(ctx, icSub, 0);
break;
case icDiv:
AddVMCommand(ctx, icDiv, 0);
break;
case icDiv_un:
AddVMCommand(ctx, icDiv_un, 0);
break;
case icMul:
AddVMCommand(ctx, icMul, 0);
break;
case icAnd:
AddVMCommand(ctx, icAnd, 0);
break;
case icOr:
AddVMCommand(ctx, icOr, 0);
break;
case icXor:
AddVMCommand(ctx, icXor, 0);
break;
case icShl:
AddVMCommand(ctx, icShl, 0);
break;
case icShr:
AddVMCommand(ctx, icShr, 0);
break;
case icShr_un:
AddVMCommand(ctx, icShr_un, 0);
break;
case icNeg:
AddVMCommand(ctx, icNeg, 0);
break;
case icNot:
AddVMCommand(ctx, icNot, 0);
break;
case icAdd_ovf:
AddVMCommand(ctx, icAdd_ovf, 0);
break;
case icAdd_ovf_un:
AddVMCommand(ctx, icAdd_ovf_un, 0);
break;
case icMul_ovf:
AddVMCommand(ctx, icMul_ovf, 0);
break;
case icMul_ovf_un:
AddVMCommand(ctx, icMul_ovf_un, 0);
break;
case icSub_ovf:
AddVMCommand(ctx, icSub_ovf, 0);
break;
case icSub_ovf_un:
AddVMCommand(ctx, icSub_ovf_un, 0);
break;
case icCeq:
AddCmpSection(ctx, icCeq);
break;
case icClt:
AddCmpSection(ctx, icClt);
break;
case icClt_un:
AddCmpSection(ctx, icClt_un);
break;
case icCgt:
AddCmpSection(ctx, icCgt);
break;
case icCgt_un:
AddCmpSection(ctx, icCgt_un);
break;
case icCall:
if (section_options_ & rtLinkedFrom) {
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icCallvm, 0);
}
else {
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icCall, 0);
}
break;
case icCallvirt:
if (section_options_ & rtLinkedFrom) {
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icCallvmvirt, 0);
}
else {
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icCallvirt, 0);
}
break;
case icNewobj:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icNewobj, 0);
break;
case icNewarr:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icNewarr, 0);
break;
case icInitobj:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icInitobj, 0);
break;
case icConstrained:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icConstrained, 0);
break;
case icStelem_i:
case icStelem_i1:
case icStelem_i2:
case icStelem_i4:
case icStelem_i8:
case icStelem_r4:
case icStelem_r8:
case icStelem_ref:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icStelem_ref, 0);
break;
case icStelem:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icStelem_ref, 0);
break;
case icLdelem_i:
case icLdelem_i1:
case icLdelem_u1:
case icLdelem_i2:
case icLdelem_u2:
case icLdelem_i4:
case icLdelem_u4:
case icLdelem_i8:
case icLdelem_r4:
case icLdelem_r8:
case icLdelem_ref:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icLdelem_ref, 0);
break;
case icLdelem:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdelem_ref, 0);
break;
case icLdelema:
AddVMCommand(ctx, icLdelema, 0);
break;
case icLdlen:
AddVMCommand(ctx, icLdlen, 0);
break;
case icLdfld:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdfld, 0);
break;
case icLdsfld:
AddVMCommand(ctx, icLdnull, 0);
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdfld, 0);
break;
case icLdflda:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdflda, 0);
break;
case icLdsflda:
AddVMCommand(ctx, icLdnull, 0);
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdflda, 0);
break;
case icStfld:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icStfld, 0);
break;
case icStsfld:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icStsfld, 0);
break;
case icLdstr:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdstr, 0);
break;
case icLdftn:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdftn, 0);
break;
case icLdvirtftn:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdvirtftn, 0);
break;
case icStind_i:
case icStind_i1:
case icStind_i2:
case icStind_i4:
case icStind_i8:
case icStind_r4:
case icStind_r8:
case icStind_ref:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icStind_ref, 0);
break;
case icStobj:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icStind_ref, 0);
break;
case icLdind_i:
case icLdind_i1:
case icLdind_i2:
case icLdind_i4:
case icLdind_i8:
case icLdind_u1:
case icLdind_u2:
case icLdind_u4:
case icLdind_r4:
case icLdind_r8:
case icLdind_ref:
AddVMCommand(ctx, icLdc_i4, internal_type->id(), 0, internal_token_reference);
AddVMCommand(ctx, icLdind_ref, 0);
break;
case icLdobj:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdind_ref, 0);
break;
case icPop:
AddVMCommand(ctx, icPop, 0);
break;
case icDup:
AddVMCommand(ctx, icDup, 0);
break;
case icBox:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icBox, 0);
break;
case icUnbox:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icUnbox, 0);
break;
case icUnbox_any:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icUnbox_any, 0);
break;
case icLdtoken:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icLdtoken, 0);
break;
case icBr:
case icBr_s:
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icBr, 0);
break;
case icBeq:
case icBeq_s:
AddCmpSection(ctx, icBeq);
AddJmpWithFlagSection(ctx, true);
break;
case icBne_un:
case icBne_un_s:
AddCmpSection(ctx, icBne_un);
AddJmpWithFlagSection(ctx, true);
break;
case icBlt:
case icBlt_s:
AddCmpSection(ctx, icBlt);
AddJmpWithFlagSection(ctx, true);
break;
case icBlt_un:
case icBlt_un_s:
AddCmpSection(ctx, icBlt_un);
AddJmpWithFlagSection(ctx, true);
break;
case icBle:
case icBle_s:
AddCmpSection(ctx, icBle);
AddJmpWithFlagSection(ctx, true);
break;
case icBle_un:
case icBle_un_s:
AddCmpSection(ctx, icBle_un);
AddJmpWithFlagSection(ctx, true);
break;
case icBgt:
case icBgt_s:
AddCmpSection(ctx, icBgt);
AddJmpWithFlagSection(ctx, true);
break;
case icBgt_un:
case icBgt_un_s:
AddCmpSection(ctx, icBgt_un);
AddJmpWithFlagSection(ctx, true);
break;
case icBge:
case icBge_s:
AddCmpSection(ctx, icBge);
AddJmpWithFlagSection(ctx, true);
break;
case icBge_un:
case icBge_un_s:
AddCmpSection(ctx, icBge_un);
AddJmpWithFlagSection(ctx, true);
break;
case icBrtrue:
case icBrtrue_s:
AddJmpWithFlagSection(ctx, true);
break;
case icBrfalse:
case icBrfalse_s:
AddJmpWithFlagSection(ctx, false);
break;
case icLeave:
case icLeave_s:
AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(operand_value_ - ctx.file->image_base()));
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icLeave, 0);
break;
case icSwitch:
AddVMCommand(ctx, icDup, 0);
AddVMCommand(ctx, icLdc_i4, operand_value_);
AddCmpSection(ctx, icBlt_un);
AddJmpWithFlagSection(ctx, true);
AddVMCommand(ctx, icPop, 0, voLinkCommand);
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icBr, 0);
AddVMCommand(ctx, icLdc_i4, 2, voLinkCommand);
AddVMCommand(ctx, icShl, 0);
AddVMCommand(ctx, icLdc_i4, 0, voLinkCommand);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icLdmem_i4, 0);
AddVMCommand(ctx, icBr, 0);
break;
case icCase:
AddVMCommand(ctx, icDword, 0, voLinkCommand);
break;
case icEndfinally:
AddVMCommand(ctx, icEndfinally, 0);
break;
case icEndfilter:
AddVMCommand(ctx, icEndfilter, 0);
break;
case icRet:
if (operand_value_) {
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icBox, 0);
}
else
AddVMCommand(ctx, icLdnull, 0);
AddVMCommand(ctx, icLdc_i4, 0);
if ((ctx.options.flags & cpMemoryProtection) && owner()->tag() != ftLoader) {
ILVMCommand *vm_command = AddVMCommand(ctx, icLdc_i4, 0);
uint64_t address = ctx.runtime->export_list()->GetAddressByType(atRandom);
ICommand *to_command = ctx.file->function_list()->GetCommandByAddress(address, true);
internal_links_.Add(vlNone, vm_command, to_command);
AddVMCommand(ctx, icCallvm, 0);
AddVMCommand(ctx, icLdc_i4, rand32());
AddVMCommand(ctx, icAdd, 0);
vm_command = AddVMCommand(ctx, icLdc_i4, 0);
internal_links_.Add(vlCRCTableCount, vm_command, NULL);
AddVMCommand(ctx, icRem_un, 0);
AddVMCommand(ctx, icLdc_i4, sizeof(CRCInfo::POD));
AddVMCommand(ctx, icMul, 0);
vm_command = AddVMCommand(ctx, icLdc_i4, 0);
internal_links_.Add(vlCRCTableAddress, vm_command, NULL);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icStarg, param_);
AddVMCommand(ctx, icLdarg, param_);
AddVMCommand(ctx, icLdmem_i4, 0);
AddCryptorSection(ctx, ctx.file->function_list()->crc_cryptor(), true);
AddVMCommand(ctx, icLdarg, param_);
AddVMCommand(ctx, icLdc_i4, offsetof(CRCInfo::POD, size));
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icLdmem_i4, 0);
AddCryptorSection(ctx, ctx.file->function_list()->crc_cryptor(), true);
vm_command = AddVMCommand(ctx, icLdc_i4, 0);
address = ctx.runtime->export_list()->GetAddressByType(atCalcCRC);
to_command = ctx.file->function_list()->GetCommandByAddress(address, true);
internal_links_.Add(vlNone, vm_command, to_command);
AddVMCommand(ctx, icCallvm, 0);
AddVMCommand(ctx, icLdarg, param_);
AddVMCommand(ctx, icLdc_i4, offsetof(CRCInfo::POD, hash));
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icLdmem_i4, 0);
AddVMCommand(ctx, icAdd, 0);
AddVMCommand(ctx, icAdd, 0);
}
AddVMCommand(ctx, icBr, 0);
break;
case icThrow:
AddVMCommand(ctx, icThrow, 0);
break;
case icRethrow:
AddVMCommand(ctx, icRethrow, 0);
break;
case icCastclass:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icCastclass, 0);
break;
case icIsinst:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icIsinst, 0);
break;
case icCkfinite:
AddVMCommand(ctx, icCkfinite, 0);
break;
case icLocalloc:
AddVMCommand(ctx, icLocalloc, 0);
break;
case icSizeof:
AddVMCommand(ctx, icLdc_i4, operand_value_, 0, token_reference_);
AddVMCommand(ctx, icSizeof, 0);
break;
default:
throw std::runtime_error("Runtime error at CompileToVM: " + text());
}
}
void ILCommand::AddExtSection(const CompileContext &ctx)
{
ext_vm_entry_ = AddVMCommand(ctx, icLdc_i4, 0);
AddVMCommand(ctx, icPop, 0);
}
void ILCommand::AddCryptorSection(const CompileContext &ctx, ValueCryptor *cryptor, bool is_decrypt)
{
if (!cryptor)
return;
CompileContext new_ctx = ctx;
new_ctx.options.flags &= ~cpMemoryProtection;
for (size_t i = 0; i < cryptor->count(); i++) {
ValueCommand *value_command = cryptor->item(i);
AddVMCommand(new_ctx, icLdc_i4, value_command->value());
switch (value_command->type(is_decrypt)) {
case ccDec:
AddVMCommand(new_ctx, icSub, 0);
break;
case ccInc:
AddVMCommand(new_ctx, icAdd, 0);
break;
case ccSub:
AddVMCommand(new_ctx, icSub, 0);
break;
case ccAdd:
AddVMCommand(new_ctx, icAdd, 0);
break;
case ccXor:
AddVMCommand(new_ctx, icXor, 0);
break;
default:
throw 1;
}
}
}
/**
* ILVMCommand
*/
ILVMCommand::ILVMCommand(ILCommand *owner, ILCommandType command_type, uint64_t value, TokenReference *token_reference)
: BaseVMCommand(owner), command_type_(command_type), value_(value), token_reference_(token_reference), address_(0),
crypt_command_(icUnknown), crypt_size_(osDWord), crypt_key_(0), link_command_(NULL)
{
}
void ILVMCommand::Compile()
{
reinterpret_cast<ILVirtualMachine *>(owner()->block()->virtual_machine())->CompileCommand(*this);
}
void ILVMCommand::WriteToFile(IArchitecture &file)
{
if (!dump_.size())
return;
file.Write(dump_.data(), dump_.size());
}
/**
* ILFunction
*/
ILFunction::ILFunction(IFunctionList *owner, const FunctionName &name, CompilationType compilation_type, uint32_t compilation_options, bool need_compile, Folder *folder)
: BaseFunction(owner, name, compilation_type, compilation_options, need_compile, folder)
{
}
ILFunction::ILFunction(IFunctionList *owner /*= NULL*/)
: BaseFunction(owner, FunctionName(""), ctVirtualization, 0, true, NULL)
{
}
ILFunction::ILFunction(IFunctionList *owner, OperandSize cpu_address_size, IFunction *parent)
: BaseFunction(owner, cpu_address_size, parent)
{
}
ILFunction::ILFunction(IFunctionList *owner, const ILFunction &src)
: BaseFunction(owner, src)
{
}
bool ILFunction::Prepare(const CompileContext &ctx)
{
size_t i, j;
if (type() == otString) {
IArchitecture *file = from_runtime() ? ctx.runtime : ctx.file;
MapFunction *map_function = file->map_function_list()->GetFunctionByAddress(address());
if (map_function) {
uint64_t address = ctx.runtime->export_list()->GetAddressByType(atDecryptStringA);
if (!address)
return false;
NETRuntimeFunctionList *runtime_function_list = reinterpret_cast<NETRuntimeFunctionList *>(ctx.file->runtime_function_list());
ILMetaData *meta = reinterpret_cast<ILMetaData *>(ctx.runtime->command_list());
ILMethodDef *decrypt_method = meta->GetMethod(address);
if (!decrypt_method)
return false;
ILFunction *decrypt_func = reinterpret_cast<ILFunction*>(ctx.runtime->function_list()->GetFunctionByAddress(address));
if (!decrypt_func)
return false;
size_t orig_count = count();
ILTable *table = meta->table(ttMethodDef);
for (i = 0; i < orig_count; i++) {
ILMethodDef *method = reinterpret_cast<ILMethodDef*>(decrypt_method->Clone(meta, table));
method->set_deleted(false);
method->declaring_type()->AddMethod(method);
size_t old_count = count();
for (j = 0; j < decrypt_func->count(); j++) {
ILCommand *src_command = decrypt_func->item(j);
ILCommand *dst_command = reinterpret_cast<ILCommand*>(src_command->Clone(this));
dst_command->set_address(0);
if (static_cast<uint32_t>(dst_command->operand_value()) == FACE_STORAGE_INFO)
dst_command->set_operand_value(0, item(i)->address() - ctx.file->image_base());
AddObject(dst_command);
CommandLink *src_link = src_command->link();
if (src_link) {
CommandLink *dst_link = src_link->Clone(link_list());
dst_link->set_from_command(dst_command);
link_list()->AddObject(dst_link);
if (src_link->parent_command())
dst_link->set_parent_command(GetCommandByAddress(src_link->parent_command()->address()));
}
if (dst_command->token_reference()) {
TokenReference *token_reference = dst_command->token_reference()->Clone(dst_command->token_reference()->owner());
token_reference->owner()->AddObject(token_reference);
dst_command->set_token_reference(token_reference);
}
}
for (j = 0; j < decrypt_func->function_info_list()->count(); j++) {
FunctionInfo *src_info = decrypt_func->function_info_list()->item(j);
FunctionInfo *dst_info = src_info->Clone(function_info_list());
function_info_list()->AddObject(dst_info);
if (dst_info->entry())
dst_info->set_entry(item(old_count + decrypt_func->IndexOf(dst_info->entry())));
dst_info->set_source(runtime_function_list->Add(0, 0, 0, method));
}
AddressRange *address_range = function_info_list()->item(0)->Add(0, 0, NULL, NULL, NULL);
for (j = old_count; j < count(); j++) {
ILCommand *command = item(j);
command->set_address_range(address_range);
command->CompileToNative();
}
ILCommand *src_command = item(i);
for (j = 0; j < map_function->reference_list()->count(); j++) {
Reference *reference = map_function->reference_list()->item(j);
if (reference->operand_address() != src_command->address())
continue;
address = reference->address();
ILCommand *command = reinterpret_cast<ILCommand *>(ctx.file->function_list()->GetCommandByAddress(address, true));
if (!command) {
if (!ctx.file->AddressSeek(address))
return false;
CommandBlock *block = AddBlock(count(), true);
block->set_address(address);
command = Add(address);
command->ReadFromFile(*file);
command->set_block(block);
command->exclude_option(roClearOriginalCode);
}
if (command->token_reference())
command->token_reference()->set_deleted(true);
command->Init(icCall, method->id());
command->set_token_reference(method->reference_list()->Add(address + 1));
command->CompileToNative();
}
}
}
}
else {
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (!info->source())
continue;
std::map<size_t, size_t> variable_map;
NETRuntimeFunction *runtime_func = reinterpret_cast<NETRuntimeFunction *>(info->source());
ILStandAloneSig *locals = runtime_func->method()->locals();
std::vector<ILMethodDef *> method_list = runtime_func->method_list();
if (locals) {
if (locals->reference_list()->count() > 1) {
ILStandAloneSig *old_locals = locals;
locals = locals->Clone(locals->meta(), locals->owner());
locals->owner()->AddObject(locals);
locals->reference_list()->clear();
for (j = 0; j < method_list.size(); j++) {
method_list[j]->set_locals(locals);
}
for (j = old_locals->reference_list()->count(); j > 0; j--) {
TokenReference *token_reference = old_locals->reference_list()->item(j - 1);
if (item(IndexOf(info->entry()) + 1)->token_reference() == token_reference)
token_reference->set_owner(locals->reference_list());
}
}
}
else if (compilation_type() != ctVirtualization) {
ILData data;
data.push_back(stLocal);
data.push_back(0);
locals = runtime_func->method()->meta()->AddStandAloneSig(data);
for (j = 0; j < method_list.size(); j++) {
method_list[j]->set_locals(locals);
}
ILCommand *command = new ILCommand(this, osDWord, icComment, 0);
size_t insert_index = IndexOf(info->entry()) + 1;
InsertObject(insert_index, command);
command->set_token_reference(locals->reference_list()->Add(0));
command->set_address_range(info->entry()->address_range());
uint32_t dump = 0;
command->set_dump(&dump, sizeof(dump));
for (j = 0; j < block_list()->count(); j++) {
CommandBlock *block = block_list()->item(j);
if (block->start_index() >= insert_index) {
block->set_start_index(block->start_index() + 1);
block->set_end_index(block->end_index() + 1);
}
}
}
else
continue;
if (compilation_type() != ctMutation)
locals->set_deleted(true);
ILSignature *signature = locals->signature();
size_t old_count = signature->count();
ILElement *random_variable = new ILElement(locals->meta(), signature);
{
ILData data;
data.push_back(ELEMENT_TYPE_U4);
random_variable->Parse(data);
}
random_variable->set_is_predicate(true);
signature->AddObject(random_variable);
std::vector<ILElement *> element_list;
std::vector<size_t> index_list;
while (signature->count()) {
ILElement *element = signature->item(0);
signature->RemoveObject(element);
index_list.push_back(element_list.size());
element_list.push_back(element);
}
for (j = 0; j < index_list.size(); j++) {
std::swap(index_list[j], index_list[rand() % index_list.size()]);
}
for (j = 0; j < index_list.size(); j++) {
size_t old_index = index_list[j];
if (old_index < old_count)
variable_map[old_index] = j;
signature->AddObject(element_list[old_index]);
}
if (variable_map.empty())
continue;
std::map<size_t, size_t>::const_iterator it;
for (j = IndexOf(info->entry()) + 2; j < count(); j++) {
ILCommand *command = item(j);
switch (command->type()) {
case icLdloc:
case icLdloc_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloc, it->second);
command->CompileToNative();
break;
case icLdloca:
case icLdloca_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloca, it->second);
command->CompileToNative();
break;
case icStloc:
case icStloc_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icStloc, it->second);
command->CompileToNative();
break;
case icLdloc_0:
it = variable_map.find(0);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloc, it->second);
command->CompileToNative();
break;
case icLdloc_1:
it = variable_map.find(1);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloc, it->second);
command->CompileToNative();
break;
case icLdloc_2:
it = variable_map.find(2);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloc, it->second);
command->CompileToNative();
break;
case icLdloc_3:
it = variable_map.find(3);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icLdloc, it->second);
command->CompileToNative();
break;
case icStloc_0:
it = variable_map.find(0);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icStloc, it->second);
command->CompileToNative();
break;
case icStloc_1:
it = variable_map.find(1);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icStloc, it->second);
command->CompileToNative();
break;
case icStloc_2:
it = variable_map.find(2);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icStloc, it->second);
command->CompileToNative();
break;
case icStloc_3:
it = variable_map.find(3);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->Init(icStloc, it->second);
command->CompileToNative();
break;
}
if (command->is_data() || command->type() == icComment)
break;
}
}
}
if (tag() == ftProcessor) {
if (ctx.runtime)
return true;
}
return BaseFunction::Prepare(ctx);
}
void ILFunction::CompileLinks(const CompileContext &ctx)
{
BaseFunction::CompileLinks(ctx);
}
void ILFunction::CompileInfo(const CompileContext &ctx)
{
BaseFunction::CompileInfo(ctx);
size_t i;
FunctionInfo *info;
AddressRange *range;
ILCommand *command;
for (i = 0; i < function_info_list()->count(); i++) {
info = function_info_list()->item(i);
if (!info->begin())
continue;
if (info->entry()) {
uint32_t code_size = static_cast<uint32_t>(info->end() - info->begin());
command = reinterpret_cast<ILCommand *>(info->entry());
Data data;
for (size_t k = 0; k < command->dump_size(); k++) {
data.PushByte(command->dump(k));
}
if (command->dump_size() == 1) {
code_size--;
data[0] = CorILMethod_TinyFormat | (code_size << CorILMethod_FormatShift);
} else {
code_size -= (data[1] >> 4) * sizeof(uint32_t);
data.WriteDWord(offsetof(IMAGE_COR_ILMETHOD_FAT, CodeSize), code_size);
}
command->set_dump(data.data(), data.size());
}
}
for (i = 0; i < range_list()->count(); i++) {
range = range_list()->item(i);
info = function_info_list()->GetItemByAddress(range->begin());
if (!info)
continue;
uint64_t base_value = info->begin() + info->base_value();
if (range->begin_entry()) {
command = reinterpret_cast<ILCommand *>(range->begin_entry());
command->set_operand_value(0, range->begin() - base_value);
command->CompileToNative();
}
if (range->size_entry()) {
command = reinterpret_cast<ILCommand *>(range->size_entry());
command->set_operand_value(0, range->end() - range->begin());
command->CompileToNative();
}
}
}
void ILFunction::ReadFromBuffer(Buffer &buffer, IArchitecture &file)
{
BaseFunction::ReadFromBuffer(buffer, file);
}
void ILFunction::ParseBeginCommands(IArchitecture &)
{
}
void ILFunction::ParseEndCommands(IArchitecture &file)
{
if (type() != otString) {
size_t i;
size_t use_address_count = 0;
for (i = 0; i < file.map_function_list()->count(); i++) {
if (file.map_function_list()->item(i)->address() == address()) {
use_address_count++;
}
}
if (use_address_count > 1) {
for (i = 0; i < count(); i++) {
item(i)->exclude_option(roClearOriginalCode);
}
}
}
}
void ILFunction::CreateBlocks()
{
CommandBlock *cur_block = NULL;
for (size_t i = 0; i < count(); i++) {
ILCommand *command = item(i);
if (command->block() || (command->options() & roNeedCompile) == 0) {
cur_block = NULL;
continue;
}
if ((!cur_block || (command->options() & roCreateNewBlock) || item(cur_block->end_index())->is_data() != command->is_data()))
cur_block = AddBlock(i, true);
cur_block->set_end_index(i);
command->set_block(cur_block);
if (command->is_end())
cur_block = NULL;
}
}
void ILFunction::CalcStack(std::map<ILCommand *, int> &stack_map)
{
size_t i, j;
ILCommand *command, *entry, *link_command;
std::set<ILCommand *> entry_stack, link_list;
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (!info->source())
continue;
entry = reinterpret_cast<ILCommand *>(info->entry());
entry_stack.insert(entry);
stack_map[entry] = 0;
}
for (i = 0; i < this->link_list()->count(); i++) {
CommandLink *link = this->link_list()->item(i);
switch (link->type()) {
case ltSEHBlock:
entry = reinterpret_cast<ILCommand *>(link->to_command());
entry_stack.insert(entry);
stack_map[entry] = 1;
break;
case ltFinallyBlock:
entry = reinterpret_cast<ILCommand *>(link->to_command());
entry_stack.insert(entry);
stack_map[entry] = 0;
break;
}
}
int stack;
while (!entry_stack.empty()) {
entry = *entry_stack.begin();
j = IndexOf(entry);
std::map<ILCommand *, int>::const_iterator s = stack_map.find(entry);
if (s == stack_map.end())
throw std::runtime_error("Incorrect stack");
stack = s->second;
for (i = j; i < count(); i++) {
command = item(i);
std::set<ILCommand *>::const_iterator c = entry_stack.find(command);
if (c != entry_stack.end())
entry_stack.erase(c);
if (command->type() == icComment || command->type() == icCase || command->is_data())
continue;
s = stack_map.find(command);
if (s != stack_map.end()) {
if (s->second != stack)
throw std::runtime_error("Incorrect stack");
}
else
stack_map[command] = stack;
stack += command->GetStackLevel();
link_list.clear();
if (command->type() == icSwitch) {
size_t case_count = static_cast<uint32_t>(command->operand_value());
for (j = 0; j < case_count; j++) {
ILCommand *case_command = item(i + 1 + j);
link_list.insert(reinterpret_cast<ILCommand *>(case_command->link()->to_command()));
}
}
else {
if (command->link() && (command->link()->type() == ltJmp || command->link()->type() == ltJmpWithFlag))
link_list.insert(reinterpret_cast<ILCommand *>(command->link()->to_command()));
}
for (std::set<ILCommand *>::const_iterator l = link_list.begin(); l != link_list.end(); l++) {
link_command = *l;
s = stack_map.find(link_command);
if (s != stack_map.end()) {
if (s->second != stack)
throw std::runtime_error("Incorrect stack");
}
else {
entry_stack.insert(link_command);
stack_map[link_command] = stack;
}
}
if (command->is_end())
break;
}
}
}
void ILFunction::Mutate(const CompileContext &ctx)
{
size_t i, j, index;
ILCommand *command, *insert_command, *entry;
std::map<ILCommand *, ILSignature *> variables_map;
uint32_t value, old_value, bit_mask;
std::map<ILCommand *, int> stack_map;
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (!info->source())
continue;
ILStandAloneSig *locals = reinterpret_cast<NETRuntimeFunction *>(info->source())->method()->locals();
entry = reinterpret_cast<ILCommand *>(info->entry());
variables_map[entry] = locals ? locals->signature() : NULL;
stack_map[entry] = 0;
}
CalcStack(stack_map);
int stack;
std::list<ICommand *> new_command_list;
std::vector<ILCommand *> insert_command_list;
std::vector<size_t> predicate_list;
std::map<size_t, uint32_t> value_list;
std::map<ILCommand *, std::map<size_t, uint32_t> > value_list_map;
ILSignature *variables = NULL;
size_t orig_count = count();
std::vector<CommandLink *> insert_link_list;
for (i = 0; i < orig_count; i++) {
command = item(i);
new_command_list.push_back(command);
if ((command->options() & roNoProgress) == 0)
ctx.file->StepProgress();
if (is_breaked_address(command->address()))
continue;
if (command->block()) {
CommandBlock *block = command->block();
size_t insert_count = new_command_list.size() - 1 - i;
for (j = block->start_index() + 1; j <= block->end_index(); j++) {
new_command_list.push_back(item(j));
}
i = block->end_index();
if (insert_count) {
block->set_start_index(block->start_index() + insert_count);
block->set_end_index(block->end_index() + insert_count);
}
ctx.file->StepProgress(block->end_index() - block->start_index());
continue;
}
else if ((command->options() & roNeedCompile) == 0)
continue;
std::map<ILCommand *, ILSignature *>::const_iterator it = variables_map.find(command);
if (it != variables_map.end()) {
predicate_list.clear();
variables = it->second;
if (variables) {
for (j = 0; j < variables->count(); j++) {
ILElement *variable = variables->item(j);
if (variable->is_predicate())
predicate_list.push_back(j);
}
}
}
if (command->type() == icComment || command->type() == icCase || command->is_data())
continue;
insert_command_list.clear();
new_command_list.pop_back();
AddressRange *address_range = command->address_range();
ILCommand *link_command = (command->link() && (command->link()->type() == ltJmp || command->link()->type() == ltJmpWithFlag)) ? reinterpret_cast<ILCommand *>(command->link()->to_command()) : NULL;
if (command->section_options() & (rtLinkedToInt | rtLinkedToExt)) {
std::map<size_t, uint32_t> dst_value_list;
if ((command->section_options() & rtLinkedToExt) == 0) {
std::map<ILCommand *, std::map<size_t, uint32_t> >::iterator it = value_list_map.find(command);
if (it != value_list_map.end()) {
dst_value_list = it->second;
if (!new_command_list.empty() && !new_command_list.back()->is_end()) {
for (std::map<size_t, uint32_t>::const_iterator d = dst_value_list.begin(); d != dst_value_list.end(); d++) {
index = d->first;
value = d->second;
std::map<size_t, uint32_t>::const_iterator v = value_list.find(index);
if (v == value_list.end()) {
insert_command_list.push_back(AddCommand(icLdc_i4, value));
}
else {
insert_command_list.push_back(AddCommand(icLdloc, index));
old_value = v->second;
if (rand() & 1) {
insert_command_list.push_back(AddCommand(icLdc_i4, value - old_value));
insert_command_list.push_back(AddCommand(icAdd, 0));
}
else {
insert_command_list.push_back(AddCommand(icLdc_i4, value ^ old_value));
insert_command_list.push_back(AddCommand(icXor, 0));
}
}
insert_command_list.push_back(AddCommand(icStloc, index));
}
}
for (j = 0; j < insert_command_list.size(); j++) {
insert_command = insert_command_list[j];
insert_command->set_address_range(address_range);
insert_command->CompileToNative();
insert_command->include_option(roNoProgress);
new_command_list.push_back(insert_command);
}
if (!insert_command_list.empty()) {
insert_command = insert_command_list[0];
for (j = 0; j < link_list()->count(); j++) {
CommandLink *link = link_list()->item(j);
if (link->next_command() == command)
link->set_next_command(insert_command);
}
}
insert_command_list.clear();
}
else if (!value_list.empty()) {
value_list_map[command] = value_list;
dst_value_list = value_list;
}
}
value_list = dst_value_list;
}
if (!predicate_list.empty() && !command->is_end()) {
if (rand() & 1) {
// modify predicate
value = rand32();
bit_mask = 0x1f;
index = predicate_list[rand() % predicate_list.size()];
std::map<size_t, uint32_t>::const_iterator it = value_list.find(index);
if (it != value_list.end()) {
old_value = it->second;
if (rand() & 1) {
insert_command_list.push_back(AddCommand(icLdc_i4, value));
insert_command_list.push_back(AddCommand(icLdloc, index));
}
else {
insert_command_list.push_back(AddCommand(icLdloc, index));
insert_command_list.push_back(AddCommand(icLdc_i4, value));
std::swap(value, old_value);
}
switch (rand() % (8 + (old_value != 0 ? 2 : 0))) {
case 0:
value += old_value;
insert_command_list.push_back(AddCommand(icAdd, 0));
break;
case 1:
value -= old_value;
insert_command_list.push_back(AddCommand(icSub, 0));
break;
case 2:
old_value &= bit_mask;
if (insert_command_list.back()->type() == icLdc_i4)
insert_command_list.back()->set_operand_value(0, old_value);
else {
insert_command_list.push_back(AddCommand(icLdc_i4, bit_mask));
insert_command_list.push_back(AddCommand(icAnd, 0));
}
value >>= old_value;
insert_command_list.push_back(AddCommand(icShr_un, 0));
break;
case 3:
old_value &= bit_mask;
if (insert_command_list.back()->type() == icLdc_i4)
insert_command_list.back()->set_operand_value(0, old_value);
else {
insert_command_list.push_back(AddCommand(icLdc_i4, bit_mask));
insert_command_list.push_back(AddCommand(icAnd, 0));
}
value <<= old_value;
insert_command_list.push_back(AddCommand(icShl, 0));
break;
case 4:
value |= old_value;
insert_command_list.push_back(AddCommand(icOr, 0));
break;
case 5:
value &= old_value;
insert_command_list.push_back(AddCommand(icAnd, 0));
break;
case 6:
value *= old_value;
insert_command_list.push_back(AddCommand(icMul, 0));
break;
case 8:
value /= old_value;
insert_command_list.push_back(AddCommand(icDiv_un, 0));
break;
case 9:
value %= old_value;
insert_command_list.push_back(AddCommand(icRem_un, 0));
break;
default:
value ^= old_value;
insert_command_list.push_back(AddCommand(icXor, 0));
break;
}
}
else {
insert_command_list.push_back(AddCommand(icLdc_i4, value));
}
insert_command_list.push_back(AddCommand(icStloc, index));
value_list[index] = value;
}
}
if (!value_list.empty()) {
index = NOT_ID;
for (std::map<size_t, uint32_t>::const_iterator it = value_list.begin(); it != value_list.end(); it++) {
index = it->first;
if (rand() & 1)
break;
}
value = value_list[index];
if (command->link() == NULL && (command->type() == icLdc_i4 || command->type() == icLdc_i4_s
|| command->type() == icLdc_i4_0 || command->type() == icLdc_i4_1 || command->type() == icLdc_i4_2 || command->type() == icLdc_i4_3
|| command->type() == icLdc_i4_4 || command->type() == icLdc_i4_5 || command->type() == icLdc_i4_6 || command->type() == icLdc_i4_7
|| command->type() == icLdc_i4_8 || command->type() == icLdc_i4_m1)) {
switch (command->type()) {
case icLdc_i4_0:
old_value = 0;
break;
case icLdc_i4_1:
old_value = 1;
break;
case icLdc_i4_2:
old_value = 2;
break;
case icLdc_i4_3:
old_value = 3;
break;
case icLdc_i4_4:
old_value = 4;
break;
case icLdc_i4_5:
old_value = 5;
break;
case icLdc_i4_6:
old_value = 6;
break;
case icLdc_i4_7:
old_value = 7;
break;
case icLdc_i4_8:
old_value = 8;
break;
case icLdc_i4_m1:
old_value = static_cast<uint32_t>(-1);
break;
case icLdc_i4_s:
old_value = static_cast<int8_t>(command->operand_value());
break;
default:
old_value = static_cast<uint32_t>(command->operand_value());
break;
}
insert_command_list.push_back(AddCommand(icLdloc, index));
switch (rand() & 3) {
case 0:
insert_command_list.push_back(AddCommand(icLdc_i4, old_value - value));
insert_command_list.push_back(AddCommand(icAdd, 0));
break;
case 1:
insert_command_list.push_back(AddCommand(icLdc_i4, value - old_value));
insert_command_list.push_back(AddCommand(icSub, 0));
break;
default:
insert_command_list.push_back(AddCommand(icLdc_i4, old_value ^ value));
insert_command_list.push_back(AddCommand(icXor, 0));
break;
}
command->clear();
command->Init(icNop);
}
else if (rand() & 1) {
stack = stack_map[command];
if (stack < 1)
{
ILCommand *random_command = NULL;
for (j = 0; j < orig_count; j++) {
insert_command = item(j);
if (insert_command->type() == icComment || insert_command->type() == icCase || insert_command->is_data())
continue;
if (insert_command != command && insert_command->address_range() == address_range) {
if (stack == stack_map[insert_command]) {
random_command = insert_command;
if (rand() & 1)
break;
}
}
while (insert_command->is_prefix() && j < orig_count) {
insert_command = item(++j);
}
}
if (random_command) {
old_value = rand32();
bit_mask = 0x1f;
if (rand() & 1) {
insert_command_list.push_back(AddCommand(icLdloc, index));
insert_command_list.push_back(AddCommand(icLdc_i4, old_value));
}
else {
insert_command_list.push_back(AddCommand(icLdc_i4, old_value));
insert_command_list.push_back(AddCommand(icLdloc, index));
std::swap(value, old_value);
}
ILCommandType branch_type;
if (rand() & 1) {
switch (rand() % 3) {
case 0:
branch_type = (value < old_value) ? icBge_un : icBlt_un;
break;
case 1:
branch_type = (value > old_value) ? icBle_un : icBgt_un;
break;
default:
branch_type = (value != old_value) ? icBeq : icBne_un;
break;
}
}
else {
switch (rand() % (8 + (old_value != 0 ? 2 : 0))) {
case 0:
value += old_value;
insert_command_list.push_back(AddCommand(icAdd, 0));
break;
case 1:
value -= old_value;
insert_command_list.push_back(AddCommand(icSub, 0));
break;
case 2:
old_value &= bit_mask;
if (insert_command_list.back()->type() == icLdc_i4)
insert_command_list.back()->set_operand_value(0, old_value);
else {
insert_command_list.push_back(AddCommand(icLdc_i4, bit_mask));
insert_command_list.push_back(AddCommand(icAnd, 0));
}
value >>= old_value;
insert_command_list.push_back(AddCommand(icShr_un, 0));
break;
case 3:
old_value &= bit_mask;
if (insert_command_list.back()->type() == icLdc_i4)
insert_command_list.back()->set_operand_value(0, old_value);
else {
insert_command_list.push_back(AddCommand(icLdc_i4, bit_mask));
insert_command_list.push_back(AddCommand(icAnd, 0));
}
value <<= old_value;
insert_command_list.push_back(AddCommand(icShl, 0));
break;
case 5:
value &= old_value;
insert_command_list.push_back(AddCommand(icAnd, 0));
break;
case 6:
value *= old_value;
insert_command_list.push_back(AddCommand(icMul, 0));
break;
case 8:
value /= old_value;
insert_command_list.push_back(AddCommand(icDiv_un, 0));
break;
case 9:
value %= old_value;
insert_command_list.push_back(AddCommand(icRem_un, 0));
break;
default:
value ^= old_value;
insert_command_list.push_back(AddCommand(icXor, 0));
break;
}
branch_type = value ? icBrfalse : icBrtrue;
}
insert_command = AddCommand(branch_type, 0);
insert_link_list.push_back(insert_command->AddLink(0, ltJmpWithFlag, random_command));
insert_command_list.push_back(insert_command);
}
}
}
}
if (command->type() == icSwitch) {
size_t case_count = static_cast<uint32_t>(command->operand_value());
std::set<ILCommand *> link_command_list;
for (j = 0; j < case_count; j++) {
ILCommand *case_command = item(i + 1 + j);
link_command_list.insert(reinterpret_cast<ILCommand *>(case_command->link()->to_command()));
}
for (std::set<ILCommand *>::const_iterator l = link_command_list.begin(); l != link_command_list.end(); l++) {
link_command = *l;
if (link_command && (link_command->section_options() & rtLinkedToExt) == 0) {
std::map<ILCommand *, std::map<size_t, uint32_t> >::const_iterator it = value_list_map.find(link_command);
if (it == value_list_map.end())
value_list_map[link_command] = value_list;
else
value_list_map.erase(it);
}
}
} else if (link_command && (link_command->section_options() & rtLinkedToExt) == 0) {
std::map<ILCommand *, std::map<size_t, uint32_t> >::const_iterator it = value_list_map.find(link_command);
if (it == value_list_map.end())
value_list_map[link_command] = value_list;
else {
// syncronize predicates
std::map<size_t, uint32_t> dst_value_list = it->second;
for (std::map<size_t, uint32_t>::const_iterator d = dst_value_list.begin(); d != dst_value_list.end(); d++) {
index = d->first;
value = d->second;
std::map<size_t, uint32_t>::const_iterator v = value_list.find(index);
if (v == value_list.end()) {
insert_command_list.push_back(AddCommand(icLdc_i4, value));
}
else {
insert_command_list.push_back(AddCommand(icLdloc, index));
old_value = v->second;
if (rand() & 1) {
insert_command_list.push_back(AddCommand(icLdc_i4, value - old_value));
insert_command_list.push_back(AddCommand(icAdd, 0));
}
else {
insert_command_list.push_back(AddCommand(icLdc_i4, value ^ old_value));
insert_command_list.push_back(AddCommand(icXor, 0));
}
}
insert_command_list.push_back(AddCommand(icStloc, index));
value_list[index] = value;
}
}
}
if (!insert_command_list.empty()) {
if (command->type() == icNop) {
new_command_list.push_back(command);
command = NULL;
} else if (command->section_options() & (rtLinkedToInt | rtLinkedToExt)) {
insert_command = command->Clone(this);
if (command->link())
command->link()->set_from_command(insert_command);
command->clear();
command->Init(icNop);
new_command_list.push_back(command);
command = insert_command;
}
for (j = 0; j < insert_command_list.size(); j++) {
insert_command = insert_command_list[j];
insert_command->set_address_range(address_range);
insert_command->CompileToNative();
new_command_list.push_back(insert_command);
}
}
if (command) {
new_command_list.push_back(command);
while (command->is_prefix() && i < orig_count) {
command = item(++i);
new_command_list.push_back(command);
}
if (command->is_end())
value_list.clear();
}
}
assign(new_command_list);
for (i = 0; i < insert_link_list.size(); i++) {
insert_link_list[i]->from_command()->PrepareLink(ctx);
}
}
void ILFunction::CompileToNative(const CompileContext &ctx)
{
size_t i, j, k;
ILCommand *command;
uint16_t variable_index;
uint16_t add_stack = (compilation_type() == ctMutation) ? 20 : 0;
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
ILCommand *entry = reinterpret_cast<ILCommand *>(info->entry());
if (entry) {
ILStandAloneSig *locals = reinterpret_cast<NETRuntimeFunction *>(info->source())->method()->locals();
Data data;
if (entry->dump_size() == 1) {
data.PushByte(CorILMethod_InitLocals | CorILMethod_FatFormat);
data.PushByte(0x30);
data.PushWord(0x08 + add_stack);
}
else {
data.PushByte(entry->dump(0));
data.PushByte(entry->dump(1));
data.PushWord(static_cast<uint16_t>(entry->dump_value(2, osWord)) + add_stack);
}
data.PushDWord(0x0);
if (!locals)
data.PushDWord(0x0);
entry->set_dump(data.data(), data.size());
}
ILCommand *data_entry = reinterpret_cast<ILCommand *>(info->data_entry());
if (!data_entry)
continue;
uint32_t flags = static_cast<uint32_t>(data_entry->operand_value());
bool fat = (flags & CorILMethod_Sect_FatFormat) != 0;
if (fat)
continue;
if (flags & CorILMethod_Sect_EHTable) {
// convert EHTable to fat format
uint32_t data_size = fat ? (flags >> 8) : static_cast<uint8_t>(flags >> 8);
uint32_t clauses = (data_size - sizeof(uint32_t)) / (fat ? 24 : 12);
k = IndexOf(data_entry) + 1;
for (j = 0; j < clauses * 6; j++) {
command = item(k + j);
command->Init(icDword, command->operand_value(), command->token_reference());
command->CompileToNative();
}
data_size = sizeof(uint32_t) + clauses * 6 * sizeof(uint32_t);
data_entry->set_operand_value(0, (data_size << 8) | (flags & 0xff) | CorILMethod_Sect_FatFormat);
data_entry->CompileToNative();
}
}
for (i = 0; i < count(); i++) {
command = item(i);
switch (command->type()) {
case icBr_s:
command->Init(icBr, command->operand_value());
command->CompileToNative();
break;
case icBrfalse_s:
command->Init(icBrfalse, command->operand_value());
command->CompileToNative();
break;
case icBrtrue_s:
command->Init(icBrtrue, command->operand_value());
command->CompileToNative();
break;
case icBeq_s:
command->Init(icBeq, command->operand_value());
command->CompileToNative();
break;
case icBge_s:
command->Init(icBge, command->operand_value());
command->CompileToNative();
break;
case icBgt_s:
command->Init(icBgt, command->operand_value());
command->CompileToNative();
break;
case icBle_s:
command->Init(icBle, command->operand_value());
command->CompileToNative();
break;
case icBlt_s:
command->Init(icBlt, command->operand_value());
command->CompileToNative();
break;
case icBne_un_s:
command->Init(icBne_un, command->operand_value());
command->CompileToNative();
break;
case icBge_un_s:
command->Init(icBge_un, command->operand_value());
command->CompileToNative();
break;
case icBgt_un_s:
command->Init(icBgt_un, command->operand_value());
command->CompileToNative();
break;
case icBle_un_s:
command->Init(icBle_un, command->operand_value());
command->CompileToNative();
break;
case icBlt_un_s:
command->Init(icBlt_un, command->operand_value());
command->CompileToNative();
break;
case icLeave_s:
command->Init(icLeave, command->operand_value());
command->CompileToNative();
break;
case icLdloc:
variable_index = static_cast<uint16_t>(command->operand_value());
switch (variable_index) {
case 0:
command->Init(icLdloc_0, 0);
command->CompileToNative();
break;
case 1:
command->Init(icLdloc_1, 0);
command->CompileToNative();
break;
case 2:
command->Init(icLdloc_2, 0);
command->CompileToNative();
break;
case 3:
command->Init(icLdloc_3, 0);
command->CompileToNative();
break;
default:
if (variable_index < 0x100) {
command->Init(icLdloc_s, variable_index);
command->CompileToNative();
}
break;
}
break;
case icLdloca:
variable_index = static_cast<uint16_t>(command->operand_value());
if (variable_index < 0x100) {
command->Init(icLdloca_s, variable_index);
command->CompileToNative();
}
break;
case icStloc:
variable_index = static_cast<uint16_t>(command->operand_value());
switch (variable_index) {
case 0:
command->Init(icStloc_0, 0);
command->CompileToNative();
break;
case 1:
command->Init(icStloc_1, 0);
command->CompileToNative();
break;
case 2:
command->Init(icStloc_2, 0);
command->CompileToNative();
break;
case 3:
command->Init(icStloc_3, 0);
command->CompileToNative();
break;
default:
if (variable_index < 0x100) {
command->Init(icStloc_s, variable_index);
command->CompileToNative();
}
break;
}
break;
}
}
CreateBlocks();
}
bool IsBoxedElementType(CorElementType type)
{
switch (type) {
case ELEMENT_TYPE_BOOLEAN: case ELEMENT_TYPE_CHAR: case ELEMENT_TYPE_I1: case ELEMENT_TYPE_U1:
case ELEMENT_TYPE_I2: case ELEMENT_TYPE_U2: case ELEMENT_TYPE_I4: case ELEMENT_TYPE_U4:
case ELEMENT_TYPE_I8: case ELEMENT_TYPE_U8: case ELEMENT_TYPE_R4: case ELEMENT_TYPE_R8:
case ELEMENT_TYPE_I: case ELEMENT_TYPE_U: case ELEMENT_TYPE_VALUETYPE:
return true;
}
return false;
}
void ILFunction::CompileToVM(const CompileContext &ctx)
{
if (function_info_list()->count() == 0)
return;
struct FunctionEntryInfo {
ILMethodDef *method;
ILToken *ret;
std::vector<ILElement *> param_list;
ILSignature *locals;
ILCommand *data_entry;
FunctionEntryInfo()
: method(NULL), ret(NULL), locals(NULL), data_entry(NULL)
{
}
FunctionEntryInfo(ILMethodDef *method_, ILToken *ret_, std::vector<ILElement *> param_list_, ILSignature *locals_, ILCommand *data_entry_)
{
method = method_;
ret = ret_;
param_list = param_list_;
locals = locals_;
data_entry = data_entry_;
}
};
size_t i, j, c, k;
ILCommand *command, *data_entry;
CommandLink *link;
CommandBlock *cur_block;
std::string type_name;
ILToken *type, *type_ret;
ILSignature *locals;
ILMetaData *meta;
ILElement *element;
ILMethodDef *method;
std::map<ICommand *, FunctionEntryInfo> entry_map;
// create internal links
c = link_list()->count();
for (i = 0; i < count(); i++) {
command = item(i);
if (command->block() || (command->options() & roNeedCompile) == 0)
continue;
switch (command->type()) {
case icSwitch:
command->AddLink(0, ltSwitch, item(i + 1));
break;
}
}
for (i = c; i < link_list()->count(); i++) {
link_list()->item(i)->from_command()->PrepareLink(ctx);
}
ILVirtualMachine *virtual_machine = reinterpret_cast<ILVirtualMachine *>(BaseFunction::virtual_machine(ctx.file->virtual_machine_list(), NULL));
for (k = 0; k < function_info_list()->count(); k++) {
FunctionInfo *info = function_info_list()->item(k);
if (!info->source() || info->entry()->block())
continue;
ICommand *entry = info->entry();
data_entry = reinterpret_cast<ILCommand *>(info->data_entry());
method = reinterpret_cast<NETRuntimeFunction *>(info->source())->method();
ILSignature *signature = method->signature();
if (!signature->ret())
throw std::runtime_error("Invalid method signature");
locals = method->locals() ? method->locals()->signature() : NULL;
meta = method->meta();
ILElement object_element(NULL, NULL);
{
ILData data;
data.push_back(ELEMENT_TYPE_OBJECT);
object_element.Parse(data);
}
ILToken *type_object = meta->ImportType(object_element);
type_ret = (signature->ret()->type() == ELEMENT_TYPE_VOID) ? NULL : meta->ImportType(*signature->ret());
bool need_try_finally = false;
size_t try_start_index = 0;
size_t try_end_index = 0;
size_t finally_start_index = 0;
size_t finally_end_index = 0;
size_t newarr_index = 0;
std::vector<ILElement *> param_list;
for (i = 0; i < signature->count(); i++) {
element = signature->item(i);
if (element->is_ref())
need_try_finally = true;
param_list.push_back(element);
}
if ((method->flags() & mdStatic) == 0)
param_list.insert(param_list.begin(), &object_element);
if (!method->is_deleted()) {
cur_block = AddBlock(count(), true);
// header
Data data;
data.PushByte(CorILMethod_InitLocals | CorILMethod_FatFormat | (need_try_finally ? CorILMethod_MoreSects : 0));
data.PushByte(0x30);
data.PushWord(0x08);
data.PushDWord(0x0);
command = AddCommand(data);
command->set_alignment(sizeof(uint32_t));
info = function_info_list()->Add(info->begin(), info->begin(), btValue, command->dump_size() + sizeof(uint32_t), 0, 0, info->source(), command);
// locals
if (need_try_finally) {
ILData locals_data;
locals_data.push_back(stLocal);
locals_data.push_back(1 + ((signature->ret()->type() == ELEMENT_TYPE_VOID) ? 0 : 1));
locals_data.push_back(ELEMENT_TYPE_SZARRAY);
locals_data.push_back(ELEMENT_TYPE_OBJECT);
if (signature->ret()->type() != ELEMENT_TYPE_VOID)
signature->ret()->WriteToData(locals_data);
ILStandAloneSig *new_locals = meta->AddStandAloneSig(locals_data);
command = AddCommand(icDword, new_locals->id());
command->set_token_reference(new_locals->reference_list()->Add(0));
}
else {
AddCommand(icDword, 0);
}
newarr_index = count();
if (!param_list.empty()) {
// load args
AddCommand(icLdc_i4, param_list.size());
command = AddCommand(icNewarr, type_object->id());
command->set_token_reference(type_object->reference_list()->Add(0));
for (i = 0; i < param_list.size(); i++) {
AddCommand(icDup, 0);
switch (i) {
case 0: AddCommand(icLdc_i4_0, 0); break;
case 1: AddCommand(icLdc_i4_1, 0); break;
case 2: AddCommand(icLdc_i4_2, 0); break;
case 3: AddCommand(icLdc_i4_3, 0); break;
case 4: AddCommand(icLdc_i4_4, 0); break;
case 5: AddCommand(icLdc_i4_5, 0); break;
case 6: AddCommand(icLdc_i4_6, 0); break;
case 7: AddCommand(icLdc_i4_7, 0); break;
case 8: AddCommand(icLdc_i4_8, 0); break;
default: AddCommand(icLdc_i4, i); break;
}
switch (i) {
case 0: AddCommand(icLdarg_0, 0); break;
case 1: AddCommand(icLdarg_1, 0); break;
case 2: AddCommand(icLdarg_2, 0); break;
case 3: AddCommand(icLdarg_3, 0); break;
default: AddCommand(icLdarg_s, i); break;
}
element = param_list[i];
if (element->is_ref()) {
switch (element->type()) {
case ELEMENT_TYPE_BOOLEAN:
case ELEMENT_TYPE_I1:
AddCommand(icLdind_i1, 0);
break;
case ELEMENT_TYPE_U1:
AddCommand(icLdind_u1, 0);
break;
case ELEMENT_TYPE_I2:
AddCommand(icLdind_i2, 0);
break;
case ELEMENT_TYPE_CHAR:
case ELEMENT_TYPE_U2:
AddCommand(icLdind_u2, 0);
break;
case ELEMENT_TYPE_I4:
AddCommand(icLdind_i4, 0);
break;
case ELEMENT_TYPE_U4:
AddCommand(icLdind_u4, 0);
break;
case ELEMENT_TYPE_I8:
AddCommand(icLdind_i8, 0);
break;
case ELEMENT_TYPE_R4:
AddCommand(icLdind_r4, 0);
break;
case ELEMENT_TYPE_R8:
AddCommand(icLdind_r8, 0);
break;
case ELEMENT_TYPE_I:
case ELEMENT_TYPE_U:
AddCommand(icLdind_i, 0);
break;
default:
AddCommand(icLdind_ref, 0);
break;
}
}
type = meta->ImportType(*element);
if (element->type() == ELEMENT_TYPE_PTR) {
uint64_t address = ctx.vm_runtime->export_list()->GetAddressByType(atBoxPointer);
ILFunction *func = reinterpret_cast<ILFunction *>(ctx.vm_runtime->function_list()->GetFunctionByAddress(address));
for (j = 0; j < func->count(); j++) {
ILCommand *src_command = func->item(j);
if (src_command->type() == icComment || src_command->type() == icLdarg_0)
continue;
if (src_command->type() == icRet)
break;
command = src_command->Clone(this);
if (command->type() == icLdtoken) {
command->set_operand_value(0, type->id());
command->set_token_reference(type->reference_list()->Add(0));
} else if (command->token_reference())
command->set_token_reference(command->token_reference()->owner()->Add(0));
AddObject(command);
}
} else if (IsBoxedElementType(element->type())) {
command = AddCommand(icBox, type->id());
command->set_token_reference(type->reference_list()->Add(0));
}
AddCommand(icStelem_ref, 0);
}
if (need_try_finally)
AddCommand(icStloc_0, 0);
}
if (need_try_finally || param_list.empty()) {
try_start_index = count();
command = AddCommand(icNewobj, virtual_machine->ctor()->id());
command->set_token_reference(virtual_machine->ctor()->reference_list()->Add(0));
if (param_list.empty())
AddCommand(icLdnull, 0);
else
AddCommand(icLdloc_0, 0);
}
else {
command = new ILCommand(this, cpu_address_size(), icNewobj, virtual_machine->ctor()->id());
command->set_token_reference(virtual_machine->ctor()->reference_list()->Add(0));
InsertObject(newarr_index, command);
}
command = AddCommand(icLdc_i4, 0);
link = command->AddLink(0, ltOffset, entry);
entry->include_section_option(rtLinkedToExt);
link->set_sub_value(ctx.file->image_base());
command = AddCommand(icCall, virtual_machine->invoke()->id());
command->set_token_reference(virtual_machine->invoke()->reference_list()->Add(0));
if (!type_ret)
AddCommand(icPop, 0);
else {
if (signature->ret()->type() == ELEMENT_TYPE_PTR) {
uint64_t address = ctx.vm_runtime->export_list()->GetAddressByType(atUnboxPointer);
ILFunction *func = reinterpret_cast<ILFunction *>(ctx.vm_runtime->function_list()->GetFunctionByAddress(address));
for (j = 0; j < func->count(); j++) {
ILCommand *src_command = func->item(j);
if (src_command->type() == icComment || src_command->type() == icLdarg_0)
continue;
if (src_command->type() == icRet)
break;
command = src_command->Clone(this);
if (command->token_reference())
command->set_token_reference(command->token_reference()->owner()->Add(0));
AddObject(command);
}
} else {
command = AddCommand(IsBoxedElementType(signature->ret()->type()) ? icUnbox_any : icCastclass, type_ret->id());
command->set_token_reference(type_ret->reference_list()->Add(0));
}
if (need_try_finally)
AddCommand(icStloc_1, 0);
}
if (need_try_finally) {
command = AddCommand(icLeave, 0);
command->AddLink(0, ltJmp);
try_end_index = count() - 1;
if (!param_list.empty()) {
// save out args
finally_start_index = count();
for (i = 0; i < param_list.size(); i++) {
element = param_list[i];
if (!element->is_ref())
continue;
switch (i) {
case 0: AddCommand(icLdarg_0, 0); break;
case 1: AddCommand(icLdarg_1, 0); break;
case 2: AddCommand(icLdarg_2, 0); break;
case 3: AddCommand(icLdarg_3, 0); break;
default: AddCommand(icLdarg_s, i); break;
}
AddCommand(icLdloc_0, 0);
switch (i) {
case 0: AddCommand(icLdc_i4_0, 0); break;
case 1: AddCommand(icLdc_i4_1, 0); break;
case 2: AddCommand(icLdc_i4_2, 0); break;
case 3: AddCommand(icLdc_i4_3, 0); break;
case 4: AddCommand(icLdc_i4_4, 0); break;
case 5: AddCommand(icLdc_i4_5, 0); break;
case 6: AddCommand(icLdc_i4_6, 0); break;
case 7: AddCommand(icLdc_i4_7, 0); break;
case 8: AddCommand(icLdc_i4_8, 0); break;
default: AddCommand(icLdc_i4, i); break;
}
AddCommand(icLdelem_ref, 0);
if (IsBoxedElementType(element->type())) {
type = meta->ImportType(*element);
command = AddCommand(icUnbox_any, type->id());
command->set_token_reference(type->reference_list()->Add(0));
}
switch (element->type()) {
case ELEMENT_TYPE_BOOLEAN:
case ELEMENT_TYPE_I1:
case ELEMENT_TYPE_U1:
AddCommand(icStind_i1, 0);
break;
case ELEMENT_TYPE_CHAR:
case ELEMENT_TYPE_I2:
case ELEMENT_TYPE_U2:
AddCommand(icStind_i2, 0);
break;
case ELEMENT_TYPE_I4:
case ELEMENT_TYPE_U4:
AddCommand(icStind_i4, 0);
break;
case ELEMENT_TYPE_I8:
AddCommand(icStind_i8, 0);
break;
case ELEMENT_TYPE_R4:
AddCommand(icStind_r4, 0);
break;
case ELEMENT_TYPE_R8:
AddCommand(icStind_r8, 0);
break;
case ELEMENT_TYPE_I:
case ELEMENT_TYPE_U:
AddCommand(icStind_i, 0);
break;
default:
AddCommand(icStind_ref, 0);
break;
}
}
AddCommand(icEndfinally, 0);
finally_end_index = count() - 1;
if (type_ret)
AddCommand(icLdloc_1, 0);
}
}
AddCommand(icRet, 0);
AddressRange *address_range = info->Add(0, 0, NULL, NULL, NULL);
for (i = cur_block->start_index(); i < count(); i++) {
command = item(i);
command->set_block(cur_block);
command->set_address_range(address_range);
command->CompileToNative();
cur_block->set_end_index(i);
}
if (need_try_finally) {
item(try_end_index)->link()->set_to_command(item(finally_end_index + 1));
address_range = function_info_list()->Add(info->begin() + 1, info->begin() + 1, btValue, 0, 0, 0, NULL, NULL)->Add(0, 0, NULL, NULL, NULL);
cur_block = AddBlock(count(), true);
command = AddCommand(icDword, (0x1c << 8) | CorILMethod_Sect_FatFormat | CorILMethod_Sect_EHTable);
command->set_alignment(sizeof(uint32_t));
AddCommand(icDword, COR_ILEXCEPTION_CLAUSE_FINALLY);
ILCommand *try_entry = AddCommand(icDword, 0);
ILCommand *try_length = AddCommand(icDword, 0);
ILCommand *finally_entry = AddCommand(icDword, 0);
link = finally_entry->AddLink(0, ltFinallyBlock, item(finally_start_index));
link->set_base_function_info(info);
ILCommand *finally_length = AddCommand(icDword, 0);
AddCommand(icDword, 0);
for (i = cur_block->start_index(); i < count(); i++) {
command = item(i);
command->set_block(cur_block);
command->set_address_range(address_range);
command->CompileToNative();
cur_block->set_end_index(i);
}
address_range = range_list()->Add(0, 0, try_entry, NULL, try_length);
for (i = try_start_index; i <= try_end_index; i++) {
command = item(i);
command->set_address_range(address_range);
}
address_range = range_list()->Add(0, 0, finally_entry, NULL, finally_length);
for (i = finally_start_index; i <= finally_end_index; i++) {
command = item(i);
command->set_address_range(address_range);
}
}
}
entry_map[entry] = FunctionEntryInfo(method, type_ret, param_list, locals, data_entry);
}
cur_block = NULL;
uint64_t image_base = ctx.file->image_base();
std::set<uint64_t> try_list;
std::map<uint32_t, uint32_t> variable_map, param_map;
size_t temp_variable_index = NOT_ID;
for (i = 0; i < count(); i++) {
command = item(i);
if ((command->options() & roNoProgress) == 0)
ctx.file->StepProgress();
if (command->block() || (command->options() & roNeedCompile) == 0) {
cur_block = NULL;
continue;
}
if (command->is_data()) {
cur_block = NULL;
continue;
}
if (!cur_block) {
cur_block = AddBlock(i, false);
cur_block->set_virtual_machine(virtual_machine);
}
cur_block->set_end_index(i);
command->set_block(cur_block);
if (command->type() == icComment) {
if (command->token_reference())
command->token_reference()->set_deleted(true);
std::map<ICommand *, FunctionEntryInfo>::const_iterator it = entry_map.find(command);
if (it != entry_map.end()) {
try_list.clear();
type_ret = it->second.ret;
method = it->second.method;
meta = method->meta();
locals = it->second.locals;
data_entry = it->second.data_entry;
std::vector<ILElement *> param_list = it->second.param_list;
variable_map.clear();
param_map.clear();
if (command->section_options() & rtLinkedToInt) {
command->AddVMCommand(ctx, icWord, param_list.size(), 0);
for (j = param_list.size(); j > 0; j--) {
element = param_list[j - 1];
type = meta->ImportType(*element);
command->AddVMCommand(ctx, icDword, type->id(), 0, type->reference_list()->Add(0));
}
if (type_ret)
command->AddVMCommand(ctx, icDword, type_ret->id(), 0, type_ret->reference_list()->Add(0));
else
command->AddVMCommand(ctx, icDword, 0, 0);
if (command->section_options() & rtLinkedToExt)
command->AddExtSection(ctx);
}
if (data_entry) {
uint32_t flags = static_cast<uint32_t>(data_entry->operand_value());
if (flags & CorILMethod_Sect_EHTable) {
bool fat = (flags & CorILMethod_Sect_FatFormat) != 0;
uint32_t data_size = fat ? (flags >> 8) : static_cast<uint8_t>(flags >> 8);
uint32_t clauses = (data_size - sizeof(uint32_t)) / (fat ? 24 : 12);
k = IndexOf(data_entry) + 1;
for (j = 0; j < clauses; j++) {
ILCommand *catch_flags = item(k++);
ILCommand *catch_try_offset = item(k++);
ILCommand *catch_try_length = item(k++);
ILCommand *catch_handler_offset = item(k++);
ILCommand *catch_handler_length = item(k++);
ILCommand *catch_filter = item(k++);
command->AddVMCommand(ctx, icInitcatchblock, 0);
command->AddVMCommand(ctx, icByte, static_cast<uint8_t>(catch_flags->operand_value()));
uint64_t try_address = method->address() + method->fat_size() + catch_try_offset->operand_value();
try_list.insert(try_address);
command->AddVMCommand(ctx, icDword, static_cast<uint32_t>(try_address - image_base));
command->AddVMCommand(ctx, icDword, static_cast<uint32_t>(try_address - image_base + catch_try_length->operand_value()));
command->AddVMCommand(ctx, icDword, 0, 0, NULL, catch_handler_offset->link() ? reinterpret_cast<ILCommand *>(catch_handler_offset->link()->to_command()) : NULL);
if (catch_filter->link()) {
command->AddVMCommand(ctx, icDword, 0, 0, NULL, reinterpret_cast<ILCommand *>(catch_filter->link()->to_command()));
}
else {
command->AddVMCommand(ctx, icDword, catch_filter->operand_value(), 0, catch_filter->token_reference());
}
}
}
}
// merge params and locals
std::vector<ILElement *> arg_list;
for (j = 0; j < param_list.size(); j++) {
arg_list.push_back(param_list[j]);
}
if (locals) {
for (j = 0; j < locals->count(); j++) {
arg_list.push_back(locals->item(j));
}
}
for (j = 0; j < arg_list.size(); j++) {
std::swap(arg_list[j], arg_list[rand() % arg_list.size()]);
}
for (j = 0; j < arg_list.size(); j++) {
element = arg_list[j];
type = meta->ImportType(*element);
uint32_t index;
std::vector<ILElement *>::const_iterator it = std::find(param_list.begin(), param_list.end(), element);
if (it != param_list.end()) {
index = static_cast<uint32_t>(it - param_list.begin());
command->AddVMCommand(ctx, icDup, 0);
command->AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(index));
command->AddVMCommand(ctx, icLdelema, 0);
command->AddVMCommand(ctx, element->is_ref() ? icDup : icLdc_i4, 0);
command->AddVMCommand(ctx, icLdc_i4, type->id(), 0, type->reference_list()->Add(0));
command->AddVMCommand(ctx, icInitarg, 0);
param_map[index] = static_cast<uint32_t>(j);
}
else {
index = static_cast<uint32_t>(locals->IndexOf(element));
command->AddVMCommand(ctx, icLdnull, 0);
command->AddVMCommand(ctx, icLdc_i4, element->is_ref() ? 1 : 0);
command->AddVMCommand(ctx, icLdc_i4, type->id(), 0, type->reference_list()->Add(0));
command->AddVMCommand(ctx, icInitarg, 0);
variable_map[index] = static_cast<uint32_t>(j);
}
}
if ((ctx.options.flags & cpMemoryProtection) && tag() != ftLoader) {
temp_variable_index = arg_list.size();
type = meta->ImportType(ELEMENT_TYPE_I4);
command->AddVMCommand(ctx, icLdnull, 0);
command->AddVMCommand(ctx, icDup, 0);
command->AddVMCommand(ctx, icLdc_i4, type->id(), 0, type->reference_list()->Add(0));
command->AddVMCommand(ctx, icInitarg, 0);
}
command->AddVMCommand(ctx, icPop, 0);
}
continue;
}
else {
std::map<uint32_t, uint32_t>::const_iterator it;
switch (command->type()) {
case icRet:
if (type_ret) {
command->set_operand_value(0, type_ret->id());
command->set_token_reference(type_ret->reference_list()->Add(0));
}
command->set_param(static_cast<uint32_t>(temp_variable_index));
break;
case icLdloc:
case icLdloc_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdloca:
case icLdloca_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icStloc:
case icStloc_s:
it = variable_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdloc_0:
it = variable_map.find(0);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdloc_1:
it = variable_map.find(1);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdloc_2:
it = variable_map.find(2);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdloc_3:
it = variable_map.find(3);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icStloc_0:
it = variable_map.find(0);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icStloc_1:
it = variable_map.find(1);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icStloc_2:
it = variable_map.find(2);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icStloc_3:
it = variable_map.find(3);
if (it == variable_map.end())
throw std::runtime_error("Invalid variable index");
command->set_param(it->second);
break;
case icLdarg:
case icLdarg_s:
it = param_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icLdarga:
case icLdarga_s:
it = param_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icStarg:
case icStarg_s:
it = param_map.find(static_cast<uint16_t>(command->operand_value()));
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icLdarg_0:
it = param_map.find(0);
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icLdarg_1:
it = param_map.find(1);
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icLdarg_2:
it = param_map.find(2);
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
case icLdarg_3:
it = param_map.find(3);
if (it == param_map.end())
throw std::runtime_error("Invalid param index");
command->set_param(it->second);
break;
}
}
std::set<uint64_t>::const_iterator it = try_list.find(command->address());
if (it != try_list.end()) {
command->AddVMCommand(ctx, icLdc_i4, static_cast<uint32_t>(command->address() - image_base));
command->AddVMCommand(ctx, icEntertry, 0);
}
command->CompileToVM(ctx);
if (command->section_options() & rtCloseSection)
cur_block = NULL;
}
}
bool ILFunction::Compile(const CompileContext &ctx)
{
if (compilation_type() != ctMutation) {
for (size_t i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (!info->source() || info->entry()->block())
continue;
ILSignature *signature = reinterpret_cast<NETRuntimeFunction *>(info->source())->method()->signature();
bool is_generic = (signature->ret()->type() == ELEMENT_TYPE_VAR || signature->ret()->type() == ELEMENT_TYPE_MVAR);
for (size_t j = 0; j < signature->count(); j++) {
ILElement *element = signature->item(j);
if (element->type() == ELEMENT_TYPE_VAR || element->type() == ELEMENT_TYPE_MVAR) {
is_generic = true;
break;
}
}
if (is_generic) {
ctx.file->Notify(mtError, this, "Method with generic parameters can't be virtualized");
return false;
}
}
}
switch (compilation_type()) {
case ctMutation:
Mutate(ctx);
CompileToNative(ctx);
break;
case ctVirtualization:
CompileToVM(ctx);
break;
case ctUltra:
Mutate(ctx);
CompileToVM(ctx);
break;
default:
return false;
}
return BaseFunction::Compile(ctx);
}
void ILFunction::AfterCompile(const CompileContext &ctx)
{
for (size_t i = 0; i < count(); i++) {
ILCommand *command = item(i);
if (!command->block())
continue;
TokenReference *reference = command->token_reference();
if (reference) {
if (command->options() & roClearOriginalCode)
reference->set_address(0);
}
}
BaseFunction::AfterCompile(ctx);
}
ILCommand *ILFunction::ParseCommand(IArchitecture &file, uint64_t address, bool dump_mode /*= false*/)
{
NETRuntimeFunction *runtime_func = reinterpret_cast<NETRuntimeFunction *>(file.runtime_function_list()->GetFunctionByAddress(address));
ILCommand *command;
if (dump_mode) {
command = Add(address);
if (!file.AddressSeek(address))
command->InitUnknown();
else if (!runtime_func || address < runtime_func->start() || (file.selected_segment()->memory_type() & mtExecutable) == 0)
command->ReadValueFromFile(file, osByte);
else {
command->ReadFromFile(file);
/*switch (command->type()) {
case cmJmp:
case cmCall:
case cmJmpWithFlag: case cmJCXZ: case cmLoop: case cmLoope: case cmLoopne:
if ((command->options() & roFar) == 0 && command->operand(0).type == otValue)
command->AddLink(0, ltNone, command->operand(0).value);
break;
}
*/
}
return command;
} else {
if (!runtime_func)
return NULL;
if (address == runtime_func->begin() && file.AddressSeek(address)) {
size_t index = count();
command = Add(address);
command->set_alignment(sizeof(uint32_t));
uint8_t format = static_cast<uint8_t>(command->ReadValueFromFile(file, osByte));
FunctionInfo *info = function_info_list()->Add(runtime_func->begin(), runtime_func->end(), btValue, runtime_func->method()->fat_size(), 0, 0, runtime_func, command);
switch (format & CorILMethod_FormatMask) {
case CorILMethod_TinyFormat:
command->InitComment(string_format(".maxstack %d", 8));
break;
case CorILMethod_FatFormat:
{
ILStandAloneSig *locals = runtime_func->method()->locals();
command->ReadValueFromFile(file, osByte);
uint16_t max_stack = static_cast<uint16_t>(command->ReadValueFromFile(file, osWord));
command->ReadValueFromFile(file, osDWord);
if (!locals)
command->ReadValueFromFile(file, osDWord);
command->InitComment(string_format(".maxstack %d", max_stack));
address = command->next_address();
if (locals) {
command = Add(address);
command->ReadValueFromFile(file, osDWord, true);
command->InitComment(string_format(".locals init %s", locals ? locals->name(true).c_str() : "()"));
address = command->next_address();
}
if (format & CorILMethod_MoreSects) {
size_t old_count = count();
uint64_t start_address = address;
address = AlignValue(runtime_func->end(), sizeof(uint32_t));
file.AddressSeek(address);
uint64_t data_address = address;
command = Add(address);
command->set_alignment(sizeof(uint32_t));
uint32_t flags = static_cast<uint32_t>(command->ReadValueFromFile(file, osDWord));
bool fat = (flags & CorILMethod_Sect_FatFormat) != 0;
uint32_t data_size = fat ? (flags >> 8) : static_cast<uint8_t>(flags >> 8);
command->set_comment(CommentInfo(ttComment, string_format("Kind: %.2X; DataSize: %.8X", flags & CorILMethod_Sect_KindMask, data_size)));
address = command->next_address();
info->set_data_entry(command);
if (flags & CorILMethod_Sect_EHTable) {
CommandLink *link;
uint64_t value;
size_t clauses = (data_size - sizeof(uint32_t)) / (fat ? 24 : 12);
for (size_t i = 0; i < clauses; i++) {
command = Add(address);
flags = static_cast<uint32_t>(command->ReadValueFromFile(file, fat ? osDWord : osWord));
command->set_comment(CommentInfo(ttComment, "Flags"));
address = command->next_address();
ILCommand *try_entry = Add(address);
value = try_entry->ReadValueFromFile(file, fat ? osDWord : osWord);
try_entry->set_comment(CommentInfo(ttComment, "TryOffset"));
address = try_entry->next_address();
uint64_t try_begin = start_address + value;
ILCommand *try_size = Add(address);
value = try_size->ReadValueFromFile(file, fat ? osDWord : osByte);
try_size->set_comment(CommentInfo(ttComment, "TryLength"));
address = try_size->next_address();
uint64_t try_end = try_begin + value;
range_list()->Add(try_begin, try_end, try_entry, NULL, try_size);
ILCommand *handler_entry = Add(address);
value = handler_entry->ReadValueFromFile(file, fat ? osDWord : osWord);
handler_entry->set_comment(CommentInfo(ttComment, "HandlerOffset"));
address = handler_entry->next_address();
link = handler_entry->AddLink(0, (flags & COR_ILEXCEPTION_CLAUSE_FINALLY) ? ltFinallyBlock : ltSEHBlock, start_address + value);
link->set_base_function_info(info);
uint64_t handler_begin = start_address + value;
ILCommand *handler_size = Add(address);
value = handler_size->ReadValueFromFile(file, fat ? osDWord : osByte);
handler_size->set_comment(CommentInfo(ttComment, "HandlerLength"));
address = handler_size->next_address();
uint64_t handler_end = handler_begin + value;
range_list()->Add(handler_begin, handler_end, handler_entry, NULL, handler_size);
if (flags & COR_ILEXCEPTION_CLAUSE_FILTER) {
command = Add(address);
value = command->ReadValueFromFile(file, osDWord);
command->set_comment(CommentInfo(ttComment, "FilterOffset"));
address = command->next_address();
link = command->AddLink(0, ltSEHBlock, start_address + value);
link->set_base_function_info(info);
} else {
command = Add(address);
command->ReadValueFromFile(file, osDWord, true);
command->set_comment(CommentInfo(ttComment, "ClassToken"));
address = command->next_address();
}
}
}
function_info_list()->Add(data_address, data_address + data_size, btValue, 0, 0, 0, NULL, NULL);
for (size_t i = old_count; i < count(); i++) {
command = item(i);
//command->exclude_option(roNeedCompile);
command->exclude_option(roClearOriginalCode);
}
}
}
break;
}
return item(index);
}
if (address < runtime_func->start())
address = runtime_func->start();
if (!file.AddressSeek(address))
return NULL;
}
command = Add(address);
command->ReadFromFile(file);
switch (command->type()) {
case icBrfalse_s: case icBrtrue_s: case icBeq_s: case icBge_s: case icBgt_s:
case icBle_s: case icBlt_s: case icBne_un_s: case icBge_un_s: case icBgt_un_s: case icBle_un_s:
case icBlt_un_s: case icBrfalse: case icBrtrue: case icBeq: case icBge: case icBgt:
case icBle: case icBlt: case icBne_un: case icBge_un: case icBgt_un: case icBle_un: case icBlt_un:
command->AddLink(0, ltJmpWithFlag, command->operand_value());
break;
case icBr: case icBr_s: case icJmp: case icLeave: case icLeave_s:
command->AddLink(0, ltJmp, command->operand_value());
break;
case icSwitch:
{
address = command->next_address();
uint32_t case_count = static_cast<uint32_t>(command->operand_value());
uint64_t add_value = address + case_count * sizeof(uint32_t);
ILCommand *case_command = NULL;
for (size_t i = 0; i < case_count; i++) {
case_command = Add(address);
case_command->ReadCaseCommand(file);
case_command->set_comment(CommentInfo(ttComment, "Case"));
CommandLink *link = case_command->AddLink(0, ltCase, case_command->operand_value() + add_value);
link->set_parent_command(command);
address = case_command->next_address();
}
if (case_command)
command = case_command;
}
break;
case icCall: case icCallvirt: case icLdftn: case icLdvirtftn: case icNewobj:
{
ILToken *token = reinterpret_cast<NETArchitecture&>(file).command_list()->token(static_cast<uint32_t>(command->operand_value()));
uint64_t to_address = 0;
LinkType link_type = ltNone;
if (token && token->type() == ttMethodDef) {
ILMethodDef *method = reinterpret_cast<ILMethodDef*>(token);
if ((method->impl_flags() & miCodeTypeMask) == miIL && method->address()) {
to_address = method->address();
if (command->type() == icCall || (command->type() == icCallvirt && (method->flags() & mdVirtual) == 0))
link_type = ltCall;
}
}
if (to_address)
command->AddLink(-1, link_type, to_address);
}
break;
}
return command;
}
ILCommand *ILFunction::AddCommand(const std::string &value)
{
ILCommand *command = new ILCommand(this, cpu_address_size(), value);
AddObject(command);
return command;
}
ILCommand *ILFunction::AddCommand(const Data &value)
{
ILCommand *command = new ILCommand(this, cpu_address_size(), value);
AddObject(command);
return command;
}
ILCommand *ILFunction::AddCommand(OperandSize value_size, uint64_t value)
{
ILCommandType command_type;
switch (value_size) {
case osWord:
command_type = icWord;
break;
case osDWord:
command_type = icDword;
break;
default:
return NULL;
}
return AddCommand(command_type, value);
}
ILCommand *ILFunction::CreateCommand()
{
return new ILCommand(this, cpu_address_size());
}
ILCommand *ILFunction::AddCommand(ILCommandType type, uint64_t operand_value, IFixup *fixup)
{
ILCommand *command = new ILCommand(this, cpu_address_size(), type, operand_value, fixup);
AddObject(command);
return command;
}
ILFunction *ILFunction::Clone(IFunctionList *owner) const
{
ILFunction *func = new ILFunction(owner, *this);
return func;
}
ILCommand *ILFunction::GetCommandByAddress(uint64_t address) const
{
return reinterpret_cast<ILCommand *>(BaseFunction::GetCommandByAddress(address));
}
ILCommand *ILFunction::GetCommandByNearAddress(uint64_t address) const
{
return reinterpret_cast<ILCommand *>(BaseFunction::GetCommandByNearAddress(address));
}
ILCommand *ILFunction::ParseString(IArchitecture &file, uint64_t address, size_t len)
{
if (!file.AddressSeek(address))
return NULL;
ILCommand *command = Add(address);
command->ReadString(file, len);
command->exclude_option(roNeedCompile);
command->exclude_option(roClearOriginalCode);
return command;
}
ILCommand *ILFunction::Add(uint64_t address)
{
ILCommand *command = new ILCommand(this, cpu_address_size(), address);
AddObject(command);
return command;
}
/**
* ILFunctionList
*/
ILFunctionList::ILFunctionList(IArchitecture *owner)
: BaseFunctionList(owner), crc_table_(NULL), runtime_crc_table_(NULL), import_(NULL)
{
crc_cryptor_ = new ValueCryptor();
}
ILFunctionList::ILFunctionList(IArchitecture *owner, const ILFunctionList &src)
: BaseFunctionList(owner, src), crc_table_(NULL), runtime_crc_table_(NULL), import_(NULL)
{
crc_cryptor_ = new ValueCryptor();
}
ILFunctionList::~ILFunctionList()
{
delete crc_cryptor_;
}
ILFunction *ILFunctionList::item(size_t index) const
{
return reinterpret_cast<ILFunction *>(BaseFunctionList::item(index));
}
ILFunction *ILFunctionList::GetFunctionByAddress(uint64_t address) const
{
return reinterpret_cast<ILFunction *>(BaseFunctionList::GetFunctionByAddress(address));
}
bool ILFunctionList::Prepare(const CompileContext &ctx)
{
OperandSize cpu_address_size = ctx.file->cpu_address_size();
crc_cryptor_->clear();
crc_cryptor_->set_size(osDWord);
crc_cryptor_->Add(ccXor, rand32());
if ((ctx.options.flags | ctx.options.sdk_flags) & cpMemoryProtection) {
crc_table_ = AddCRCTable(cpu_address_size);
} else {
crc_table_ = NULL;
}
if (ctx.runtime) {
ILFunctionList *function_list = reinterpret_cast<ILFunctionList *>(ctx.runtime->function_list());
ILFunction *func;
size_t i, j;
// remove DecryptString template
uint64_t address = ctx.runtime->export_list()->GetAddressByType(atDecryptStringA);
if (address) {
func = function_list->GetFunctionByAddress(address);
if (func)
func->set_need_compile(false);
}
NETArchitecture *runtime = reinterpret_cast<NETArchitecture *>(ctx.runtime);
NETRuntimeFunctionList *runtime_function_list = runtime->runtime_function_list();
for (i = 0; i < function_list->count(); i++) {
func = function_list->item(i);
if (func->compilation_type() != ctMutation && func->entry_type() == etNone) {
NETRuntimeFunction *runtime_func = runtime_function_list->GetFunctionByAddress(func->address());
if (runtime_func) {
std::vector<ILMethodDef *> method_list = runtime_func->method_list();
for (j = 0; j < method_list.size(); j++) {
ILMethodDef *method = method_list[j];
if (method->flags() & mdSpecialName)
return false;
method->set_deleted(true);
}
}
}
}
APIType crc_helpers[] = { atRandom, atCalcCRC };
if (ctx.runtime->segment_list()->count() > 0) {
// add runtime functions
for (i = 0; i < function_list->count(); i++) {
func = function_list->item(i);
if (func->tag() == ftProcessor)
continue;
if (func->need_compile()) {
func = func->Clone(this);
AddObject(func);
if (func->compilation_type() != ctMutation && func->entry_type() == etNone)
func->entry()->include_section_option(rtLinkedToInt);
for (j = 0; j < func->count(); j++) {
ILCommand *command = func->item(j);
if (command->type() == icLdstr) {
ILData dump = runtime->command_list()->GetUserData(TOKEN_VALUE(command->operand_value()), &address);
for (size_t k = 0; k < MESSAGE_COUNT; k++) {
os::unicode_string unicode_message =
#ifdef VMP_GNU
os::FromUTF8(default_message[k]);
#else
default_message[k];
#endif
if (dump.size() == unicode_message.size() * sizeof(os::unicode_char) && memcmp(dump.data(), unicode_message.c_str(), dump.size()) == 0) {
MapFunction *map_function = runtime->map_function_list()->Add(address, address + dump.size(), otString, FunctionName());
map_function->reference_list()->Add(command->address(), address);
ILFunction *str_function = reinterpret_cast<ILFunction *>(function_list->AddByAddress(address, ctVirtualization, 0, true, NULL));
if (str_function) {
str_function->set_from_runtime(true);
os::unicode_string str = os::FromUTF8(ctx.options.messages[k]);
if (str.empty())
str.push_back(0);
str_function->item(0)->set_dump(reinterpret_cast<const uint8_t*>(str.c_str()), str.size() * sizeof(os::unicode_char));
}
break;
}
}
}
}
for (j = 0; j < func->count(); j++) {
func->item(j)->CompileToNative();
}
} else {
if (func->tag() != ftLoader) {
NETRuntimeFunction *runtime_func = runtime_function_list->GetFunctionByAddress(func->address());
if (runtime_func) {
std::vector<ILMethodDef *> method_list = runtime_func->method_list();
for (j = 0; j < method_list.size(); j++) {
method_list[j]->set_deleted(true);
}
}
}
if (!func->FreeByManager(ctx))
return false;
}
}
for (i = 0; i < _countof(crc_helpers); i++) {
address = ctx.runtime->export_list()->GetAddressByType(crc_helpers[i]);
func = reinterpret_cast<ILFunction *>(ctx.file->function_list()->GetFunctionByAddress(address));
if (!func)
return false;
// exclude from memory protection
func->entry()->include_section_option(rtLinkedToInt);
func->set_tag(ftLoader);
}
AddRuntimeData(cpu_address_size);
}
else {
// remove references to VMProtect.Core
address = ctx.runtime->export_list()->GetAddressByType(atSetupImage);
func = function_list->GetFunctionByAddress(address);
if (!func)
return false;
for (i = 0; i < func->count(); i++) {
ILCommand *command = func->item(i);
if (command->token_reference()) {
ILToken *token = command->token_reference()->owner()->owner();
switch (token->type()) {
case ttField:
if (reinterpret_cast<ILField *>(token)->declaring_type()->full_name() == "VMProtect.Core")
command->Init(icNop);
break;
case ttMethodDef:
if (reinterpret_cast<ILMethodDef *>(token)->declaring_type()->full_name() == "VMProtect.Core")
command->Init(icNop);
break;
}
}
}
if (ctx.options.flags & cpMemoryProtection) {
// add CRC helpers
for (i = 0; i < _countof(crc_helpers); i++) {
address = ctx.runtime->export_list()->GetAddressByType(crc_helpers[i]);
func = function_list->GetFunctionByAddress(address);
if (!func)
return false;
func = func->Clone(this);
AddObject(func);
for (j = 0; j < func->count(); j++) {
ILCommand *command = func->item(j);
command->exclude_option(roClearOriginalCode);
command->CompileToNative();
}
// exclude from memory protection
func->entry()->include_section_option(rtLinkedToInt);
func->set_tag(ftLoader);
}
}
}
}
if (ctx.options.flags & cpImportProtection) {
import_ = AddImport(cpu_address_size);
} else {
import_ = NULL;
}
AddSDK(cpu_address_size);
AddWatermark(cpu_address_size, ctx.options.watermark, ctx.runtime ? 8 : 10);
return BaseFunctionList::Prepare(ctx);
}
void ILFunctionList::CompileLinks(const CompileContext &ctx)
{
if (ctx.options.flags & cpMemoryProtection) {
runtime_crc_table_ = AddRuntimeCRCTable(ctx.file->cpu_address_size());
runtime_crc_table_->Compile(ctx);
}
else {
runtime_crc_table_ = NULL;
}
return BaseFunctionList::CompileLinks(ctx);
}
void ILFunctionList::CompileInfo(const CompileContext &ctx)
{
return BaseFunctionList::CompileInfo(ctx);
}
void ILFunctionList::ReadFromBuffer(Buffer &buffer, IArchitecture &file)
{
BaseFunctionList::ReadFromBuffer(buffer, file);
// add loader stubs
size_t i, j;
std::set<ILTypeDef*> class_list;
for (i = 0; i < count(); i++) {
ILFunction *func = item(i);
if (func->tag() != ftLoader)
continue;
for (j = 0; j < func->count(); j++) {
ILCommand *command = func->item(j);
if (command->token_reference()) {
ILToken *token = command->token_reference()->owner()->owner();
if (token->type() == ttMethodDef)
class_list.insert(reinterpret_cast<ILMethodDef*>(token)->declaring_type());
}
}
}
for (std::set<ILTypeDef*>::const_iterator it = class_list.begin(); it != class_list.end(); ) {
ILTypeDef *type_def = *it;
if (type_def->full_name() == "VMProtect.Core") {
class_list.erase(it++);
continue;
}
it++;
}
ILTable *method_table = reinterpret_cast<NETArchitecture&>(file).command_list()->table(ttMethodDef);
for (i = 0; i < method_table->count(); i++) {
ILMethodDef *method = reinterpret_cast<ILMethodDef*>(method_table->item(i));
uint64_t address = method->address();
if (!address || class_list.find(method->declaring_type()) == class_list.end() || GetFunctionByAddress(address))
continue;
ILFunction *new_func = reinterpret_cast<ILFunction *>(AddByAddress(address, ctMutation, 0, false, NULL));
if (new_func) {
new_func->set_tag(ftLoader);
for (j = 0; j < new_func->count(); j++) {
new_func->item(j)->exclude_option(roClearOriginalCode);
}
}
}
}
ILFunction *ILFunctionList::Add(const std::string &name, CompilationType compilation_type, uint32_t compilation_options, bool need_compile, Folder *folder)
{
ILFunction *func = new ILFunction(this, name, compilation_type, compilation_options, need_compile, folder);
AddObject(func);
return func;
}
ILFunctionList * ILFunctionList::Clone(IArchitecture *owner) const
{
ILFunctionList *list = new ILFunctionList(owner, *this);
return list;
}
IFunction * ILFunctionList::CreateFunction(OperandSize cpu_address_size)
{
return new ILFunction(this, cpu_address_size);
}
bool ILFunctionList::Compile(const CompileContext &ctx)
{
return BaseFunctionList::Compile(ctx);
}
ILSDK *ILFunctionList::AddSDK(OperandSize cpu_address_size)
{
ILSDK *func = new ILSDK(this, cpu_address_size);
AddObject(func);
return func;
}
ILRuntimeData *ILFunctionList::AddRuntimeData(OperandSize cpu_address_size)
{
ILRuntimeData *func = new ILRuntimeData(this, cpu_address_size);
AddObject(func);
return func;
}
ILCRCTable *ILFunctionList::AddCRCTable(OperandSize cpu_address_size)
{
ILCRCTable *func = new ILCRCTable(this, cpu_address_size);
AddObject(func);
return func;
}
ILImport *ILFunctionList::AddImport(OperandSize cpu_address_size)
{
ILImport *func = new ILImport(this, cpu_address_size);
AddObject(func);
return func;
}
ILFunction *ILFunctionList::AddWatermark(OperandSize cpu_address_size, Watermark *watermark, int copy_count)
{
ILFunction *func = new ILFunction(this, cpu_address_size);
func->set_compilation_type(ctMutation);
func->set_memory_type(mtNone);
func->AddWatermark(watermark, copy_count);
AddObject(func);
return func;
}
ILRuntimeCRCTable *ILFunctionList::AddRuntimeCRCTable(OperandSize cpu_address_size)
{
ILRuntimeCRCTable *func = new ILRuntimeCRCTable(this, cpu_address_size);
AddObject(func);
return func;
}
ILVirtualMachineProcessor *ILFunctionList::AddProcessor(OperandSize cpu_address_size)
{
ILVirtualMachineProcessor *func = new ILVirtualMachineProcessor(this, cpu_address_size);
AddObject(func);
return func;
}
/**
* ILSDK
*/
ILSDK::ILSDK(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size)
{
set_compilation_type(ctMutation);
}
bool ILSDK::Init(const CompileContext &ctx)
{
MapFunctionList *map_function_list;
MapFunction *map_function;
IFunctionList *function_list;
size_t i, c, j, n, f;
uint64_t address;
IArchitecture *file;
IImportList *import_list;
IImport *import;
IImportFunction *import_function;
ILCommand *command;
CommandBlock *block;
uint64_t api_address;
f = (ctx.runtime && ctx.runtime->segment_list()->count() > 0) ? 2 : 1;
for (n = 0; n < f; n++) {
file = (n == 0) ? ctx.file : ctx.runtime;
map_function_list = file->map_function_list();
function_list = file->function_list();
import_list = file->import_list();
for (i = 0; i < import_list->count(); i++) {
import = import_list->item(i);
if (!import->is_sdk())
continue;
for (j = 0; j < import->count(); j++) {
import_function = import->item(j);
if (import_function->type() == atNone)
continue;
map_function = import_function->map_function();
for (c = 0; c < map_function->reference_list()->count(); c++) {
address = map_function->reference_list()->item(c)->address();
command = reinterpret_cast<ILCommand *>(ctx.file->function_list()->GetCommandByNearAddress(address, true));
if (command) {
delete command->link();
} else {
if (!file->AddressSeek(address))
return false;
block = AddBlock(count(), true);
block->set_address(address);
command = Add(address);
command->ReadFromFile(*file);
command->set_block(block);
command->include_option(roFillNop);
command->exclude_option(roClearOriginalCode);
}
if (command->token_reference()) {
command->token_reference()->set_deleted(true);
command->set_token_reference(NULL);
}
switch (import_function->type()) {
case atBegin:
command->Init(icPop);
break;
case atEnd:
command->Init(icNop);
break;
case atDecryptStringA:
case atDecryptStringW:
command->Init(icNop);
break;
case atFreeString:
Notify(mtWarning, command, "VMProtect.SDK.FreeString is deprecated");
// fall through
case atIsDebuggerPresent:
case atIsVirtualMachinePresent:
case atIsValidImageCRC:
case atActivateLicense:
case atDeactivateLicense:
case atGetOfflineActivationString:
case atGetOfflineDeactivationString:
case atSetSerialNumber:
case atGetSerialNumberState:
case atGetSerialNumberData:
case atGetCurrentHWID:
case atIsProtected:
if (!ctx.runtime || ctx.runtime->segment_list()->count() == 0) {
switch (import_function->type()) {
case atFreeString:
command->Init(icNop);
break;
case atIsProtected:
command->Init(icLdc_i4_1);
break;
default:
// other APIs can not work without runtime
return false;
}
} else {
api_address = ctx.runtime->export_list()->GetAddressByType(import_function->type());
if (!api_address)
return false;
ILMethodDef *method = reinterpret_cast<ILMetaData *>(ctx.runtime->command_list())->GetMethod(api_address);
if (!method)
return false;
command->set_operand_value(0, method->id());
command->set_token_reference(method->reference_list()->Add(command->address() + 1));
}
break;
default:
throw std::runtime_error("Unknown API from SDK: " + import_function->name());
}
command->CompileToNative();
}
}
}
}
for (i = 0; i < count(); i++) {
item(i)->CompileToNative();
}
return true;
}
/**
* ILCRCTable
*/
ILCRCTable::ILCRCTable(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size)
{
set_compilation_type(ctMutation);
}
bool ILCRCTable::Init(const CompileContext &ctx)
{
size_t i, c, n, f;
c = 10;
f = (ctx.runtime && ctx.runtime->segment_list()->count() > 0) ? 2 : 1;
for (n = 0; n < f; n++) {
IArchitecture *file = (n == 0) ? ctx.file : ctx.runtime;
c += ctx.file->segment_list()->count();
}
for (i = 0; i < c; i++) {
AddCommand(icDword, 0);
AddCommand(icDword, 0);
AddCommand(icDword, 0);
}
size_entry_ = AddCommand(icDword, 0);
size_entry_->include_option(roCreateNewBlock);
hash_entry_ = AddCommand(icDword, 0);
hash_entry_->include_option(roCreateNewBlock);
for (i = 0; i < count(); i++) {
ILCommand *command = item(i);
command->CompileToNative();
command->include_option(roWritable);
}
return true;
}
/**
* ILRuntimeData
*/
ILRuntimeData::ILRuntimeData(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size), strings_entry_(NULL), strings_size_(0), trial_hwid_entry_(NULL), trial_hwid_size_(0),
resources_entry_(NULL), resources_size_(0)
#ifdef ULTIMATE
, license_data_entry_(NULL), license_data_size_(0)
#endif
{
set_compilation_type(ctMutation);
rc5_key_.Create();
}
bool ILRuntimeData::CommandCompareHelper::operator()(const ILCommand *left, ILCommand *right) const
{
return (left->address() < right->address());
}
bool ILRuntimeData::Init(const CompileContext &ctx)
{
ILFunctionList *function_list;
size_t i, j, index;
std::vector<ILCommand *> string_command_list;
ILCommand *command, *string_command;
CommandLink *link;
ILCommand *key_entry;
Data key;
uint32_t data_key;
uint64_t image_base = ctx.file->image_base();
key.PushBuff(rc5_key_.Value, sizeof(rc5_key_.Value));
data_key = key.ReadDWord(0);
resources_entry_ = NULL;
resources_size_ = 0;
#ifdef ULTIMATE
if (ctx.options.file_manager) {
FileManager *file_manager = ctx.options.file_manager;
if (!file_manager->OpenFiles())
return false;
InternalFile *internal_file;
index = count();
// write entry
for (i = 0; i < file_manager->count(); i++) {
internal_file = file_manager->item(i);
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, internal_file->stream()->Size());
AddCommand(osDWord, 0);
}
// end entry
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
resources_entry_ = item(index);
resources_entry_->include_option(roCreateNewBlock);
resources_size_ = static_cast<uint32_t>((count() - index) * OperandSizeToValue(osDWord));
PEFile pe_file;
for (i = 0; i < file_manager->count(); i++) {
internal_file = file_manager->item(i);
std::string file_name = internal_file->absolute_file_name();
std::string assembly_name;
try
{
if (pe_file.Open(file_name.c_str(), foRead | foHeaderOnly) == osSuccess && pe_file.count() > 1)
assembly_name = reinterpret_cast<NETArchitecture *>(pe_file.item(1))->full_name();
}
catch (...)
{
}
pe_file.Close();
if (assembly_name.empty()) {
Notify(mtError, internal_file, string_format(language[lsFileHasIncorrectFormat].c_str(), file_name.c_str(), ".NET"));
file_manager->CloseFiles();
return false;
}
// write name
Data data;
os::unicode_string name = os::FromUTF8(assembly_name.c_str());
const os::unicode_char *p = name.c_str();
for (j = 0; j < name.size() + 1; j++) {
data.PushWord(static_cast<uint16_t>(p[j] ^ (_rotl32(data_key, static_cast<int>(j)) + j)));
}
command = AddCommand(data);
command->include_option(roCreateNewBlock);
link = item(index + i * 4)->AddLink(0, ltOffset, command);
link->set_sub_value(image_base);
// write data
Notify(mtInformation, NULL, string_format("%s %s", language[lsLoading].c_str(), os::ExtractFileName(file_name.c_str()).c_str()));
data.resize(static_cast<size_t>(internal_file->stream()->Size()));
internal_file->stream()->Read(&data[0], data.size());
for (j = 0; j < data.size(); j++) {
data[j] = (data[j] ^ static_cast<uint8_t>(_rotl32(data_key, static_cast<int>(j)) + j));
}
command = AddCommand(data);
command->include_option(roCreateNewBlock);
link = item(index + i * 4 + 1)->AddLink(0, ltOffset, command);
link->set_sub_value(image_base);
}
file_manager->CloseFiles();
}
#endif
if ((ctx.options.flags & cpResourceProtection) && ctx.file->resource_list()->count()) {
std::string assembly_name;
NETArchitecture *file = reinterpret_cast<NETArchitecture *>(ctx.file);
while (true) {
assembly_name = string_format("%.8X", rand32());
if (!file->command_list()->GetAssemblyRef(assembly_name))
break;
}
Data data;
if (file->WriteResources(assembly_name, data)) {
ILAssemblyRef *assembly_ref = file->command_list()->GetAssemblyRef(assembly_name);
if (!assembly_ref)
return false;
if (resources_entry_) {
index = IndexOf(resources_entry_) + (resources_size_ - 4 * sizeof(uint32_t)) / sizeof(uint32_t);
// insert entry
InsertObject(index + 0, new ILCommand(this, cpu_address_size(), icDword, 0));
InsertObject(index + 1, new ILCommand(this, cpu_address_size(), icDword, 0));
InsertObject(index + 2, new ILCommand(this, cpu_address_size(), icDword, data.size()));
InsertObject(index + 3, new ILCommand(this, cpu_address_size(), icDword, 0));
resources_size_ += 4 * sizeof(uint32_t);
}
else {
// write entry
index = count();
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, data.size());
AddCommand(osDWord, 0); // action
// end entry
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
AddCommand(osDWord, 0);
resources_entry_ = item(index);
resources_entry_->include_option(roCreateNewBlock);
resources_size_ = static_cast<uint32_t>((count() - index) * OperandSizeToValue(osDWord));
}
// write name
Data str;
os::unicode_string name = os::FromUTF8(assembly_ref->full_name().c_str());
const os::unicode_char *p = name.c_str();
for (i = 0; i < name.size() + 1; i++) {
str.PushWord(static_cast<uint16_t>(p[i] ^ (_rotl32(data_key, static_cast<int>(i)) + i)));
}
command = AddCommand(str);
command->include_option(roCreateNewBlock);
link = item(index)->AddLink(0, ltOffset, command);
link->set_sub_value(image_base);
// write data
for (i = 0; i < data.size(); i++) {
data[i] = data[i] ^ static_cast<uint8_t>(_rotl32(data_key, static_cast<int>(i)) + i);
}
command = AddCommand(data);
command->include_option(roCreateNewBlock);
link = item(index + 1)->AddLink(0, ltOffset, command);
link->set_sub_value(image_base);
}
}
function_list = reinterpret_cast<ILFunctionList *>(ctx.file->function_list());
for (i = 0; i < function_list->count(); i++) {
ILFunction *func = function_list->item(i);
if (func->need_compile() && func->type() == otString) {
for (j = 0; j < func->count(); j++) {
string_command_list.push_back(func->item(j));
}
}
}
key_entry = AddCommand(key);
key_entry->include_option(roCreateNewBlock);
strings_entry_ = NULL;
strings_size_ = 0;
if (string_command_list.size()) {
std::sort(string_command_list.begin(), string_command_list.end(), CommandCompareHelper());
index = count();
// create directory
AddCommand(osDWord, string_command_list.size());
AddCommand(osDWord, 0);
for (i = 0; i < string_command_list.size(); i++) {
string_command = string_command_list[i];
// create string entry
AddCommand(osDWord, string_command->address() - ctx.file->image_base());
command = AddCommand(osDWord, 0);
link = command->AddLink(0, ltOffset);
link->set_sub_value(ctx.file->image_base());
AddCommand(osDWord, string_command->dump_size());
AddCommand(osDWord, 0);
}
strings_entry_ = item(index);
strings_entry_->include_option(roCreateNewBlock);
strings_size_ = static_cast<uint32_t>((count() - index) * OperandSizeToValue(osDWord));
i = AlignValue(strings_size_, 8);
if (i > strings_size_) {
Data tmp;
tmp.resize(i - strings_size_, 0);
AddCommand(tmp);
strings_size_ = (uint32_t)i;
}
// create string values
Data data;
for (i = 0; i < string_command_list.size(); i++) {
string_command = string_command_list[i];
data.clear();
for (j = 0; j < string_command->dump_size(); j++) {
data.PushByte(string_command->dump(j) ^ static_cast<uint8_t>(_rotl32(data_key, static_cast<int>(j)) + j));
}
command = AddCommand(data);
command->include_option(roCreateNewBlock);
item(index + 2 + i * 4 + 1)->link()->set_to_command(command);
}
}
#ifdef ULTIMATE
license_data_entry_ = NULL;
license_data_size_ = 0;
if (ctx.options.licensing_manager) {
Data license_data;
if (ctx.options.licensing_manager->GetLicenseData(license_data)) {
license_data_entry_ = AddCommand(license_data);
license_data_entry_->include_option(roCreateNewBlock);
license_data_size_ = static_cast<uint32_t>(license_data.size());
}
}
#endif
trial_hwid_entry_ = NULL;
trial_hwid_size_ = 0;
#ifdef DEMO
if (true)
#else
if (ctx.options.flags & cpUnregisteredVersion)
#endif
{
size_t size = VMProtectGetCurrentHWID(NULL, 0);
std::vector<char> hwid;
hwid.resize(size);
VMProtectGetCurrentHWID(hwid.data(), (int)hwid.size());
std::vector<uint8_t> binary;
binary.resize(size);
Base64Decode(hwid.data(), hwid.size(), binary.data(), size);
Data data;
data.PushBuff(binary.data(), binary.size());
data.resize(64);
trial_hwid_size_ = static_cast<uint32_t>(std::min(size, data.size()));
trial_hwid_entry_ = AddCommand(data);
trial_hwid_entry_->include_option(roCreateNewBlock);
}
#ifdef ULTIMATE
else if (!ctx.options.hwid.empty()) {
std::string hwid = ctx.options.hwid;
size_t size = hwid.size();
std::vector<uint8_t> binary;
binary.resize(size);
Base64Decode(hwid.data(), hwid.size(), binary.data(), size);
if (size & 3) {
Notify(mtError, NULL, "Invalid HWID");
return false;
}
Data data;
data.PushBuff(binary.data(), binary.size());
data.resize(64);
trial_hwid_size_ = static_cast<uint32_t>(std::min(size, data.size()));
trial_hwid_entry_ = AddCommand(data);
trial_hwid_entry_->include_option(roCreateNewBlock);
}
#endif
for (i = 0; i < count(); i++) {
item(i)->CompileToNative();
}
// setup faces for common runtime functions
ILCRCTable *il_crc = reinterpret_cast<ILFunctionList *>(ctx.file->function_list())->crc_table();
for (i = 0; i < function_list->count(); i++) {
ILFunction *func = function_list->item(i);
if (!func->from_runtime())
continue;
for (j = 0; j < func->count(); j++) {
ILCommand *command = func->item(j);
if (command->type() != icLdc_i4)
continue;
uint32_t value = static_cast<uint32_t>(command->operand_value());
if ((value & 0xFFFF0000) == 0xFACE0000) {
switch (value) {
case FACE_STRING_DECRYPT_KEY:
command->set_operand_value(0, data_key);
command->CompileToNative();
break;
case FACE_RC5_P:
command->set_operand_value(0, rc5_key_.P);
command->CompileToNative();
break;
case FACE_RC5_Q:
command->set_operand_value(0, rc5_key_.Q);
command->CompileToNative();
break;
case FACE_KEY_INFO:
if (key_entry) {
link = command->AddLink(0, ltOffset, key_entry);
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_STRING_INFO:
if (strings_entry_) {
link = command->AddLink(0, ltOffset, strings_entry_);
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_RESOURCE_INFO:
if (resources_entry_) {
link = command->AddLink(0, ltOffset, resources_entry_);
link->set_sub_value(image_base);
}
else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
#ifdef ULTIMATE
case FACE_LICENSE_INFO:
if (license_data_entry_) {
link = command->AddLink(0, ltOffset, license_data_entry_);
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_LICENSE_INFO_SIZE:
command->set_operand_value(0, license_data_size_);
command->CompileToNative();
break;
#else
case FACE_LICENSE_INFO:
case FACE_LICENSE_INFO_SIZE:
command->set_operand_value(0, 0);
command->CompileToNative();
break;
#endif
case FACE_TRIAL_HWID:
if (trial_hwid_entry_) {
link = command->AddLink(0, ltOffset, trial_hwid_entry_);
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_TRIAL_HWID_SIZE:
command->set_operand_value(0, trial_hwid_size_);
command->CompileToNative();
break;
case FACE_CRC_TABLE_ENTRY:
if (il_crc) {
link = command->AddLink(0, ltOffset, il_crc->table_entry());
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_CRC_TABLE_SIZE:
if (il_crc) {
link = command->AddLink(0, ltOffset, il_crc->size_entry());
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_CRC_TABLE_HASH:
if (il_crc) {
link = command->AddLink(0, ltOffset, il_crc->hash_entry());
link->set_sub_value(image_base);
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_CORE_OPTIONS:
{
uint32_t options = 0;
if (ctx.options.flags & cpInternalMemoryProtection)
options |= CORE_OPTION_MEMORY_PROTECTION;
if (ctx.options.flags & cpCheckDebugger)
options |= CORE_OPTION_CHECK_DEBUGGER;
command->set_operand_value(0, options);
}
command->CompileToNative();
break;
}
}
}
}
return true;
}
size_t ILRuntimeData::WriteToFile(IArchitecture &file)
{
size_t res = ILFunction::WriteToFile(file);
CipherRC5 cipher(rc5_key_);
for (size_t i = 0; i < 3; i++) {
ILCommand *command;
size_t size;
switch (i) {
case 0:
command = strings_entry_;
size = strings_size_;
break;
case 1:
command = trial_hwid_entry_;
size = trial_hwid_entry_ ? trial_hwid_entry_->dump_size() : 0;
break;
case 2:
command = resources_entry_;
size = resources_size_;
break;
#ifdef ULTIMATE
/*
case 2:
command = license_data_entry_;
size = license_data_size_;
break;
*/
#endif
default:
command = NULL;
size = 0;
}
if (size) {
std::vector<uint8_t> buff;
buff.resize(size);
file.AddressSeek(command->address());
uint64_t pos = file.Tell();
file.Read(buff.data(), buff.size());
cipher.Encrypt(buff.data(), buff.size());
file.Seek(pos);
file.Write(buff.data(), buff.size());
}
}
return res;
}
/**
* ILRuntimeCRCTable
*/
ILRuntimeCRCTable::ILRuntimeCRCTable(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size), cryptor_(NULL)
{
set_compilation_type(ctMutation);
}
void ILRuntimeCRCTable::clear()
{
region_info_list_.clear();
ILFunction::clear();
}
bool ILRuntimeCRCTable::Compile(const CompileContext &ctx)
{
ILFunctionList *function_list = reinterpret_cast<ILFunctionList *>(ctx.file->function_list());
cryptor_ = function_list->crc_cryptor();
size_t block_size, i, j, k;
uint64_t block_address;
MemoryManager manager(ctx.file);
for (i = 0; i < function_list->count(); i++) {
ILFunction *func = function_list->item(i);
if (!func->need_compile())
continue;
for (j = 0; j < func->block_list()->count(); j++) {
CommandBlock *block = func->block_list()->item(j);
if (block->type() & mtExecutable) {
// native block
block_size = 0;
block_address = 0;
for (k = block->start_index(); k <= block->end_index(); k++) {
ILCommand *command = func->item(k);
if (command->options() & roWritable)
continue;
if (block_address && (block_address + block_size) != command->address()) {
if (block_size)
manager.Add(block_address, block_size, mtReadable);
block_address = 0;
block_size = 0;
}
if (!block_address)
block_address = command->address();
block_size += command->dump_size();
}
if (block_size)
manager.Add(block_address, block_size, mtReadable);
}
else {
// VM block
block_size = 0;
block_address = 0;
for (k = block->start_index(); k <= block->end_index(); k++) {
ILCommand *command = func->item(k);
if (block_address && (block_address + block_size) != command->vm_address()) {
if (block_size)
manager.Add(block_address, block_size, mtReadable);
block_address = 0;
block_size = 0;
}
if (!block_address)
block_address = command->vm_address();
block_size += command->vm_dump_size();
}
if (block_size)
manager.Add(block_address, block_size, mtReadable);
}
}
}
if (manager.count() == 0)
return true;
manager.Pack();
for (i = 0; i < manager.count(); i++) {
MemoryRegion *region = manager.item(i);
uint64_t block_address = region->address();
size_t region_size, block_size;
for (region_size = region->size(); region_size != 0; region_size -= block_size, block_address += block_size) {
block_size = 0x1000 - (rand() & 0xff);
if (block_size > region_size)
block_size = region_size;
region_info_list_.push_back(RegionInfo(block_address, static_cast<uint32_t>(block_size), false));
}
}
for (i = 0; i < region_info_list_.size(); i++) {
std::swap(region_info_list_[i], region_info_list_[rand() % region_info_list_.size()]);
}
size_t self_crc_offset = 0;
size_t self_crc_size = 0;
for (i = 0; i < region_info_list_.size(); i++) {
self_crc_size += sizeof(CRCInfo::POD);
if (self_crc_size > 0x1000 && (rand() & 1)) {
region_info_list_.insert(region_info_list_.begin() + i + 1, RegionInfo(self_crc_offset, (uint32_t)self_crc_size, true));
self_crc_offset += self_crc_size;
self_crc_size = 0;
}
}
if (self_crc_size)
region_info_list_.push_back(RegionInfo(self_crc_offset, (uint32_t)self_crc_size, true));
for (i = 0; i < region_info_list_.size(); i++) {
AddCommand(icDword, 0);
AddCommand(icDword, 0);
AddCommand(icDword, 0);
}
set_entry(item(0));
for (i = 0; i < count(); i++) {
item(i)->CompileToNative();
}
CreateBlocks();
for (i = 0; i < block_list()->count(); i++) {
block_list()->item(i)->Compile(*ctx.manager);
}
return true;
}
size_t ILRuntimeCRCTable::WriteToFile(IArchitecture &file)
{
size_t res = ILFunction::WriteToFile(file);
if (entry()) {
uint64_t address = entry()->address();
std::vector<CRCInfo> crc_info_list;
std::vector<uint8_t> dump;
for (size_t i = 0; i < region_info_list_.size(); i++) {
RegionInfo region_info = region_info_list_[i];
dump.resize(region_info.size);
if (region_info.is_self_crc) {
memcpy(&dump[0], reinterpret_cast<uint8_t *>(&crc_info_list[0]) + region_info.address, dump.size());
region_info.address += address;
}
else {
file.AddressSeek(region_info.address);
file.Read(&dump[0], dump.size());
}
CRCInfo crc_info(static_cast<uint32_t>(region_info.address - file.image_base()), dump);
if (cryptor_) {
crc_info.pod.address = static_cast<uint32_t>(cryptor_->Encrypt(crc_info.pod.address));
crc_info.pod.size = static_cast<uint32_t>(cryptor_->Encrypt(crc_info.pod.size));
}
crc_info.pod.hash = 0 - crc_info.pod.hash;
crc_info_list.push_back(crc_info);
}
file.AddressSeek(address);
file.Write(&crc_info_list[0], crc_info_list.size() * sizeof(CRCInfo::POD));
}
return res;
}
/**
* ILFileHelper
*/
ILFileHelper::ILFileHelper()
: IObject(), marker_index_(0), marker_name_list_(NULL)
{
}
ILFileHelper::~ILFileHelper()
{
}
void ILFileHelper::Parse(NETArchitecture &file)
{
size_t i, j;
ILToken *token;
ILFunction func(NULL, file.cpu_address_size());
std::set<uint64_t> address_list;
size_t c = file.map_function_list()->count();
file.StartProgress(string_format("%s %s...", language[lsLoading].c_str(), os::ExtractFileName(file.owner()->file_name().c_str()).c_str()), c);
for (i = 0; i < c; i++) {
MapFunction *map_function = file.map_function_list()->item(i);
file.StepProgress();
if (map_function->type() != otCode && map_function->type() != otMarker && map_function->type() != otExport)
continue;
if (address_list.find(map_function->address()) != address_list.end())
continue;
address_list.insert(map_function->address());
func.ReadFromFile(file, map_function->address());
for (j = 0; j < func.count(); j++) {
ILCommand *command = func.item(j);
uint32_t value = static_cast<uint32_t>(command->operand_value());
if (command->type() == icComment) {
if (command->comment().value.substr(0, 7) == ".locals") {
token = file.command_list()->token(value);
if (token)
token->reference_list()->Add(command->address());
}
} else if (command->type() == icDword) {
if (command->comment().value == "ClassToken") {
token = file.command_list()->token(value);
if (token)
token->reference_list()->Add(command->address());
}
} else switch (ILOpCodes[command->type()].operand_type) {
case InlineString:
token = file.command_list()->token(value);
if (!token)
token = file.command_list()->us_table()->Add(value);
command->set_token_reference(token->reference_list()->Add(command->address() + command->operand_pos()));
if (map_function->strings_protection())
AddString(file, static_cast<uint32_t>(command->operand_value()), command->address());
break;
case InlineField:
case InlineMethod:
case InlineSig:
case InlineTok:
case InlineType:
token = file.command_list()->token(value);
if (token) {
command->set_token_reference(token->reference_list()->Add(command->address() + command->operand_pos(), command->type()));
NETImportFunction *import_function = file.import_list()->GetFunctionByToken(value);
if (import_function) {
import_function->map_function()->reference_list()->Add(command->address(), 0);
if (ILOpCodes[command->type()].flow_type == Call) {
if (j > 0 && ILOpCodes[func.item(j - 1)->type()].opcode_type == Prefix)
import_function->include_option(ioHasCallPrefix);
}
if (import_function->owner()->is_sdk()) {
switch (import_function->type()) {
case atDecryptStringA:
case atDecryptStringW:
if (command->type() == icCall && j > 0) {
ILCommand *prev = func.item(j - 1);
if (prev->type() == icLdstr)
AddString(file, static_cast<uint32_t>(prev->operand_value()), prev->address());
}
break;
}
}
}
}
break;
}
}
ILCommandBlock block;
if (block.Parse(func)) {
for (j = 0; j < block.count(); j++) {
ILCommandNode *node = block.item(j);
ILCommand *command = node->command();
switch (command->type()) {
case icCall: case icNewobj:
{
std::string name = node->token_name();
if (name == "instance void System.Runtime.InteropServices.ComAwareEventInfo::.ctor(class System.Type, string)") {
auto stack = node->stack();
token = block.GetTypeOf(stack[0]);
if (token && token->type() == ttTypeDef) {
ILTypeDef *type_def = reinterpret_cast<ILTypeDef*>(token);
if (stack[1]->command()->type() == icLdstr) {
if (ILEvent *event = type_def->GetEvent(reinterpret_cast<ILUserString*>(stack[1]->command()->token_reference()->owner()->owner())->name()))
event->set_can_rename(false);
}
else {
ILTable *ref_table = file.command_list()->table(ttEvent);
for (size_t r = 0; r < ref_table->count(); r++) {
ILEvent *event = reinterpret_cast<ILEvent *>(ref_table->item(r));
if (event->declaring_type() == type_def)
event->set_can_rename(false);
}
}
}
}
else if (name.size() > 58 && name.substr(0, 58) == "string Newtonsoft.Json.JsonConvert::SerializeObject(object") {
auto stack = node->stack();
token = block.GetTypeFromStack(stack[0]);
if (token && token->type() == ttTypeDef) {
ILTypeDef *type_def = reinterpret_cast<ILTypeDef*>(token);
ILTable *ref_table = file.command_list()->table(ttProperty);
for (size_t r = 0; r < ref_table->count(); r++) {
ILProperty *property = reinterpret_cast<ILProperty*>(ref_table->item(r));
if (property->declaring_type() == type_def)
property->set_can_rename(false);
}
}
}
}
break;
}
}
}
}
file.EndProgress();
}
void ILFileHelper::AddString(NETArchitecture &file, uint32_t token, uint64_t reference)
{
if (TOKEN_TYPE(token) != ttUserString)
return;
uint64_t address;
ILData dump = file.command_list()->GetUserData(TOKEN_VALUE(token), &address);
if (dump.empty())
return;
std::string name = "string \"" + file.command_list()->GetUserString(TOKEN_VALUE(token)) + "\"";
MapFunction *map_function = file.map_function_list()->Add(address, address + dump.size(), otString, name);
map_function->reference_list()->Add(reference, address);
}
/**
* ILCommandNode
*/
ILCommandNode::ILCommandNode(ILCommandBlock *owner, ILCommand *command)
: IObject(), owner_(owner), command_(command)
{
}
ILCommandNode::~ILCommandNode()
{
if (owner_)
owner_->RemoveObject(this);
}
std::string ILCommandNode::token_name() const
{
if (command_->token_reference()) {
ILToken *token = command_->token_reference()->owner()->owner();
if (token->type() == ttMethodSpec)
token = reinterpret_cast<ILMethodSpec*>(token)->parent();
switch (token->type()) {
case ttMemberRef:
return reinterpret_cast<ILMemberRef*>(token)->full_name();
}
}
return std::string();
}
/**
* ILCommandBlock
*/
ILCommandBlock::ILCommandBlock()
: ObjectList<ILCommandNode>(), method_(NULL)
{
}
ILCommandNode *ILCommandBlock::Add(ILCommand *command)
{
ILCommandNode *node = new ILCommandNode(this, command);
AddObject(node);
return node;
}
ILCommandNode *ILCommandBlock::GetNodeByCommand(ILCommand *command) const
{
auto it = map_.find(command);
return (it != map_.end()) ? it->second : NULL;
}
void ILCommandBlock::AddObject(ILCommandNode *node)
{
ObjectList<ILCommandNode>::AddObject(node);
if (node->command())
map_[node->command()] = node;
}
ILToken *ILCommandBlock::GetTypeOf(ILCommandNode *node) const
{
if (node->command()->type() == icCall) {
if (node->token_name() == "class System.Type System.Type::GetTypeFromHandle(valuetype System.RuntimeTypeHandle)") {
auto stack = node->stack();
if (stack[0]->command()->type() == icLdtoken)
return stack[0]->command()->token_reference()->owner()->owner();
}
}
return NULL;
}
ILToken *ILCommandBlock::GetTypeFromStack(ILCommandNode *node) const
{
switch (node->command()->type()) {
case icLdarg_0:
if (method_ && method_->signature()->has_this())
return method_->declaring_type();
break;
case icDup:
return GetTypeFromStack(node->stack()[0]);
case icNewobj:
{
ILToken *token = node->command()->token_reference()->owner()->owner();
switch (token->type()) {
case ttMethodDef:
return reinterpret_cast<ILMethodDef*>(token)->declaring_type();
case ttMemberRef:
return reinterpret_cast<ILMemberRef*>(token)->declaring_type();
}
}
break;
}
return NULL;
}
bool ILCommandBlock::Parse(ILFunction &func)
{
size_t i, j, c;
ILCommand *command;
std::set<ILCommand *> entry_list;
std::vector<ILCommandNode *> command_stack;
std::map<ILCommand *, std::vector<ILCommandNode *> > stack_map;
for (i = 0; i < func.function_info_list()->count(); i++) {
FunctionInfo *info = func.function_info_list()->item(i);
if (!info->source())
continue;
command = reinterpret_cast<ILCommand *>(info->entry());
entry_list.insert(command);
method_ = reinterpret_cast<NETRuntimeFunction*>(info->source())->method();
}
command_stack.push_back(NULL);
for (i = 0; i < func.link_list()->count(); i++) {
CommandLink *link = func.link_list()->item(i);
switch (link->type()) {
case ltSEHBlock:
command = func.GetCommandByAddress(link->to_address());
if (command) {
entry_list.insert(command);
stack_map[command] = command_stack;
}
break;
case ltFinallyBlock:
command = func.GetCommandByAddress(link->to_address());
if (command)
entry_list.insert(command);
break;
}
}
while (!entry_list.empty()) {
command = *entry_list.begin();
j = func.IndexOf(command);
auto it = stack_map.find(command);
if (it != stack_map.end())
command_stack = it->second;
else
command_stack.clear();
for (i = j; i < func.count(); i++) {
command = func.item(i);
std::set<ILCommand *>::const_iterator it = entry_list.find(command);
if (it != entry_list.end())
entry_list.erase(it);
ILCommandNode *node = Add(command);
if (command->type() == icRet) {
if (command_stack.size() > 1)
return false;
node->set_stack(command_stack);
break;
}
size_t pop;
int stack = command->GetStackLevel(&pop);
if (pop) {
if (pop > command_stack.size())
return false;
std::vector<ILCommandNode *> node_stack;
node_stack.insert(node_stack.begin(), command_stack.end() - pop, command_stack.end());
command_stack.erase(command_stack.end() - pop, command_stack.end());
node->set_stack(node_stack);
}
for (c = 0; c < stack + pop; c++) {
command_stack.push_back(node);
}
if (command->link() && (command->link()->type() == ltJmp || command->link()->type() == ltJmpWithFlag)) {
if (ILCommand *link_command = func.GetCommandByAddress(command->link()->to_address())) {
if (!GetNodeByCommand(link_command)) {
entry_list.insert(link_command);
stack_map[link_command] = command_stack;
}
}
}
if (command->is_end())
break;
}
}
return true;
}
/**
* ILImport
*/
ILImport::ILImport(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size)
{
set_compilation_type(ctMutation);
}
bool ILImport::Init(const CompileContext &ctx)
{
NETArchitecture *file = reinterpret_cast<NETArchitecture *>(ctx.file);
NETArchitecture *runtime = reinterpret_cast<NETArchitecture *>(ctx.runtime);
ILTypeDef *module = reinterpret_cast<ILTypeDef*>(file->command_list()->token(ttTypeDef | 1));
ILTypeDef *runtime_loader = runtime->command_list()->GetTypeDef("VMProtect.Loader");
if (!runtime_loader)
return false;
ILField *iat_field = runtime_loader->GetField("IAT");
if (!iat_field)
return false;
size_t i, j, k, c;
std::map<ILToken *, bool> value_type_map;
ILTokenType table_types[] = {ttMemberRef, ttStandAloneSig, ttField, ttMethodDef};
for (i = 0; i < _countof(table_types); i++) {
ILTable *table = module->meta()->table(table_types[i]);
for (k = 0; k < table->count(); k++) {
ILToken *token = table->item(k);
ILSignature *signature = NULL;
switch (token->type()) {
case ttMemberRef:
signature = reinterpret_cast<ILMemberRef *>(token)->signature();
break;
case ttStandAloneSig:
signature = reinterpret_cast<ILStandAloneSig *>(token)->signature();
break;
case ttField:
signature = reinterpret_cast<ILField *>(token)->signature();
break;
case ttMethodDef:
signature = reinterpret_cast<ILMethodDef *>(token)->signature();
break;
}
if (!signature)
continue;
std::vector<ILElement *> element_stack;
element_stack.push_back(signature->ret());
for (c = 0; c < signature->count(); c++) {
element_stack.push_back(signature->item(c));
}
for (c = 0; c < element_stack.size(); c++) {
ILElement *element = element_stack[c];
if (element->type() == ELEMENT_TYPE_CLASS)
value_type_map[element->token()] = false;
else if (element->type() == ELEMENT_TYPE_VALUETYPE)
value_type_map[element->token()] = true;
if (element->next())
element_stack.push_back(element->next());
}
}
}
std::vector<NETImportFunction *> import_list;
for (i = 0; i < file->import_list()->count(); i++) {
NETImport *import = file->import_list()->item(i);
// APIs processed by ILSDK
if (import->is_sdk())
continue;
if (import->excluded_from_import_protection())
continue;
for (j = 0; j < import->count(); j++) {
NETImportFunction *import_function = import->item(j);
if (import_function->options() & ioHasCallPrefix)
continue;
import_list.push_back(import_function);
}
}
for (i = 0; i < import_list.size(); i++) {
std::swap(import_list[i], import_list[rand() % import_list.size()]);
}
size_t iat_index = 0;
ILTypeRef *delegate_base_type = NULL;
for (i = 0; i < import_list.size(); i++) {
NETImportFunction *import_function = import_list[i];
ILToken *import_token = file->command_list()->token(import_function->token());
ILSignature *signature;
ILToken *declaring_type;
switch (import_token->type()) {
case ttMemberRef:
signature = reinterpret_cast<ILMemberRef*>(import_token)->signature();
declaring_type = reinterpret_cast<ILMemberRef*>(import_token)->declaring_type();
break;
default:
continue;
}
if (declaring_type->type() != ttTypeRef)
continue;
MapFunction *map_function = import_function->map_function();
ReferenceList *reference_list = map_function->reference_list();
std::map<ILCommandType, ILMethodDef *> proxy_map;
for (j = 0; j < reference_list->count(); j++) {
uint64_t address = reference_list->item(j)->address();
if (!file->AddressSeek(address))
return false;
ILCommand *src_command = reinterpret_cast<ILCommand *>(ctx.file->function_list()->GetCommandByAddress(address, true));
size_t index = count();
ILCommand *ref_command = Add(address);
ref_command->ReadFromFile(*file);
ILCommandType ref_type = static_cast<ILCommandType>(ref_command->type());
if (ref_type != icNewobj && ref_type != icCall && ref_type != icCallvirt && ref_type != icLdsfld && ref_type != icLdfld) {
delete ref_command;
continue;
}
ILMethodDef *proxy_method;
std::map<ILCommandType, ILMethodDef *>::const_iterator it = proxy_map.find(ref_type);
if (it == proxy_map.end()) {
ILSignature *new_signature;
if (signature->is_field()) {
ILData data;
data.push_back(stDefault);
data.push_back(0);
signature->ret()->WriteToData(data);
new_signature = new ILSignature(module->meta());
new_signature->Parse(data);
if (ref_type == icLdfld)
new_signature->set_has_this(true);
}
else {
new_signature = signature->Clone(module->meta());
}
if (new_signature->has_this()) {
std::map<ILToken *, bool>::const_iterator it = value_type_map.find(declaring_type);
if (it == value_type_map.end()) {
delete ref_command;
continue;
}
bool is_value_type = it->second;
ILData data;
if (is_value_type) {
if (ref_type != icNewobj && ref_type != icLdfld)
data.push_back(ELEMENT_TYPE_BYREF);
data.push_back(ELEMENT_TYPE_VALUETYPE);
}
else {
data.push_back(ELEMENT_TYPE_CLASS);
}
uint32_t value = declaring_type->value() << 2;
switch (declaring_type->type()) {
case ttTypeDef:
value |= 0;
break;
case ttTypeRef:
value |= 1;
break;
case ttTypeSpec:
value |= 2;
break;
}
data.push_back(static_cast<uint8_t>(value >> 24) | 0xc0);
data.push_back(static_cast<uint8_t>(value >> 16));
data.push_back(static_cast<uint8_t>(value >> 8));
data.push_back(static_cast<uint8_t>(value));
if (ref_type == icNewobj)
new_signature->ret()->Parse(data);
else {
ILElement *element = new ILElement(module->meta(), new_signature);
element->Parse(data);
new_signature->InsertObject(0, element);
}
}
ILData proxy_signature;
{
ILData data;
data.push_back(ELEMENT_TYPE_OBJECT);
ILElement *element = new_signature->ret();
switch (element->type()) {
case ELEMENT_TYPE_CLASS:
case ELEMENT_TYPE_SZARRAY:
element->Parse(data);
break;
case ELEMENT_TYPE_GENERICINST:
if (element->next()->type() == ELEMENT_TYPE_CLASS)
element->Parse(data);
break;
}
if (new_signature->has_this()) {
new_signature->set_has_this(false);
for (k = 0; k < new_signature->count(); k++) {
element = new_signature->item(k);
switch (element->type()) {
case ELEMENT_TYPE_CLASS:
case ELEMENT_TYPE_SZARRAY:
element->Parse(data);
break;
case ELEMENT_TYPE_GENERICINST:
if (element->next()->type() == ELEMENT_TYPE_CLASS)
element->Parse(data);
break;
}
}
}
}
new_signature->WriteToData(proxy_signature);
delete new_signature;
if (!delegate_base_type) {
ILAssemblyRef *assembly_ref = module->meta()->GetCoreLib();
delegate_base_type = module->meta()->ImportTypeRef(assembly_ref, "System", "MulticastDelegate");
}
ILTypeDef *delegate_type = module->meta()->AddTypeDef(delegate_base_type, "", "", (CorTypeAttr)(tdNotPublic | tdSealed));
file->RenameToken(delegate_type);
ILData ctor_signature; // instance void (object, native int)
ctor_signature.push_back(stDefault | stHasThis);
ctor_signature.push_back(2);
ctor_signature.push_back(ELEMENT_TYPE_VOID);
ctor_signature.push_back(ELEMENT_TYPE_OBJECT);
ctor_signature.push_back(ELEMENT_TYPE_I);
ILMethodDef *delegate_ctor = module->meta()->AddMethod(delegate_type, ".ctor", ctor_signature, (CorMethodAttr)(mdAssem | mdHideBySig | mdRTSpecialName | mdSpecialName), miRuntime);
ILMethodDef *delegate_invoke = module->meta()->AddMethod(delegate_type, "Invoke", proxy_signature, (CorMethodAttr)(mdAssem | mdHideBySig | mdVirtual | mdNewSlot), miRuntime);
delegate_invoke->signature()->set_has_this(true);
uint32_t call_type;
switch (ref_type) {
case icNewobj:
call_type = 1;
break;
case icCallvirt:
call_type = 2;
break;
case icLdsfld:
call_type = 8;
break;
case icLdfld:
call_type = 9;
break;
default:
call_type = 0;
break;
}
info_list_.push_back(ImportDelegateInfo(import_token, delegate_invoke, call_type));
proxy_method = module->meta()->AddMethod(module, "", proxy_signature, (CorMethodAttr)(mdAssem | mdStatic), miIL);
file->RenameToken(proxy_method);
size_t entry_index = count();
// header
ILCommand *command = AddCommand(icComment, 0);
command->include_section_option(rtLinkedToExt);
Data data;
data.PushByte(CorILMethod_TinyFormat);
command->set_dump(data.data(), data.size());
command->set_alignment(sizeof(uint32_t));
// code
command = AddCommand(icLdsfld, 0);
command->set_token_reference(iat_field->reference_list()->Add(0));
AddCommand(icLdc_i4, static_cast<uint32_t>(iat_index++));
AddCommand(icLdelem_ref, 0);
for (k = 0; k < proxy_method->signature()->count(); k++) {
AddCommand(icLdarg, static_cast<uint32_t>(k));
}
command = AddCommand(icCall, 0);
command->set_token_reference(delegate_invoke->reference_list()->Add(0));
AddCommand(icRet, 0);
FunctionInfo *info = function_info_list()->Add(0, 0, btValue, item(entry_index)->dump_size(), 0, 0, runtime->runtime_function_list()->Add(0, 0, 0, proxy_method), item(entry_index));
AddressRange *range = info->Add(0, 0, NULL, NULL, NULL);
for (k = entry_index; k < count(); k++) {
item(k)->set_address_range(range);
}
proxy_map[ref_type] = proxy_method;
}
else {
proxy_method = it->second;
}
if (src_command) {
delete ref_command;
src_command->Init(icCall, 0, src_command->token_reference());
src_command->CompileToNative();
src_command->token_reference()->set_owner(proxy_method->reference_list());
}
else {
ref_command->exclude_option(roClearOriginalCode);
ref_command->Init(icCall, 0, ref_command->token_reference());
ref_command->CompileToNative();
ref_command->token_reference()->set_owner(proxy_method->reference_list());
CommandBlock *block = AddBlock(index, true);
block->set_address(address);
ref_command->set_block(block);
}
}
}
for (i = 0; i < count(); i++) {
item(i)->CompileToNative();
}
return ILFunction::Init(ctx);
}
/**
* ILVirtualMachine
*/
ILVirtualMachine::ILVirtualMachine(ILVirtualMachineList *owner, uint8_t id, ILVirtualMachineProcessor *processor)
: BaseVirtualMachine(owner, id), processor_(processor), ctor_(NULL), invoke_(NULL)
{
}
void ILVirtualMachine::Init(const CompileContext &ctx)
{
processor_->InitCommands(ctx);
ctor_ = processor_->function_info_list()->count() > 0 ? reinterpret_cast<NETRuntimeFunction *>(processor_->function_info_list()->item(0)->source())->method() : NULL;
invoke_ = processor_->function_info_list()->count() > 1 ? reinterpret_cast<NETRuntimeFunction *>(processor_->function_info_list()->item(1)->source())->method() : NULL;
size_t i, j, c;
ILCommand *command;
ILOpcodeInfo *opcode;
size_t template_index, template_count;
template_index = 0;
template_count = 0;
for (i = 0; i < processor_->count(); i++) {
command = processor_->item(i);
if (command->type() != icLdftn)
continue;
template_index = i - 4;
for (j = i; j < processor_->count(); j++) {
command = processor_->item(j);
if (command->type() != icLdftn)
continue;
if (!template_count && j > i)
template_count = j - i;
opcode_list_.Add(static_cast<ILCommandType>(processor_->item(j - 2)->operand_value()), command->token_reference()->owner()->owner());
}
break;
}
for (i = template_index + template_count * opcode_list_.count(); i > template_index + template_count; i--) {
command = processor_->item(i - 1);
if (command->link())
delete command->link();
if (command->token_reference())
delete command->token_reference();
delete command;
}
// randomize opcodes
c = opcode_list_.count();
for (i = 0; i < opcode_list_.count(); i++) {
opcode_list_.SwapObjects(i, rand() % c);
}
for (i = opcode_list_.count(); i < 0x100; i++) {
opcode = opcode_list_.item(rand() % i);
opcode_list_.Add(opcode->command_type(), opcode->entry());
}
opcode_stack_.clear();
for (i = 0; i < opcode_list_.count(); i++) {
opcode = opcode_list_.item(i);
opcode->set_opcode(static_cast<uint8_t>(i));
opcode_stack_[opcode->Key()].push_back(opcode);
for (j = 0; j < template_count; j++) {
command = processor_->item(template_index + j);
if (i > 0) {
command = command->Clone(processor_);
processor_->InsertObject(template_index + i * template_count + j, command);
}
switch (command->type()) {
case icLdc_i4: case icLdc_i4_s:
command->Init(icLdc_i4, opcode->opcode());
command->CompileToNative();
break;
case icLdftn:
command->set_operand_value(0, opcode->entry()->id());
command->set_token_reference(opcode->entry()->reference_list()->Add(0));
command->CompileToNative();
break;
default:
if (command->token_reference())
command->set_token_reference(command->token_reference()->owner()->Add(0));
}
}
}
}
ILOpcodeInfo *ILVirtualMachine::GetOpcode(ILCommandType command_type)
{
ILOpcodeInfo *res = NULL;
auto it = opcode_stack_.find(command_type);
if (it != opcode_stack_.end())
res = it->second.Next();
return res;
}
static void EncryptBuffer(uint32_t *buffer, uint64_t key)
{
uint32_t key0 = static_cast<uint32_t>(key >> 32);
uint32_t key1 = static_cast<uint32_t>(key);
buffer[0] = _rotr32(buffer[0] - key0, 7) ^ key1;
buffer[1] = _rotr32(buffer[1] - key0, 11) ^ key1;
buffer[2] = _rotr32(buffer[2] - key0, 17) ^ key1;
buffer[3] = _rotr32(buffer[3] - key0, 23) ^ key1;
}
void ILVirtualMachine::CompileCommand(ILVMCommand &vm_command)
{
Data dump;
ILOpcodeInfo *opcode = NULL;
uint64_t value = vm_command.value();
if (vm_command.crypt_command() == icAdd) {
uint32_t crypted_value[4];
size_t i;
for (i = 0; i < _countof(crypted_value); i++) {
crypted_value[i] = rand32();
}
switch (vm_command.crypt_size()) {
case osDWord:
crypted_value[3] = static_cast<uint32_t>(value);
break;
case osQWord:
*reinterpret_cast<uint64_t*>(&crypted_value[2]) = value;
break;
}
uint32_t dw = 0;
for (i = 1; i < 4; i++) {
dw += crypted_value[i];
}
crypted_value[0] = 0 - dw;
EncryptBuffer(crypted_value, vm_command.crypt_key());
ILVMCommand *link_command = vm_command.link_command();
for (i = 3; i > 0; i--) {
link_command->set_value(crypted_value[i - 1]);
link_command->Compile();
link_command = link_command->link_command();
}
value = crypted_value[3];
}
switch (vm_command.command_type()) {
case icLdarg: case icLdarga:
case icStarg:
case icLdloc: case icLdloca:
case icStloc:
opcode = GetOpcode(vm_command.command_type());
dump.PushWord(static_cast<uint16_t>(vm_command.value()));
break;
case icLdc_i4:
opcode = GetOpcode(vm_command.command_type());
dump.PushDWord(static_cast<uint32_t>(value));
break;
case icLdc_i8:
opcode = GetOpcode(vm_command.command_type());
dump.PushQWord(value);
break;
case icLdc_r4:
opcode = GetOpcode(vm_command.command_type());
dump.PushDWord(static_cast<uint32_t>(value));
break;
case icLdc_r8:
opcode = GetOpcode(vm_command.command_type());
dump.PushQWord(value);
break;
case icByte:
dump.PushByte(static_cast<uint8_t>(vm_command.value()));
break;
case icWord:
dump.PushWord(static_cast<uint16_t>(vm_command.value()));
break;
case icDword:
dump.PushDWord(static_cast<uint32_t>(vm_command.value()));
break;
case icQword:
dump.PushQWord(vm_command.value());
break;
default:
opcode = GetOpcode(vm_command.command_type());
break;
}
if (opcode) {
dump.InsertByte(0, opcode->opcode());
}
else if (!vm_command.is_data()) {
throw std::runtime_error("Runtime error at CompileToVM: " + std::string(ILOpCodes[vm_command.command_type()].name));
}
vm_command.set_dump(dump);
}
/**
* ILOpcodeInfo
*/
ILOpcodeInfo::ILOpcodeInfo(ILOpcodeList *owner, ILCommandType command_type, ILToken *entry)
: IObject(), owner_(owner), command_type_(command_type), entry_(entry), opcode_(0)
{
}
ILOpcodeInfo::~ILOpcodeInfo()
{
if (owner_)
owner_->RemoveObject(this);
}
ILOpcodeInfo *ILOpcodeInfo::circular_queue::Next()
{
ILOpcodeInfo *res = NULL;
if (size())
res = this->operator[](position_++ % size());
return res;
}
/**
* ILOpcodeList
*/
ILOpcodeList::ILOpcodeList()
: ObjectList<ILOpcodeInfo>()
{
}
ILOpcodeInfo *ILOpcodeList::Add(ILCommandType command_type, ILToken *entry)
{
ILOpcodeInfo *opcode = new ILOpcodeInfo(this, command_type, entry);
AddObject(opcode);
return opcode;
}
ILOpcodeInfo *ILOpcodeList::GetOpcodeInfo(ILCommandType command_type) const
{
for (size_t i = 0; i < count(); i++) {
ILOpcodeInfo *opcode = item(i);
if (opcode->command_type() == command_type)
return opcode;
}
return NULL;
}
/**
* ILVirtualMachineList
*/
IVirtualMachineList * ILVirtualMachineList::Clone() const
{
ILVirtualMachineList *list = new ILVirtualMachineList();
return list;
}
void ILVirtualMachineList::Prepare(const CompileContext &ctx)
{
ILFunctionList *function_list = reinterpret_cast<ILFunctionList *>(ctx.file->function_list());
ILVirtualMachineProcessor *processor = function_list->AddProcessor(ctx.file->cpu_address_size());
ILVirtualMachine *virtual_machine = new ILVirtualMachine(this, 1, processor);
AddObject(virtual_machine);
virtual_machine->Init(ctx);
}
/**
* ILVirtualMachineProcessor
*/
ILVirtualMachineProcessor::ILVirtualMachineProcessor(ILFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size)
{
set_compilation_type(ctMutation);
set_tag(ftProcessor);
}
void ILVirtualMachineProcessor::InitCommands(const CompileContext &ctx)
{
size_t i, j, k;
ILCommand *command, *src_command, *dst_command;
CommandLink *src_link, *dst_link;
ILFunctionList *runtime_function_list = reinterpret_cast<ILFunctionList *>(ctx.vm_runtime->function_list());
for (i = 0; i < runtime_function_list->count(); i++) {
ILFunction *func = runtime_function_list->item(i);
if (func->tag() != ftProcessor)
continue;
//func->Init(ctx);
size_t orig_function_info_count = function_info_list()->count();
for (j = 0; j < func->function_info_list()->count(); j++) {
FunctionInfo *info = func->function_info_list()->item(j);
function_info_list()->AddObject(info->Clone(function_info_list()));
}
for (j = 0; j < func->range_list()->count(); j++) {
AddressRange *range = func->range_list()->item(j);
range_list()->AddObject(range->Clone(range_list()));
}
for (j = 0; j < func->count(); j++) {
src_command = func->item(j);
dst_command = src_command->Clone(this);
AddressRange *address_range = src_command->address_range();
if (address_range) {
FunctionInfo *info = function_info_list()->item(orig_function_info_count + func->function_info_list()->IndexOf(address_range->owner()));
dst_command->set_address_range(info->item(address_range->owner()->IndexOf(address_range)));
}
AddObject(dst_command);
src_link = src_command->link();
if (src_link) {
dst_link = src_link->Clone(link_list());
dst_link->set_from_command(dst_command);
link_list()->AddObject(dst_link);
if (src_link->parent_command())
dst_link->set_parent_command(GetCommandByAddress(src_link->parent_command()->address()));
if (src_link->base_function_info())
dst_link->set_base_function_info(function_info_list()->GetItemByAddress(src_link->base_function_info()->begin()));
}
}
}
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (info->entry())
info->set_entry(GetCommandByAddress(info->entry()->address()));
if (info->data_entry())
info->set_data_entry(GetCommandByAddress(info->data_entry()->address()));
for (j = 0; j < info->count(); j++) {
AddressRange *dest = info->item(j);
for (k = 0; k < range_list()->count(); k++) {
AddressRange *range = range_list()->item(k);
if (range->begin() <= dest->begin() && range->end() > dest->begin())
dest->AddLink(range);
}
}
}
for (i = 0; i < range_list()->count(); i++) {
AddressRange *range = range_list()->item(i);
if (range->begin_entry())
range->set_begin_entry(GetCommandByAddress(range->begin_entry()->address()));
if (range->end_entry())
range->set_end_entry(GetCommandByAddress(range->end_entry()->address()));
if (range->size_entry())
range->set_size_entry(GetCommandByAddress(range->size_entry()->address()));
}
for (i = 0; i < count(); i++) {
command = item(i);
dst_link = command->link();
if (dst_link) {
if (dst_link->to_address())
dst_link->set_to_command(GetCommandByAddress(dst_link->to_address()));
}
if (command->token_reference())
command->token_reference()->set_address(0);
command->exclude_option(roClearOriginalCode);
command->CompileToNative();
}
}
void ILVirtualMachineProcessor::AfterCompile(const CompileContext &ctx)
{
ILFunction::AfterCompile(ctx);
if (ctx.runtime)
set_need_compile(false);
}
void ILVirtualMachineProcessor::CompileLinks(const CompileContext &ctx)
{
if (!need_compile())
return;
ILFunction::CompileLinks(ctx);
}
void ILVirtualMachineProcessor::CompileInfo(const CompileContext &ctx)
{
if (!need_compile())
return;
ILFunction::CompileInfo(ctx);
}
size_t ILVirtualMachineProcessor::WriteToFile(IArchitecture &file)
{
if (!need_compile())
return 0;
return ILFunction::WriteToFile(file);
}
/**
* NETLoader
*/
NETLoader::NETLoader(IFunctionList *owner, OperandSize cpu_address_size)
: ILFunction(owner, cpu_address_size), file_crc_entry_(NULL), file_crc_size_(0),
import_entry_(NULL), import_size_(0), iat_entry_(NULL), iat_size_(0), loader_crc_entry_(NULL), loader_crc_size_(0),
file_crc_size_entry_(NULL), loader_crc_size_entry_(NULL), loader_crc_hash_entry_(NULL),
pe_entry_(NULL), strong_name_signature_entry_(NULL), tls_entry_(NULL), tls_size_(0)
{
set_compilation_type(ctVirtualization);
set_tag(ftLoader);
}
void NETLoader::AddAVBuffer(const CompileContext &ctx)
{
ILCommand *command;
uint32_t sum = 0;
CommandBlock *block = AddBlock(count(), true);
for (size_t i = 0; i < 64; i++) {
uint32_t value = (i == 0) ? 0 : rand32();
sum += value;
command = AddCommand(icDword, value);
command->CompileToNative();
command->set_block(block);
}
block->set_end_index(count() - 1);
command = item(block->start_index());
command->set_operand_value(0, 0xB7896EB5 - sum);
command->CompileToNative();
uint64_t address = ctx.manager->Alloc((block->end_index() - block->start_index() + 1) * sizeof(uint32_t), mtReadable);
block->set_address(address);
}
Data EncryptString(const char *str, uint32_t key);
Data EncryptString(const os::unicode_char *str, uint32_t key);
bool NETLoader::Prepare(const CompileContext &ctx)
{
size_t i, j, k, old_count, index, import_index, start_index;
NETArchitecture *file, *runtime;
ILFunctionList *runtime_function_list;
ILFunction *func;
CommandLink *link, *src_link, *dst_link;
ILCommand *src_command, *dst_command, *command;
ILCRCTable *il_crc;
PEImportList new_import_list(NULL);
std::vector<ImportInfo> import_info_list;
std::vector<ImportFunctionInfo> import_function_info_list;
PEImport *import;
PEImportFunction *import_function;
ImportInfo import_info;
PESegment *segment;
ILImport *il_import;
uint64_t tls_index_address;
file = reinterpret_cast<NETArchitecture *>(ctx.file);
runtime = reinterpret_cast<NETArchitecture *>(ctx.runtime);
il_crc = reinterpret_cast<ILFunctionList *>(file->function_list())->crc_table();
il_import = reinterpret_cast<ILFunctionList *>(file->function_list())->import();
PEArchitecture *pe_file = file->pe();
uint32_t string_key = rand32();
ILCommandType value_command_type = (cpu_address_size() == osDWord) ? icDword : icQword;
// create AV signature buffer
AddAVBuffer(ctx);
start_index = count();
import_size_ = 0;
import_entry_ = 0;
PEImportFunction *entry_point_import_function = NULL;
if ((ctx.options.flags & cpPack) && pe_file->import_list()->count()) {
for (i = 0; i < pe_file->import_list()->count(); i++) {
import = pe_file->import_list()->item(i);
new_import_list.AddObject(import->Clone(&new_import_list));
}
const std::string import_name = "mscoree.dll";
const std::string import_function_name = (pe_file->image_type() == itExe) ? "_CorExeMain" : "_CorDllMain";
import = reinterpret_cast<PEImport *>(new_import_list.GetImportByName(import_name));
if (!import) {
import = new PEImport(&new_import_list, import_name);
new_import_list.AddObject(import);
}
entry_point_import_function = NULL;
for (i = 0; i < import->count(); i++) {
import_function = import->item(i);
if (import_function->name() == import_function_name) {
entry_point_import_function = import_function;
break;
}
}
if (!entry_point_import_function) {
entry_point_import_function = new PEImportFunction(import, import_function_name);
import->AddObject(entry_point_import_function);
}
// create import directory
for (i = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
// IMAGE_IMPORT_DESCRIPTOR.OriginalFirstThunk
command = AddCommand(icDword, 0);
command->AddLink(0, ltOffset);
import_info.original_first_thunk = command;
// IMAGE_IMPORT_DESCRIPTOR.TimeDateStamp
AddCommand(icDword, 0);
// IMAGE_IMPORT_DESCRIPTOR.ForwarderChain
AddCommand(icDword, 0);
// IMAGE_IMPORT_DESCRIPTOR.Name
command = AddCommand(icDword, 0);
command->AddLink(0, ltOffset);
import_info.name = command;
// IMAGE_IMPORT_DESCRIPTOR.FirstThunk
command = AddCommand(icDword, 0);
command->AddLink(0, ltOffset);
import_info.first_thunk = command;
import_info_list.push_back(import_info);
}
// end of import directory
for (j = 0; j < 5; j++) {
AddCommand(icDword, 0);
}
import_size_ = static_cast<uint32_t>((count() - start_index) * sizeof(uint32_t));
import_entry_ = item(start_index);
import_entry_->set_alignment(OperandSizeToValue(pe_file->cpu_address_size()));
// create IAT
uint64_t ordinal_mask = (cpu_address_size() == osDWord) ? IMAGE_ORDINAL_FLAG32 : IMAGE_ORDINAL_FLAG64;
size_t name_index = count();
for (i = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
index = count();
for (j = 0; j < import->count(); j++) {
import_function = import->item(j);
if (import_function->is_ordinal()) {
command = AddCommand(value_command_type, ordinal_mask | import_function->ordinal());
} else {
command = AddCommand(value_command_type, 0);
command->AddLink(0, ltOffset);
}
ImportFunctionInfo import_function_info(import_function);
import_function_info.name = command;
import_function_info_list.push_back(import_function_info);
}
AddCommand(value_command_type, 0);
import_info_list[i].original_first_thunk->link()->set_to_command(item(index));
}
command = item(name_index);
command->set_alignment(OperandSizeToValue(cpu_address_size()));
size_t iat_index = count();
for (i = 0, import_index = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
bool is_delay_import = (import_info_list[i].name == NULL);
index = count();
for (j = 0; j < import->count(); j++, import_index++) {
import_function = import->item(j);
if (import_function->is_ordinal()) {
command = AddCommand(value_command_type, ordinal_mask | import_function->ordinal());
} else {
command = AddCommand(value_command_type, 0);
command->AddLink(0, ltOffset);
}
import_function_info_list[import_index].thunk = command;
}
if (is_delay_import)
continue;
AddCommand(value_command_type, 0);
import_info_list[i].first_thunk->link()->set_to_command(item(index));
}
iat_entry_ = item(iat_index);
iat_size_ = static_cast<uint32_t>((count() - iat_index) * OperandSizeToValue(cpu_address_size()));
iat_entry_->set_alignment(file->segment_alignment());
iat_entry_->include_option(roCreateNewBlock);
// IAT size must be aligned by page size
j = AlignValue(iat_size_, file->segment_alignment());
if (j > iat_size_) {
std::string buffer;
buffer.resize(j - iat_size_, 0);
AddCommand(buffer);
}
// create import DLL names
uint32_t string_key = rand32();
for (i = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
command = NULL;
for (j = 0; j < i; j++) {
if (new_import_list.item(j)->CompareName(import->name())) {
command = reinterpret_cast<ILCommand *>(import_info_list[j].name->link()->to_command());
break;
}
}
if (command == NULL) {
command = AddCommand(import->name());
command->include_option(roCreateNewBlock);
command->set_alignment(sizeof(uint16_t));
}
import_info_list[i].name->link()->set_to_command(command);
}
// create import function names
for (i = 0, import_index = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
for (j = 0; j < import->count(); j++, import_index++) {
import_function = import->item(j);
if (import_function->is_ordinal())
continue;
command = AddCommand(icWord, 0);
command->include_option(roCreateNewBlock);
command->set_alignment(sizeof(uint16_t));
AddCommand(import_function->name());
import_function_info_list[import_index].name->link()->set_to_command(command);
import_function_info_list[import_index].thunk->link()->set_to_command(command);
}
}
// update links for PE structures
for (i = 0; i < count(); i++) {
link = item(i)->link();
if (!link)
continue;
link->set_sub_value(file->image_base());
}
}
pe_entry_ = NULL;
strong_name_signature_entry_ = NULL;
if ((ctx.options.flags & cpPack)) {
if (pe_file->entry_point()) {
// create native entry point
segment = file->segment_list()->GetSectionByAddress(pe_file->entry_point());
if (segment && !segment->excluded_from_packing()) {
std::vector<ImportFunctionInfo>::const_iterator it = std::find(import_function_info_list.begin(), import_function_info_list.end(), entry_point_import_function);
if (it == import_function_info_list.end())
return false;
index = count();
CommandBlock *block = AddBlock(count(), true);
AddCommand(icWord, 0x25ff);
command = AddCommand(icDword, 0);
command->AddLink(0, ltOffset, it->thunk);
pe_entry_ = item(index);
for (i = index; i < count(); i++) {
command = item(i);
command->set_block(block);
block->set_end_index(i);
}
}
}
IMAGE_DATA_DIRECTORY strong_name_signature = file->header().StrongNameSignature;
if (strong_name_signature.VirtualAddress) {
// create blank StrongNameSignature
segment = file->segment_list()->GetSectionByAddress(strong_name_signature.VirtualAddress + file->image_base());
if (segment && !segment->excluded_from_packing()) {
Data data;
data.resize(strong_name_signature.Size ? strong_name_signature.Size : 0x80);
strong_name_signature_entry_ = AddCommand(data);
strong_name_signature_entry_->include_option(roCreateNewBlock);
strong_name_signature_entry_->set_alignment(sizeof(uint32_t));
}
}
}
// create tls structure
tls_entry_ = NULL;
tls_size_ = 0;
tls_call_back_entry_ = NULL;
tls_index_address = 0;
if (pe_file->tls_directory()->address() && (pe_file->tls_directory()->count() || (ctx.options.flags & cpPack))) {
size_t tls_index = count();
PETLSDirectory *tls_directory = pe_file->tls_directory();
if (ctx.options.flags & cpPack) {
if (file->AddressSeek(tls_directory->address_of_index()) && !file->selected_segment()->excluded_from_packing())
tls_index_address = tls_directory->address_of_index();
}
ILCommand *start_address_entry = AddCommand(value_command_type, tls_directory->start_address_of_raw_data(), NEED_FIXUP);
ILCommand *end_address_entry = AddCommand(value_command_type, tls_directory->end_address_of_raw_data(), NEED_FIXUP);
AddCommand(value_command_type, tls_directory->address_of_index(), NEED_FIXUP);
ILCommand *call_back_entry = AddCommand(value_command_type, 0);
AddCommand(icDword, tls_directory->size_of_zero_fill());
AddCommand(icDword, tls_directory->characteristics());
if (tls_directory->end_address_of_raw_data() > tls_directory->start_address_of_raw_data()) {
Data data;
data.resize(static_cast<size_t>(tls_directory->end_address_of_raw_data() - tls_directory->start_address_of_raw_data()));
if (file->AddressSeek(tls_directory->start_address_of_raw_data()))
file->Read(&data[0], data.size());
ILCommand *data_entry = AddCommand(data);
start_address_entry->AddLink(0, ltOffset, data_entry);
link = end_address_entry->AddLink(0, ltOffset, data_entry);
link->set_sub_value(0 - (uint64_t)data.size());
}
index = count();
if (pe_file->tls_directory()->count()) {
tls_call_back_entry_ = AddCommand(value_command_type, 0, NEED_FIXUP);
tls_call_back_entry_->AddLink(0, ltGateOffset);
}
for (i = 0; i < tls_directory->count(); i++) {
AddCommand(icDword, tls_directory->item(i)->address(), NEED_FIXUP);
}
if (count() > index) {
AddCommand(value_command_type, 0);
call_back_entry->AddLink(0, ltOffset, item(index));
call_back_entry->set_operand_fixup(NEED_FIXUP);
}
tls_entry_ = item(tls_index);
tls_entry_->include_option(roCreateNewBlock);
tls_entry_->set_alignment(OperandSizeToValue(cpu_address_size()));
for (i = tls_index; i < count(); i++) {
command = item(i);
tls_size_ += OperandSizeToValue(command->type() == icDword ? osDWord : osQWord);
}
}
// create watermarks
AddWatermark(ctx.options.watermark, 2);
// create section list for setting WRITABLE flag
std::vector<PESegment *> writable_section_list;
for (i = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
for (j = 0; j < import->count(); j++) {
import_function = import->item(j);
segment = file->segment_list()->GetSectionByAddress(import_function->address());
if (!segment)
continue;
if (std::find(writable_section_list.begin(), writable_section_list.end(), segment) == writable_section_list.end())
writable_section_list.push_back(segment);
}
}
std::vector<PackerInfo> packer_info_list;
PEFixupList loader_fixup_list;
bool pack_resources = false;
ILCommand *packer_props = NULL;
vtable_fixups_entry_ = NULL;
if (ctx.options.flags & cpPack) {
std::set<PESegment *> skip_segment_list;
if (file->header().VTableFixups.VirtualAddress) {
uint64_t address = file->image_base() + file->header().VTableFixups.VirtualAddress;
segment = file->segment_list()->GetSectionByAddress(address);
if (segment && !segment->excluded_from_packing() && file->AddressSeek(address)) {
index = count();
j = file->header().VTableFixups.Size;
for (i = 0; i < j; i += sizeof(IMAGE_COR_VTABLEFIXUP)) {
IMAGE_COR_VTABLEFIXUP fixup;
file->Read(&fixup, sizeof(fixup));
AddCommand(icDword, fixup.RVA);
AddCommand(icWord, fixup.Count);
AddCommand(icWord, fixup.Type);
if (PESegment *tmp = file->segment_list()->GetSectionByAddress(file->image_base() + fixup.RVA))
skip_segment_list.insert(tmp);
}
if (index < count()) {
vtable_fixups_entry_ = item(index);
vtable_fixups_entry_->include_option(roCreateNewBlock);
vtable_fixups_entry_->set_alignment(sizeof(uint64_t));
}
}
}
PackerInfo packer_info;
for (i = 0; i < file->segment_list()->count(); i++) {
segment = file->segment_list()->item(i);
if (segment->excluded_from_packing())
continue;
bool can_be_packed = true;
if ((segment->memory_type() & (mtWritable | mtShared)) == (mtWritable | mtShared)) {
can_be_packed = false;
} else if (skip_segment_list.find(segment) != skip_segment_list.end()) {
can_be_packed = false;
}
if (!can_be_packed) {
//file->Notify(mtWarning, NULL, string_format(language[lsSegmentCanNotBePacked].c_str(), segment->name().c_str()));
continue;
}
if (segment->physical_size()) {
packer_info.section = segment;
packer_info.address = segment->address();
packer_info.size = static_cast<size_t>(segment->physical_size());
packer_info.data = NULL;
packer_info_list.push_back(packer_info);
// need add packed section into WRITABLE section list
if (std::find(writable_section_list.begin(), writable_section_list.end(), segment) == writable_section_list.end())
writable_section_list.push_back(segment);
}
}
if ((ctx.options.flags & cpStripFixups) == 0) {
PEFixupList *fixup_list = file->pe()->fixup_list();
for (i = 0; i < fixup_list->count(); i++) {
PEFixup *fixup = fixup_list->item(i);
if (fixup->is_deleted())
continue;
segment = file->segment_list()->GetSectionByAddress(fixup->address());
if (!segment || std::find(packer_info_list.begin(), packer_info_list.end(), segment) == packer_info_list.end())
continue;
loader_fixup_list.AddObject(fixup->Clone(&loader_fixup_list));
fixup->set_deleted(true);
// need add section into WRITABLE section list
if (std::find(writable_section_list.begin(), writable_section_list.end(), segment) == writable_section_list.end())
writable_section_list.push_back(segment);
}
}
// packing sections
j = 0;
for (i = 0; i < packer_info_list.size(); i++) {
j += packer_info_list[i].size;
}
/*
if (file->resource_list()->size() > file->resource_list()->store_size())
j += file->resource_list()->size() - file->resource_list()->store_size();
*/
file->StartProgress(string_format("%s...", language[lsPacking].c_str()), j);
Data data;
Packer packer;
if (!packer.WriteProps(&data))
throw std::runtime_error("Packer error");
packer_props = AddCommand(data);
packer_props->include_option(roCreateNewBlock);
for (i = 0; i < packer_info_list.size(); i++) {
packer_info = packer_info_list[i];
if (!file->AddressSeek(packer_info.address))
return false;
if (!packer.Code(file, packer_info.size, &data))
throw std::runtime_error("Packer error");
command = AddCommand(data);
command->include_option(roCreateNewBlock);
packer_info_list[i].data = command;
}
// remove packed sections from file
uint32_t physical_offset = 0;
for (i = 0; i < file->segment_list()->count(); i++) {
segment = file->segment_list()->item(i);
if (segment->physical_offset() > 0 && segment->physical_size() > 0) {
physical_offset = static_cast<uint32_t>(segment->physical_offset());
break;
}
}
for (i = 0; i < file->segment_list()->count(); i++) {
segment = file->segment_list()->item(i);
uint32_t physical_size = segment->physical_size();
bool is_packed = false;
std::vector<PackerInfo>::iterator it = std::find(packer_info_list.begin(), packer_info_list.end(), segment);
if (it != packer_info_list.end()) {
physical_size = static_cast<uint32_t>(it->address - segment->address());
is_packed = true;
}
if (physical_size > 0 && segment->physical_offset() != physical_offset) {
uint8_t *buff = new uint8_t[physical_size];
file->Seek(segment->physical_offset());
file->Read(buff, physical_size);
file->Seek(physical_offset);
file->Write(buff, physical_size);
delete [] buff;
}
segment->set_physical_offset(physical_offset);
segment->set_physical_size(physical_size);
if (is_packed) {
j = physical_offset + physical_size;
file->Seek(j);
physical_offset = (uint32_t)AlignValue(j, file->file_alignment());
for (k = j; k < physical_offset; k++) {
file->WriteByte(0);
}
} else {
physical_offset += physical_size;
}
}
file->Resize(physical_offset);
}
// create packer info for loader
std::vector<LoaderInfo> loader_info_list;
index = count();
if (packer_props) {
command = AddCommand(icDword, 0);
link = command->AddLink(0, ltOffset, packer_props);
link->set_sub_value(file->image_base());
AddCommand(icDword, packer_props->dump_size());
for (i = 0; i < packer_info_list.size(); i++) {
command = AddCommand(icDword, 0);
link = command->AddLink(0, ltOffset, packer_info_list[i].data);
link->set_sub_value(file->image_base());
AddCommand(icDword, packer_info_list[i].address - file->image_base());
}
}
command = (count() == index) ? NULL : item(index);
if (command)
command->include_option(roCreateNewBlock);
loader_info_list.push_back(LoaderInfo(command, (count() - index) * OperandSizeToValue(osDWord)));
// create file CRC info for loader
index = count();
if (((ctx.options.flags | ctx.options.sdk_flags) & cpMemoryProtection)) {
AddCommand(icDword, 0);
for (i = 0; i < 10; i++) {
AddCommand(icDword, 0);
AddCommand(icDword, 0);
AddCommand(icDword, 0);
}
}
file_crc_entry_ = (count() == index) ? NULL : item(index);
if (file_crc_entry_)
file_crc_entry_->include_option(roCreateNewBlock);
file_crc_size_ = static_cast<uint32_t>((count() - index) * OperandSizeToValue(osDWord));
loader_info_list.push_back(LoaderInfo(file_crc_entry_, file_crc_size_));
file_crc_size_entry_ = file_crc_entry_ ? AddCommand(icDword, 0) : NULL;
if (file_crc_size_entry_)
file_crc_size_entry_->include_option(roCreateNewBlock);
// create header and loader CRC info for loader
index = count();
if (((ctx.options.flags | ctx.options.sdk_flags) & cpMemoryProtection) || (ctx.options.flags & cpLoaderCRC)) {
// calc CRC blocks count
k = 20 + new_import_list.count();
if ((ctx.options.flags & cpStripFixups) == 0) {
PEFixupList *fixup_list = file->pe()->fixup_list();
for (i = 0; i < fixup_list->count(); i++) {
PEFixup *fixup = fixup_list->item(i);
if (!fixup->is_deleted())
k++;
}
}
for (i = 0; i < k; i++) {
AddCommand(icDword, 0);
AddCommand(icDword, 0);
AddCommand(icDword, 0);
}
}
loader_crc_entry_ = (count() == index) ? NULL : item(index);
if (loader_crc_entry_)
loader_crc_entry_->include_option(roCreateNewBlock);
loader_crc_size_ = static_cast<uint32_t>((count() - index) * OperandSizeToValue(osDWord));
loader_info_list.push_back(LoaderInfo(loader_crc_entry_, loader_crc_size_));
loader_crc_size_entry_ = loader_crc_entry_ ? AddCommand(icDword, 0) : NULL;
if (loader_crc_size_entry_)
loader_crc_size_entry_->include_option(roCreateNewBlock);
loader_crc_hash_entry_ = loader_crc_entry_ ? AddCommand(icDword, 0) : NULL;
if (loader_crc_hash_entry_)
loader_crc_hash_entry_->include_option(roCreateNewBlock);
// create section info for loader
index = count();
for (i = 0; i < writable_section_list.size(); i++) {
segment = writable_section_list[i];
AddCommand(icDword, segment->address() - file->image_base());
AddCommand(icDword, segment->size());
AddCommand(icDword, segment->flags());
}
command = (count() == index) ? NULL : item(index);
if (command)
command->include_option(roCreateNewBlock);
loader_info_list.push_back(LoaderInfo(command, (count() - index) * OperandSizeToValue(osDWord)));
// create fixup info for loader
if (loader_fixup_list.count() > 0) {
Data data;
loader_fixup_list.WriteToData(data, file->image_base());
command = AddCommand(data);
} else {
command = NULL;
}
if (command)
command->include_option(roCreateNewBlock);
loader_info_list.push_back(LoaderInfo(command, (command) ? command->dump_size() : 0));
// create relocation info for loader
loader_info_list.push_back(LoaderInfo(NULL, 0));
// create IAT info for loader
index = count();
for (i = 0, import_index = 0; i < new_import_list.count(); i++) {
import = new_import_list.item(i);
if (import->count() == 0)
continue;
import_function = import->item(0);
command = AddCommand(icDword, 0);
link = command->AddLink(0, ltOffset, import_function_info_list[import_index].thunk);
link->set_sub_value(file->image_base());
AddCommand(icDword, import_function->address() - file->image_base());
AddCommand(icDword, import->count() * OperandSizeToValue(file->pe()->cpu_address_size()));
import_index += import->count();
}
command = (count() == index) ? NULL : item(index);
if (command)
command->include_option(roCreateNewBlock);
loader_info_list.push_back(LoaderInfo(command, (count() - index) * OperandSizeToValue(osDWord)));
index = count();
if (il_import) {
std::vector<ImportDelegateInfo> info_list = il_import->info_list();
for (i = 0; i < info_list.size(); i++) {
ImportDelegateInfo info = info_list[i];
AddCommand(icDword, info.method->id() ^ string_key);
AddCommand(icDword, info.invoke->id() ^ string_key);
AddCommand(icDword, info.call_type);
}
}
command = (count() == index) ? NULL : item(index);
if (command)
command->include_option(roCreateNewBlock);
loader_info_list.push_back(LoaderInfo(command, (count() - index) * OperandSizeToValue(osDWord)));
loader_info_list.push_back(LoaderInfo(NULL, 0));
// create memory CRC info for loader
if (il_crc) {
command = il_crc->table_entry();
i = static_cast<size_t>(il_crc->size_entry()->operand_value());
} else {
command = NULL;
i = 0;
}
loader_info_list.push_back(LoaderInfo(command, i));
// create strings for loader
std::map<uint32_t, ILCommand *> loader_string_list;
loader_string_list[FACE_FILE_CORRUPTED] = AddCommand(EncryptString((ctx.options.flags & cpMemoryProtection) ? os::FromUTF8(ctx.options.messages[MESSAGE_FILE_CORRUPTED]).c_str() : os::unicode_string().c_str(), string_key));
loader_string_list[FACE_DEBUGGER_FOUND] = AddCommand(EncryptString(os::FromUTF8(ctx.options.messages[MESSAGE_DEBUGGER_FOUND]).c_str(), string_key));
loader_string_list[FACE_VIRTUAL_MACHINE_FOUND] = AddCommand(EncryptString(os::FromUTF8(ctx.options.messages[MESSAGE_VIRTUAL_MACHINE_FOUND]).c_str(), string_key));
loader_string_list[FACE_INITIALIZATION_ERROR] = AddCommand(EncryptString(os::FromUTF8("Initialization error {0}").c_str(), string_key));
VMProtectBeginVirtualization("Loader Strings");
loader_string_list[FACE_UNREGISTERED_VERSION] = AddCommand(EncryptString(
#ifdef DEMO
true
#else
(ctx.options.flags & cpUnregisteredVersion)
#endif
? os::FromUTF8(VMProtectDecryptStringA("This application is protected with unregistered version of VMProtect.")).c_str() : os::unicode_string().c_str(), string_key));
VMProtectEnd();
loader_string_list[FACE_NTDLL_NAME] = AddCommand(EncryptString("ntdll.dll", string_key));
loader_string_list[FACE_NT_SET_INFORMATION_THREAD_NAME] = AddCommand(EncryptString("NtSetInformationThread", string_key));
loader_string_list[FACE_NT_QUERY_INFORMATION_PROCESS_NAME] = AddCommand(EncryptString("NtQueryInformationProcess", string_key));
loader_string_list[FACE_NT_CLOSE] = AddCommand(EncryptString("NtClose", string_key));
loader_string_list[FACE_NT_VIRTUAL_PROTECT_NAME] = AddCommand(EncryptString("NtProtectVirtualMemory", string_key));
loader_string_list[FACE_NT_ALLOCATE_VIRTUAL_MEMORY_NAME] = AddCommand(EncryptString("NtAllocateVirtualMemory", string_key));
loader_string_list[FACE_NT_FREE_VIRTUAL_MEMORY_NAME] = AddCommand(EncryptString("NtFreeVirtualMemory", string_key));
loader_string_list[FACE_NT_UNMAP_VIEW_OF_SECTION] = AddCommand(EncryptString("NtUnmapViewOfSection", string_key));
loader_string_list[FACE_NT_OPEN_FILE_NAME] = AddCommand(EncryptString("NtOpenFile", string_key));
loader_string_list[FACE_NT_RAISE_HARD_ERROR_NAME] = AddCommand(EncryptString("NtRaiseHardError", string_key));
loader_string_list[FACE_NT_CREATE_SECTION_NAME] = AddCommand(EncryptString("NtCreateSection", string_key));
loader_string_list[FACE_KERNEL32_NAME] = AddCommand(EncryptString("kernel32.dll", string_key));
loader_string_list[FACE_ENUM_SYSTEM_FIRMWARE_NAME] = AddCommand(EncryptString("EnumSystemFirmwareTables", string_key));
loader_string_list[FACE_GET_SYSTEM_FIRMWARE_NAME] = AddCommand(EncryptString("GetSystemFirmwareTable", string_key));
loader_string_list[FACE_NT_MAP_VIEW_OF_SECTION] = AddCommand(EncryptString("MapViewOfFile", string_key));
for (std::map<uint32_t, ILCommand *>::const_iterator it = loader_string_list.begin(); it != loader_string_list.end(); it++) {
it->second->include_option(roCreateNewBlock);
}
// append loader
old_count = count();
runtime_function_list = reinterpret_cast<ILFunctionList *>(runtime->function_list());
for (size_t n = 0; n < 2; n++) {
for (i = 0; i < runtime_function_list->count(); i++) {
func = runtime_function_list->item(i);
if (func->tag() != ftLoader)
continue;
if (func->compilation_type() == ctMutation) {
if (n != 0)
continue;
} else {
if (n != 1)
continue;
}
func->Init(ctx);
size_t orig_function_info_count = function_info_list()->count();
for (j = 0; j < func->function_info_list()->count(); j++) {
FunctionInfo *info = func->function_info_list()->item(j);
function_info_list()->AddObject(info->Clone(function_info_list()));
}
for (j = 0; j < func->range_list()->count(); j++) {
AddressRange *range = func->range_list()->item(j);
range_list()->AddObject(range->Clone(range_list()));
}
for (j = 0; j < func->count(); j++) {
src_command = func->item(j);
dst_command = src_command->Clone(this);
AddressRange *address_range = src_command->address_range();
if (address_range) {
FunctionInfo *info = function_info_list()->item(orig_function_info_count + func->function_info_list()->IndexOf(address_range->owner()));
dst_command->set_address_range(info->item(address_range->owner()->IndexOf(address_range)));
}
AddObject(dst_command);
src_link = src_command->link();
if (src_link) {
dst_link = src_link->Clone(link_list());
dst_link->set_from_command(dst_command);
link_list()->AddObject(dst_link);
if (src_link->parent_command())
dst_link->set_parent_command(GetCommandByAddress(src_link->parent_command()->address()));
if (src_link->base_function_info())
dst_link->set_base_function_info(function_info_list()->GetItemByAddress(src_link->base_function_info()->begin()));
}
command = dst_command;
if (command->type() == icLdc_i4) {
uint32_t value = static_cast<uint32_t>(command->operand_value());
if ((value & 0xFFFF0000) == 0xFACE0000) {
switch (value) {
case FACE_LOADER_OPTIONS:
value = 0;
if (ctx.options.flags & cpMemoryProtection)
value |= LOADER_OPTION_CHECK_PATCH;
if (ctx.options.flags & cpCheckDebugger)
value |= LOADER_OPTION_CHECK_DEBUGGER;
if (ctx.options.flags & cpCheckKernelDebugger)
value |= LOADER_OPTION_CHECK_KERNEL_DEBUGGER;
if (ctx.options.flags & cpCheckVirtualMachine)
value |= LOADER_OPTION_CHECK_VIRTUAL_MACHINE;
command->set_operand_value(0, value);
command->CompileToNative();
break;
case FACE_STRING_DECRYPT_KEY:
command->set_operand_value(0, string_key);
command->CompileToNative();
break;
case FACE_PACKER_INFO:
case FACE_FILE_CRC_INFO:
case FACE_LOADER_CRC_INFO:
case FACE_SECTION_INFO:
case FACE_FIXUP_INFO:
case FACE_RELOCATION_INFO:
case FACE_IAT_INFO:
case FACE_IMPORT_INFO:
case FACE_INTERNAL_IMPORT_INFO:
case FACE_MEMORY_CRC_INFO:
case FACE_DELAY_IMPORT_INFO:
dst_command = loader_info_list[(value & 0xff) >> 1].data;
if (dst_command) {
link = command->AddLink(0, ltOffset, dst_command);
link->set_sub_value(file->image_base());
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_PACKER_INFO_SIZE:
case FACE_SECTION_INFO_SIZE:
case FACE_FIXUP_INFO_SIZE:
case FACE_RELOCATION_INFO_SIZE:
case FACE_IAT_INFO_SIZE:
case FACE_IMPORT_INFO_SIZE:
case FACE_INTERNAL_IMPORT_INFO_SIZE:
case FACE_MEMORY_CRC_INFO_SIZE:
case FACE_DELAY_IMPORT_INFO_SIZE:
command->set_operand_value(0, loader_info_list[(value & 0xff) >> 1].size);
command->CompileToNative();
break;
case FACE_LOADER_CRC_INFO_SIZE:
if (loader_crc_size_entry_) {
link = command->AddLink(0, ltOffset, loader_crc_size_entry_);
link->set_sub_value(file->image_base());
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_LOADER_CRC_INFO_HASH:
if (loader_crc_hash_entry_) {
link = command->AddLink(0, ltOffset, loader_crc_hash_entry_);
link->set_sub_value(file->image_base());
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_FILE_CRC_INFO_SIZE:
if (file_crc_size_entry_) {
link = command->AddLink(0, ltOffset, file_crc_size_entry_);
link->set_sub_value(file->image_base());
} else {
command->set_operand_value(0, 0);
command->CompileToNative();
}
break;
case FACE_MEMORY_CRC_INFO_HASH:
command->set_operand_value(0, il_crc ? il_crc->hash_entry()->operand_value() : 0);
command->CompileToNative();
break;
case FACE_CRC_INFO_SALT:
command->set_operand_value(0, file->function_list()->crc_cryptor()->item(0)->value());
command->CompileToNative();
break;
case FACE_FILE_BASE:
command->Init(icLdc_i8, file->image_base());
command->CompileToNative();
break;
case FACE_TLS_INDEX_INFO:
command->set_operand_value(0, tls_index_address ? tls_index_address - file->image_base() : 0);
command->CompileToNative();
break;
case FACE_VAR_IS_PATCH_DETECTED:
case FACE_VAR_IS_DEBUGGER_DETECTED:
case FACE_VAR_LOADER_CRC_INFO:
case FACE_VAR_LOADER_CRC_INFO_SIZE:
case FACE_VAR_LOADER_CRC_INFO_HASH:
case FACE_VAR_SESSION_KEY:
case FACE_VAR_LOADER_STATUS:
case FACE_VAR_SERVER_DATE:
command->set_operand_value(j, ctx.runtime_var_index[(value & 0xff) >> 4]);
command->CompileToNative();
break;
case FACE_VAR_IS_PATCH_DETECTED_SALT:
case FACE_VAR_IS_DEBUGGER_DETECTED_SALT:
case FACE_VAR_LOADER_CRC_INFO_SALT:
case FACE_VAR_LOADER_CRC_INFO_SIZE_SALT:
case FACE_VAR_LOADER_CRC_INFO_HASH_SALT:
case FACE_VAR_SERVER_DATE_SALT:
command->set_operand_value(j, ctx.runtime_var_salt[value & 0xff]);
command->CompileToNative();
break;
default:
std::map<uint32_t, ILCommand *>::const_iterator it = loader_string_list.find(value);
if (it != loader_string_list.end()) {
link = command->AddLink(0, ltOffset, it->second);
link->set_sub_value(file->image_base());
} else {
throw std::runtime_error(string_format("Unknown loader string: %X", value));
}
}
}
}
}
}
if (n == 0) {
// create native blocks
for (j = 0; j < count(); j++) {
item(j)->include_option(roNoProgress);
}
CompileToNative(ctx);
for (j = 0; j < count(); j++) {
item(j)->exclude_option(roNoProgress);
}
}
}
for (i = 0; i < function_info_list()->count(); i++) {
FunctionInfo *info = function_info_list()->item(i);
if (info->entry())
info->set_entry(GetCommandByAddress(info->entry()->address()));
if (info->data_entry())
info->set_data_entry(GetCommandByAddress(info->data_entry()->address()));
for (j = 0; j < info->count(); j++) {
AddressRange *dest = info->item(j);
for (k = 0; k < range_list()->count(); k++) {
AddressRange *range = range_list()->item(k);
if (range->begin() <= dest->begin() && range->end() > dest->begin())
dest->AddLink(range);
}
}
}
for (i = 0; i < range_list()->count(); i++) {
AddressRange *range = range_list()->item(i);
if (range->begin_entry())
range->set_begin_entry(GetCommandByAddress(range->begin_entry()->address()));
if (range->end_entry())
range->set_end_entry(GetCommandByAddress(range->end_entry()->address()));
if (range->size_entry())
range->set_size_entry(GetCommandByAddress(range->size_entry()->address()));
}
for (i = old_count; i < count(); i++) {
command = item(i);
dst_link = command->link();
if (dst_link) {
if (dst_link->to_address())
dst_link->set_to_command(GetCommandByAddress(dst_link->to_address()));
}
}
// create code for <Module>::.cctor()
old_count = count();
command = AddCommand(icComment, 0);
Data data;
data.PushByte(0);
command->set_dump(data.data(), data.size());
command->set_alignment(4);
command = AddCommand(icCall, 0);
command->AddLink(0, ltCall, GetCommandByAddress(runtime->export_list()->GetAddressByType(atSetupImage)));
// search user <Module>::.cctor
ILMethodDef *module_cctor = runtime->entry_point_method();
ILTypeDef *module = module_cctor->declaring_type();
ILTable *table = file->command_list()->table(ttMethodDef);
for (i = table->IndexOf(module->method_list()); i < table->count(); i++) {
ILMethodDef *method = reinterpret_cast<ILMethodDef *>(table->item(i));
if (method->declaring_type() != module || method == module_cctor)
break;
if (method->name() == ".cctor") {
method->set_flags((CorMethodAttr)(method->flags() & ~(mdSpecialName | mdRTSpecialName)));
file->RenameToken(method);
command = AddCommand(icCall, method->id());
break;
}
}
AddCommand(icRet, 0);
FunctionInfo *info = function_info_list()->Add(0, 0, btValue, 1, 0, 0, file->runtime_function_list()->Add(0, 0, 0, runtime->entry_point_method()), item(old_count));
AddressRange *range = info->Add(0, 0, NULL, NULL, NULL);
for (i = old_count; i < count(); i++) {
item(i)->set_address_range(range);
}
for (i = 0; i < count(); i++) {
command = item(i);
command->CompileToNative();
}
// prepare ranges
std::vector<IFunction *> function_list = ctx.file->function_list()->processor_list();
function_list.push_back(this);
for (i = 0; i < function_list.size(); i++) {
ILFunction *func = reinterpret_cast<ILFunction *>(function_list[i]);
func->range_list()->Prepare();
func->function_info_list()->Prepare();
}
PrepareExtCommands(ctx);
for (i = 0; i < link_list()->count(); i++) {
CommandLink *link = link_list()->item(i);
link->from_command()->PrepareLink(ctx);
}
return true;
}
bool NETLoader::Compile(const CompileContext &ctx)
{
size_t i, j;
std::vector<CommandBlock*> block_list;
ILCommand *command;
std::vector<IFunction *> function_list = ctx.file->function_list()->processor_list();
for (i = 0; i < function_list.size(); i++) {
function_list[i]->set_need_compile(true);
}
function_list.push_back(this);
// update tokens
for (i = 0; i < function_list.size(); i++) {
ILFunction *func = reinterpret_cast<ILFunction *>(function_list[i]);
for (j = 0; j < func->count(); j++) {
command = func->item(j);
TokenReference *token_reference = command->token_reference();
if (!token_reference)
continue;
uint32_t id = token_reference->owner()->owner()->id();
if (command->type() == icComment) {
Data data;
data.PushDWord(id);
command->set_dump(data.data(), data.size());
}
else {
command->set_operand_value(0, id);
command->CompileToNative();
}
}
}
if (!ILFunction::Compile(ctx))
return false;
ILFunction::AfterCompile(ctx);
for (i = 0; i < count(); i++) {
command = item(i);
if (!command->block() || (command->block()->type() & mtExecutable) == 0 || !command->token_reference())
continue;
ILToken *token = command->token_reference()->owner()->owner();
if (token->type() == ttStandAloneSig) {
ILTable *table = token->owner();
uint32_t id = token->type() | 1;
for (j = 0; j < table->count(); j++) {
ILToken *tmp = table->item(j);
if (tmp == token)
break;
if (!tmp->is_deleted())
id++;
}
if (token->id() != id) {
token->set_value(TOKEN_VALUE(id));
command->set_operand_value(0, id);
command->CompileToNative();
}
}
}
for (i = 0; i < function_list.size(); i++) {
IFunction *func = function_list[i];
for (j = 0; j < func->block_list()->count(); j++) {
block_list.push_back(func->block_list()->item(j));
}
}
for (i = 0; i < block_list.size(); i++) {
std::swap(block_list[i], block_list[rand() % block_list.size()]);
}
if (ctx.file->runtime_function_list() && ctx.file->runtime_function_list()->count()) {
// sort blocks by address range
std::sort(block_list.begin(), block_list.end(), CommandBlockListCompareHelper());
}
for (i = 0; i < block_list.size(); i++) {
block_list[i]->Compile(*ctx.manager);
}
for (i = 0; i < function_list.size(); i++) {
function_list[i]->CompileInfo(ctx);
}
for (i = 0; i < function_list.size(); i++) {
function_list[i]->CompileLinks(ctx);
}
return true;
}
size_t NETLoader::WriteToFile(IArchitecture &file)
{
size_t res = BaseFunction::WriteToFile(file);
if (pe_entry_) {
PEArchitecture *pe = reinterpret_cast<NETArchitecture &>(file).pe();
if (pe->cpu_address_size() == osDWord) {
IFixup *fixup = pe->fixup_list()->AddDefault(osDWord, true);
fixup->set_address(pe_entry_->address() + pe_entry_->dump_size());
}
}
return res;
}