Win32汇编学习笔记03.RadAsm和补丁-C/C++基础-断点社区-专业的老牌游戏安全技术交流社区 - BpSend.net
扫雷游戏啊下补丁
在扫雷游戏中,点关闭弹出一个确认框,确认之后再关闭,取消就不关闭
data:image/s3,"s3://crabby-images/717e1/717e1374fd71b8f101c12b87d0fd878cd1d9c336" alt=""
- 首先第一步就是确认关闭按钮响应的位置,一般都是 WM_CLOSE 的消息 ,消息响应一般都在过程函数,所以就是要定位到过程函数,我们知道 MC 项目中 ,如果是重叠类的窗口,注册窗口类的时候会用到过程函数, 如果是对话框的 那就是调 DialogBox
- 先假设扫雷是一个 重叠类窗口 那我们就要知道程序中那个地方调用了 RegisterClass (注册窗口类)
最笨的方法就是一步步跟,但比较耗时间
- 使用OD自带的功能,帮助我们定位在哪些地方,调用了哪些导入函数,只显示当前加载的
data:image/s3,"s3://crabby-images/9d115/9d115f2d3a6f9864fe882874cd8e92d6c3f75504" alt=""
data:image/s3,"s3://crabby-images/f13ab/f13ab64f4840cf47fd24304972374da5a4d730ad" alt=""
data:image/s3,"s3://crabby-images/c4c05/c4c0586c1e36959c8b9c1f6871ebe68bc77ae370" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/8fe6e/8fe6eb2888cb9a4ffa2cf7dc2393dc6a590b6b65" alt=""
在所有调用地方设置断点
data:image/s3,"s3://crabby-images/6b283/6b283810dab02503cbd1f9070551f5ad4cdff6df" alt=""
- 下好断点后开始调试, F9 运行到断点处
注册窗口类第一个参数是窗口类的地址,
data:image/s3,"s3://crabby-images/ba43c/ba43c30be9a951c970155444fd35a0a3720ec207" alt=""
窗口类第二个参数是函数地址,第一个是风格
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
- 寻找 WM_CLOSE 消息,可以通过条件断点寻找
data:image/s3,"s3://crabby-images/3bf97/3bf97df8936dc4d057a0d75f407b469bee15a8e3" alt=""
分析可知 , 参数入栈顺序是 最后一个参数,倒数第二个参数........,第一个参数,返回地址,二消息id是通过 第三个参数 WPARAM 所以参数位置在 esp + 8 处
data:image/s3,"s3://crabby-images/1906d/1906dcf67ba3bb6028b233cea86c27a9031d764a" alt=""
data:image/s3,"s3://crabby-images/35614/35614aa23e6572e24a412a0fae4449c36ec33869" alt=""
- 在关闭游戏,这样就可以触发 WM_CLOSE 消息,从而触发断点,在 F8 单步运行,看到哪部跳走
data:image/s3,"s3://crabby-images/5b2df/5b2df927b8f63e3633b04f626f15bb56679ac460" alt=""
可以看出它调用了 默认过程函数,并没有自己处理关闭消息,所以我们只能 自己来 判断是不是WM_CLOSE 消息,然后我们自己来处理 ,如果是 WM_CLOSE ,那么就需要我们去弹窗,不是的话就交给过程函数去处理
- 我们要处理 WM_CLOSE 消息,那么就需要去写代码,但是又不能覆盖他原本的代码,因此他们有代码地方我们都不能用,只能找空白的地方写
data:image/s3,"s3://crabby-images/2dc7f/2dc7f933d9829fc7a5132b5d90f63cfa7d3ffa00" alt=""
记录一下我们们写代码的地方
我们汇编代码开始位置
01004A60 00 db 00
data:image/s3,"s3://crabby-images/d0bbe/d0bbebc373eead6a5dc654c6e87d9aadf8448145" alt=""
通过之前我们跟代码,可以知道,程序是从该处跳转到 默认过程函数的,因此我们可以将此处改成 跳转到我们的代码处,然后再由我们跳转回来
原来的跳转地址
01001C16 |. /0F85 8D050000 jnz 010021A9
函数退出位置
010021BB |> \5F pop edi
通过空格单击, ,修改汇编代码,来修改跳转位置
data:image/s3,"s3://crabby-images/50cab/50cabf12619b50caeecd255bcfd9e9ba0c0d1b4c" alt=""
选中修改后的地址,跳转过去开始写我们的汇编代码
data:image/s3,"s3://crabby-images/e0feb/e0feb7e4199a2f4ea73f4068273a66a3d6af0ba2" alt=""
data:image/s3,"s3://crabby-images/36b69/36b69889de30f963333c489a4e7c6f19f0c5e678" alt=""
接下来相等就要弹窗 ,首先要有提示语,需要保存到文本
data:image/s3,"s3://crabby-images/e272b/e272b93d51e6a3cd4fa6a6d8bb6d0ff5c50ddfc6" alt=""
data:image/s3,"s3://crabby-images/195f5/195f5ded9024eab92049b356898a6c7fb541526f" alt=""
data:image/s3,"s3://crabby-images/73444/734441768e899b574a4f23df1e0413f345c57382" alt=""
- 调用弹窗函数
data:image/s3,"s3://crabby-images/1fc3f/1fc3f933b5078818ee81bdc69217b8fe47cf95f6" alt=""
标题,文本入栈
data:image/s3,"s3://crabby-images/826c9/826c919ab6dbfcf2053fef9655c9c76d3c485882" alt=""
data:image/s3,"s3://crabby-images/fed35/fed353a294f0f9d256fafc59cd66ed989e871499" alt=""
窗口句柄入栈
data:image/s3,"s3://crabby-images/dbd59/dbd5959a81388b0f9fb1ecc433a865750893b98a" alt=""
调用函数
data:image/s3,"s3://crabby-images/35d85/35d85c6416bb6b1428837fcab88942257ed6d106" alt=""
判断点击按钮
data:image/s3,"s3://crabby-images/5d103/5d103f1b6f4cfc69059dd71686299a7ab533d20d" alt=""
判断,是确定就挑转到 默认过程处理函数,否则退出函数
data:image/s3,"s3://crabby-images/4d33b/4d33b45c42c87644a7b356a29782e4ae4df37196" alt=""
data:image/s3,"s3://crabby-images/5b145/5b14526e8bc73bae978fc4d3ce12c04438762b99" alt=""
data:image/s3,"s3://crabby-images/12cc6/12cc6dba9388ead176be44ea13979142300c87f1" alt=""
但是这个时候我们写的代码还只是在内存里面,程序一旦重启 就都没了,因此我们要保存到可执行文件
data:image/s3,"s3://crabby-images/9612b/9612bb47ef3ec8961057b969c5456b7d7e00396f" alt=""
data:image/s3,"s3://crabby-images/95787/9578729725e2bb9f047377da8fac2e13fee7ace3" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
- 保存文件运行
这是点击退出,发现出现了弹窗提示
data:image/s3,"s3://crabby-images/8bd07/8bd07941be2462aebf3542218a2d687cfa92018e" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
注意: 找空白位置不要到最后,不然按可能报错,位置有规律,但是要学pe
data:image/s3,"s3://crabby-images/e0344/e0344dd3ae72eb32404717c6648ba6723050ab95" alt=""
通过OD 下断点/取消断点 alt +b
data:image/s3,"s3://crabby-images/6b067/6b06741414c7848e8e36a70d6c96da75b3814518" alt=""
如果有断点 ,选中 del
程序 F9 运行后
data:image/s3,"s3://crabby-images/32d4b/32d4b9afafc3eb7d36bfa1a3661ff627e48d570e" alt=""
data:image/s3,"s3://crabby-images/be18a/be18ad59634aa0160aaba12544870c1a82a706df" alt=""
data:image/s3,"s3://crabby-images/38368/3836839aa160efdd3789afbc6df772b40012d9e3" alt=""
data:image/s3,"s3://crabby-images/27211/272111d41b440435a5c6355e86bf002f7b79eafc" alt=""
data:image/s3,"s3://crabby-images/2b65e/2b65e2e4db824b3c5557208bf3bb11878a8759e0" alt=""
在Crack Me 下补丁
使其提示成功
Brad Soblesky.1.zip(4 KB)
data:image/s3,"s3://crabby-images/5ad6e/5ad6ee3f05a97408a08b56362ad08fa2125c22cf" alt=""
方法1: 定位 command 消息
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/766b8/766b81808ab524b331b069d8c36725f8360d97bb" alt=""
或者
data:image/s3,"s3://crabby-images/11560/115605e501f5c52017d8ecc8a57e3ad3b7ff19c6" alt=""
data:image/s3,"s3://crabby-images/26f87/26f87f5c8fede1b3cf9bbcb990316d5d8dda47e0" alt=""
f9 运行 到断点 ,在 f8 单步运行
data:image/s3,"s3://crabby-images/2cc72/2cc723a8a9cd5ff399273104e2d1b1b47bdb515d" alt=""
可以看到,运行到系统的 dll 中,所以这不是一个 重叠类窗口,所以过程函数不是自己的,无法用上面方法,所以要换个f方法
因为 他失败成功都会弹出一个对话框,里面有字符串,因此我们可以通过 OD 的字符串查找功能
data:image/s3,"s3://crabby-images/9e67c/9e67cf53a23887bad8f7b7cf0d48262a16cfbe76" alt=""
data:image/s3,"s3://crabby-images/00d1e/00d1eaac44a3e5b9439dfd0e37d6ae409a464ce9" alt=""
data:image/s3,"s3://crabby-images/cc15e/cc15e57bdbccc3f7855d8f5cddbd517568e31bf1" alt=""
data:image/s3,"s3://crabby-images/fb12d/fb12da8c990c91344fac1bfb4263a3b4bd5f925b" alt=""
查看上面的代码
data:image/s3,"s3://crabby-images/30373/30373b8d4c4744b2ef94adb770bdaa7b551f35ec" alt=""
在此处下点,就可以看到比较的字符串,即正确的字符串
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
或者直接去掉比较,那么不管输出字符串是啥,都可以成功
data:image/s3,"s3://crabby-images/f6028/f6028903ff785af173d669267441ba068cf3de55" alt=""
data:image/s3,"s3://crabby-images/acb1c/acb1c6c65181359569bc7e080aaa3bcae9aecb05" alt=""
将文件保存
data:image/s3,"s3://crabby-images/cb95f/cb95ff6a72e565add7b5cb646a911e1e2ff30202" alt=""
运行我们保存的文件,发下不管输入什么字符串都可以通过了,即破解成功了
注册机
当遇到非明码比较时,上面的方法就行不通,因为密码是通过 加密解密得到的
data:image/s3,"s3://crabby-images/8a00b/8a00b4193a9c3c51b937b0dce9289686ba68cf8a" alt=""
输入2行字符串,验证通过
Brad Soblesky.2.zip(4 KB)
首先定位到字符串,下断点
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
往上找代码,在函数入口出下断点
data:image/s3,"s3://crabby-images/f3bb2/f3bb2952e4fdb2f1844748140bb5766c3f1334f3" alt=""
跟一下大体流程,通过跟代码,我们可以知道
data:image/s3,"s3://crabby-images/90e07/90e0766da82f86d91cfc298eae494a53ebc747dd" alt=""
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/fc0e9/fc0e961b4391e9c0d0040fa1fa3bddce2874f2a2" alt=""
F4 运行到光标处,运行完循环 ,继续运行
data:image/s3,"s3://crabby-images/6f5ed/6f5edb67f1eccadd62da29682feb13b1447d556f" alt=""
data:image/s3,"s3://crabby-images/d992c/d992ccc1e92c2003e991349033cba461faa232d5" alt=""
断点继续跟
data:image/s3,"s3://crabby-images/8c5b6/8c5b6af7fa08accc761f6e72edceeafd4e89f85c" alt=""
data:image/s3,"s3://crabby-images/386ec/386ecdfe30d0d000684d2a236af411760a494e30" alt=""
data:image/s3,"s3://crabby-images/c9adb/c9adbf850f2d7dcc8e30f7bdc8a53985a4d539db" alt=""
加注释于0040163B
data:image/s3,"s3://crabby-images/a06f4/a06f47940cc1d170dea552046fffc101df54b645" alt=""
因此可以分析出,程序是那我们输入的字符串做一系列运算,再把结果格式化后去跟第二个字符串 作比较,我们把上面格式化后的字符串拿去测试,可以发现通过了,所以猜想是正确的
data:image/s3,"s3://crabby-images/74c95/74c95867f7f3270aed2acfcf25fd6c5d5d6a8458" alt=""
上面软件已经提供了加密算法,因此写该软件的注册机可以直接把他的代码复制出来来就可以了
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
L000:
mov edx, dword ptr [ebp-20]
add edx, 1
mov dword ptr [ebp-20], edx
mov eax, dword ptr [ebp-20]
cmp eax, dword ptr [ebp-1C]
jge short 0040161A
mov ecx, dword ptr [ebp-20]
push ecx
lea ecx, dword ptr [ebp-14]
call 00401900
movsx edx, al
mov eax, dword ptr [ebp-10]
add eax, edx
mov dword ptr [ebp-10], eax
mov ecx, dword ptr [ebp-20]
shl ecx, 8
mov edx, dword ptr [ebp-10]
xor edx, ecx
mov dword ptr [ebp-10], edx
mov eax, dword ptr [ebp-20]
add eax, 1
mov ecx, dword ptr [ebp-1C]
imul ecx, dword ptr [ebp-20]
not ecx
imul eax, ecx
mov edx, dword ptr [ebp-10]
imul edx, eax
mov dword ptr [ebp-10], edx
jmp L000
上面还有很多参数,必须我们去分析
data:image/s3,"s3://crabby-images/7630a/7630ab629d96b8cda8f7d7bcdacde8994f8a6b26" alt=""
新建工程
data:image/s3,"s3://crabby-images/1f29d/1f29ddafdbf318efeabe3b9a90fd748fa7eabf00" alt=""
模板,根据自己需要选择
剩下的不用动,一直下一步完成就行
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
GIST\REGIST.ASM
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
S\ADMINISTRATOR\DESKTOP\03 补丁\REGIST\REGIST.ASM]
data:image/s3,"s3://crabby-images/d8140/d814050c81a11d0b9252b99f1d5bc806fe6502a6" alt=""
编译和链接,链接之前记得编译资源脚本,或者直接构建
添加资源
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
如果给控件添加名字需要自己手动加到头文件
data:image/s3,"s3://crabby-images/0a332/0a332c98c765f6ccd68a8795670e82e7eb5a563b" alt=""
data:image/s3,"s3://crabby-images/489ca/489ca6ad53f8c45885932ea2fd3e08037b4affa7" alt=""
下面就开始写注册机了
L000:
mov edx, dword ptr [ebp-20]
add edx, 1
mov dword ptr [ebp-20], edx
mov eax, dword ptr [ebp-20]
cmp eax, dword ptr [ebp-1C]
jge short 0040161A
mov ecx, dword ptr [ebp-20]
push ecx
lea ecx, dword ptr [ebp-14]
call 00401900
movsx edx, al
mov eax, dword ptr [ebp-10]
add eax, edx
mov dword ptr [ebp-10], eax
mov ecx, dword ptr [ebp-20]
shl ecx, 8
mov edx, dword ptr [ebp-10]
xor edx, ecx
mov dword ptr [ebp-10], edx
mov eax, dword ptr [ebp-20]
add eax, 1
mov ecx, dword ptr [ebp-1C]
imul ecx, dword ptr [ebp-20]
not ecx
imul eax, ecx
mov edx, dword ptr [ebp-10]
imul edx, eax
mov dword ptr [ebp-10], edx
jmp L000
首选要对函数进行分析
通过跟代码发现 ebp-20 是一个局部变量,作用是计数器
ebp-1c 是输入字符长度
data:image/s3,"s3://crabby-images/6fc4e/6fc4eecc18167efe95bb20735b7f9346852560ad" alt=""
这里是去字符串读取 所以可以用 lodsb 替换
几位下来就剩 ebp-10 ,ebp-1c .找到最开始出现的地方,来观察数据变化
data:image/s3,"s3://crabby-images/e03e3/e03e3e1bcf80f2f3dcfc3d480cfa712d5ba1e549" alt=""
通过几次不同测试,发现值 是固定的
data:image/s3,"s3://crabby-images/f1537/f1537029a604a0b871906b02f0b1006509ce5939" alt=""
data:image/s3,"s3://crabby-images/9546d/9546d75380f4c93ac3dfeca67d2fbc1c2f424a21" alt=""
注册机源码
.386
.model flat, stdcall ;32 bit memory model
option casemap :none ;case sensitive
include Regist.inc
.data
g_szUserName db 50 dup(0) ;用户名
g_szSerial db 50 dup(0) ;注册码
g_szFmt db "%lu" ,0 ;格式化字符串
.code
start:
invoke GetModuleHandle,NULL
mov hInstance,eax
invoke InitCommonControls
invoke DialogBoxParam,hInstance,IDD_DIALOG1,NULL,addr DlgProc,NULL
invoke ExitProcess,0
;########################################################################
Make proc hWin:HWND ;加密
LOCAL @dwIdx:DWORD
LOCAL @dwCnt:DWORD
LOCAL @dwEdp_10:DWORD
LOCAL @dwEdp_14:DWORD
;回去文本输入的值
invoke GetDlgItemText,hWin,EDT_USERNAME,offset g_szUserName,size g_szUserName ;获取用户名内容
mov @dwCnt,eax
mov @dwIdx,0
mov esi ,offset g_szUserName
mov @dwEdp_10,81276345H
jmp LSATART
;对用户名加密解密
L000:
mov edx, @dwIdx
add edx, 1
mov @dwIdx, edx
LSATART:
mov eax, @dwIdx
cmp eax, @dwCnt
jge LEND
;mov ecx, @dwIdx
;push ecx
;lea ecx, dword ptr [ebp-14]
;call 00401900
lodsb ;上面四行作用
movsx edx, al
mov eax, @dwEdp_10
add eax, edx
mov @dwEdp_10, eax
mov ecx, @dwIdx
shl ecx, 8
mov edx, @dwEdp_10
xor edx, ecx
mov @dwEdp_10, edx
mov eax, @dwIdx
add eax, 1
mov ecx, dword ptr @dwCnt
imul ecx, @dwIdx
not ecx
imul eax, ecx
mov edx, @dwEdp_10
imul edx, eax
mov @dwEdp_10,edx
jmp L000
LEND:
;格式化字符串
invoke wsprintf,offset g_szSerial,offset g_szFmt,@dwEdp_10
;显示到界面
invoke SetDlgItemText,hWin,EDT_SERUNMBER,offset g_szSerial
Make endp
DlgProc proc hWin:HWND,uMsg:UINT,wParam:WPARAM,lParam:LPARAM
mov eax,uMsg
.if eax==WM_INITDIALOG
.elseif eax==WM_COMMAND
mov eax,wParam
.if eax == BTN_MAKE
invoke Make,hWin
.endif
.elseif eax==WM_CLOSE
invoke EndDialog,hWin,0
.else
mov eax,FALSE
ret
.endif
mov eax,TRUE
ret
DlgProc endp
end start
作业
1. 修改机器码指令对照表.exe的字体
思路:
\1. sendmessage给callwindow发消息 然后绘制界面,sendmessage的第一个消息就是设置字体的
\2. 调用 CreateFontA 它有个设置字体的那块 设置完毕后会发个消息 找消息循环上面的发消息那块 就能看见有个函数,在获取系统字体 然后把这个函数换掉,换成跳转到自己的代码处 改完再跳回来
HFONT CreateFontA(
[in] int cHeight,
[in] int cWidth,
[in] int cEscapement,
[in] int cOrientation,
[in] int cWeight,
[in] DWORD bItalic,
[in] DWORD bUnderline,
[in] DWORD bStrikeOut,
[in] DWORD iCharSet,
[in] DWORD iOutPrecision,
[in] DWORD iClipPrecision,
[in] DWORD iQuality,
[in] DWORD iPitchAndFamily,
[in] LPCSTR pszFaceName
);
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/67311/67311274fe4ad44c35b0c649ceb66c3f7b5dd775" alt=""
- COM Explorer_Setup 注册机,汇编实现。
详情见 day05 补课
思路: 根据弹窗标题搜字符串,往上找到分支跳,定位到关键函数,找到算法代码,(算法代码使用四则运算和位运算比较多),然后把代码抽出来去分析,失败分支那里肯定有校验的算法
data:image/s3,"s3://crabby-images/689c5/689c5665b2eab45bbb82e0fcee84c50469d2cff4" alt=""
data:image/s3,"s3://crabby-images/4e62a/4e62a682fa6cfbfdd85677481cdb0d6ce4b4f7b9" alt=""
找到成功或失败分支
data:image/s3,"s3://crabby-images/4ad9b/4ad9bf1f863098f1c84b1cfe7ee4d515791d3a8b" alt=""
L000:
mov ebp, dword ptr [463E48]
or ecx, FFFFFFFF
mov edi, ebp
xor eax, eax
repne scas byte ptr es:[edi]
not ecx
dec ecx
mov eax, esi
xor edx, edx
mov edi, ebp
div ecx
mov ecx, dword ptr [esp+14]
mov eax, esi
movsx ebx, byte ptr [edx+ebp]
xor edx, edx
div ecx
movsx eax, byte ptr [edx+469968]
lea edx, dword ptr [eax+eax*2]
lea eax, dword ptr [eax+edx*4]
mov edx, esi
imul edx, esi
imul edx, esi
imul edx, ecx
add ebx, eax
or ecx, FFFFFFFF
xor eax, eax
add ebx, edx
repne scas byte ptr es:[edi]
not ecx
dec ecx
mov eax, ebx
xor edx, edx
div ecx
lea ecx, dword ptr [esp+10]
mov al, byte ptr [edx+ebp]
push eax
call 0042F29A
test esi, esi
je L048
mov eax, esi
xor edx, edx
mov ecx, 3
div ecx
test edx, edx
jnz L048
push 004640F8
lea ecx, dword ptr [esp+14]
call 0042F273
L048:
inc esi
cmp esi, 9
jb L000
data:image/s3,"s3://crabby-images/ff858/ff858224f697c6576dd4ccc8e577da2dfbe7f562" alt=""
data:image/s3,"s3://crabby-images/3f7f6/3f7f618c09b1f28a664c4de1fc83f06e83fb2093" alt=""
data:image/s3,"s3://crabby-images/f2775/f27752198d6d54690bb84097444e0b38182bdb9c" alt=""