我想用Makefile gcc构建共享库(.so, .dylib)
。
我的库有来自其他共享库的引用(符号)(其他C项目编译为共享库)。
我有这样的目录结构:
/c/
/Project1/...
/Project2/...
...
/unit_tests/...
/comparers/...
我已经尝试过编译unit_test共享库,它依赖于comparers共享库,这样做:
$(SHARED_LIBRARY): assertion.o
$(CC) $(CFLAGS) -shared -o $(OUTPUT_LIB_DIR)/$(SHARED_LIBRARY) $(OUTPUT_DIR)/assertion.o $(COMPARERS_SHARED_LIB_PATH)
其中:
COMPARERS_SHARED_LIB_PATH= ../comparers/output/debug/lib/libcomparers.so
但如果我使用静态库路径而不是正确编译它就不起作用。为什么这不起作用以及如何使其正常工作?
更新:错误消息
上面正确构建,但我认为它会在运行时导致错误!当我以相同的方式链接我的可执行目标时,它也会构建,但是在运行它时会出现这样的错误消息:
dyld: Library not loaded: output/debug/lib/libcomparers.so
Referenced from: /Users/michzio/Developer/MyProjects/BachelorDegree/c/unit_tests/./output/debug/bin/unit_tests
Reason: image not found
Abort trap: 6
答案 0 :(得分:0)
虽然您没有发布错误消息,但有一些错误很明显。
一个错误是您使用和不使用路径混合和匹配文件名。例如。 $(SHARED_LIBRARY)
与$(OUTPUT_LIB_DIR)/$(SHARED_LIBRARY)
不同,assertion.o
与$(OUTPUT_DIR)/assertion.o
不同。您需要学习如何使用GNU make's automatic variables来防止此类错误。
另一个错误是链接到另一个共享库的方式:运行时链接程序搜索../comparers/output/debug/lib/libcomparers.so
,并且该路径仅在当前目录是特定的时才有效。
链接通常使用LDFLAGS
,而非CFLAGS
。
一个修复可能是:
COMPARERS_SHARED_LIB_PATH := $(abspath ../comparers/output/debug/lib/libcomparers.so)
$(OUTPUT_LIB_DIR)/(SHARED_LIBRARY): $(OUTPUT_DIR)/assertion.o $(COMPARERS_SHARED_LIB_PATH)
$(CC) -shared -o $@ $(LDFLAGS) $^
对于生产版本,它需要使用-rpath
相对路径为其他共享库指定$ORIGIN
,以便运行时链接程序找到相对于其用户所需的共享库。并使用-L<path> -l<lib>
链接到该共享库,而不是完整路径。