[這個(gè)貼子最后由商朝子在 2003/08/20 12:16pm 第 3 次編輯] For Delphi,讓你的注冊(cè)機(jī)變小一些 請(qǐng)拋棄VCL,如果你嫌自己用Delphi編譯出來(lái)的注冊(cè)機(jī)個(gè)頭兒過(guò)大的話(huà)... 事實(shí)上這種事情確實(shí)在發(fā)生,就在今天,就在剛才,在偶還沒(méi)吃飯的時(shí)候就看到一個(gè)網(wǎng)友在這樣抱怨... 的確如此,KeyGen的個(gè)頭兒與它的界面一樣讓人失望,簡(jiǎn)單的form再來(lái)上兩個(gè)Edit,三個(gè)Button。普普通通的界面有著360K這樣的大體積,而上過(guò)殼后那150K的體積同樣不能讓人滿(mǎn)意... 會(huì)造成這樣的結(jié)果,與VCL是脫不了關(guān)系的,事實(shí)上你用Delphi編譯一個(gè)什么也不做的程序也會(huì)有351K,否則你以為為何用Delphi寫(xiě)程序是一件相當(dāng)輕松的事情? 不過(guò)這樣沉重的代價(jià)付到我們可愛(ài)的KeyGen上就有點(diǎn)兒太過(guò)殘忍了,很難想像某天要給一個(gè)大小200K左右的軟件寫(xiě)注冊(cè)機(jī)時(shí)會(huì)是怎樣一個(gè)情景... ![]() 來(lái)吧,讓我們想點(diǎn)兒辦法,目前有一些控件可以很好的完成這個(gè)工作,用了它們后你編譯出來(lái)的可執(zhí)行文件的個(gè)頭兒要比平常小很多,不過(guò)我并不打算介紹它們 ![]() 這并不是什么太高深的東西,相信也不會(huì)很難 ![]() 扔掉了VCL,我們能使用的東西就只有Pascal語(yǔ)言與API函數(shù)了,但實(shí)際上這就已經(jīng)足夠了:) 這之前先要找個(gè)軟件做例子,大大大大大大大前天看到一個(gè)雜志在海吹敏思硬盤(pán)衛(wèi)士,就Down了一個(gè)下來(lái),后來(lái)才發(fā)現(xiàn)它對(duì)我沒(méi)任何用處。呵呵,好老的一個(gè)軟件了,一年多沒(méi)更新了吧,估計(jì)注冊(cè)機(jī)漫天飛的都有 ![]() 先用TRW2000來(lái)擺平它,啟動(dòng)軟件,我的機(jī)器碼是yjeyj239416,隨便輸入一個(gè)注冊(cè)碼,然后用hmemcpy做斷點(diǎn)...... 在004a4165處的時(shí)候,我們會(huì)發(fā)現(xiàn)程序會(huì)把正確的注冊(cè)碼裝入edx中,比如這我邊兒的是3887361024,不過(guò)我們對(duì)它不感興趣,在它之上的的004a4160處,才是我們真正感興趣的地方,正確的注冊(cè)碼就是這個(gè)CALL來(lái)計(jì)算的。 按F8跟進(jìn)去,來(lái)到004a3a08處: 0167:004a3a08 push ebp 0167:004a3a09 mov ebp,esp 0167:004a3a0b push byte +00 0167:004a3a0d push byte +00 0167:004a3a0f push byte +00 0167:004a3a11 push ebx 0167:004a3a12 push esi 0167:004a3a13 mov esi,ecx 0167:004a3a15 mov ebx,edx 0167:004a3a17 mov [ebp-04],eax 0167:004a3a1a mov eax,[ebp-04] 0167:004a3a1d call 00404050 0167:004a3a22 xor eax,eax 0167:004a3a24 push ebp 0167:004a3a25 push dword 004a3a6e 0167:004a3a2a push dword [fs:eax] 0167:004a3a2d mov [fs:eax],esp <--掛SEH,不關(guān)咱的事兒 ![]() 0167:004a3a30 lea edx,[ebp-0c] 0167:004a3a33 mov eax,[ebp-04] 0167:004a3a36 call 004085a8 <--將機(jī)器碼轉(zhuǎn)換為大寫(xiě)字符 0167:004a3a3b mov eax,[ebp-0c] 0167:004a3a3e lea ecx,[ebp-08] 0167:004a3a41 movzx edx,bx <--edx裝入Dh,即十進(jìn)制數(shù)13 0167:004a3a44 call 004a3958 <--完成計(jì)算正確注冊(cè)碼的第一步 0167:004a3a49 mov edx,esi 0167:004a3a4b mov eax,[ebp-08] 0167:004a3a4e call 004a3bdc <-完成計(jì)算正確注冊(cè)碼的第二步 0167:004a3a53 xor eax,eax 0167:004a3a55 pop edx 0167:004a3a56 pop ecx 0167:004a3a57 pop ecx 0167:004a3a58 mov [fs:eax],edx 0167:004a3a5b push dword 004a3a75 0167:004a3a60 lea eax,[ebp-0c] 0167:004a3a63 mov edx,03 0167:004a3a68 call 00403c40 0167:004a3a6d ret 0167:004a3a6e jmp 00403630 0167:004a3a73 jmp short 004a3a60 0167:004a3a75 pop esi 0167:004a3a76 pop ebx 0167:004a3a77 mov esp,ebp 0167:004a3a79 pop ebp 0167:004a3a7a ret 004a3a44處的所謂計(jì)算注冊(cè)碼的第一步就是指先對(duì)機(jī)器碼進(jìn)行一些處理,我們按F8跟進(jìn)去: 0167:004a3958 push ebp 0167:004a3959 mov ebp,esp 0167:004a395b add esp,byte -14 0167:004a395e push ebx 0167:004a395f push esi 0167:004a3960 push edi 0167:004a3961 xor ebx,ebx 0167:004a3963 mov [ebp-14],ebx 0167:004a3966 mov [ebp-0c],ecx 0167:004a3969 mov [ebp-08],edx 0167:004a396c mov [ebp-04],eax 0167:004a396f mov eax,[ebp-04] 0167:004a3972 call 00404050 0167:004a3977 xor eax,eax 0167:004a3979 push ebp 0167:004a397a push dword 004a39f9 0167:004a397f push dword [fs:eax] 0167:004a3982 mov [fs:eax],esp 0167:004a3985 mov eax,[ebp-0c] 0167:004a3988 call 00403c1c 0167:004a398d mov eax,[ebp-08] 0167:004a3990 add eax,[ebp-08] 0167:004a3993 mov [ebp-10],eax 0167:004a3996 mov eax,[ebp-04] 0167:004a3999 call 00403e9c 0167:004a399e mov edi,eax <--eax中此時(shí)裝的是機(jī)器碼的位數(shù) 0167:004a39a0 test edi,edi <--測(cè)試edi 0167:004a39a2 jng 004a39db <--為0就跳到004a39db處 0167:004a39a4 mov esi,01 <--esi置1,用于后邊兒的循環(huán) 0167:004a39a9 mov eax,[ebp-04] <--機(jī)器碼的地址裝入eax中,此時(shí)的機(jī)器碼已轉(zhuǎn)換為大寫(xiě)字符 0167:004a39ac xor ebx,ebx <--ebx清0 0167:004a39ae mov bl,[eax+esi-01] <--得到機(jī)器碼 0167:004a39b2 test esi,01 <--test esi,01可以理解為測(cè)試esi中裝的值是否為雙數(shù) ![]() 0167:004a39b8 jnz 004a39bf <--不為雙數(shù)就跳到004a39bf處 0167:004a39ba add ebx,[ebp-08] <--ebp-08處裝的是Dh,也就是那個(gè)13,用其加上ebx中的機(jī)器碼的ASCII碼 0167:004a39bd jmp short 004a39c2 <--無(wú)條件跳轉(zhuǎn)到004a39bd處 0167:004a39bf add ebx,[ebp-10] <--ebp-10處裝的是1Ah,即十進(jìn)制數(shù)26,用其加上ebx中裝的機(jī)器碼的ASCII碼 0167:004a39c2 lea eax,[ebp-14] 0167:004a39c5 mov edx,ebx 0167:004a39c7 call 00403dc4 0167:004a39cc mov edx,[ebp-14] 0167:004a39cf mov eax,[ebp-0c] 0167:004a39d2 call 00408594 <--將得到的ASCII碼與26或13相加后的結(jié)果保存起來(lái) 0167:004a39d7 inc esi <--esi加上1 0167:004a39d8 dec edi <--edi減去1 0167:004a39d9 jnz 004a39a9 <--edi不為零,也就是機(jī)器碼還沒(méi)有全計(jì)算完,就跳回004a39a9處繼續(xù) 0167:004a39db xor eax,eax 0167:004a39dd pop edx 0167:004a39de pop ecx 0167:004a39df pop ecx 0167:004a39e0 mov [fs:eax],edx 0167:004a39e3 push dword 004a3a00 0167:004a39e8 lea eax,[ebp-14] 0167:004a39eb call 00403c1c 0167:004a39f0 lea eax,[ebp-04] 0167:004a39f3 call 00403c1c 0167:004a39f8 ret 0167:004a39f9 jmp 00403630 0167:004a39fe jmp short 004a39e8 0167:004a3a00 pop edi 0167:004a3a01 pop esi 0167:004a3a02 pop ebx 0167:004a3a03 mov esp,ebp 0167:004a3a05 pop ebp 0167:004a3a06 ret 這個(gè)CALL的作用呢,就是這樣的:將機(jī)器碼中的每個(gè)字符的ASCII碼,加上26或13(單位,如第1、3、5位加上26,雙位,如第2、4、6位加上13),并將之保存起來(lái),以便后面的應(yīng)用。與我的機(jī)器對(duì)應(yīng)的是:sW_fd?MFN>P 好的,我們繼續(xù),等到返回后,接著走兩步,我們就會(huì)到達(dá)004a3a4e處,在這個(gè)CALL里,會(huì)完成計(jì)算注冊(cè)碼的最后一步,一起去看看吧: 0167:004a3bdc push ebp 0167:004a3bdd mov ebp,esp 0167:004a3bdf add esp,byte -10 0167:004a3be2 push ebx 0167:004a3be3 mov [ebp-08],edx 0167:004a3be6 mov [ebp-04],eax 0167:004a3be9 mov eax,[ebp-04] 0167:004a3bec call 00404050 0167:004a3bf1 xor eax,eax 0167:004a3bf3 push ebp 0167:004a3bf4 push dword 004a3c62 0167:004a3bf9 push dword [fs:eax] 0167:004a3bfc mov [fs:eax],esp 0167:004a3bff mov eax,[ebp-08] 0167:004a3c02 call 00403c1c 0167:004a3c07 mov ebx,1e61 <--ebx裝入1e61h,即十進(jìn)值數(shù)7777 0167:004a3c0c mov eax,[ebp-04] 0167:004a3c0f call 00403e9c 0167:004a3c14 mov edx,eax 0167:004a3c16 test edx,edx 0167:004a3c18 jna 004a3c32 0167:004a3c1a mov eax,01 <--eax中裝入1,用于下邊兒的循環(huán) 0167:004a3c1f mov ecx,[ebp-04] <--ebp-04中裝的正是字符串sW_fd?MFN>P的地址 0167:004a3c22 movzx ecx,byte [ecx+eax-01] <--ecx中裝入本輪用于計(jì)算的字符 0167:004a3c27 add ecx,eax <--ecx加上eax的值 0167:004a3c29 imul ecx,ebx <--乘以ebx,第一次的時(shí)候ebx中的值為前邊裝入的7777 0167:004a3c2c mov ebx,ecx <--結(jié)果裝入ebx中 0167:004a3c2e inc eax <--eax加上1,用于得到下一個(gè)字符 0167:004a3c2f dec edx <--edx減去1 0167:004a3c30 jnz 004a3c1f <--不為零就跳回去繼續(xù) 0167:004a3c32 mov eax,[ebp-08] 0167:004a3c35 push eax 0167:004a3c36 mov [ebp-10],ebx 0167:004a3c39 mov byte [ebp-0c],00 0167:004a3c3d lea edx,[ebp-10] 0167:004a3c40 xor ecx,ecx 0167:004a3c42 mov eax,004a3c78 0167:004a3c47 call 004098ec 0167:004a3c4c xor eax,eax 0167:004a3c4e pop edx 0167:004a3c4f pop ecx 0167:004a3c50 pop ecx 0167:004a3c51 mov [fs:eax],edx 0167:004a3c54 push dword 004a3c69 0167:004a3c59 lea eax,[ebp-04] 0167:004a3c5c call 00403c1c 0167:004a3c61 ret 0167:004a3c62 jmp 00403630 0167:004a3c67 jmp short 004a3c59 0167:004a3c69 pop ebx 0167:004a3c6a mov esp,ebp 0167:004a3c6c pop ebp 0167:004a3c6d ret 嘿嘿,“平民算法”嘛 ![]() 現(xiàn)在我們手里所掌握的,已足夠?qū)懗鲎?cè)機(jī)了,通常情況下,我們可以聲明這樣一個(gè)函數(shù)用于計(jì)算正確的注冊(cè)碼: function GetKey(Name: String): String; var User,Serial:String; i:integer; ASC,jia,Zhi:Cardinal; begin zhi:=7777; User:=UpperCase(Name); if Length(User)=0 then Result:=‘請(qǐng)輸入您的機(jī)器碼...‘ else begin for i:=1 to Length(User) do begin ASC:=Ord(User[i]); if i mod 2 =0 then ASC:=ASC+13 else ASC:=ASC+26; ASC:=ASC+i; jia:=ASC*zhi; zhi:=jia; end; Serial:=inttostr(zhi); Result:=Serial; end; end; 然后我們就可以在Delphi中通過(guò)VCL來(lái)增大我們KeyGen的體積了,但今天我們不準(zhǔn)備這么做 ![]() OK,啟動(dòng)你的Delphi,然后新建一個(gè)應(yīng)用程序,然后就在代碼窗口中右鍵Unit1單元->Close Page,接著選No來(lái)把Unit1.Pas給刪除掉。 接著選菜單View->Units,雙擊Project1打開(kāi)它,我們就靠它吃飯了 ![]() 然后我們可以把它的內(nèi)容給全部刪掉,只留下三行: program KeyGen; begin end. 呵呵,按F9吧,你會(huì)什么也感覺(jué)不到,但事實(shí)上這已經(jīng)是一個(gè)程序了。 如果你看一下所生成的KeyGen.exe的大小,你會(huì)發(fā)現(xiàn),盡管我們什么也沒(méi)有寫(xiě),可它已經(jīng)有了8K大。這是因?yàn)樵谌魏蜠ELPHI的目標(biāo)程序中,都會(huì)自動(dòng)包含System單元中的代碼,如果你用Delphi5的話(huà),它會(huì)是16K。在這小小的8K中,包含了支撐整座DELPHI大廈的基石,VeryVery牢靠。 以后隨著我們代碼的增加,程序的大小也會(huì)從8K開(kāi)始慢慢跟著增加 ![]() 讓我們開(kāi)始吧,我把注冊(cè)機(jī)與資源文件的代碼貼出來(lái)(修正了dREAMtHEATER指出的兩處錯(cuò)誤,同時(shí)代碼更緊湊了一些^_^): {-<KeyGen.dpr>------------------------------------Code made >> Suunb[CCG]-----.} program KenGen; {$R ‘KenGen.res‘ ‘KenGen.rc‘} (*感謝dREAMtHEATER老哥的指正:)*) uses Windows, Messages; {---------------------------------------------------------<引入相關(guān)單元>------;} {----Windows.pas里面包含了常量與API的定義,Messages.pas中則包含了消息的定義----;} const IDI_CCG = 1 ; IDC_EDIT_CODE = 3001 ; IDC_EDIT_SERIAL = 3002 ; IDC_BUTTON_GENERATE = 3003 ; IDC_BUTTON_ABOUT = 3004 ; IDC_BUTTON_EXIT = 3005 ; IDC_ABOUT_BUTTON_GREETING = 3006 ; szDlgName = ‘KeyGenMain‘; szAboutDlg = ‘About‘; {---------------------------------------------------------<以上為常量定義>----;} function GetKey(S: String): String; var Code,Serial:String; i:integer; ASC,Q,P:Cardinal; NN:array [0..512] of Char; begin P:=7777; Code:=S; if Length(Code)=0 then Result:=‘請(qǐng)輸入您的機(jī)器碼...‘ else begin CharUpperBuff(PChar(Code),Length(S)); for i:=1 to Length(Code) do begin ASC:=Ord(Code[i]); if i mod 2 =0 then ASC:=ASC+13 else ASC:=ASC+26; ASC:=ASC+i; Q:=ASC*P; P:=Q; end; if P>2147483647 then begin if (P>2147483647) and (P<3000000000) then begin P:=P-2000000000; if P<100000000 then wvsprintf(NN,‘0%d‘,@P) else wvsprintf(NN,‘%d‘,@P); Serial:=‘2‘+String(NN); end else if (P>3000000000) and (P<4000000000) then begin P:=P-3000000000; if P<100000000 then wvsprintf(NN,‘0%d‘,@P) else wvsprintf(NN,‘%d‘,@P); Serial:=‘3‘+String(NN); end else if (P>4000000000) and (P<5000000000) then begin P:=P-4000000000; if P<100000000 then wvsprintf(NN,‘0%d‘,@P) else wvsprintf(NN,‘%d‘,@P); Serial:=‘4‘+String(NN); end end else begin wvsprintf(NN,‘%d‘,@P); Serial:=String(NN); end; Result:=Serial; end; end; {-------------------------------<上面的這個(gè)函數(shù)就是用來(lái)計(jì)算正確的注冊(cè)碼的>----;} function AboutProc (Wnd: hWnd; Msg, wParam: Word; lParam: LongInt) :LongInt; stdcall; begin Case Msg of WM_CLOSE:EndDialog(Wnd,0); WM_COMMAND: begin if (LOWORD(wParam))=IDC_ABOUT_BUTTON_GREETING then EndDialog(Wnd,0) else Result:=1; end; end; Result:=0; end; {-<上面的函數(shù)用于處理“關(guān)于”對(duì)話(huà)框接收到的消息,stdcall表示其為一個(gè)回調(diào)函數(shù)>-;} function DlgProc (Wnd: hWnd; Msg, wParam: Word; lParam: LongInt) :LongInt; stdcall; var Code:array[0..512] of Char; Serial:String; begin Case Msg of WM_CLOSE: EndDialog(Wnd,0); WM_INITDIALOG: begin SendMessage(Wnd,WM_SETICON, ICON_SMALL, LoadIcon(hInstance, PChar(IDI_CCG))); end; WM_COMMAND: begin case (LOWORD(wParam)) of IDC_BUTTON_about:DialogBoxParam(hInstance,szAboutDlg,Wnd, @AboutProc, 0); IDC_BUTTON_GENERATE: begin GetDlgItemText(Wnd,IDC_EDIT_CODE,Code,512); Serial:=GetKey(Code); SetDlgItemText(Wnd,IDC_EDIT_SERIAL,PChar(Serial)); end; IDC_BUTTON_EXIT:SendMessage(Wnd,WM_CLOSE, 0, 0); end; end; else Result:=1; end; Result:=0; end; {---------------------------------<主窗口的消息循環(huán),同樣為一個(gè)回調(diào)函數(shù)>------;} begin DialogBoxParam(hInstance,szDlgName,0,@DlgProc,0); (*程序開(kāi)始,調(diào)用DialogBoxParam來(lái)創(chuàng)建程序的主對(duì)話(huà)框,hInstance是Delphi提供 給我們的全局變量,不必再去調(diào)用GetModuleHandle函數(shù)*) {經(jīng)dREAMtHEATER指正,Delphi會(huì)自動(dòng)調(diào)用ExitProcess函數(shù),所以將其免去...} end. //------------------------------------------------------------------------------ //------------------the Shang Dynasty (16th―11th century B.C.)----------------- //--------------------------------------------------------------------Code end-- 下面再把資源文件也貼出來(lái): //-KeyGen.rc-------------------------------------------------by Suunb[CCG]-- #define CCG_ICO 1 #define Su_ICO 2 #define IDC_EDIT_CODE 3001 #define IDC_EDIT_SERIAL 3002 #define IDC_BUTTON_GENERATE 3003 #define IDC_BUTTON_ABOUT 3004 #define IDC_BUTTON_EXIT 3005 #define IDC_ABOUT_BUTTON_GREETING 3006 //-------------------------------------------<以上是一些常量的定義>--------- KeyGenMain DIALOGEX 10, 10, 163, 94 style DS_CENTER | WS_MINIMIZEBOX | WS_VISIBLE | WS_CAPTION | WS_SYSMENU CAPTION "CCG KeyGen for 敏思硬盤(pán)衛(wèi)士 v2.2" FONT 9, "Arial" BEGIN GROUPBOX "register",-1,5,3,151,44,BS_FLAT LTEXT "Code:",-1,10,14,23,10 EDITTEXT IDC_EDIT_CODE,35,13,115,12,ES_AUTOHSCROLL,WS_EX_DLGMODALFRAME LTEXT "Serial:",-1,10,29,23,10 EDITTEXT IDC_EDIT_SERIAL,35,29,115,12,ES_AUTOHSCROLL,WS_EX_DLGMODALFRAME PUSHBUTTON "gENERATE",IDC_BUTTON_GENERATE,13,54,39,11,BS_CENTER | BS_FLAT PUSHBUTTON "aBOUT",IDC_BUTTON_ABOUT,61,54,39,11,BS_CENTER | BS_FLAT PUSHBUTTON "eXIT",IDC_BUTTON_EXIT,108,54,39,11,BS_CENTER | BS_FLAT LTEXT "wish it becorning more prosperous every day!",-1,10,77, 143,9 GROUPBOX "Lovely CCG",-1,6,69,151,21,BS_CENTER | BS_FLAT END //--------------------------------------------<以上是程序的的主窗口對(duì)話(huà)框>--------- About DIALOGEX 10, 10, 143, 141 style DS_MODALFRAME | DS_CENTER | WS_POPUP | WS_VISIBLE | WS_CAPTION | WS_SYSMENU CAPTION "The Shang Dynasty (16th - 11th century B.C.)" FONT 9, "Arial" BEGIN ICON 1,-1,13,14,20,20 LTEXT "Code made:",-1,25,3,41,8 LTEXT "by",-1,40,13,8,8 LTEXT "Date:April 26t,2003",-1,48,25,61,9 CONTROL "",-1,"Static",SS_ETCHEDHORZ,4,36,132,1 LTEXT "Personal Individual Greetings go to:",-1,19,41,114,8 LTEXT "Sun Bird - JOJO - KANXUE",-1,11,51,85,9 LTEXT "pll621 - zmworm - yyxzz",-1,46,62,73,9,SS_CENTERIMAGE LTEXT "Personal Group Greetings go to:",-1,22,87,105,8 LTEXT "BCG FCG iPB DFCG DCM",-1,28,101,85,9,WS_BORDER LTEXT "Suunb[CCG]",-1,50,12,45,11,0,WS_EX_DLGMODALFRAME LTEXT "and other CCG guys...",-1,29,74,92,9,SS_CENTERIMAGE ICON 2,-1,112,14,20,20 PUSHBUTTON "gREETING",IDC_ABOUT_BUTTON_GREETING,50,120,41,12,BS_FLAT END //--------------------------------------------<這個(gè)是“關(guān)于”對(duì)話(huà)框>-------- 你可以用記事本將這兩個(gè)文件分別保存為KeyGen.dpr、KeyGen.rc,之后將KeyGen.rc與KeyGen.dpr放置同一目錄下用Delphi編譯KeyGen.dpr即可... 看一下編譯后的KeyGen的大小吧,20.5K,再來(lái)個(gè)殼就只有10幾K了,雖然界面并不怎么漂亮 ![]() OK,就到這兒。 |
|
來(lái)自: intruder > 《技術(shù)文章》