4.4.1模块入口代码
编译器为每个Delphi模块(可执行程序、动态链接库或包)生成模块入口代码。同类型的程序或模块,其入口代码是相同的。
编译器为可执行程序.EXE)生成的入口代码为:
Project1.dpr.9:begin
0044CA9855push ebp
0044CA99 8BEC mov ebp,esp
0044CA9B 83C4F0 add esp,-$10
0044CA9EB8B8C84400 mov eax,$0044c8b8//传入单元初始化表地址
0044CAA3 E80080FBFF cal10InitExe //调用模块初始化例程
为动态链接库(.DLL)生成的入口代码为:
Project2.dpr.19:begin
00B81FD4 55 push ebp
00B81FD5 8BEC mov ebp,esp
00B81FD7 83C4C4add esp,-$3c
00B81FDA B85C1FB800 mov eax,$00b81f5c//传入单元初始化表地址
00B81FDP E8F435FFEF call InitLib //调用模块初始化例程
为包(.BPL)生成的入口代码为:
Packagel.dpk.33:end,
//跳转到模块初始化例程,是一个相对寻址的短跳转
04A512B0 E9C3FEFFFF jmp eInitPkg
除了包之外,入口代码都主要负责传入单元初始化表的地址。其后的所有初始化工作,全部交由System.pas中的模块初始化例程完成。这些例程分别是:
//define in SysInit. pas
procedure _InitLib;
procedure _InitExe(InitTable: Pointer);
function InitPkg(Hinst: Integer; Reason: Integer;Resvd: pointer): LongBool; stdcall;
4.4.2编译器决定的程序执行流程
编译器总是按照固定的流程和代码来生成项目的二进制文件。例如一个简单的项目:
Drogram GuiProg;
uses
Dialogs;
begin
ShowMessage(' Test');
end.
它编译的代码为:
GuiProg. dpr.4:begin
00451F5C 55push ebp
00451F5D 8BEC mov ebp, esp
00451F5F 83C4F0 add esp,-$10
00451F62 B8841D4500 mov eax,$00451d84
00451F67 E8F43CFBFF ca11 eInitExe
GuiProg. dpr.5:ShowMessage(' Test');
00451F6C B8841F4500 mov eax,$00451f84
00451F71 E8D2FBFFFF call ShowMessage
Guiprog.dpr.6:end.
00451F76 E8E11DFBFF call @Halto
上面的汇编代码是顺序执行的,编译器只是在这个流程的结尾加入了一行“call
@Halt0”。Delphi并没有刻意为模块准备结束化代码或者指定结束化代码的地址,它只须按照这样的方式来编译即可。
对于可执行程序(.EXE)来说,用户在项目文件(.DPA)的begin..end之间加入的代码决定了其所有的功能,而编译器只是把.DPR文件按照.EXE方式生成,并将用户编写的代码编译成二进制代码,放到入口代码和结束时的“cal1 @Halto”之间。
其他类型的文件与可执行程序并不一致。在下一章“面向Windows开发的基本实现”中,将详细地讲述所有的模块初始化与结束化例程,以及它们不同执行流程。