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.so(libb.so的生成不使用libtool),而且由于某种原因,a对b的依赖并没有写入到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(这种依赖可以用readelf读liba.so的ELF文件看到),而上面的命令行中,并没有出现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”传递给gcc?libtool中有一个变量“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
相关推荐
在Qt编译时,确保libtool已正确安装并配置,然后按照Qt的编译指南进行操作,libtool会自动处理与库相关的任务。 总的来说,libtool-1.5.22在Linux高版本中的作用不容忽视,尤其对于依赖于特定版本库的开发工作,它...
在多版本的系统中,libtool可以帮助处理版本依赖问题,确保软件正确地链接到正确的库版本。 **libtool的主要功能** 1. **动态链接库管理**:Libtool支持创建和使用动态链接库,允许在不重新编译整个项目的情况下...
同样,对于mpeg库,libtool可以解决库的编译和链接问题,尤其是在处理复杂的依赖关系和平台差异时,大大简化了工作流程。 至于motion功能函数,这是一个常见的用于视频流处理的函数库。在libtool的帮助下,开发者...
4. **版本管理**:LIBTOOL可以帮助管理库的版本信息,确保正确链接到正确版本的库,避免因版本冲突导致的问题。 5. **多线程支持**:它能处理多线程库的构建,确保在多核或多处理器系统上的正确编译和链接。 6. **...
本文将详细解析libtool-1.5的使用方法和相关知识点。 首先,libtool-1.5的主要作用在于管理和编译库文件,它通过提供一套统一的接口,使得开发者无需关心底层平台的具体细节,如静态库和动态库的创建、版本管理等。...
书中首先回顾了Unix系统多样性、早期的configure程序、configure和Automake的发展历程,以及Libtool的演变,同时也涉及了微软Windows平台的相关内容。在操作层面,作者详细解释了如何运行configure脚本,以及标准...
在QT项目中,确保`tslib`的`autoconf`、`automake`和`libtool`设置正确,以便在编译和运行时能正确处理触摸屏相关的功能。 总的来说,`autoconf`、`automake` 和 `libtool`是构建和管理开源软件项目的重要工具,...
Autoconf, Automake and Libtool are packages for making your software more portable and to simplify building it—usually on someone else’s system. Software portability and effective build systems are ...
标题中的"apr", "aprutil" 和 "libtool" 是三个关键的开源库,它们在IT行业中,特别是在服务器端软件开发,尤其是与Apache Tomcat相关的领域中起着重要作用。现在让我们详细了解一下这三个组件以及它们与Tomcat的...
有关 Libtool 的最新更改的描述,请参阅文件 。 请注意,只要这是官方发布的 tarball,您就可以使用供应商 Make 程序从该目录构建 GNU Libtool; 否则,您将需要 GNU Make 来获得合理的 VPATH 支持。 有关如何构建...
4. **Libtool**:Libtool是为了解决库的编译和链接问题而设计的,特别是对于混合静态和动态库的支持。它提供了一致的接口来管理库的版本、编译、链接和安装,避免了直接使用编译器命令时可能出现的错误。Libtool可以...
- **版本控制**:建议将所有Autotools相关的文件纳入版本控制系统管理,便于团队协作和历史记录追踪。 - **文档编写**:编写清晰的README文件,说明如何使用Autotools构建项目,特别是对于不常见的配置或依赖项。 - ...
libtool.pdf 可能涵盖了如何配置 Libtool,创建共享库和静态库,以及如何在项目中集成 Libtool 的内容。 4. **M4** M4 是一个预处理器,用于扩展和生成 configure 脚本。它提供了宏语言,使得开发者可以编写复杂的...
Autoconf用于生成可以在多种Unix-like系统上配置的Makefile模板,Automake用于从Makefile模板中生成最终的Makefile,而Libtool则用来处理库文件的跨平台兼容性问题。下面将详细介绍这些工具的关键知识点。 Autoconf...
3. **libtool**: `libtool`是一个通用的库管理工具,它可以处理静态库和动态库的编译、链接和版本控制问题。在`tslib`的开发和打包过程中,`libtool`确保了库文件的正确生成和版本兼容性。 4. **m4**: `m4`是一个预...
总之,《Autoconf、Automake与Libtool手册》是一本非常有价值的书籍,无论是对于初学者还是经验丰富的开发者而言,都能够从中获得有关如何使用这些工具来提高开发效率的重要知识。通过阅读本书,开发者将能够更加...
了解Linux环境下库文件的存放路径、库文件依赖关系,以及各种编译工具如autoconf、automake、libtool等的作用和使用方法,对解决安装时遇到的问题大有帮助。 总结来说,PHP在Linux系统下的安装问题涵盖了广泛的知识...
这些问题通常与环境配置、依赖库版本不匹配等因素有关。本篇文章将详细介绍如何解决这些问题,并提供一个逐步的操作指南。 #### 问题一:Import cv2 报错 当尝试在Python环境中导入cv2模块时,可能会遇到以下错误...
本文将详细介绍如何在Linux环境下,特别是Redhat Linux 5.5上安装Redmine 2.0及其相关组件,以及解决可能出现的问题。 **安装环境与所需软件** 在开始安装前,确保你的系统满足以下条件: - **操作系统**: Redhat ...
- **Libtool C API (ltdl)**:Libtool提供了C API,允许其他程序动态加载和卸载共享库。 #### 四、构建项目 1. **运行`configure`**:此脚本会检查系统特性并生成`Makefile`文件。 2. **远程构建目录**:允许在...