交叉编译中libtool相关的问题

交叉编译中libtool相关的问题

1 libtool的工作原理

libtool 是一个通用库支持脚本,将使用动态库的复杂性隐藏在统一、可移植的接口中;使用libtool的标准方法,可以在不同平台上创建并调用动态库。可以认为libtool是gcc的一个抽象,其包装了gcc(或者其他的编译器),用户无需知道细节,只要告诉libtool需要编译哪些库即可,libtool将处理库的依赖等细节。libtool只与后缀名为lo、la为的libtool文件打交道。(lo是目标文件,la是库文件

libtool主要的一个作用是在编译大型软件的过程中解决了库的依赖问题;将繁重的库依赖关系的维护工作承担下来,从而释放了程序员的人力资源。libtool提供统一的接口,隐藏了不同平台间库的名称的差异等细节,0生成一个抽象的后缀名为la高层库libxx.la(其实是个文本文件),并将该库对其它库的依赖关系,都写在该la的文件中。该文件中的dependency_libs记录该库依赖的所有库(其中有些是以.la文件的形式加入的);libdir则指出了库的安装位置;library_names记录了共享库的名字;old_library记录了静态库的名字。

当编译过程到link阶段的时候,如果有下面的命令:

$libtool --mode=link gcc -o myprog -rpath /usr/lib –L/usr/lib –la

libtool会到/usr/lib路径下去寻找liba.la,然后从中(也就是从liba.la中)读取实际的共享库的名字(library_names中记录了该名字,比如liba.so)和路径(lib_dir中记录了,比如libdir=’/usr/lib’),返回诸如/usr/lib/liba.so的参数给激发出的gcc命令行。

如果liba.so依赖于库/usr/lib/libb.so,则在liba.la中将会有dependency_libs=’-L/usr/lib -lb’或者dependency_libs=’/usr/lib/libb.la’的行,如果是前者,其将直接把“-L/usr/lib –lb”当作参数传给gcc命令行;如果是后者,libtool将从/usr/lib/libb.la中读取实际的libb.so的库名称和路径,然后组合成参数“/usr/lib/libb.so”传递给gcc命令行。

当要生成的文件是诸如libmylib.la的时候,比如:

$libtool --mode=link gcc -o libmylib.la -rpath /usr/lib –L/usr/lib –la

其依赖的库的搜索基本类似,只是在这个时候会根据相应的规则生成相应的共享库和静态库。

注意:libtool在链接的时候只会涉及到后缀名为la的libtool文件;实际的库文件名称和库安装路径以及依赖关系是从该文件中读取的。

2 为何使用 -Wl,--rpath-link -Wl,DIR

使用libtool解决编译问题看上去没什么问题:库的名称、路径、依赖都得到了很好的解决。但下结论不要那么着急,一个显而易见的问题就是:并不是所有的库都是用libtool编译的。

比如上面那个例子,

$libtool --mode=link gcc -o myprog -rpath /usr/lib –L/usr/lib –la

如果liba.so不是使用libtool工具生成的,则libtool此时根本找不到liba.la文件(不存在该文件)。这种情况下,libtool只会把“–L/usr/lib –la”当作参数传递给gcc命令行。

考虑以下情况:要从myprog.o文件编译生成myprog,其依赖于库liba.so(使用libtool生成),liba.so又依赖于libb.solibb.so的生成不使用libtool),而且由于某种原因,ab的依赖并没有写入到liba.la中,那么如果用以下命令编译:

$libtool --mode=link gcc -o myprog -rpath /usr/lib –L/usr/lib –la

激发出的gcc命令行类似于下面:

              gcc –o myprog /usr/lib/liba.so

由于liba.so依赖于libb.so(这种依赖可以用readelfliba.soELF文件看到),而上面的命令行中,并没有出现libb.so,于是,可能会出现问题。

       说“可能”,是因为如果在本地编译的情况下,gcc在命令行中找不到一个库(比如上面的liba.so)依赖的其它库(比如libb.so),链接器会按照某种策略到某些路径下面去寻找需要的共享库:

1. 所有由'-rpath-link'选项指定的搜索路径.

2. 所有由'-rpath'指定的搜索路径. '-rpath''-rpath_link'的不同之处在于,'-rpath'指定的路径被包含在可执行文件中,并在运行时使用, '-rpath-link'选项仅仅在连接时起作用.

3. 在一个ELF系统中, 如果'-rpath''rpath-link'选项没有被使用, 会搜索环境变量'LD_RUN_PATH'的内容.它也只对本地连接器起作用.

4. SunOS, '-rpath'选项不使用, 只搜索所有由'-L'指定的目录.

5. 对于一个本地连接器,环境变量'LD_LIBRARY_PATH'的内容被搜索.

6. 对于一个本地ELF连接器,共享库中的`DT_RUNPATH'`DT_RPATH'操作符会被需要它的共享库搜索. 如果'DT_RUNPATH'存在了, 'DT_RPATH'就会被忽略.

7. 缺省目录, 常规的,'/lib''/usr/lib'.

8. 对于ELF系统上的本地连接器, 如果文件'/etc/ld.so.conf'存在, 这个文件中有的目录会被搜索.

从以上可以看出,在使用本地工具链进行本地编译情况下,只要库存在于某个位置,gcc总能通过如上策略找到需要的共享库。但在交叉编译下,上述八种策略,可以使用的仅仅有两个:-rpath-link-rpath。这两个选项在上述八种策略当中优先级最高,当指定这两个选项时,如果链接需要的共享库找不到,链接器会优先到这两个选项指定的路径下去搜索需要的共享库。通过上面的描述可以看到:-rpath指定的路径将被写到可执行文件中;-rpath-link则不会;我们当然不希望交叉编译情况下使用的路径信息被写进最终的可执行文件,所以我们选择使用选项-rpath-link

       gcc的选项“-Wl,--rpath-link –Wl,DIR”会把-rpath-link选项及路径信息传递给链接器。回到上面那个例子,如果命令行中没有出现libb.so,但gcc指定了“-Wl,--rpath-link –Wl,DIR”,则链接器找不到libb.so的时候,会首先到后面-rpath-link指定的路径去寻找其依赖的库。此处我们使用的编译命令的示例是使用unicore平台的工具链。

$ unicore32-linux-gcc –o myprog /usr/lib/liba.so \

-Wl,--rpath-link -Wl,/home/UNITY_float/install/usr/lib

这样,编译器会首先到“/home/UNITY_float/install/usr/lib”下面去搜索libb.so

       libtool如何把选项“-Wl,--rpath-link –Wl,DIR”传递给gcclibtool中有一个变量“hardcode_libdir_flag_spec”,该变量本来是传递“-rpath”选项的,但我们可以修改它,添加我们需要的路径,传递给unicore32-linux-gcc

       hardcode_libdir_flag_spec”原来的定义如下:

hardcode_libdir_flag_spec="\${wl}--rpath \${wl}\$libdir"

我们修改后的定义如下:

hardcode_libdir_flag_spec="\${wl}—rpath-link \${wl}\$libdir \

-Wl,--rpath-link -Wl,/home/UNITY_float/install/usr/lib \

                    -Wl,--rpath-link -Wl,/home/UNITY_float/install/usr/X11R6/lib "

这样,当libtool在“--mode=link”的模式下,就会把选项“-Wl,--rpath-link –Wl,DIR”传递给gcc编译器了。

============================================

Libtool创建和使用linux下动态库

文件fun.c,fun.h,hello.c,hello.h,main.c动态库函数都在fun.c和hello.c里面
fun.c:

int add(int a, int b)
{
return a+b;
}

fun.h:

#ifndef _FUN_H_11
#define _FUN_H_11
int add(int a, int b);
#endif
----------------------------

hello.c:

#i nclude <stdio.h>
void output(char *ss)
{
printf("HELLO   %s\n", ss);
}

hello.h

#ifndef HELLO_H_111
#define HELLO_H_111
void output(char *ss);
#endif
----------------------------

main.c:

#i nclude <stdio.h>
#i nclude "hello.h"
#i nclude "fun.h"
void
main()
{
output("world");
printf("Test Value:%d\n", add(1, 2));
}

使用libtools创建和使用安装动态库步骤:

(1)
libtool --mode=compile gcc -g -O -c hello.c
libtool --mode=compile gcc -g -O -c fun.c
libtool --mode=compile gcc -g -O -c main.c
#生成各自的o文件

(2)
libtool --mode=link gcc -g -O -o libhello.la hello.lo fun.lo -rpath /usr/local/lib -lm
#连接成动态库文件

(3)
libtool --mode=link gcc -g -O -o test main.o libhello.la -lm
#连接生成可执行文件test

(4)
libtool --mode=install cp libhello.la /usr/local/lib/libhello.la
libtool --mode=install install -c libhello.la /usr/local/lib/libhello.la
libtool -n --mode=finish /usr/local/lib
libtool install -c test /usr/local/bin/test
#安装动态库

然后就可以运行/usr/local/bin/test了,当然路径可以任意设置,这是手动过程,写成Makefile文件为:

OBJS = fun.o hello.o
LO_OBJS = main.lo fun.lo hello.lo
PACKAGE_VERSION = 1:1:1
LIBDIR=/usr/local/lib

all : test

install : libhello.la

test : libhello.la main.o
libtool --mode=link gcc -g -O -o test main.o libhello.la -lm

libhello.la : $(OBJS)
libtool gcc -g -O -o libhello.la $(LO_OBJS) -rpath ${LIBDIR} -lm -version-info ${PACKAGE_VERSION}

main.o : main.c fun.h hello.h
libtool --mode=compile gcc -g -O -c main.c

fun.o : fun.c fun.h
libtool --mode=compile gcc -g -O -c fun.c

hello.o : hello.c hello.h
libtool --mode=compile gcc -g -O -c hello.c

clean :
@rm -f OBJ/* lib*.a *~ *core *.lo *.o *.la
@rm -rf .libs



=============================

libtool使用-L/mypath/usr/lib -lsomelib去寻找当前正在编译的库编译依赖的库,其有两种模式:

(1)libtool --mode=compile(编译)

(2)libtool --mode=link(链接)

为防止libtool激发出的gcc命令将库找到系统的/usr/lib去,我们用sed替换了Makefile中所有的-lsomelib为/mypath/usr/lib/libmylib.so;在模式(1)编译模式下,不会出现什么问题,因为此时根本不需要libsomelib.so;在模式(2)下,就会有问题,又分两种情况:

其一是要生成的是可执行的二进制文件,此时libtool会把后面需要的所有.so动态库都传递到激发出的gcc命令行中,不会出问题,所有依赖的动态库都会正确的链接进来;ldd查看该二进制文件可以看到其依赖的所有动态库;

其二是要生成.so的动态库,此时libtool不会把其依赖的所有.so动态库传递到其激发出的gcc命令行中,生成的.so动态库用readelf查看,不会看到那些依赖的库,用ldd也看不到这些库。

问题描述如下:

1、在不替换所有的-l的情况下

/bin/libtool --mode=link …… -o libmylib.la -L/mypath/usr/lib -lsomelib

gcc ……/mypath/usr/lib/libsomelib.so -o libmylib.so (该行由libtool激发生成libmylib.so,ldd该库,可以看到其依赖的/usr/lib/libsomelib.so)

ar cru *.o libmylib.a(该行由libtool激发生成libmylib.a)

2、在替换-l的情况下

/bin/libtool --mode=linc …… -o libmylib.la /mypath/usr/lib/libsomelib.so

gcc …… -o libmylib.so (注意:libtool的/mypath/usr/lib/libsomelib.so并没有传递到该行,ldd生成的库,看不到/usr/lib/libsomelib.so)

ar cru *.o libmylib.a (该行产生的libmylib.a与正常的libmylib.a与上个.a应该没什么区别)

问题就是:生成动态库的时候,libtool不能将替换后的.so动态库作为参数传递到其激发出的gcc命令行当中 。

我们为什么要把 -L/mypath/usr/lib -lsomelib 替换为“/mypath/usr/lib/libsomelib.so”?

libtool激发出的gcc命令行会在我们给出的路径前面加上一个系统的库路径,比如:

/bin/libtool --mode=link -o test …… -L/mypath/usr/lib -lsomelib ……

其激发出的gcc命令可能就是:

unicore32-linux-gcc -o test …… -L/usr/lib -L/mypath/usr/lib -lsomelib

这样由于把系统的库路径放到了我们的库路径前面,导致编译器找到了系统的/usr/lib/libsomelib.so而不是/mypath/usr/lib/libsomelib.so。libtool当中如何添加-L /usr/lib路径不是很清楚,但是可以肯定不是一种单一机制(我们已经找到一些可能导致添加路径的机制,但仍然不能解决这种路径的问题,防不胜防)。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值