在Mac上使用CMake和GCC构建静态库?

17

大家好,

我有一个静态库,然后我会将其与我的应用程序链接。我的开发环境是CMake,GCC(Linux、Mac)和MinGW(Windows)。

我可以在Linux和Windows上编译静态库而没有任何问题。 (在我的应用程序中甚至可以构建共享库)。

编辑:我将库编译为共享库,它也能够正常工作了!

我已经按照以下方式配置了CMakeFile来构建静态库:

add_library(centi STATIC ${base_srcs} ${crv_srcs} ${node_srcs} ${trnk_srcs} ${defl_srcs} ${infl_srcs} ${track_srcs} ${callback_srcs} ${extract_srcs})
在链接阶段,它会给出以下错误并以某种方式构建“libcenti.a”。
Linking C static library
lib/libcenti.a /usr/bin/ranlib: file:
lib/libcenti.a(crv_in_rgn_to_bnry_img.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_initialize_by_circle.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_initialize_flgs.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_nodal_interval_min_and_max.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_remove_all_nodes.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_reset_nodal_forces.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_set_center_coords.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_set_left_ptch_rgn_pixs.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_set_out_rgn_mean_and_var.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_set_para.c.o) has
no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_set_right_ptch_rgn_pixs.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_to_in_rgn_hist.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(crv_to_out_rgn_pixs.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(trnk_initialize_by_circle.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(trnk_initialize_by_image_frame.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(trnk_stk_paint_nodes_and_pixs.c.o)
has no symbols /usr/bin/ranlib: file:
lib/libcenti.a(trnk_stk_to_inner_defl_ordn.c.o)
has no symbols
但是当我将上述库与我的应用程序链接时,它会给出"未定义符号"错误:
Undefined symbols:  
"_setActiveDrawingTrunk", referenced
from:
      RzPluginAlgoCnty::initCallBacks()     
in RzPluginAlgoCnty.cpp.o
      RzPluginAlgoCnty::clearCallBacks()    
in RzPluginAlgoCnty.cpp.o
      _trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _bg_trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _bg_trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _extract_contour_update_tracking in
libcenti.a(extract_contour_update_tracking.c.o)
      _extract_contour_update_tracking in
libcenti.a(extract_contour_update_tracking.c.o)
"_updateCurveUICallBack", referenced
from:
      RzPluginAlgoCnty::initCallBacks()     
in RzPluginAlgoCnty.cpp.o
      RzPluginAlgoCnty::initBulkCallBacks() 
in RzPluginAlgoCnty.cpp.o
      RzPluginAlgoCnty::clearCallBacks()    
in RzPluginAlgoCnty.cpp.o
      _bg_trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _bg_trnk_trck_ordn in libcenti.a(trnk_trck_ordn.c.o)
      _crv_update_1time in libcenti.a(crv_update_ordn.c.o)
      _crv_update_1time in libcenti.a(crv_update_ordn.c.o) ld:
symbol(s) not found

有什么建议吗?在 Mac 上构建静态库时,是否需要添加任何特殊参数?

编辑: 我将库编译为共享库,它可以正常工作了!


也通过编译为共享库来解决了这个问题。 - Greg M. Krsak
在链接阶段,它会给出以下错误并构建“libcenti.a”....没有符号。这些不是错误,它们是[无用的]警告。您可能应该在OS X上使用libtool而不是arranlib。我认为有一个选项可以抑制它,但我目前找不到它。还请参见Silence “table of contents is empty” warning - jww
3个回答

14

很遗憾,仅设置-no_warning_for_no_symbols并不足够。在Mac上,ar命令会为您运行ranlib,这会导致显示“没有符号/usr/bin/ranlib”的消息。 您可以通过使用以下CMake规则来停止在Mac上显示此消息:

-no_warning_for_no_symbols 不足以解决问题。在 Mac 上,ar 命令将为您运行 ranlib,从而导致显示“has no symbols /usr/bin/ranlib”消息。您可以使用以下 CMake 规则停止在 Mac 上显示此消息:

SET(CMAKE_C_ARCHIVE_CREATE   "<CMAKE_AR> Scr <TARGET> <LINK_FLAGS> <OBJECTS>")
SET(CMAKE_CXX_ARCHIVE_CREATE "<CMAKE_AR> Scr <TARGET> <LINK_FLAGS> <OBJECTS>")
SET(CMAKE_C_ARCHIVE_FINISH   "<CMAKE_RANLIB> -no_warning_for_no_symbols -c <TARGET>")
SET(CMAKE_CXX_ARCHIVE_FINISH "<CMAKE_RANLIB> -no_warning_for_no_symbols -c <TARGET>")

设置ARCHIVE_CREATE的"Scr"选项将防止ar运行ranlib。设置ARCHIVE_FINISH选项将禁止ranlib抱怨缺少符号。

你需要同时使用这两个规则来避免出现“没有符号”的消息。

对于CMake新手,还有一件事:CMake为arranlib命令生成的命令被转储到link.txt文件中。如果你在生成档案时遇到问题,可以查看link.txt以了解CMake为你做了什么。


10
ranlib的警告表明这些目标文件不包含操作代码。您需要查看每个相应的源文件,以查看其中是否应该在Mac上编译的代码。可能是代码被移植到其他地方,但条件编译无法正确识别Mac。
这并不是一个“完全确定”的诊断方式;缺少的符号明显分为两组 - 与trnk相关和与crv相关的文件同样分为两组(前缀为trnkcrv),但是它可以是文件定义不同的功能。因此,我将去查看与空目标文件相对应的源文件,看看是否可以看到缺失的函数(可能要忽略所报告的缺失符号名称前面的下划线)。如果这些函数在这些文件中定义,则必须查看来自ranlib的投诉。
偶尔,如果您在不方便的时候磁盘空间用尽,您可能会有零大小的目标文件(因此不包含任何符号)。在这种情况下,只需删除空的目标文件(并释放一些空间,但您必须已经这样做了)并再次构建即可。

1
有没有办法告诉ranlib忽略这些不包含操作代码的目标文件?我有一些这样的目标文件,因为它们是为其他平台准备的(例如,在编译为OSX时仅适用于Windows的实现)。我想在我的cmake中保留其他平台的实现,因为我想能够在我的IDE(我使用QtCreator)中看到文件,以便我可以编辑它们。不幸的是,QtCreator认为它们是错误。 - mchiasson

7

1
可能会存在问题,因为Mac的ar不是标准的。 - osgx
谢谢,但是那个链接已经失效了。而且除了MacPorts之外,有另一个解决方案会更好。有人可以提供吗? - Oscar

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接