|
0x00前言
如题所言,本次的文章可能比较长,但是内容比较详细,适合新手学习,但是同样考验耐心。这次的文章严格说来是我上一篇文章遗留下来的历史问题,上一篇文章请参考《【又见AES】第十届全国大学生信息安全大赛之逆向--欢迎来到加基森》,同时这篇文章中的字节码解释器与《【i春秋】第十届全国大学生信息安全大赛之逆向--apk题目》有异曲同工之妙。在文章开始之前,我首先要感谢一下@veritas501大神给的一些灵感。
0x01静态分析
本次要分析的代码段位于0x80495C0,如图是IDA还原出的关键伪C代码
data:image/s3,"s3://crabby-images/1a228/1a228bcef8afeba7d7bbccfca256b3df0157eda0" alt=""
这次我们要分析的就是disasm函数(这是改过的函数名,位于0x80495C0,),在分析这个函数之前,我们需要先分析一下传入的参数v4,图中已高亮显示,很容易看出v4可能是一个长度为4的数组(下标为0的变量存放一个数组的首地址,这个数组是一个指令集,其他3个我们在动态调试中说明)
disasm函数里面是一个有大量分支的switch分支语句,这里截取部分
data:image/s3,"s3://crabby-images/82d53/82d53d0c1ff68711f36ff0e802d4ef64b3ef7ae7" alt=""
0x02动态分析
在这里我是用的是Ubuntu虚拟机(该文件是ELF程序)+IDA,这里输入以0123456789012345为例,直接断点到0x0804964A(call disasm)
data:image/s3,"s3://crabby-images/ae85a/ae85aa8a32e1d3dcdcf9cfa865f2da4d63a2dc28" alt=""
data:image/s3,"s3://crabby-images/21066/21066898425a07852d6243f105f0a74e344cb30e" alt=""
此时传入的参数位于栈顶
data:image/s3,"s3://crabby-images/d12e9/d12e958b1962f0a4350938133ab6c674355dfee6" alt=""
然后在Hex View中查看内存
data:image/s3,"s3://crabby-images/fbcd2/fbcd2323cd20cb33b5786a6fa1a79e42cd4a0932" alt=""
选中的部分就是v4[0],紧跟的三个不为零的数据分别是v4[1],v4[2],v4[3]
v4[1]存放的地址指向(其实也是一个数组)
data:image/s3,"s3://crabby-images/ec79f/ec79fc20bae011f62bf9667c8c295dc0360412dd" alt=""
v4[2]存放的地址指向(虽然现在指向数组尾,但是后面会发现这是一个数据临时存放区)
data:image/s3,"s3://crabby-images/9f01a/9f01afcf5052f9a1295fe70e264345b3404c5c8c" alt=""
v4[3]存放的地址指向(待处理的数据)
data:image/s3,"s3://crabby-images/ff861/ff86165e0c54828f0384a59dd0ff4e867a2a748f" alt=""
v4与其解析为一个数组,不如解析为一个结构体来的简洁明了
[C] 纯文本查看 复制代码struct code_vm{ unsigned int *code_ip; unsigned int *array; unsigned int temp; unsigned int *input;}code_vm;
这时我们把v4转换为code_vm,其中code_ip(等同v4[0])保存指令集的游标,array(等同v4[1])保存一个数组(保存一些数据)的头地址,temp(等同v4[2])保存临时数据,input(等同v4[3])指向待处理的数据头
了解了这些内容,我们开始分析disasm函数
data:image/s3,"s3://crabby-images/8d370/8d370d50fb1f34ced03a0f6d5068af2324d1c1fd" alt="" data:image/s3,"s3://crabby-images/5ce3f/5ce3f764507ce691a12bfff970669f4be3385eaa" alt=""
进入后,获取code_ip游标处的数据(寄存器ebx即为code_ip)
字节码0x64
此时读取1个字节(0x64)后保存到eax中,然后再减去0xC(十进制12),最后我们得到0x64,然后又与0xF3(0xFF-0xC)比较,这个比较并没有什么重要作用,这里我们可以忽略
data:image/s3,"s3://crabby-images/e825c/e825cac0a728fc2ea169b1de56e1717600414ad8" alt=""
向下单步跟进开始跳转
data:image/s3,"s3://crabby-images/cc757/cc7575c45ad846c57a2935ab5110727b74fd4b4d" alt=""
接下来获取code_ip+1和code_ip+2处的数据0和0,由于都是0所以不好辨别
继续单步进入call switch_64(同样段名经过了修改)
data:image/s3,"s3://crabby-images/d21c2/d21c2687bb61df2a7cb6c482791fe34dd27cfa33" alt=""
这里有个判断,大于7的话就会跳走,但是这里一般不会跳走,先不管它(后面讲解)
往下ecx被赋值4
继续单步往下走进入call get_array(同样改过名字)
data:image/s3,"s3://crabby-images/e2487/e2487d96a2c41d8ec3c63376f853281476c45745" alt=""
执行操作ecx=ecx-1,此时ecx=3,然后与3比较,大于3就跳转
紧接着下面是mov eax,[eax+4],在执行之前有
data:image/s3,"s3://crabby-images/e3feb/e3febc61e7e09782ca033cf4b111b532a72e0f69" alt="" data:image/s3,"s3://crabby-images/19150/1915021f6ad487a4edb6e23fd35ba6affa57922d" alt="" data:image/s3,"s3://crabby-images/9c414/9c4144480baaa9226b16c1c1c0f0d7978db9ddd0" alt=""
此时eax指向图2中指向的位置(code_ip),由于该程序为32位,寄存器都是32位的,也就是4个字节,那么eax+4就是图3指向的位置(array),那么等同于eax=array
继续往下,ecx被赋值,但是赋值的结果与ecx有关,我们定位到dword_80BDA30,
data:image/s3,"s3://crabby-images/16ddb/16ddbbe4634aff88f16228d07f546edff8089514" alt=""
这是一个长度为4的数组,所以才有了前面与3比较的过程,是为了检测是否数组越界
这里要使用的是下标为3的数据,也就是0xFFFFFFFF,那么就有ecx=0xFFFFFFFF
紧接着就是一个ADD操作
data:image/s3,"s3://crabby-images/9d728/9d728514e56ca4a7ceff4a78efd7e4380faf968c" alt=""
此时寄存器里的数据如上图,有ecx=ecx and array[edx]即ecx=array[edx]
最后有eax=array[edx],此时段名最好改为get_array,以便识别
data:image/s3,"s3://crabby-images/aa3b0/aa3b094c0591904236e703fe01364ec68e971eb4" alt="" data:image/s3,"s3://crabby-images/f3f84/f3f840b79b21dfa38c0197d4ef8b7feff8ab65a4" alt=""
跳出get_array段后,往下走又是一个get_array(图1)
这时我们注意到有两个与7比较的指令(同样是检测数组越界,这里就可以推断出array的长度为8),一个是esi,另一个是ebp,看我标出的红箭头,esi就是第一个参数*(code_ip+1),ebp是第二个参数*(code_ip+2),由于都为0所以不是很好分辨
第一个get_array的返回值给了ebx
第二个get_array的返回值存放于eax
后面就是ebx=ebx^eax,即ebx=array[*(code_ip+1)]^array[*(code_ip+2)]
往下走,进入call set_array(同样修改了名字)
data:image/s3,"s3://crabby-images/29363/2936350b27575e721ed126c09b786b64d125c5e2" alt="" data:image/s3,"s3://crabby-images/37f65/37f65edeeed4dcf65490ee12e1ca6c02a59fd37e" alt="" data:image/s3,"s3://crabby-images/232f1/232f15f3996916c4f45d610662ac6da97c4f4928" alt=""
这里分别有xor,and,xor,eax异或了两次,所以与eax无关,最后有eax=ecx
edx=array
最后有array[*(code_ip+1)]=array[*(code_ip+1)]^array[*(code_ip+2)]
data:image/s3,"s3://crabby-images/d5a72/d5a72a17782d221a622581a8b3ecb32e5b0d3187" alt=""
code_ip+=3
整理后有
data:image/s3,"s3://crabby-images/cde31/cde31f6b293d1e1a95d956fa307e8aef0ac061eb" alt=""
运行输出和内存数据对比,主要看array,现在对比还不明显
data:image/s3,"s3://crabby-images/01f17/01f17cd9ce5e6f31b0086493fbb4a04409869460" alt="" data:image/s3,"s3://crabby-images/74fd1/74fd100e40477a21c467f20bf4b61ec84d1674bb" alt=""
字节码0xD9
再次读取一个字节0xD9
data:image/s3,"s3://crabby-images/34905/34905d10e809ee12a8c39801a3f9b2aad1798384" alt=""
data:image/s3,"s3://crabby-images/e38ac/e38acb9f930229adc50645984b9b82bbb37682b1" alt="" data:image/s3,"s3://crabby-images/a4241/a42414332547df1a98418a53c664ea1ebb091ae6" alt="" data:image/s3,"s3://crabby-images/6256c/6256c16e645c2934b454bb0f032ad3c1b75f4eb1" alt=""
这个就比较简单了
根据分析有temp=*(code_ip+1)
整理后有
data:image/s3,"s3://crabby-images/01737/0173771bdc1b8854d843d784986a98453f6c3e19" alt=""
程序输出和内存数据对比
data:image/s3,"s3://crabby-images/9c7e0/9c7e0e83a60250c3556243a466768bc662389b1e" alt="" data:image/s3,"s3://crabby-images/be0a5/be0a5f10e4cfafcfc6e9888f3aae47d17fbf0b2a" alt=""
字节码0xBF
再往下走就是0xBF
data:image/s3,"s3://crabby-images/46002/46002ae449a1fe8cea49d3feb801060df2c38064" alt="" data:image/s3,"s3://crabby-images/69b90/69b9041d92dc888e9f21aa5dded90dd41aa75363" alt=""
继续进入call
data:image/s3,"s3://crabby-images/5f868/5f8680d86d98764905f2bdda047bb8cc67f2b153" alt="" data:image/s3,"s3://crabby-images/1c80a/1c80ab468869a80584f9c4d4ec5dbaaeb1f63cb0" alt=""
在这里有edx=*(code_ip+1)
code_ip-=1
ebx=temp
temp=ebx+0x10=temp+0x10
再往下就是set_array
这里不再进入,分析后,有array[edx]=ebx
最后整理后有
data:image/s3,"s3://crabby-images/67bbf/67bbff359fa80980bc498f2634bd811eb65cc23f" alt=""
输出结果与内存数据对比
data:image/s3,"s3://crabby-images/44a21/44a21d1e934c7415460efbbe4ca3fd6284f2e507" alt="" data:image/s3,"s3://crabby-images/98e69/98e69940db9b1359d38bb6e8165e37c5481d3c3c" alt=""
向下走又是0xD9
data:image/s3,"s3://crabby-images/0f4d5/0f4d56675360b31f3fce7e15b96023c1219cda02" alt=""
这里不再赘述,直接给出输出结果和内存数据的对比
data:image/s3,"s3://crabby-images/8c5f7/8c5f7beaf72c04776889230bc5f922e6c333827c" alt="" data:image/s3,"s3://crabby-images/ecab5/ecab5267b493b70a782b76b1ce01c37bd8682051" alt=""
再次走到switch跳转处,0xBF
data:image/s3,"s3://crabby-images/dfe01/dfe014ef8c5b9872e7609dc4cbb5484e341c33a4" alt=""
输出结果和内存数据对比
data:image/s3,"s3://crabby-images/296bd/296bd87a42298ec8c0fc3d44e1f8eab21ccaec50" alt="" data:image/s3,"s3://crabby-images/ba42d/ba42d14c2725691d2320621d9a32e799272b950e" alt=""
字节码0xB7
data:image/s3,"s3://crabby-images/c778d/c778d81af7c42b50ba393f00ed4af7eeefd47d6a" alt=""
再次走到switch跳转处,0xB7
data:image/s3,"s3://crabby-images/74cd6/74cd65acebbed499003aff5c83c6907efce91d3d" alt=""
首先获取了*(code_ip+1),*(code_ip+2)分别为0,2
data:image/s3,"s3://crabby-images/aa86d/aa86dfb5d9bfe3eac21792cabeeea05a2dd3180f" alt=""
一般检测后,调用了get_array
执行后esi=array[*(code_ip+1)]
data:image/s3,"s3://crabby-images/03366/03366c3832f1eb6b5fbabd3df682843232c7069a" alt=""
这里再次调用get_array
执行后eax=array[*(code_ip+2)]
往下走
data:image/s3,"s3://crabby-images/36dbb/36dbbaa7dad0245ecb6b6736b7a722f452bd8eb9" alt=""
在这里,eax如果小于等于esi都会跳走,这里没有跳走
没有跳走直接将array[7]置0
整理后有
data:image/s3,"s3://crabby-images/79c79/79c79086ac473b3c348a7c1a55bb2aaede33a98e" alt=""
将输出结果与内存数据对比
data:image/s3,"s3://crabby-images/7c501/7c501686372ee5cde2652c937c3ebc7ca05dc1ec" alt="" data:image/s3,"s3://crabby-images/36b0b/36b0bcf1cb10a078363699ce86c34cb8644ea5a8" alt=""
字节码0x67
走到switch跳转处,0x67
data:image/s3,"s3://crabby-images/0919f/0919f685767cd320c6871df649be0e8e43a5dbad" alt=""
data:image/s3,"s3://crabby-images/1609b/1609b13969707ff0e84e26eab10891e94c700b8e" alt=""
根据上图有ecx=array[7]
如果ecx==0,就有code_ip+=5
否则会跳走,这里没有跳走,所以我们不再分析跳走的情况
整理后有
data:image/s3,"s3://crabby-images/3c0af/3c0af416fe9bf96b73d30eb88bb5fc76dcaf4a02" alt=""
输出结果与内存数据对比
data:image/s3,"s3://crabby-images/67c67/67c676b5709005d9d7d723ccbf53bcf5d2cebd99" alt="" data:image/s3,"s3://crabby-images/3202d/3202dc7f20f49a0aebaea4d979a3c906e4a964c1" alt=""
字节码0x65
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/ba953/ba95301143b370b5e6655f1dcc2d314f5f89b857" alt=""
走到关键代码处
data:image/s3,"s3://crabby-images/785d3/785d3015c1f0b79fe3991c9a255d80e0906925d8" alt=""
首先eax=*(code_ip+1)=0
进入call
data:image/s3,"s3://crabby-images/c5c41/c5c4199378e19bce6e441e9d19c7d72cc472f8c0" alt=""
这里再次遇到get_array
那么就有eax=array[*(code_ip+1)]
往下走
data:image/s3,"s3://crabby-images/be5a4/be5a4e4b08dd0b5c6394fd3ec4e981f8a89a7827" alt=""
这里是edx=code_ip-1
往下走
data:image/s3,"s3://crabby-images/498e5/498e548958326ef464d78a0ff3f35a0e1ec1a88d" alt=""
在这里ecx=edx
所以最终有*(code_ip+4)=eax=array[*(code_ip+1)]
整理后有
data:image/s3,"s3://crabby-images/2d2d7/2d2d7d9921ed1f28f310b431baef215794bb2625" alt=""
输出结果与内存数据对比
data:image/s3,"s3://crabby-images/a6f45/a6f454c813ce36cb8a7b466a47bd920986039414" alt="" data:image/s3,"s3://crabby-images/9d048/9d048dac3459c4896931b458dbb87d5d2b538b9f" alt=""
字节码0x7E
走到switch跳转处,0x7E
data:image/s3,"s3://crabby-images/5444d/5444d9c1cf8f8e30b533245a5455c97018753f96" alt=""
走到关键代码处
data:image/s3,"s3://crabby-images/1a802/1a8023f7098f70542d174a1f3057f9d22ecbc2e3" alt=""
同样获取*(code_ip+1)和*(code_ip+2)分别是3,0
进入call
data:image/s3,"s3://crabby-images/3dc5c/3dc5c6725d272bd7342730de461485aa8f7d793e" alt=""
在这里,ebx=input[*(code_ip+2)],从这里就开始对input进行操作了
这里同样注意code_ip+=3
下面调用set_array
通过跟进有array[*(code_ip+1)]=input[*(code_ip+2)]
整理后有
data:image/s3,"s3://crabby-images/b105c/b105c1783f9e4d334e7b4e476e4297f30f9578e7" alt=""
对比输出结果与内存数据
data:image/s3,"s3://crabby-images/9fcc0/9fcc018e8d78843a585e9c57d75adf6f07121108" alt="" data:image/s3,"s3://crabby-images/9fd76/9fd76fb3995fe8edc2d0fe6f718bdc03039890f2" alt=""
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/35d95/35d95867c6ab1fbfef7d969d9187ca6d0edb6d01" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/e8add/e8add279a903f8e7a4b8f9e45c9fdab4240d2c32" alt="" data:image/s3,"s3://crabby-images/ee9fa/ee9fadda937b098fd72ebdf8bbc0862b79948949" alt=""
走到switch跳转处,0x7E
data:image/s3,"s3://crabby-images/d1c07/d1c07dd5e3f2ed1a555798523544efdf74f3602f" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/660b9/660b97908f6c629ce559c3430e87bc15dccd8ffc" alt="" data:image/s3,"s3://crabby-images/fe0c4/fe0c449d40c8b200dde027b423efdaf2cb78e7ac" alt=""
走到switch跳转处,0x64
data:image/s3,"s3://crabby-images/a1ede/a1eded932010dd04f8072a6f215a0e0e92bddc43" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/f11d9/f11d9ffd4ec313a054b1b1a5e04ba802458766f1" alt="" data:image/s3,"s3://crabby-images/f2fca/f2fca0e17449c39f8fdb4ab404cb39682acb46b8" alt=""
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/9784e/9784e82768787c3c656f396db108a29aaa64438b" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/40775/40775501ddf7d8a39e6fac7ac6862b8737f8ce24" alt="" data:image/s3,"s3://crabby-images/8314e/8314e0faebe4e8fcdae673214a59d4dbdeaa1d4f" alt=""
字节码0x77
走到switch跳转处,0x77
data:image/s3,"s3://crabby-images/fd01c/fd01c908b07ff1537935884338dd25e3c71a1c3c" alt=""
data:image/s3,"s3://crabby-images/ac141/ac141e32bc2b5666b4c1743821af06a251f89fd0" alt=""
首先是获取*(code_ip+1)和*(code_ip+2)分别是3,0
走到关键代码处
data:image/s3,"s3://crabby-images/4af74/4af74fe7dcebf951a2d958026e3b220499b4cda4" alt=""
这里再次看见了get_array
eax=array[*(code_ip+1)]
data:image/s3,"s3://crabby-images/88790/88790b5693581e8f37d23fdcc02edb7b98f435c3" alt=""
这里有input[*(code_ip+2)]=eax
整理后有
data:image/s3,"s3://crabby-images/646a6/646a6f2003464d499dfb2cb8c01ab90e55dd8532" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/b68c4/b68c4e3fda0483ed089bb1ffd80f7445222f164d" alt="" data:image/s3,"s3://crabby-images/8a939/8a939f842a3c7c27519d142d7d829c2681ab8ab3" alt=""
data:image/s3,"s3://crabby-images/25924/2592488ca65335b3c2d9421d56c5b46778f00877" alt="" data:image/s3,"s3://crabby-images/b5baf/b5bafe31349cf1846d192c1a5367e5964e805e14" alt=""
到目前为止,我们终于看到了数据被操作,这里可以很容易看出数据处理逻辑是
input[0]=input[0]^input[0x10]
同样也可以得到一个假设,在执行0x77时就会修改input数据
走到switch跳转处,0xD9
data:image/s3,"s3://crabby-images/b0b3a/b0b3ab904bd8a3d3cea553e5b80db7d72b697018" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/4a39f/4a39fc1b174f4d6fc3a9fb7430b13baa7734e17c" alt="" data:image/s3,"s3://crabby-images/d3e89/d3e890dceee068d67dd208de8ba7bbade45003ea" alt=""
走到switch跳转处,0xBF
data:image/s3,"s3://crabby-images/54f34/54f3468ed4b375d739e863fe23e97d845847d1c4" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/6915f/6915fcaa84d334091daa0db908a02172e8253205" alt="" data:image/s3,"s3://crabby-images/f7cfc/f7cfc8d204fc95b8784f65909d509edce5d8efa2" alt=""
走到switch跳转处,0xD9
data:image/s3,"s3://crabby-images/88cf2/88cf2111b9e1b4ae0befe9408594aa9fc5fa577e" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/1c110/1c1103ee0ba455374a207d6ed352df514717086e" alt="" data:image/s3,"s3://crabby-images/b0597/b05979216ff202579b5d93ddc212b548587df126" alt=""
走到switch跳转处,0xBF
data:image/s3,"s3://crabby-images/5719b/5719b2d05dbd7fc6525b4de076363456fdcfd149" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/f5784/f5784b5feaa88b7082a151f81ed06852678fce64" alt="" data:image/s3,"s3://crabby-images/b6a0e/b6a0e0cbd055844d01d59223a527947d7b9f08f6" alt=""
字节码0x79
走到switch跳转处,0x79
data:image/s3,"s3://crabby-images/6cd3f/6cd3f0121c397ab29969910ba85699f38de90f5f" alt=""
走进去看
data:image/s3,"s3://crabby-images/fa595/fa5951ff744f278f6996f5139d37521580058edf" alt=""
获取了*(code_ip+1)和*(code_ip+2)分别是3,0
进入call
data:image/s3,"s3://crabby-images/d42ff/d42ff2c1bfdc59f85f3a609025a31c238006aace" alt=""
这里调用了两次get_array,有ebx=array[*(code_ip+1)]
eax=array[*(code_ip+2)]
data:image/s3,"s3://crabby-images/87959/8795931c249411ca496af4d0553246c351c22ec1" alt=""
这里有ebx-=eax
调用set_array,array[*(code_ip+1)]=ebx
整理后有
data:image/s3,"s3://crabby-images/72f1d/72f1d42ffeb5d66f495d2cfd3e10f530270d40c4" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/0914e/0914ede782876100e8367dec68cea6713427c88b" alt="" data:image/s3,"s3://crabby-images/046e3/046e3822a916d5eaffaa2595373b9bd872636137" alt=""
走到switch跳转处,0x79
data:image/s3,"s3://crabby-images/bd862/bd8620effc3221a638ee31f4b5aa9a9a110116d6" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/146f3/146f3d57d02c5d8b7ac84cad664d5465e2fcd11e" alt="" data:image/s3,"s3://crabby-images/3c3f1/3c3f1787878d2bca8a2c1f8d072a0a5b199c09be" alt=""
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/06c7e/06c7edca2ed7bf5e59dd886bb8dd8291d3b50f17" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/daf77/daf77257df447ae1f9149fa532c42da6e33596b5" alt="" data:image/s3,"s3://crabby-images/48f95/48f952917d7ca9b0d3fd6036c825c5dfc030acfc" alt=""
走到switch跳转处,0x7E
data:image/s3,"s3://crabby-images/df2bb/df2bb280294347ebf7205d4471fa159fafb0c43a" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/28916/2891657986ef78023e34e5c2cd7072a0e86fe2ad" alt="" data:image/s3,"s3://crabby-images/b2f0a/b2f0af1a8ef53677b545452fca6bbe3b7f899fc5" alt=""
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/94bf5/94bf59cf0f87fcf3f4c679b82548121a4ba00141" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/e40a9/e40a95889479a20bb5417ce2b976261aa3a974fd" alt="" data:image/s3,"s3://crabby-images/0d9d2/0d9d20d9577904a4a73699258e25e26eeff8ec26" alt=""
走到switch跳转处,0x7E
data:image/s3,"s3://crabby-images/3e5fb/3e5fb08b6ad8428305fb24606f83357f4054e2ee" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/692b5/692b56a7c61e6c532e91f3aa52c8c70790cdba53" alt="" data:image/s3,"s3://crabby-images/d7df0/d7df04acc37999cb9ca7306581932038be5ade66" alt=""
这时已将input[0xF],input[0xE]放入了array中
走到switch跳转处,0x64
data:image/s3,"s3://crabby-images/e07f9/e07f95e45317d076f1b2d304bbcb239ab0b3a700" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/5877d/5877d6d35641e7202ec39d238e7b400775661fca" alt="" data:image/s3,"s3://crabby-images/12764/127642cbeb58f163bbb8f5db222309f434dd21c8" alt=""
走到switch跳转处,0x65
data:image/s3,"s3://crabby-images/2cac4/2cac417bd2b1db83811ac77a7534793e451ce8ea" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/9417d/9417d2859ec768ca98f3ec8ef97678e0915bb497" alt="" data:image/s3,"s3://crabby-images/ad281/ad28110d649179d6c870fa5205ab2db6315c7460" alt=""
走到switch跳转处,0x77
data:image/s3,"s3://crabby-images/a8dd6/a8dd648564d1ad24e9ef7bbf8a4b2727e4de77fe" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/7186f/7186f3c414f63a119ab8872457adcb5070e5527b" alt="" data:image/s3,"s3://crabby-images/83c27/83c27e9737ed290b1a22fa334d3d1ef74e82452b" alt=""
data:image/s3,"s3://crabby-images/ba1e7/ba1e77bf4f02a3fc6c4a970510719e8ea4313109" alt="" data:image/s3,"s3://crabby-images/73606/73606d9be750c063db11a9398f2de96f831426c2" alt=""
这里的数据处理逻辑是input[0xE]^=input[0xF]
所以在这里可以做个假设,假设处理整个逻辑是
input[index]^=input[0x10+index](1)
input[0xE-index]^=input[0xF-index](2)
index是从0开始的,但是不清楚i到那里结束,在这里我们也可以猜想一下
可以看出第一个式子最多可以执行0xF次,第二个式子最多可以执行0xE次
所以i的结束条件可以是0xF,也可以是0xE
毕竟这些都是猜想,需要验证是否正确
这中题目类似于找规律,求归纳式子
那么要找规律就必须至少验证两次循环了
所以我们要验证处理过程是否正确,就需要再跟进一次循环过程
下面继续分析
字节码0x00
走到switch跳转处,0x00
data:image/s3,"s3://crabby-images/4b0f8/4b0f80180bfb6c85c7b2ae6b5c8a56fbf4543129" alt=""
进入跳转
data:image/s3,"s3://crabby-images/225f6/225f6c8b18c5cf2ffbf828190e8a33aa7980e72a" alt=""
这里获取了*(code_ip+1)是0
进入call
data:image/s3,"s3://crabby-images/1dff2/1dff24076004dbfafe086221ff5a259714760d1a" alt=""
这里面调用了get_array和set_array
分析有
eax=array[*(code_ip+1)]
esi=eax+1
array[*(code_ip+1)]=esi
整理后有
data:image/s3,"s3://crabby-images/25576/25576913925bd5de9384ffbb1a6dd975da417157" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/a0854/a0854f1b115c1231ec386a430b68ac7847152cde" alt="" data:image/s3,"s3://crabby-images/5c608/5c608d25fd2a03feb575fe8878e5c165649dd9b1" alt=""
走到switch跳转处,0x00
data:image/s3,"s3://crabby-images/0bfea/0bfea3151a8a3e24b8db4d79c231ab5ce1d5e624" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/03c7b/03c7b79dc4890a95f3d5e702ce6cd9ee4dfd9b3b" alt="" data:image/s3,"s3://crabby-images/fb662/fb662dce7c5e9ffeccd5603ac25b9189a5684699" alt=""
字节码0xA0
走到switch跳转处,0xA0
data:image/s3,"s3://crabby-images/eddcd/eddcdcab0311e8ea11c0f08b57face1bd1ff15e2" alt=""
进入跳转
data:image/s3,"s3://crabby-images/e329b/e329b254227bbb7d1d23fefbaed46711db417a52" alt=""
到这里,eax=0x0FFFFFFAA=-86(十进制)
eax=edx-89
这里就可以理解为回到了循环的开始
整理后有
data:image/s3,"s3://crabby-images/39b1a/39b1a5cdc554da1373bb079d0742e14a91995aaa" alt=""
对比输出结果和内存数据
data:image/s3,"s3://crabby-images/8c861/8c861728da54b00c5ac64e321f869056426b8c6b" alt="" data:image/s3,"s3://crabby-images/3204e/3204e0ca8d52167408e37d9d5ce44b29441b459e" alt=""
接下来,我们分析一下执行的流程
程序在0xA0处回到了循环的开始
那么0xA0就可以作为一次循环结束的标志
然后我们通过修改我们编写的程序来进行模拟循环的执行流程
于是我们可以有
data:image/s3,"s3://crabby-images/b30e7/b30e796d2c1bd7b5036a30a12cf1cf54344a2b76" alt=""
仔细观察,我们可以发现两次循环有相同的执行流程
我们设置可以查看循环3次,4次的输出结果,会更加明显
data:image/s3,"s3://crabby-images/e5a83/e5a83591736f16723593faa84c33ad60dbdef914" alt=""
这是我们发现第二次,第三次,第四次循环的流程是一模一样的
那么我们假设的归纳式子就是成立的了
但是我们还是不知道结束的标志在哪里
其实细心的朋友可能已经知道了
i应该执行到0xE就结束,为什么呢?
我们仔细分析了12个有用的字节码,而其他的都是根本用不到的
而这里面的操作都是不同的
那么哪个操作码是判断结束的呢?
我们可以大致浏览一下每个字节码的作用
[C] 纯文本查看 复制代码 case 0u: printf("0x00->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->array[*(codeVM->code_ip+1)]++; codeVM->code_ip+=2; break; case 0x64u: printf("0x64->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->array[*(codeVM->code_ip+1)]^=codeVM->array[*(codeVM->code_ip+2)]; codeVM->code_ip+=3; break; case 0x65u: printf("0x65->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); *(codeVM->code_ip+4)=codeVM->array[*(codeVM->code_ip+1)]; codeVM->code_ip+=2; break; case 0x67u: printf("0x65->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); if(codeVM->array[7]==0){ codeVM->code_ip+=5; } break; case 0x77u: printf("0x77->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->input[*(codeVM->code_ip+2)]=codeVM->array[*(codeVM->code_ip+1)]; codeVM->code_ip+=3; break; case 0x79u: printf("0x79->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->array[*(codeVM->code_ip+1)]-=codeVM->array[*(codeVM->code_ip+2)]; codeVM->code_ip+=3; break; case 0x7Eu: printf("0x7E->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->array[*(codeVM->code_ip+1)]=codeVM->input[*(codeVM->code_ip+2)]; codeVM->code_ip+=3; break; case 0xA0u: printf("0xA0->"); if(icode_ip+1),*(codeVM->code_ip+2)); codeVM->code_ip-=0x100-*(codeVM->code_ip+1); i++; break; case 0xB7u: printf("0xB7->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); if(codeVM->array[*(codeVM->code_ip+1)]array[*(codeVM->code_ip+2)]){ codeVM->array[7]=0; }else{ return; } codeVM->code_ip+=3; break; case 0xBFu: printf("0xBF->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->array[*(codeVM->code_ip+1)]=codeVM->temp; codeVM->temp+=0x10; codeVM->code_ip+=2; break; case 0xD9u: printf("0xD9->");// printf("\t%X %X\n",*(codeVM->code_ip+1),*(codeVM->code_ip+2)); codeVM->temp=*(codeVM->code_ip+1); codeVM->code_ip+=5; break; }
经过分析我们可以看到在字节码0xB7处,有一个判断,我们可以看一下它需要的参数
我们单独输出0xB7的参数
data:image/s3,"s3://crabby-images/7af39/7af395003d3dd3b8b0bd61442828e6ca068d0698" alt=""
这是执行三次循环后的输出,我们发现参数是固定的,都是0和2
那么array[0]和array[2],里面又是什么呢?我们同样把它们输出
data:image/s3,"s3://crabby-images/e4bd1/e4bd183e957c03afcf31713bf11eb75eeec97224" alt=""
<font face="微软雅黑"><font size="4"><font color="#000000">这样的话就十分清晰了,那么就有index |
本帖子中包含更多资源
您需要 登录 才可以下载或查看,没有帐号?立即注册
x
|