java opcode 反汇编,如何将VM的opcode嵌入汇编源码中

这次来一个关于VM的混淆办法,可能只是个小trick,仅仅来自胡思乱想

也许你会觉得很奇怪,一个VM能有啥新鲜的,对,单纯来说VM保护源代码已经非常的成熟了,所以在这里只做最基本的介绍,而且这次的重点不在于VM。

VM就是Virtual Machine(虚拟机器),这里和vmware不是一样的,这里指的是自己写的引擎和opcode类型,比如Java就是基于JVM的语言,通过将代码编译成字节码,再运行在JVM上。

所以我们对于一个加密算法可以先翻译成opcode,再运行在我们自己写的引擎上,这样就大大增加逆向成本。

d529f78e0dc2d6f47bc91bb4d15e7379.png                上图就是一个典型的VM执行引擎结构图(有点像控制流平坦化 ).....

再来个源码样式:

3e6116c64aeb7686a5d2edb458bc3896.png

那么介绍完了VM,我们了解到VM保护的逆向一般都得先找到opcode,再分析引擎,再写parser,再人肉看,我们从后三个方向都容易让人恶心,这里介绍一种从Opcode这里恶心人的操作。

我们知道CTF的VM一般将opcode单独放到一个字节数组里,有一个指针慢慢的扫描过去来进行解释执行,完全是虚拟化的环境,但是有没有想过,这里就导致了opcode极其容易dump出来,要是能够让opcode和汇编指令混在一起就好了。

所以这里说的就是如何将Opcode和汇编指令放在一起,混杂着执行就很恶心了。

首先,你的静态分析会炸掉,因为混杂了不可反汇编的指令,可能会因为上下文使得这一块的代码和数据类型区分不开甚至指令分析错误,F5铁定是废了,再者,Opcode提取比较困难,你可能得手工提取那些穿插着Opcode。

其实还有一种好处,涉及到原理才能知道为什么了。

这里我们使用windows下的环境,使用的是windows 强大的调试API,具体的实现逻辑是:运行时碰到非法指令,会触发异常,通过调试器调试子进程截获异常,获取上下文进行操作

所以我们得先创造子进程,子进程被调试,本身则相当于系统处理不了的字节码的解释器(有点类似linux下基于信号的VM,大佬敏锐的见解)

STARTUPINFO si;

memset(&si,0,sizeof(si));

si.cb=sizeof(si);

PROCESS_INFORMATION pi;

HMODULE h=GetModuleHandleW(0);

CHAR Filename[260];

GetModuleFileNameA(h,(LPSTR)&Filename,260);

BOOL result=CreateProcessA(NULL,(LPSTR)&Filename,NULL,NULL,FALSE,DEBUG_PROCESS | DEBUG_ONLY_THIS_PROCESS,NULL,NULL,(LPSTARTUPINFOA)&si,&pi);

这里就是创建子进程的代码,而且是以调试模式启动创建,所以会拥有许多权限。

这里实现VM最重要的API就是SetThreadContext,因为可以实现寄存器等的获取与修改,还有一个就是ReadProcessMemory和WriteProcessMemory,因为需要对内存进行读写。

最后再来个while(true)里面随时截获异常消息进行处理,处理完之后再SetThreadContext将目前执行完的寄存器的值设置。

还有一点细节,就是写个IsDebuggerPresent来判断自己是被调试的还是父进程,切勿陷入死循环,疯狂CreateProcess,那CPU直接暴毙。

void CreateSubProcess()

{

STARTUPINFO si;

memset(&si,0,sizeof(si));

si.cb=sizeof(si);

PROCESS_INFORMATION pi;

HMODULE h=GetModuleHandleW(0);

CHAR Filename[260];

GetModuleFileNameA(h,(LPSTR)&Filename,260);

BOOL result=CreateProcessA(NULL,(LPSTR)&Filename,NULL,NULL,FALSE,DEBUG_PROCESS | DEBUG_ONLY_THIS_PROCESS,NULL,NULL,(LPSTARTUPINFOA)&si,&pi);

VMState vms;

if(result)

{

//ShowWindow(GetConsoleWindow(),SW_HIDE);

DEBUG_EVENT de;

while(WaitForDebugEvent(&de,INFINITE)!=0)

{

if(de.dwDebugEventCode==EXCEPTION_DEBUG_EVENT)

{

EXCEPTION_DEBUG_INFO di=de.u.Exception;

if(di.ExceptionRecord.ExceptionCode==EXCEPTION_ILLEGAL_INSTRUCTION)

{

CONTEXT context;

memset(&context,0,sizeof(context));

context.ContextFlags=CONTEXT_FULL;

GetThreadContext(pi.hThread,&context);

if(ReadRemoteByte(pi.hProcess,context.Eip)==0xC7)

{

vms.context=&context;

vms.pi=π

RunVM(vms);

SetThreadContext(pi.hThread,&context);

}

else

{

TerminateProcess(pi.hProcess,0);

exit(0);

}

}

}

if(de.dwDebugEventCode==EXIT_PROCESS_DEBUG_EVENT)

exit(0);

ContinueDebugEvent(de.dwProcessId,de.dwThreadId,DBG_CONTINUE);

}

}

return;

}

所以主要代码就在这里了,然后完善下RunVM函数就可以了。如下就是完整代码,有个Opcode.h没给出不过问题不大,我们不在于写VM而在于怎么实现opcode和asm混杂

还有一个重要细节,就是每个VM的opcode之前必须要有个前缀,而且这个前缀必须不能对应任何的汇编指令,不然就不会触发非法指令的异常,最后的EIP设置也记得要跳过最开头这个字节,在代码中我选取的是0xC7

// SEHVM.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。

//

#include

#include

#include

#include"Opcodes.h"

using namespace std;

struct VMState

{

PCONTEXT context;

PPROCESS_INFORMATION pi;

};

BYTE ReadRemoteByte(HANDLE hProcess,int addr)

{

BYTE data;

ReadProcessMemory(hProcess,(LPVOID)addr,&data,1,NULL);

return data;

}

DWORD ReadRemoteDword(HANDLE hProcess,int addr)

{

int data;

ReadProcessMemory(hProcess,(LPVOID)addr,&data,4,NULL);

return data;

}

void WriteRemoteDword(HANDLE hProcess,int addr,DWORD data)

{

WriteProcessMemory(hProcess,(LPVOID)addr,(LPVOID)&data,4,NULL);

}

void pop(VMState vms,DWORD *val)

{

*val=ReadRemoteDword(vms.pi->hProcess,vms.context->Esp);

vms.context->Esp+=4;

}

void push(VMState vms,int val)

{

vms.context->Esp-=4;

WriteRemoteDword(vms.pi->hProcess,vms.context->Esp,val);

}

DWORD *getReg(VMState vms,BYTE x)

{

if(x==Reg1)

return &vms.context->Eax;

if(x==Reg2)

return &vms.context->Ebx;

if(x==Flag)

return &vms.context->Ecx;

if(x==LoopReg)

return &vms.context->Edx;

printf("Unknow reg id: 0x%X\n",x);

exit(0);

}

void RunVM(VMState vms)

{

DWORD *eip=&(vms.context->Eip);

BYTE op=ReadRemoteByte(vms.pi->hProcess,*eip+1);

int arg1;

BYTE arg2;

switch(op)

{

case PushImm:

arg1=ReadRemoteDword(vms.pi->hProcess,*eip+2);

push(vms,arg1);

vms.context->Eip+=6;

break;

case PopReg:

arg2=ReadRemoteByte(vms.pi->hProcess,*eip+2);

pop(vms,getReg(vms,arg2));

vms.context->Eip+=3;

break;

default:

TerminateProcess(vms.pi->hProcess,0);

exit(0);

break;

}

}

void CreateSubProcess()

{

STARTUPINFO si;

memset(&si,0,sizeof(si));

si.cb=sizeof(si);

PROCESS_INFORMATION pi;

HMODULE h=GetModuleHandleW(0);

CHAR Filename[260];

GetModuleFileNameA(h,(LPSTR)&Filename,260);

BOOL result=CreateProcessA(NULL,(LPSTR)&Filename,NULL,NULL,FALSE,DEBUG_PROCESS | DEBUG_ONLY_THIS_PROCESS,NULL,NULL,(LPSTARTUPINFOA)&si,&pi);

VMState vms;

if(result)

{

//ShowWindow(GetConsoleWindow(),SW_HIDE);

DEBUG_EVENT de;

while(WaitForDebugEvent(&de,INFINITE)!=0)

{

if(de.dwDebugEventCode==EXCEPTION_DEBUG_EVENT)

{

EXCEPTION_DEBUG_INFO di=de.u.Exception;

if(di.ExceptionRecord.ExceptionCode==EXCEPTION_ILLEGAL_INSTRUCTION)

{

CONTEXT context;

memset(&context,0,sizeof(context));

context.ContextFlags=CONTEXT_FULL;

GetThreadContext(pi.hThread,&context);

if(ReadRemoteByte(pi.hProcess,context.Eip)==0xC7)

{

vms.context=&context;

vms.pi=π

RunVM(vms);

SetThreadContext(pi.hThread,&context);

}

else

{

TerminateProcess(pi.hProcess,0);

exit(0);

}

}

}

if(de.dwDebugEventCode==EXIT_PROCESS_DEBUG_EVENT)

exit(0);

ContinueDebugEvent(de.dwProcessId,de.dwThreadId,DBG_CONTINUE);

}

}

return;

}

int main()

{

if(!IsDebuggerPresent())

{

CreateSubProcess();

return 0;

}

printf("I am sub process\n");

int t=0;

_asm

{

_emit 0xC7 //pushImm 0xDDCCBBAA

_emit 0x17

_emit 0xAA

_emit 0xBB

_emit 0xCC

_emit 0xDD

_emit 0xC7 //pushImm 0xEECCBBAA

_emit 0x17

_emit 0xAA

_emit 0xBB

_emit 0xCC

_emit 0xEE

pop eax

pop ebx

mov t,eax

}

printf("the value of eax: %X\n",t); //eax=0xEECCBBAA

MessageBoxA(NULL,"GOT","YES",MB_OK);

return 0;

}

这里的VM只实现了最简单的两条指令,所以等着大佬们完善,这里主要是阐述一种操作,只是个雏形罢了。

最后再说一下在反调试处的作用,由于程序分为了调试进程与非调试进程,那么当你用ida等调试器执行时则会使得程序误认为自己为子进程。

就会开始按正常操作执行代码,当执行到那个无法识别的指令时,则会触发异常,传递给ida,然而ida并不是flag父进程,不知道如何处理这些opcode

所以则会执行失败,使得做题人无法理解发生了啥,所以一定要将创建子进程和VM处理代码藏起了,才可以发挥最大效果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/442254.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

php ip 合法,什么是合法ip地址

合法的IP地址中,每个三位数都是在0~254之间的,不可能是大于254就连255都不行。这才是合法的IP地址,还有 IP地址有A\B\C类IP。iPv4的ip地址都是(1~255).(0~255).(0~255).(0~255)的格式。A类的IP地址范围为0.0.0.0-127.255.255.255B类的IP地址范…

php session和cookie区别,php中session和cookie的区别是什么?

一、Session(1)Session的由来以及介绍Session:在计算机中,尤其是在网络应用中,称为“会话控制”,生存时间为用户在浏览某个网站时,从进入网站到关闭这个网站所经过的这段时间,也就是用户浏览这个网站所花费的时间。由于Http是一种…

matlab 泡泡图,使用matlab绘制2维、3维气泡图

在学习模糊c均值聚类时,突然想到能否将每个样本对所属簇的奴属度(C)用气泡图的形式表示出来,这样就可以在一张图上同时获得分类与奴属度(C)两类信息。在matlab中没有绘制气泡图的专用函数,不过可以通过手动设置参数,来最终达到气泡…

用vscode可以开发php,【编程开发工具】vscode能够编写php吗

Visual Studio Code一个轻量且壮大的代码编辑器,支撑Windows,OS X和Linux。内置JavaScript、TypeScript和Node.js支撑,而且具有雄厚的插件生态系统,可通过装置插件来支撑C、C#、Python、PHP等其他言语。Visual Studio Code设置php…

ant混淆编译java web,Android中使用ant混淆编译

搞了好几天,查看了上百个网站,最后摸索出一套很简单的ant混淆编译的方法。下面开始:1.拿一个普通项目来说,首先为它加上ant编译功能。android update project --name project_name -t 3 -p D:/temp/project_name此时会在项目根目录…

matlab漂亮图表,漂亮,美观的图表之Matlab强势回归~~~~走你8

今天来画3D plot% Load the dataload latticeExamplexx x(:);yy y(:);zz z(:);% locate the non-zero pointsa find(T~0);% plot the non-zero points using a scatter plot% use the values of T to represent both color and size of symbolsscatter3(xx(a),yy(a),zz(a),…

100转换成二进制 java,一段简单的java代码,十进制转二进制

一段简单的java代码,十进制转二进制mip版 关注:188 答案:5 悬赏:40解决时间 2021-01-23 23:14已解决2021-01-23 05:43代码如下,希望可以帮我说明白点这段代码。factorOfTwo到底是指的什么?这段代码是如何运行的?多谢int number…

linux 挂在iso文件,linux 挂载iso文件安装文件 与 网络yum的搭建

挂载1.首先,创建文件作为挂载点mkdir /文件名/2.mount /iso/rhel-server-7.3-x86_64-dvd.iso /dir/将/iso目录下的镜像挂载到 /dir/文件上3. cd /etc/yum.repos.d/ 进入文件4. rm -fr * 删除目录下所有文件5.vim yum.repo 编辑文件内容6. rhe17.3] 说明namerhe17.3 …

微软引入linux内核,微软内部已在讨论Linux内核中加入exFAT的可能性

微软的exFAT文件系统已经有十多年的历史了,虽然已经有了一些突破,但是主流Linux内核仍然不支持此文件系统,即使它出现在更多的SD卡和其他设备上。但现在又重新努力将exFAT驱动程序引入Linux内核,有一个开发人员对为exFAT添加一个新…

linux内核ddr初始化,X-007-UBOOT-DDR的初始化(Bubblegum-96平台)

X-007-UBOOT-DDR的初始化(Bubblegum-96平台)作者:wowo 发布于:2016-7-21 22:47分类:X Project1. 前言到目前为止,“X Project”在Bubblegum-96平台上的代码,都是运行在SRAM中。由于SRAM的size很小(最多也就96KB)&#…

linux 4.4内核特性,Linux Kernel 4.4.19 LTS长期支持版发布

原标题:Linux Kernel 4.4.19 LTS长期支持版发布摘要:近日,内核开发者Greg Kroah-Hartman公布了长期支持的Linux 4.4 Kernel系列第19个维护版本的细节。Linux 4.4是目前最新的LTS内核分支,被Arch Linux, Solus和Ubuntu Linux等众多…

linux 软件源 概念,Linux 软件源介绍

陈科肇1.软件源简介软件源是Linux系统免费的应用程序安装仓库,很多的应用软件都会这收录到这个仓库里面,按类型分则有:1).软件仓库:各类软件的二进制包和源代码2).ISO镜像:发行版的ISO文件软件源可以是网络服务器&…

c语言汇编混编,c语言与汇编混编写法

R13 (stack pointer)R14 (Link Register)R15 (program counter)c语言中内嵌汇编的规则:内嵌汇编器与armasm的区别:访问全局变量:unsigned char LDRB/ STRBunsigned short LDRH/STRHunsigned int LDR/…

买iphone不买android,为何宁可用4年前的苹果6s,也不买两三千的安卓呢?理由很真实...

在这个信息时代,几乎每个人都离不开手机,手机渗入到了我们生活的方方面面,不管是社交聊天、还是影音娱乐,甚至是办公出行,没有手机,很多人都会寸步难行。全球的智能手机市场竞争处于白热化的阶段&#xff0…

html5画图作品,8款最佳HTML5绘图工具

HTML5无疑是当前最受宠的一项技术,今天推荐8款HTML5绘图工具,同样惊艳你的眼球!这些绘图工具大多数是用HTML5画布(Canvas)实现的,部分辅以Javascript。对每一个web设计者来说,制图和草绘是他们工作中最喜欢的一件事情了。草绘和制…

vivoiqooz1鸿蒙系统,iQOOZ1评测:vivo新一代性价比神机

【手机中国评测】如果说2019是5G发展的元年,那么2020就是5G技术和终端奋起发力的一年。纵观目前市面上的5G机型我们不难发现,“涨价”已经成为了2020年的新关键词。技术成本的提升,让5G手机似乎失去了我们传统意义上的性价比。但这世间从来不…

qt解析html的数据,windows下用QTwebkit解析html

环境windows7 VS2010 QT5.2_opengl配置开发环境1、安装VS20102、安装QT 5.23、安装Visual Studio Add-in for QT54、配置VS 2010配置路径:QT5 > "Qt Options" > "Qt Versions" > Add默认路径为:C:\Qt\Qt5.2.0\5.2.0\msvc…

计算机网络拓扑图的描述,计算机网络拓扑结构 以下关于星型网络拓扑结构的描述正确的是______。 (多选题 )...

题目和答案在最下面!全文转自百度,自己总结方便自己以后查找!常见类型:星型拓扑总线拓扑▪ 环型拓扑▪ 树型拓扑▪ 混合型拓▪ 网型拓扑开关电源拓扑简单介绍的:星型优点:可靠性高,方便管理&…

考研生物和计算机结合的专业,2020考研:生物医学工程,考研是考原专业还是跨专业考计算机好?...

中公浙江研招网温馨提醒您关注专业解析:【2020考研:考研想考编程专业但没基础,该怎么准备?】2019浙江考研复试群:8663839642020浙江考研交流群:416469560许多考生在报考专业的时候会很茫然,怕考原专业毕业…

vue项目编写html,从头搭建、编写一个VUE项目

一、创建VUE项目1、新建一个vue项目进入工作目录,新建一个vue项目:vue init webpack 项目名vue init webpack vue-project-demoimage.png说明:Vue build > 打包方式,回车即可;Install vue-router > 是否要安装 v…