编译&link cygwin下调用winapi的静态库

Compile & link static library calling winapi under cygwin

我有一个 C 语言的项目,我正在将其从 unix/linux 移植到 Windows/Cygwin(使用 autotools)。 我需要调用一些 winapi 函数。这些函数需要包含一些 windows header。但是当我指定那些包括一些结构或函数的重新定义引起的很多错误时会被引发。

所以我决定创建一个静态库,其中将调用所有 winapi 函数以避免这些错误。但是我在让它工作时遇到了一些困难。

目前静态库仅包含一次 winapi 函数调用 GetAdaptersAddresses 并且如下所示:

#include <windows.h>
#include <winsock2.h>
#include <iphlpapi.h>

void win_if_scan(void)
{
  ULONG size = 10 * sizeof(IP_ADAPTER_ADDRESSES);
  IP_ADAPTER_ADDRESSES *addresses = (IP_ADAPTER_ADDRESSES *)malloc(size);

  GetAdaptersAddresses(AF_INET,
    GAA_FLAG_INCLUDE_ALL_INTERFACES |
    GAA_FLAG_INCLUDE_PREFIX |
    GAA_FLAG_SKIP_DNS_SERVER,
    NULL,
    addresses,
    &size);
}

我正在像这样编译库 libwin.a

gcc libwin.c -liphlpapi -o libwin.o
ar crv libwin.a libwin.o

从项目中我调用了 win_if_scan 函数(并且我为 libwin 包含了 header)。但是我收到错误:undefined reference to WinMain,完整消息是:

$ make > /dev/null
/usr/lib/gcc/x86_64-pc-cygwin/4.9.2/../../../../lib/libcygwin.a(libcmain.o): In function `main':
/usr/src/debug/cygwin-1.7.33-1/winsup/cygwin/lib/libcmain.c:39: undefined reference to `WinMain'
/usr/src/debug/cygwin-1.7.33-1/winsup/cygwin/lib/libcmain.c:39:(.text.startup+0x7e): relocation truncated to fit: R_X86_64_PC32 against undefined symbol `WinMain'
collect2: error: ld returned 1 exit status
make[1]: *** [libwin.o] Error 1
make: *** [all] Error 2

我不明白这里发生了什么。它是一个库,为什么它需要 main 功能?当我尝试添加时:

int main(void) { return 0; }

int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd) { return 0; }

它确实如预期的那样说 main 的多重定义。

如何获得这项工作? 或者有没有更好的方法来处理 cygwin 和 windows headers 之间的重新定义错误?

第一步是只从 libwin 生成目标文件,正如我在评论中提到的那样。 其次,库的规范-liphlpapi 必须放在项目链接在一起的地方,而不是生成目标文件的地方。这是有道理的,因为那里没有链接。 因此库存档是通过以下命令创建的:

gcc -c libwin.c -o libwin.o
ar crv libwin.a libwin.o

并且项目是这样链接在一起的:

$(exedir)/bin: $(bin-dep)
    $(CC) $(LDFLAGS) -o $@ $^ ../libwin/libwin.a $(LIBS) -liphlpapi

这是一个 Makefile 规则,你可以看到我在哪里添加了 ../libwin/libwin.a-liphlpapi

这项工作仍在进行中,很抱歉没有很好或系统的解决方案,但想法就在那里。

重要的是要记住 -liphlpapi 在 Makefile 规则中的位置,如前所述 a here