来自C中存档(.a)文件的共享库

时间:2015-02-11 14:24:36

标签: c linux makefile shared-libraries

我被赋予了从对象文件创建存档文件 .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

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

2 个答案:

答案 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