TL; DR:我正在进行使用dlfcn.h
打开共享库的C练习。尽管根据其他帖子添加了(我认为是)正确的标记,但undefined reference to
,dlopen
以及dlsym
中定义的一些其他函数仍然出现dlfcn.h
错误1}}(错误消息和make文件包含在下面)。
我在.c
文件的开头包含以下内容:#include <dlfcn.h>
我做错了什么?
详细版本:
我正在处理exercise 30 of Learn C The Hard Way
,并且为了获得libex29_tests.c
程序(页面上的最后一段代码)正确编译,我还需要做些什么。正如您可能已经知道的那样,共享库/ makefile /编译器标志对我来说是新的。
到目前为止我尝试过的内容:根据以下帖子,我尝试通过向各种-ldl
添加LIBS=-ldl fPIC
和/或LDFLAGS+=-ldl
来添加-ldl
标记我的Makefile的一部分,但仍然有问题。本书中的makefile版本(包含在下面,作为参考)确实包含makefile
标志,尽管语法略有不同。在任何情况下,我都会继续收到相同的错误消息。
有什么建议吗?
这些是我得到的错误。这些错误假定下面包含的-ldl
版本。但是,正如我所提到的,更改~/.../lchw/ex30_automated$ make
cc -std=gnu99 -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG -fPIC -c -o src/libex29.o src/libex29.c
src/libex29.c: In function ‘fail_on_purpose’:
src/libex29.c:42:33: warning: unused parameter ‘msg’ [-Wunused-parameter]
int fail_on_purpose(const char *msg)
^
ar rcs build/libYOUR_LIBRARY.a src/libex29.o
ranlib build/libYOUR_LIBRARY.a
cc -shared -o build/libYOUR_LIBRARY.so src/libex29.o
cc -std=gnu99 -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG build/libYOUR_LIBRARY.a tests/libex29_tests.c -o tests/libex29_tests
In file included from tests/libex29_tests.c:1:0:
tests/libex29_tests.c: In function ‘main’:
src/minunit.h:14:38: warning: parameter ‘argc’ set but not used [-Wunused-but-set-parameter]
#define RUN_TESTS(name) int main(int argc, char *argv[]) {\
^
tests/libex29_tests.c:64:1: note: in expansion of macro ‘RUN_TESTS’
RUN_TESTS(all_tests);
^
/tmp/dchaudh/ccwzxpC3.o: In function `check_function':
/home/dchaudh/Dropbox/dchaudh/wc/lchw/ex30_automated/tests/libex29_tests.c:10: undefined reference to `dlsym'
/home/dchaudh/Dropbox/dchaudh/wc/lchw/ex30_automated/tests/libex29_tests.c:11: undefined reference to `dlerror'
/tmp/dchaudh/ccwzxpC3.o: In function `test_dlopen':
/home/dchaudh/Dropbox/dchaudh/wc/lchw/ex30_automated/tests/libex29_tests.c:23: undefined reference to `dlopen'
/tmp/dchaudh/ccwzxpC3.o: In function `test_dlclose':
/home/dchaudh/Dropbox/dchaudh/wc/lchw/ex30_automated/tests/libex29_tests.c:46: undefined reference to `dlclose'
collect2: error: ld returned 1 exit status
make: *** [tests/libex29_tests] Error 1
标志的语法会导致几乎相同的错误消息。
makefile
根据本书,这里是最新的-ldl
。它包含CFLAGS=-std=gnu99 -g -O2 -Wall -Wextra -Isrc -rdynamic -DNDEBUG $(OPTFLAGS)
LIBS=-ldl $(OPTLIBS)
PREFIX?=/usr/local
SOURCES=$(wildcard src/**/*.c src/*.c)
OBJECTS=$(patsubst %.c,%.o,$(SOURCES))
TEST_SRC=$(wildcard tests/*_tests.c)
TESTS=$(patsubst %.c,%,$(TEST_SRC))
TARGET=build/libYOUR_LIBRARY.a
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)
tests: $(TESTS)
sh ./tests/runtests.sh
valgrind:
VALGRIND="valgrind --log-file=/tmp/valgrind-%p.log" $(MAKE)
# The Cleaner
clean:
rm -rf build $(OBJECTS) $(TESTS)
rm -f tests/tests.log
find . -name "*.gc*" -exec rm {} \;
rm -rf `find . -name "*.dSYM" -print`
# The Install
install: all
install -d $(DESTDIR)/$(PREFIX)/lib/
install $(TARGET) $(DESTDIR)/$(PREFIX)/lib/
# The Checker
BADFUNCS='[^_.>a-zA-Z0-9](str(n?cpy|n?cat|xfrm|n?dup|str|pbrk|tok|_)|stpn?cpy|a?sn?printf|byte_)'
check:
@echo Files with potentially dangerous functions.
@egrep $(BADFUNCS) $(SOURCES) || true
标志,如第二行所示。
{{1}}
答案 0 :(得分:6)
${LIBS}
(以及您的-ldl
)在makefile中没有使用。看来,您正在使用内置规则进行编译,并且可以使用LDLIBS
代替LIBS
作为此变量的名称。
除此之外,
tests: CFLAGS += $(TARGET)
您的示例中的是错误的,因为TARGET
是一个必须进入(LD)LIBS
的库。
$(TARGET): build $(OBJECTS)
build:
@mkdir -p build
也不健全。这将在并行构建中中断($(OBJECTS)
应该依赖于build
)并且将导致不需要的构建(build
是一个目录并且由每个编译/链接操作改变)。简单build
对于VPATH
版本会有问题,我建议使用
$(OBJECTS): | build/.dirstamp
build/.dirstamp:
mkdir ${@D}
这里。