如果有问题,请加QQ群 891339868 进行交流
在使用mingw进行编译时,出现了以下的链接错误:
警告: 通过链结到 _SKF_EnumDev 以解决 _SKF_EnumDev@12
使用 --enable-stdcall-fixup 来禁用这些警告
使用 --disable-stdcall-fixup 来禁用这些修正
什么意思呢?
这要从mingw说起:
mingw是gcc的Windows版本,功能和性能都挺好,很多win下面的IDE都是将mingw封装了进去,但是mingw下使用和创建DLL倒要特别注意,问题主要集中在gcc编译器对于DLL的函数输入及输出的名称修饰、调用协议上和VC编译器是有很大的区别的。
一、mingw怎么使用一个标准的windows动态库呢?
gcc/g++编译器遇到代码中__stdcall修饰的函数名,会自动将其函数名在链接时设置为函数名@nn,nn是函数参数栈字节数,另外,gcc/g++编译器/链接器在链接时,其实不需要DLL的导入库(import lib),因为它们可以直接从DLL链接,这样更方便,省去了很多从DLL如何生成符合格式要求的.a导入库等问题。只需要在gcc/g++参数中加入-Wl,--enable-stdcall-fixup 即可直接从DLL文件本身完成链接。这里要注意,-Wl参数指示gcc链接器需要采用后面的链接控制参数(以逗号分割),--enable-stdcall-fixup告诉gcc链接器需要导入的DLL函数的名字需要自动在尾部加上@nn格式的后缀,以便符合gcc/g++对__stdcall函数名的扩展规范。
如果不加--enable-stdcall-fixup,gcc/g++总是会报链接错误,因为gcc/g++将代码中需要从DLL导入的函数名后面都强制加了@nn,但DLL中的函数名不带@nn,没有-enable-stdcall-fixup,很有可能就会出错,即使不出错,也会有很多警告。
二、mingw如何创建一个标准的windows的动态库呢?
mingw中创建标准的DLL,应该使用__declspec(dllexport),包括extern "C"等都和VC一样的,但是要注意,这样生成的DLL,导入的函数名尾部都带有@nn,为了要去除它们,必须在链接器参数设置中使用-Wl,--kill-at,它告诉链接器创建DLL时导出的函数名尾部不要带@nn。
从上面两点可以看出来,mingw默认在使用和生成windows动态库时,都在函数名后面增加了函数参数栈字节数,而在windows的标准动态库中,是不包含这些东西的,所以,为了要和windows标准的动态库兼容,创建库时,需要使用-Wl,--kill-at参数,使用库时,需要使用-Wl,enable-stdcall-fixup,为了使用简单,最好在创建和使用DLL时,都加入-Wl,--kill-at,enable-stdcall-fixup这些参数。
OK,今天就记录到这里!