在main函数启动前执行代码-Release版本
这里记录一下在使用VC编写在main函数前执行代码时遇到的坑。
由于项目需要,需要在main函数之前执行一些代码初始化环境变量。于是在网上找到了一个解决方案。代码如下:
int __cdecl beforeMain()
{
printf(__FUNCTION__ "\n");
return 0;
}
typedef int(__cdecl* _PIFV)(void);
#pragma section(".CRT$XIU", long, read)
EXTERN_C __declspec(allocate(".CRT$XIU")) _PIFV bf[] = { beforeMain };
然后debug模式下运行,果然出结果了。正当兴高采烈的去发布release版本后,运行发现,和debug版本的结果不一样。beforeMain函数竟然没运行。奇了怪了。于是乎各种搜索,终于,找到了问题所在。release下的版本需要加上这么一句
#pragma comment(linker,"/INCLUDE:_bf")
终于可以了。完整代码如下(包含TLS部分,一样的问题,一样的处理方式):
#include <stdio.h>
#include <Windows.h>
//TLS部分
void NTAPI __stdcall TLS_CALLBACK(PVOID DllHandle, DWORD Reason, PVOID Reserved)
{
///DLL_PROCESS_ATTACH、DLL_THREAD_ATTACH、DLL_THREAD_DETACH和DLL_PROCESS_DETACH
printf(__FUNCTION__ " Reason:%d\n", Reason);
}
#pragma comment(linker,"/INCLUDE:__tls_used")
#pragma section(".CRT$XLB",long,read)
EXTERN_C __declspec(allocate(".CRT$XLB")) PIMAGE_TLS_CALLBACK _tls_callback[] = { TLS_CALLBACK };
#pragma comment(linker,"/INCLUDE:__tls_callback")
int __cdecl beforeMain()
{
printf(__FUNCTION__ "\n");
return 0;
}
typedef int(__cdecl* _PIFV)(void);
#pragma section(".CRT$XIU", long, read)
EXTERN_C __declspec(allocate(".CRT$XIU")) _PIFV bf[] = { beforeMain };
#pragma comment(linker,"/INCLUDE:_bf")
class A
{
public:
A()
{
printf(__FUNCTION__ "\n");
}
};
A a;
int main()
{
printf("Hello World!\n");
}
为什么加上
#pragma comment(linker,"/INCLUDE:_bf")
就可以了呢。原来在release编译下,默认打开了
/OPT:REF
链接开关,而beforeMain
函数没有任何地方调用,因此编译器把beforeMain函数给优化掉了。因此也可以把/OPT:REF
开关去掉,也可以达到相同效果。
改成
而#pragma comment(linker,"/INCLUDE:_bf")
就是告诉链接器,这部分代码不能被优化掉。个人比较推荐 #pragma comment(linker,"/INCLUDE:_bf")
的办法。
希望以上能帮到有需要的朋友