2015-02-11 69 views
2

我得到了一个任务,从对象文件创建归档文件.a,并从归档.a文件创建共享库文件。我已经试过有实验以下文件:
foo.h中来自存档(.a)文件的共享库C

#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; 
} 


的Makefile

CFLAGS = -Wall -Werror 
LDFLAGS = -L/home/betatest/Public/implicit-rule-archive -Wl,-rpath,'$$ORIGIN' 

all : run 

run : main.o libfoo.so 
    $(CC) $(LDFLAGS) -o [email protected] $^ 

libfoo.so : CFLAGS += -fPIC# Build objects for .so with -fPIC. 
libfoo.so : libfoo.a 
    $(CC) -shared -o [email protected] $^ 

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 [email protected] $< 

#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 

有人请指出我在哪里出错?非常感谢。

+1

你不链接共享库:尝试' -lfoo'而不是'libfoo.so' – 2015-02-11 14:29:46

+0

cc正在获取$ ORIGIN附近的文字单引号。尝试省略它们。 – 2015-02-11 14:34:48

回答

4

如果您希望从静态库构建共享库,则必须告诉链接器使用静态库(.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 [email protected] $^ 

libfoo.so : CFLAGS += -fPIC# Build objects for .so with -fPIC. 
libfoo.so : libfoo.a 
    $(CC) -shared -o [email protected] -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 [email protected] $< 

#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 
+0

谢谢@ctheo先生!!!!!!编译顺利进行,编译输出为:'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 -Wl, - whole-archive libfoo.a -Wl, - no-whole-archive; CC -L/home/betatest/Public /隐式规则归档-Wl,-rpath,'$ ORIGIN'-o运行main.o libfoo.so;' 我想问一下main.o是如何产生的隐式规则或模式定义如下?谢谢!! – 2015-02-11 17:54:00

+0

main.o是由命令'cc -c -Wall -Werror -o main.o main.c'构建的。它使用不带'-fPIC'的CFLAGS,因为'libfoo.so'不依赖于它。 – ztik 2015-02-12 08:05:53

+1

它意味着它看到一个依赖main.o并调用定义的模式规则'%.o:%.c'。谢谢 – 2015-02-12 09:02:18

1

你错就错在这个问题:不要归档使共享库,从对象组成的,即改变的Makefile这样

libfoo.so : foo.o bar.o 
    $(CC) -shared -o [email protected] foo.o bar.o 

你也可以使用%.o意味着所有对象

也全局需要CFLAGS = -fPIC,以影响编译命令,而不仅仅是链接步骤,那就是你的.o目标文件应该是位置无关的。

也在这里extern是不需要的,所有的C函数隐式的extern

extern void foo(void); 

我也不会建议使用rpath更好

export LD_LIBRARY_PATH=. 

从当前目录加载的.so(或出口指向.so目录的路径)