8

次の問題があります。

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 は次のとおりです。

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)

Debian6上のVirtualBoxの下でこれを実行しWindows7ます。システム仕様:

$ 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) 

PS は、Zed Shaw の Learn C The Hard Way の演習 33からのものです。宿題としてマークする必要があるかどうかわかりません:)

4

2 に答える 2

6

構築中のmakefileルールは表示されませんがtests/list_tests、組み込みのルールであるかのように見えます。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のビルドに使用されます。ただし、1つのテストだけに関心がある場合は、リンクライブラリがコマンドに含まれていないため、失敗tests/list_testsすることがわかります。make tests/list_tests(詳細については、この回答を参照してください。)

于 2012-05-10T18:49:17.350 に答える
2

私は初心者で、同じ本を読み進めて、次のように作成しました。

行を変更しました:

テスト:CFLAGS + = $(TARGET)#この行は今は役に立たないと思います

テスト:CFLAGS + = $(SO_TARGET)

于 2013-01-01T07:18:11.703 に答える