WinMain是如何被调用的


WinMain函数

WinMain函数原型

Win32应用程序的入口函数为WinMain,函数原型在WinBase.h文件中:

int WINAPI WinMain (

    _In_ HINSTANCE hInstance,
    _In_opt_ HINSTANCE hPrevInstance,
    _In_ LPSTR lpCmdLine,
    _In_ int nShowCmd
);

WinMain函数原型中的符号

符号 描述 其它
int 返回值  
WINAPI 函数调用约定

WINAPI宏展开:

#define WINAPI      __stdcall

WinMain 函数名  
HINSTANCE hInstance 当前应用程序实例句柄

HINSTANCE宏展开:

DECLARE_HANDLE(HINSTANCE);

#define DECLARE_HANDLE(name) struct name##__{int unused;}; typedef struct name##__ *name

HINSTANCE hPrevInstance 没有意义。 它用于 16 位Windows,但现在始终为零  
LPSTR lpCmdLine 命令行参数  
int nShowCmd 窗口显示控制参数  

程序入口

Win32编程入口为什么是WinMain呢?

C/C++语言中,编程入口为什么是main,而不是其它的呢?

CPU控制权传递

CPU是计算机的大脑,CPU读取指令后执行指令,即:控制CPU读指令的位置,就控制了CPU。

计算机加电后,CPU从BIOS指定位置读取并执行开机程序第一条指令,即,CPU被开机程序控制,开机程序获得了CPU的控制权。

开机程序获得CPU控制权后,其它程序如何获得CPU控制权呢?开机程序约定,我会去读取哪个地址的指令来执行。

其它程序把指令写入开机程序约定的地方,CPU的控制权就到了其它程序。

CPU控制权一步一步传递到操作系统。

CPU控制权到了操作系统后,其它程序想要获得控制权,就需要遵循操作系统的约定。

操作系统约定包括PE格式等,首先按PE格式生成可执行文件,如.exe。

双击或者调用进程创建函数启动程序,操作系统将可执行程序装载到内存,将下一条读取地址指向程序的入口地址,这样CPU的控制权传递到了可执行程序。

PE文件生成

要执行自定义的指令,就要遵循与操作系统的约定。

操作系统要求装载PE格式的文件,可以通过编译工具编译生成PE格式文件,如.exe。

编程语言

编译工具负责生成PE格式文件。

编译工具约定了输入的格式,即编程语言。

程序入口

C/C++编程语言的编译工具约定,可执行程序的入口默认是mainCRTStartup,该入口函数可以通过设置重新指定。

入口函数

C/C++编程语言约定了程序的入口函数,如main。程序入口(mainCRTStartup)调用入口函数main。

不论是main还是WinMain都只是一种约定,可以根据需要制定约定。

WinMain和main

C/C++语言的入口函数都是main,为什么Win32是WinMain?

程序入口

程序入口可以通过编译器指定,根据需要指定入口点,控制台默认入口为mainCRTStartup,Win32默认入口为WinMainCRTStartup,也设置为自定义函数。

main调用

main调用可以通过exe_main.cpp,exe_common.inl文件查看

mainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->main()。

WinMain调用

WinMain调用可以通过exe_winmain.cpp,exe_common.inl文件查看

WinMainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->WinMain()。

为什么是WinMain不是main?

不管是main还是WinMain都是编程语言的约定,约定内容包括返回值,调用约定,函数名,函数参数。

main和WinMain参数都包含命令行参数。

对Win32应用程序来说,主函数大概率会用到应用程序实例,即HINSTANCE hInstance参数。

为了给Win32应用程序开发提供便捷,制定了更适合Win32应用程序开发的接口约定WinMain。

应用程序实例也可以通过GetModuleHandle获取,也就是说,只要通过设置,用main或者自定义的函数作为主函数也可以编写Win32应用程序。

构建原理

入口函数缺失

在控制台程序中,不写main函数,编译报错:

1>MSVCRTD.lib(exe_main.obj) : error LNK2019: 无法解析的外部符号 main,函数 "int __cdecl invoke_main(void)" (?invoke_main@@YAHXZ) 中引用了该符号

在Win32程序中,不写WinMain函数,编译报错:

1>MSVCRTD.lib(exe_winmain.obj) : error LNK2019: 无法解析的外部符号 WinMain,函数 "int __cdecl invoke_main(void)" (?invoke_main@@YAHXZ) 中引用了该符号

LIB文件

上面的MSVCRTD.lib(exe_main.obj) 是什么意思?

.lib文件可以是静态链接库,也可以是导入库。

静态链接库包含可执行代码,导入库包含导入库信息。可以通过lib命令查看lib文件的类型。

MSVCRTD.lib是一个静态链接库。

exe_main.obj是MSVCRTD.lib中的一部分。

链接

链接时,由于程序入口为mainCRTStartup或WinMainCRTStartup,因此链接时会对程序入口的依赖进行检查。

main链接时,exe_main.obj包含mainCRTStartup()。

mainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->main()。

如果不写main函数,链接找不到main函数,链接报错。

main链接时,exe_winmain.obj包含WinMainCRTStartup()。

WinMainCRTStartup()->__scrt_common_main()->__scrt_common_main_seh()->invoke_main()->WinMain()。

如果不写WinMain函数,链接找不到WinMain函数,链接报错。