我被赋予了从对象文件创建存档文件 .a 的任务,并从存档.a文件创建共享库文件。我尝试过以下文件的实验:
的 foo.h中
#ifndef _foo_h__
#define _foo_h__
extern void foo(void);
extern void bar(void);
#endif //_foo_h__
的 foo.c的
#include<stdio.h>
void foo(void)
{
puts("Hello, I'm a shared library");
}
的 bar.c
#include<stdio.h>
void bar(void)
{
puts("This is bar function call.");
}
的 main.c中
#include <stdio.h>
#include"foo.h"
int main(void)
{
puts("This is a shared library test...");
foo();
bar();
return 0;
}
的生成文件
CFLAGS = -Wall -Werror
LDFLAGS = -L/home/betatest/Public/implicit-rule-archive -Wl,-rpath,'$$ORIGIN'
all : run
run : main.o libfoo.so
$(CC) $(LDFLAGS) -o $@ $^
libfoo.so : CFLAGS += -fPIC # Build objects for .so with -fPIC.
libfoo.so : libfoo.a
$(CC) -shared -o $@ $^
libfoo.a : foo.o bar.o
ar cvq libfoo.a foo.o bar.o
# ar cr libfoo.a foo.o bar.o
# Compile any .o from .c. Also make dependencies automatically.
%.o : %.c
$(CC) -c $(CFLAGS) -o $@ $<
#Include dependencies on subsequent builds.
.PHONY : all clean
clean :
-rm -f *.o *.d run libfoo.*
这个简单的测试程序似乎运行正常,但在编译时使用它会产生错误:
cc -c -Wall -Werror -o main.o main.c
cc -c -Wall -Werror -fPIC -o foo.o foo.c
cc -c -Wall -Werror -fPIC -o bar.o bar.c
ar cvq libfoo.a foo.o bar.o
a - foo.o
a - bar.o
cc -shared -o libfoo.so libfoo.a
cc -L/home/betatest/Public/implicit-rule-archive -Wl,-rpath,'$ORIGIN' -o run main.o libfoo.so
main.o: In function `main':
main.c:(.text+0xf): undefined reference to `foo'
main.c:(.text+0x14): undefined reference to `bar'
collect2: ld returned 1 exit status
Makefile-test:7: recipe for target 'run' failed
make: *** [run] Error 1
有人请指出我哪里出错?非常感谢。
答案 0 :(得分:5)
如果您希望从静态库构建共享库,则必须告诉链接器使用静态库(.a)中包含的所有函数/符号。否则,共享库(.so)中不会包含任何内容。
您必须在链接中使用--whole-archive/--no-whole-archive
对。
CFLAGS = -Wall -Werror
LDFLAGS = -L/home/betatest/Public/implicit-rule-archive -Wl,-rpath,'$$ORIGIN'
all : run
run : main.o libfoo.so
$(CC) $(LDFLAGS) -o $@ $^
libfoo.so : CFLAGS += -fPIC # Build objects for .so with -fPIC.
libfoo.so : libfoo.a
$(CC) -shared -o $@ -Wl,--whole-archive $^ -Wl,--no-whole-archive
libfoo.a : foo.o bar.o
ar cvq libfoo.a foo.o bar.o
# Compile any .o from .c. Also make dependencies automatically.
%.o : %.c
$(CC) -c $(CFLAGS) -o $@ $<
#Include dependencies on subsequent builds.
.PHONY : all clean
clean :
-rm -f *.o *.d run libfoo.*
您可以使用nm
命令检查导出的函数:
$ nm -D libfoo.so | grep ' T '
0000000000000702 T bar
0000000000000714 T _fini
00000000000006f0 T foo
0000000000000590 T _init
当没有使用--whole-archive/--no-whole-archive
对时,您会得到:
$ nm -D libfoo.so | grep ' T '
0000000000000714 T _fini
0000000000000590 T _init
答案 1 :(得分:1)
你的错误就在于这个问题:不要从归档创建共享库,从对象创建共享库,即像这样更改Makefile
libfoo.so : foo.o bar.o
$(CC) -shared -o $@ foo.o bar.o
您也可以使用%.o
表示所有对象
你全局需要CFLAGS = -fPIC
来影响编译命令,不仅仅是链接步骤,也就是你.o目标文件应该与位置无关。
此处不需要extern
,所有C函数都是隐式extern
extern void foo(void);
我也不建议使用rpath
和更好的
export LD_LIBRARY_PATH=.
从当前目录(或导出路径到.so的目录)加载.so