|
 |
推荐文章 |
|
|
|
|
|
|
|
|
|
|
Quickness 3.1 注册算法分析 + 注册机源代码(tc2) |
|
Quickness 3.1 注册算法分析 + 注册机源代码(tc2)
破解目标:Quickness 3.1 官方主页:http://www.qwertysoft.com/ 软件简介:Quickness 是一个方便用户输入的小工具。如果有些文本内容是我们经常要用到的,就可以利用 Quickness 将其保存起来,并为其指定热键,这样,当我们需要用的时候,直接按热键就可以调出内容了。特别是在我们聊天的时候,有了 Quickness 的相助,可是会大大提高速度的哟。 下载页面:http://www.hanzify.org/detail.asp?SOFT_ID=6938 (汉化版)
使用工具:PEiD 0.8、W32Dasm、Ollydbg 1.09b 汉化版、Windows 自带的计算器、32bit Calculator 1.6 by cybult、UltraEdit,另外还要保存大脑清醒^^
作者:炎之川[BCG] 时间:2003.4.13 主页:http://skipli.yeah.net/
声明:此文仅用于学习及交流,若要转载请保持文章完整。
首先说明,我下载的是汉化版,官方主页上不去,天空软件站好像也没有,所以英文原版没有办法找到,下面的分析都是基于汉化版的,当然从分析的角度说,无论汉化版还是英文版都是一样的。
其实头脑清醒的话,软件的算法并不是很难,只是在一个循环中要同时考虑三个寄存器中的数字的变化,以及其他一些变化的量,稍不留神就会乱掉。最好在分析的时候,随时纪录一下值的变化。
用 PEiD 0.8 征测可知软件无壳,使用 W32Dasm 分析,可以找到注册成功、失败等提示信息,稍作分析后用 OD 装入程序,在 4029A7 处下断点,然后 Ctrl+F2 重新载入程序,F9 运行,输入注册名及假注册码并点击“注册”: Name: lovefire[BCG] Serial: 123-456-789
004029A7 > 8B7C24 14 MOV EDI,DWORD PTR SS:[ESP+14] ; Case 1 of switch 00402965 /在这里下断点 004029AB . 68 1A040000 PUSH 41A ; /ControlID = 41A (1050.) 004029B0 . 57 PUSH EDI ; |hWnd 004029B1 . FF15 5C654100 CALL DWORD PTR DS:[<&USER32.GetDlgItem>] ; \GetDlgItem 004029B7 . 8BD8 MOV EBX,EAX 004029B9 . 53 PUSH EBX ; /hWnd 004029BA . FF15 AC644100 CALL DWORD PTR DS:[<&USER32.GetWindowTex>; \GetWindowTextLengthA 004029C0 . 8BE8 MOV EBP,EAX 004029C2 . 8D45 01 LEA EAX,DWORD PTR SS:[EBP+1] 004029C5 . 50 PUSH EAX 004029C6 . E8 35500000 CALL qns31chs.00407A00 004029CB . 83C4 04 ADD ESP,4 004029CE . 8BF0 MOV ESI,EAX 004029D0 . 8D45 01 LEA EAX,DWORD PTR SS:[EBP+1] 004029D3 . 50 PUSH EAX ; /Count 004029D4 . 56 PUSH ESI ; |Buffer 004029D5 . 53 PUSH EBX ; |hWnd 004029D6 . FF15 B4644100 CALL DWORD PTR DS:[<&USER32.GetWindowTex>; \GetWindowTextA 004029DC . 56 PUSH ESI 004029DD . 68 BC0C4100 PUSH qns31chs.00410CBC ; ASCII "user_name" 004029E2 . C6042E 00 MOV BYTE PTR DS:[ESI+EBP],0 004029E6 . E8 C5E9FFFF CALL qns31chs.004013B0 004029EB . 83C4 08 ADD ESP,8 004029EE . 56 PUSH ESI 004029EF . E8 FC4F0000 CALL qns31chs.004079F0 004029F4 . 83C4 04 ADD ESP,4 004029F7 . B9 743F4100 MOV ECX,qns31chs.00413F74 004029FC . 57 PUSH EDI 004029FD . E8 6E000000 CALL qns31chs.00402A70 00402A02 . E8 C9FDFFFF CALL qns31chs.004027D0 //关键call!F7 跟进 00402A07 . 85C0 TEST EAX,EAX //比较eax是否为0,为0则注册失败 00402A09 . 6A 00 PUSH 0 ; /Style = MB_OK|MB_APPLMODAL 00402A0B . 68 A00B4100 PUSH qns31chs.00410BA0 ; |Title = "Quickness" 00402A10 . 74 28 JE SHORT qns31chs.00402A3A ; | //这里不能跳!! 00402A12 . 68 E80C4100 PUSH qns31chs.00410CE8 ; |Text = "正确。感谢。" 00402A17 . 57 PUSH EDI ; |hOwner 00402A18 . FF15 84644100 CALL DWORD PTR DS:[<&USER32.MessageBoxA>>; \MessageBoxA 00402A1E . 6A 01 PUSH 1 ; /Result = 1 00402A20 . 57 PUSH EDI ; |hWnd 00402A21 . C705 50084100 >MOV DWORD PTR DS:[410850],0 ; | 00402A2B . FF15 58654100 CALL DWORD PTR DS:[<&USER32.EndDialog>] ; \EndDialog 00402A31 . 33C0 XOR EAX,EAX 00402A33 . 5F POP EDI 00402A34 . 5E POP ESI 00402A35 . 5D POP EBP 00402A36 . 5B POP EBX 00402A37 . C2 1000 RETN 10 00402A3A > 68 C80C4100 PUSH qns31chs.00410CC8 ; |Text = "注册失败。继续试用。" 00402A3F . 57 PUSH EDI ; |hOwner 00402A40 . FF15 84644100 CALL DWORD PTR DS:[<&USER32.MessageBoxA>>; \MessageBoxA 00402A46 . 33C0 XOR EAX,EAX 00402A48 . 5F POP EDI 00402A49 . 5E POP ESI 00402A4A . 5D POP EBP 00402A4B . 5B POP EBX 00402A4C . C2 1000 RETN 10 00402A4F > 8B5424 14 MOV EDX,DWORD PTR SS:[ESP+14] 00402A53 . 68 F80C4100 PUSH qns31chs.00410CF8 ; ASCII "a_register.htm" 00402A58 . 52 PUSH EDX 00402A59 . E8 92420000 CALL qns31chs.00406CF0 00402A5E . 83C4 08 ADD ESP,8 00402A61 > 5F POP EDI ; Default case of switch 00402965 00402A62 . 5E POP ESI 00402A63 . 5D POP EBP 00402A64 . 33C0 XOR EAX,EAX 00402A66 . 5B POP EBX 00402A67 . C2 1000 RETN 10
---------------------------------------------------------- 跟进 402A02 的关键call:
004027D0 /$ 81EC 94010000 SUB ESP,194 004027D6 |. 8D4424 04 LEA EAX,DWORD PTR SS:[ESP+4] 004027DA |. 53 PUSH EBX 004027DB |. 55 PUSH EBP 004027DC |. 56 PUSH ESI 004027DD |. 57 PUSH EDI 004027DE |. 68 C8000000 PUSH 0C8 004027E3 |. 50 PUSH EAX 004027E4 |. 68 BC0C4100 PUSH qns31chs.00410CBC ; ASCII "user_name" 004027E9 |. E8 32EBFFFF CALL qns31chs.00401320 004027EE |. 83C4 0C ADD ESP,0C 004027F1 |. 85C0 TEST EAX,EAX 004027F3 |. 75 0B JNZ SHORT qns31chs.00402800 004027F5 |. 5F POP EDI 004027F6 |. 5E POP ESI 004027F7 |. 5D POP EBP 004027F8 |. 5B POP EBX 004027F9 |. 81C4 94010000 ADD ESP,194 004027FF |. C3 RETN 00402800 |> 8D8C24 DC00000>LEA ECX,DWORD PTR SS:[ESP+DC] 00402807 |. 68 C8000000 PUSH 0C8 0040280C |. 51 PUSH ECX 0040280D |. 68 B40C4100 PUSH qns31chs.00410CB4 ; ASCII "reg_key" 00402812 |. E8 09EBFFFF CALL qns31chs.00401320 00402817 |. 83C4 0C ADD ESP,0C 0040281A |. 85C0 TEST EAX,EAX 0040281C |. 75 0B JNZ SHORT qns31chs.00402829 0040281E |. 5F POP EDI 0040281F |. 5E POP ESI 00402820 |. 5D POP EBP 00402821 |. 5B POP EBX 00402822 |. 81C4 94010000 ADD ESP,194 00402828 |. C3 RETN 00402829 |> 8D7C24 14 LEA EDI,DWORD PTR SS:[ESP+14] 0040282D |. 83C9 FF OR ECX,FFFFFFFF 00402830 |. 33C0 XOR EAX,EAX 00402832 |. F2:AE REPNE SCAS BYTE PTR ES:[EDI] 00402834 |. F7D1 NOT ECX 00402836 |. 49 DEC ECX //得到注册名长度 00402837 |. 8BC1 MOV EAX,ECX //长度放入eax 00402839 |. 83F8 01 CMP EAX,1 //比较是否大于1 0040283C |. 894424 10 MOV DWORD PTR SS:[ESP+10],EAX 00402840 |. 73 0D JNB SHORT qns31chs.0040284F //大于1则继续 00402842 |. 33C0 XOR EAX,EAX 00402844 |. 5F POP EDI 00402845 |. 5E POP ESI 00402846 |. 5D POP EBP 00402847 |. 5B POP EBX 00402848 |. 81C4 94010000 ADD ESP,194 0040284E |. C3 RETN 0040284F |> 33C9 XOR ECX,ECX //ecx 清零,做计数器 00402851 |. BE 01000000 MOV ESI,1 //esi 赋值 1 00402856 |. 85C0 TEST EAX,EAX 00402858 |. BD 03000000 MOV EBP,3 //ebp 赋值 3 0040285D |. BF 05000000 MOV EDI,5 //edi 赋值 5 00402862 |. 76 4F JBE SHORT qns31chs.004028B3
下面开始就是计算注册码的循环了,开始的时候我没有理清思路,在这里转来转去,头都大了(汗…典型的菜鸟-_-b),后来经过仔细分析,发现这个循环的任务实际上是在计算三个值,并保存到不同的三个寄存器中,所以其他值都是无关紧要的中间变量,分析的时候,只要注意三个值的变化情况即可。
为了便于理解,我把这个循环分成三个部分,每一部分都写出了这一部分重要的中间变量和重要的寄存器的具体变化和算法。
下面均使用C语言的运算符表述,记住 ESI、EBP、EDI 这三个寄存器的值是我们需要掌握的,EAX 和 EDX 是关键的中间变量,name[i] 指用户名的 ASCII 值。
另外,从C语言运算符的优先级角度讲,其实并不需要用那么多的括号,但是我为了自己不看乱掉,所以每一运算均使用括号来分隔,不要数错了^_^ (最后差点数错的好像是我自己…狂汗-_-bbb)
00402864 |> 0FBE5C0C 14 /MOVSX EBX,BYTE PTR SS:[ESP+ECX+14] //逐位取注册名的ASCII值放入ebx 00402869 |. 8D0476 |LEA EAX,DWORD PTR DS:[ESI+ESI*2] //eax=esi*2+esi=esi*3 0040286C |. 33D2 |XOR EDX,EDX //edx清零 0040286E |. C1E0 04 |SHL EAX,4 //eax 左移4,等于4次*2运算 00402871 |. 2BC6 |SUB EAX,ESI //eax=eax-esi 00402873 |. BE E8030000 |MOV ESI,3E8 //给 esi 赋值为 3E8 00402878 |. 03C3 |ADD EAX,EBX //eax=eax+ebx,ebx中是注册名的ASCII值 0040287A |. F7F6 |DIV ESI //eax/esi=eax/3E8,余数放入edx
eax=(((esi*3)<<4-esi)+name[i])/0x3E8 edx=(((esi*3)<<4-esi)+name[i])%0x3E8 = 值1
0040287C |. 8BC5 |MOV EAX,EBP //eax=ebp 0040287E |. C1E0 04 |SHL EAX,4 //eax再次左移4 00402881 |. 03C5 |ADD EAX,EBP //eax=eax+ebp 00402883 |. BD E8030000 |MOV EBP,3E8 //ebp=3E8,为下面的计算再次赋值 00402888 |. 8BF2 |MOV ESI,EDX //esi=edx,edx是上面 eax mod 3E8 得到的值 0040288A |. 8D1443 |LEA EDX,DWORD PTR DS:[EBX+EAX*2] //edx=ebx+eax*2,ebx是注册名ASCII 0040288D |. 03C2 |ADD EAX,EDX //eax=eax+edx 0040288F |. 33D2 |XOR EDX,EDX //edx 清零 00402891 |. F7F5 |DIV EBP //eax=eax/ebp=eax/3E8,余数放入edx
eax=((((ebp<<4)+ebp)*2)+name[i])+((ebp<<4)+ebp) edx=(((ebp<<4)+ebp)+(name[i]+(((ebp<<4)+ebp)*2)))%0x3E8 = 值2 esi=值1 (!)
00402893 |. 8D04FF |LEA EAX,DWORD PTR DS:[EDI+EDI*8] //eax=edi+edi*8=edi*9 00402896 |. C1E0 03 |SHL EAX,3 //eax 左移3 00402899 |. 2BC7 |SUB EAX,EDI //eax=eax-edi 0040289B |. BF E8030000 |MOV EDI,3E8 //edi=3E8 004028A0 |. 03C3 |ADD EAX,EBX //eax=eax+ebx 004028A2 |. 8BEA |MOV EBP,EDX //ebp=edx,edx是上面 eax mod 3E8 得到的值 004028A4 |. 33D2 |XOR EDX,EDX //edx 清零 004028A6 |. F7F7 |DIV EDI //eax=eax/edi
eax=(edi*8<<3-edi+name[i])/0x3E8 ebp=值2 (!) edx=(edi*8<<3-edi+name[i])%3E8 = 值3
004028A8 |. 8B4424 10 |MOV EAX,DWORD PTR SS:[ESP+10] //注册名长度放入eax 004028AC |. 41 |INC ECX //计数器+1 004028AD |. 3BC8 |CMP ECX,EAX //比较计数器与注册名长度 004028AF |. 8BFA |MOV EDI,EDX //edi=edx 004028B1 |.^72 B1 \JB SHORT qns31chs.00402864 //没有取完就跳回去继续循环
edi=值3 (!)
004028B3 |> 57 PUSH EDI 004028B4 |. 55 PUSH EBP 004028B5 |. 56 PUSH ESI 004028B6 |. 8D4424 20 LEA EAX,DWORD PTR SS:[ESP+20] 004028BA |. 68 A40C4100 PUSH qns31chs.00410CA4 ; ASCII "%03d-%03d-%03d" //注册码格式,十进制输出,宽度为3 004028BF |. 50 PUSH EAX 004028C0 |. E8 BB500000 CALL qns31chs.00407980 //将 esi、ebp、edi 分别作为注册码的第一、二、三部分,合并起来成为完整的注册码。 004028C5 |. 83C4 14 ADD ESP,14 004028C8 |. 8DB424 DC00000>LEA ESI,DWORD PTR SS:[ESP+DC] //假码放入esi 004028CF |. 8D4424 14 LEA EAX,DWORD PTR SS:[ESP+14] //真码放入eax 004028D3 |> 8A10 /MOV DL,BYTE PTR DS:[EAX] //下面开始比较注册码 004028D5 |. 8A1E |MOV BL,BYTE PTR DS:[ESI] 004028D7 |. 8ACA |MOV CL,DL 004028D9 |. 3AD3 |CMP DL,BL 004028DB |. 75 30 |JNZ SHORT qns31chs.0040290D 004028DD |. 84C9 |TEST CL,CL 004028DF |. 74 16 |JE SHORT qns31chs.004028F7 004028E1 |. 8A50 01 |MOV DL,BYTE PTR DS:[EAX+1] 004028E4 |. 8A5E 01 |MOV BL,BYTE PTR DS:[ESI+1] 004028E7 |. 8ACA |MOV CL,DL 004028E9 |. 3AD3 |CMP DL,BL 004028EB |. 75 20 |JNZ SHORT qns31chs.0040290D 004028ED |. 83C0 02 |ADD EAX,2 004028F0 |. 83C6 02 |ADD ESI,2 004028F3 |. 84C9 |TEST CL,CL 004028F5 |.^75 DC \JNZ SHORT qns31chs.004028D3 004028F7 |> 33C0 XOR EAX,EAX 004028F9 |. 33C9 XOR ECX,ECX 004028FB |. 85C0 TEST EAX,EAX 004028FD |. 0F94C1 SETE CL 00402900 |. 8BC1 MOV EAX,ECX 00402902 |. 5F POP EDI 00402903 |. 5E POP ESI 00402904 |. 5D POP EBP 00402905 |. 5B POP EBX 00402906 |. 81C4 94010000 ADD ESP,194 0040290C |. C3 RETN
算法总结如下: 注册码分三个部分,分别用 esi、ebp、edi 代表。每个部分为三位十进制数字,中间用“-”分隔。 设初始值:esi=1、ebp=3、edi=5,name[i]为取得的注册用户名的 ASCII 值:
第一部分算法: esi=((((esi*3)<<4)-esi)+name[i])%0x3E8
第二部分算法: ebp=(((((ebp<<4)+ebp)*2)+name[i])+((ebp<<4)+ebp))%0x3E8
第三部分算法: edi=((((edi*9)<<3)-edi)+name[i])%0x3E8
至此 Quickness 3.1 注册算法分析完成,一组可用的注册码:Name: lovefire[BCG] Serial: 409-351-613
注册信息保存: 软件安装目录下的 qns31.ini 文件中,注册信息保存如下: [x] user_name=lovefire[BCG] reg_key=409-351-613
---------------------------------------------------------- 注册机源代码(TC 2.0)
顺便再写一下注册机,TC 2.0 编译通过。
/* KeyGen by 炎之川[BCG],2003.4.12 */
#include <stdio.h> #include <string.h> main() { char name[255]; int name_len,i; unsigned long int esi=1; unsigned long int ebp=3; unsigned long int edi=5; clrscr(); printf("\n _/_/_/ _/_/_/ _/_/_/\n _/ _/ _/ _/\n _/_/_/ _/ _/ _/_/\n _/ _/ _/ _/ _/\n_/_/_/ _/_/_/ _/_/_/\n\n -= Quickness v3.1 KeyGen by lovefire[BCG] =-\n\n\nPlease enter your name: "); gets(name); name_len=strlen(name); if (name_len>0) { for (i=0;i<name_len;i++) { esi=((((esi*3)<<4)-esi)+name[i])%0x3E8; ebp=(((((ebp<<4)+ebp)*2)+name[i])+((ebp<<4)+ebp))%0x3E8; edi=((((edi*9)<<3)-edi)+name[i])%0x3E8; } printf("\nok, try this serial: %03ld-%03ld-%03ld\n",esi,ebp,edi); printf("\n\nNOTE: serial only for test!"); printf("\nIf you like it, buy it to support the soft's author!"); } else { printf("\nI think you should tell me your name first ;)\n"); } printf("\n\nhave fun^^\nwelcome to http://skipli.yeah.net/"); getch(); } ----------------------------------------------------------
炎之川 属于中国破解组织BCG(Beginner's Cracking Group)
_/_/_/ _/_/_/ _/_/_/ _/ _/ _/ _/ _/_/_/ _/ _/ _/_/ _/ _/ _/ _/ _/ _/_/_/ _/_/_/ _/_/_/ |
|
|
|
|
|
特别声明: 本站除部分特别声明禁止转载的专稿外的其他文章可以自由转载,但请务必注明出处和原始作者。文章版权归文章原始作者所有。对于被本站转载文章的个人和网站,我们表示深深的谢意。如果本站转载的文章有版权问题请联系编辑人员,我们尽快予以更正。 |
|
|
|
|
|
责任编辑: 原点 |
投稿作者: 本站收集 |
|
|
信息来源: 网络 |
录入时间: 2005-6-1 |
|
|
|
| |
|