data:image/s3,"s3://crabby-images/b21b0/b21b080dc3213dd37f65c7d558c7d426c5f51af9" alt="enter description here"
接着上一篇的总结,ONDragon的第二课。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
| #define _CRT_SECURE_NO_WARNINGS #include <windows.h> #include <stdio.h> #include "resource.h" char name[1024] = "ONDragon"; //设置用户名、密码 char pass[1024] = "666"; BOOL check(char userName[], char passWord[]) //检测判断用户名和密码是否和设置一样 { if (userName == NULL || passWord == NULL) { return FALSE; } else { if (*userName == *name) { if (*passWord == *pass) { return TRUE; } else { return FALSE; } } else { return FALSE; } } } BOOL CALLBACK MainDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { BOOL bRet = FALSE; switch (uMsg) { case WM_CLOSE: { EndDialog(hDlg, 0); break; } case WM_INITDIALOG: { SetWindowPos(hDlg, HWND_NOTOPMOST, 400, 300, 410, 180, SWP_SHOWWINDOW); break; } case WM_COMMAND: switch (LOWORD(wParam)) { case IDC_BUTTON_LOGIN: { char userName[1024]; char passWord[1024]; GetDlgItemText(hDlg, IDC_EDIT_USER, userName, 1024); GetDlgItemText(hDlg, IDC_EDIT_PASS, passWord, 1024); if (check(userName,passWord)) { MessageBox(hDlg, "Login Successful", "Congratulation", MB_OK); //登录成功 } else { MessageBox(hDlg, "Login Failed", "Sorry", MB_OK); //登录失败 } return TRUE; }
} break; }
return bRet; } int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG_MAIN), NULL, MainDlgProc); return 0; }
|
这次依旧是登录小程序
本次编译的是 release版本
data:image/s3,"s3://crabby-images/e3712/e37124afa48a36b10a47158d52a321d5d7055d77" alt="enter description here"
未破解的
data:image/s3,"s3://crabby-images/8cda5/8cda54f5eff55b7794db0c43620c77ba93832779" alt="enter description here"
破解后的
data:image/s3,"s3://crabby-images/fc96d/fc96d75e46ab9756913bae7690679e4795a1b7bc" alt="enter description here"
载入Ollydebug
这里我们载入OD
data:image/s3,"s3://crabby-images/0f5dd/0f5ddc64890bedd47291759587a52283e1a1c8d3" alt="enter description here"
和上篇一样,我这里依旧使用的是查找字符串
data:image/s3,"s3://crabby-images/760ed/760ede2013bb1ee1b6b5b63613e6d4d9eaeaecd3" alt="enter description here"
在这里只能短短的看到 登录失败和登录成功几个字,我们双击其中一个字符串,进行跟踪
data:image/s3,"s3://crabby-images/50997/50997637233ce8119fc5d4ba8260a24e667f9175" alt="enter description here"
data:image/s3,"s3://crabby-images/b87a0/b87a0783fd807ac6e74decb1baa52414e295c59e" alt="enter description here"
我们来简单看下。
data:image/s3,"s3://crabby-images/60a50/60a5029bdb32b98b9e8f1f0a536aeb5d65fed648" alt="enter description here"
1. 简单思考与分析
这两条指令 (cmp)将 ss:[ebp-0x404]的值和0x4F做比较,两个值相减为0,(ZF ,零标志,标明结果为0。真置1,假置0。)
反汇编窗口
这里也就是将第一个字节 与4F 做判断是否非为0。第一个cmp指的是用户名的字符串
data:image/s3,"s3://crabby-images/f106a/f106aa2400dcc740b2f0e3c76a3f7c3b0ce8440b" alt="enter description here"
那么这里我们不能让下条指令跳转到,登录失败。 第二个cmp指的是密码的字符串
data:image/s3,"s3://crabby-images/029e2/029e2f3e87f92de77d8d34e40ed3c8110c6ce4d7" alt="enter description here"
这里同样也是将第一个字节 与4F 做判断是否非为0。
data:image/s3,"s3://crabby-images/1a9a5/1a9a58c9bc3d5cd2141a47346ae0fe2fac681b65" alt="enter description here"
寄存器窗口
data:image/s3,"s3://crabby-images/d14e5/d14e557059204e280a66cf460519aa548ee9916e" alt="enter description here"
2.修改
我们如何不能让他跳转到登录失败呢
这里我们下个断点,在第一个cmp处。也就是判断用户名是否正确的地方。(按F2设置断点)
data:image/s3,"s3://crabby-images/e5255/e5255baf39345cf1a1dd9f7e79091fe6e7b56997" alt="enter description here"
并且运行程序
data:image/s3,"s3://crabby-images/f7245/f72452d1623f00bdce38edbe42c1f3df00ee03bd" alt="enter description here"
键入用户名和密码后,已断在断点处。
data:image/s3,"s3://crabby-images/0320a/0320acf347cf4cf9d0a80a9276d457c88137a135" alt="enter description here"
选中第一个cmp处,查看内存地址对应键入用户名的字符串 (右键->数据窗口中跟随->内存地址)
data:image/s3,"s3://crabby-images/95831/95831d520993c2f0a20d871085438affa0416965" alt="enter description here"
我们在数据窗口中可以看到,我们键入的用户名为demon ,第一个字节d 对应的则为64 十六进制 (在ASCLL码表即可找到相应的,其他依次类推)由于在cmp 设置的为bype 一个字节,后门判断 4F也为一个字节,cmp需将其判断为0,则我们也可以将其cmp处的代码 中的0x4F,改为0x64
data:image/s3,"s3://crabby-images/4477a/4477aec0959bcd540ed077e633bcd5a6137c174f" alt="enter description here"
data:image/s3,"s3://crabby-images/4ec40/4ec4042bd9db3ad9777047a5db1a8676510a4396" alt="enter description here"
或将数据窗口的 64 改成4F即可。
data:image/s3,"s3://crabby-images/c2ce6/c2ce6404856b34dc22eaca1829398cf054a7d5a8" alt="enter description here"
第二处 cmp以此类推。
修改完两处之后,我们运行看看,可以成功得到login successful 的弹窗!
data:image/s3,"s3://crabby-images/c8fdb/c8fdb0e13749c7420fe53bdf9dd02bcc347b156e" alt="enter description here"
3.NOP大法
还有另一种修改方案则是NOP大法。
在cmp处下断点,并将两处cmp 将其NOP掉。
双击CMP 汇编代码 ,将其代码填写为nop
data:image/s3,"s3://crabby-images/d466b/d466b2f916da98c9281417faffc346d748c82385" alt="enter description here"
填充nop
data:image/s3,"s3://crabby-images/76944/769440bfe25a02dc44117f094ad0e39b82ebadf0" alt="enter description here"
nop后
data:image/s3,"s3://crabby-images/cf336/cf336993d413967e0af25bf3a9219bd8484b69cc" alt="enter description here"
运行后,一样提示得到login successful 的弹窗!
data:image/s3,"s3://crabby-images/278be/278bea9eaddf5cac4be561d03d45a93583f365c1" alt="enter description here"
4.修改跳转指令
先上几张跳转指令基础知识图
data:image/s3,"s3://crabby-images/ce823/ce823c31ed22abdee35dcd917030326ae2f11804" alt="enter description here"
data:image/s3,"s3://crabby-images/4066f/4066fd138e9bfaf1001a13402c6a2c26fe55d3d7" alt="enter description here"
今天就先看这几个跳转指令,与ZF标志位相关
data:image/s3,"s3://crabby-images/707c5/707c5f3672e9a74997230ef092ebbffc569b27ea" alt="enter description here"
这里我们依旧还是在cmp处下断点,并运行,执行到断点处
data:image/s3,"s3://crabby-images/4e587/4e587da6de03c276942e51e21de2509b97d5f230" alt="enter description here"
两处jnz 改为 jz 或者je (改成jz 或自动填写je)
data:image/s3,"s3://crabby-images/4e0bd/4e0bd78f857b0834f612dcff74109a29b6184fe7" alt="enter description here"
以下是debug版本的IDA载入图,破解方法和之前的类似,这里就不再重复多说了,工具和课程视频咋即刻安全网盘即可下载。
data:image/s3,"s3://crabby-images/0d551/0d55108d2f76e11e108dae7343f07c9f37fe4600" alt="enter description here"