【问题标题】:How to follow linking order when linking against static library with gnu-make?使用 gnu-make 链接静态库时如何遵循链接顺序?
【发布时间】:2023-03-29 09:24:01
【问题描述】:

我遇到了以下问题:

cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG  build/liblcthw.a    tests/list_tests.c   -o tests/list_tests
/tmp/ccpvGjZp.o: In function `test_create':
~/lcthw/tests/list_tests.c:12: undefined reference to `List_create'
collect2: ld returned 1 exit status
make: *** [tests/list_tests] Error 1

但是

cc -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG tests/list_tests.c  build/liblcthw.a -o tests/list_tests

运行良好,nm 显示预期内容,测试运行,每个人都很高兴,等等。

我搜索了 SO 并找到了很多答案(例如 Linker order - GCC),因此很明显链接器可以正常工作。那么,我应该如何修改我的 makefile 以遵循顺序?

这是目前的 Makefile:

CFLAGS=-g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG $(OPTFLAGS)
LIBS=$(OPTLIBS)
PREFIX?=/usr/local
BUILD=build

SOURCES=$(wildcard src/**/*.c src/*.c)
OBJECTS=$(patsubst %.c,%.o,$(SOURCES))

TEST_SRC=$(wildcard tests/*_tests.c)
TESTS=$(patsubst %.c,%,$(TEST_SRC))

TARGET=$(BUILD)/liblcthw.a
TARGET_LINK=lcthw
SO_TARGET=$(patsubst %.a,%.so,$(TARGET))

#The Target Build
all: $(TARGET) $(SO_TARGET) tests

dev: CFLAGS=-g -Wall -Isrc -Wall -Wextra $(OPTFLAGS)
dev: all

$(TARGET): CFLAGS += -fPIC
$(TARGET): build $(OBJECTS)
    ar rcs $@ $(OBJECTS)
    ranlib $@

$(SO_TARGET): $(TARGET) $(OBJECTS)
    $(CC) -shared -o $@ $(OBJECTS)

build:
    @mkdir -p $(BUILD)
    @mkdir -p bin

#The Unit Tests
.PHONY: tests
tests: CFLAGS+=$(TARGET)     #I think this line is useless now
tests: $(TESTS)
    sh ./tests/runtests.sh

#some other irrelevant targets

尝试了一些奇怪且明显错误的事情,例如递归调用

$(TESTS):
    $(MAKE) $(TESTS) $(TARGET)

Windows7 的VirtualBox 下的Debian6 中运行它。系统规格:

$ uname -a
Linux VMDebian 2.6.32-5-686 #1 SMP Mon Mar 26 05:20:33 UTC 2012 i686 GNU/Linux
$ gcc -v
Using built-in specs.
Target: i486-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Debian 4.4.5-8' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.4 --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --libdir=/usr/lib --enable-nls --enable-clocale=gnu --enable-libstdcxx-debug --enable-objc-gc --enable-targets=all --with-arch-32=i586 --with-tune=generic --enable-checking=release --build=i486-linux-gnu --host=i486-linux-gnu --target=i486-linux-gnu
Thread model: posix
gcc version 4.4.5 (Debian 4.4.5-8) 

附:它来自 Zed Shaw 的 Learn C The Hard Way,exercise 33。不知道我是否应该将其标记为作业:)

【问题讨论】:

  • 它们是如何连接的?前两个(一个好,一个坏)示例构建 list_tests。 makefile 构建 liblcthw.a.
  • 尝试在CFLAGS中的$(OPTFLAGS)之后添加-Wl,--no-as-needed

标签: c makefile gnu-make static-linking


【解决方案1】:

您没有显示正在构建 tests/list_tests 的 makefile 规则,但它看起来好像只是内置规则。使用 GNU Make,您可以使用 -p 打印出该规则,它将显示:

# default
LINK.c = $(CC) $(CFLAGS) $(CPPFLAGS) $(LDFLAGS) $(TARGET_ARCH)
[...]
.c:
#  recipe to execute (built-in):
    $(LINK.c) $^ $(LOADLIBES) $(LDLIBS) -o $@

通过将库添加到$(CFLAGS)(通过特定于目标的变量tests: CFLAGS+=$(TARGET)),您将在生成的命令中将其放置在$^ 之前。相反,您应该将其添加到 $(LDLIBS) 以便它出现在目标文件之后:

tests: LDLIBS+=$(TARGET)

但是请注意,像这样依赖特定于目标的变量的传播在实践中并不是特别有效。当您键入 make tests 时,该库将用于构建 tests/list_tests 等。但是,当您只对一项测试感兴趣时,您会发现make tests/list_tests 失败,因为命令中未包含链接库。 (详情请参阅this answer。)

【讨论】:

  • 内置规则也是listed in the manual。如果您猜到内置的隐式规则恰好包含$(LDLIBS)$(LDFLAGS),以便您可以将这些加载程序库放在命令的末尾,那么您是对的!
【解决方案2】:

我是一个菜鸟,正在阅读同一本书,我得到了这样的构建方式:

我换行了:

tests: CFLAGS+=$(TARGET) #我觉得这条线现在没用了

测试:CFLAGS+=$(SO_TARGET)

【讨论】:

  • 供任何人将来参考这样做并在$(OPTFLAGS) 的第一行中添加-Wl,--no-as-neededMakefile 的第一行启用以编译本书的此练习
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-02-25
  • 2014-11-17
  • 1970-01-01
相关资源
最近更新 更多