libtool 创建库的工具

libtool 创建库的工具

1. 背景

  在不同的系统中建立动态链接库的方法有很大的差别,这主要是因为每个系统对动态链接库的用法和实现并不相同,以及编译器对动态链接库支持的选项也不太一样。

  对于开发人员,如果尝试将使用动态库的软件在这些系统之间移植,需要参考枯涩难懂的系统手册,以及修改相应的 Makefile,这一工作是乏味的,并且具有一定的难度。

  使用 GNU Libtool 可以容易的在不同的系统中建立动态链接库。它通过一个称为 Libtool 库的抽象,隐藏了不同系统之间的差异,给开发人员提供了一致的的接口。对于大部分情况,开发人员甚至不用去查看相应的系统手册,只需要掌握 GNU Libtool 的用法就可以了。并且,使用 Libtool 的 Makefile 也只需要编写一次就可以在多个系统上使用

  Libtool 库可以是一个静态链接库,可以是一个动态链接库,也可以同时包含两者。

  在这篇文档中,我们围绕 Libtool 库的建立和使用,只是在适当的说明 Libtool 库和系统动态或者静态链接库之间的映射关系

2. Libtool 是一个工具

  虽然 Libtool 隐藏了在不同平台创建链接库的复杂性,但其最终还是需要底层系统对链接库的支持,它不能超越系统的限制,例如,Libtool 并不能在不支持动态链接库的系统中创建出动态链接库。

3.  Libtool 基本用法

  以实例来说明如何使用 Libtool 从源代码创建最终链接库以及执行程序的完整步骤,这是软件开发过程中经常使用的内容,包括 :

  • 创建 Libtool 对象文件 ;
  • 创建 Libtool 库;
  • 安装 Libtool 库 ;
  • 使用 Libtool 库 ;
  • 卸载 Libtool 库 ;

  首先需要准备一个源文件 compress.c,代码如下

#include <sys/mman.h> 
#include <sys/stat.h> 
#include <fcntl.h> 
#include <stdio.h> 
#include <stdlib.h> 
#include <unistd.h> 
#include <string.h> 
#include <limits.h> 
#include <assert.h> 
#include <zlib.h> 
 
/* 一个简单的文件压缩函数 */ 
int compress_file (const char *filename) 
{ 
 int src_fd, dest_fd; 
 struct stat sb; 
 Bytef *src, *dest; 
 uLong dest_len; 
 char dest_file[PATH_MAX]; 
 
 src_fd = open (filename, O_RDONLY); 
 assert (dest_fd != -1); 
 
 assert (fstat (src_fd, &sb) != -1); 
 
 src = mmap (NULL, sb.st_size, PROT_READ, MAP_PRIVATE, src_fd, 0); 
 assert (src != MAP_FAILED); 
 
 dest_len = compressBound (sb.st_size); 
 dest = malloc (dest_len); 
 assert (dest); 
 
 assert (compress (dest, &dest_len, src, sb.st_size) == Z_OK); 
 
 munmap (src, sb.st_size); 
 close (src_fd); 
 
 snprintf (dest_file, sizeof (dest_file), "%s.z", filename); 
 dest_fd = creat (dest_file, S_IRUSR | S_IWUSR); 
 assert (dest_fd != -1); 
 
 write (dest_fd, dest, dest_len); 
 close (dest_fd); 
 free (dest); 
 
 return 0; 
}

  这个文件实现了一个函数 compress_file(),它接收一个文件名作为参数,然后对文件进行压缩,生成一个 .z结尾的压缩文件。在这个文件中使用了 compress()函数,这个函数是有由 libz 提供的

  从源文件建立 Libtool 库需要经过两个步骤,先建立 Libtool 对象文件,再建立 Libtool 库。

(1) Libtool 对象文件

  如果使用传统的方式,建立对象文件通常使用下面的命令 

$ gcc -c compress.c

  使用 Libtool 则使用下面的命令 :

$ libtool --mode=compile gcc -c foo.c    

  可以看到,使用 Libtool 只需要将“传统”的命令 (gcc -c foo.c) 作为参数传递给 Libtool 即可。

  在上面的命令中,libtool 使用 compile模式 (--mode=compile 选项 ),这是建立对象文件的模式,Libtool 还有其它的模式,后面将介绍。

  上面的命令输出如下 :

mkdir .libs 
gcc -c compress.c  -fPIC -DPIC -o .libs/compress.o 
gcc -c compress.c -o compress.o >/dev/null 2>&1

  它建立了两个文件,一个是 .libs/compress.o,在建立这个文件时,Libtool 自动插入了 -fPIC和 -DPIC选项,告诉编译器生成位置独立的代码,之后将用这个文件来建立动态链接库。生成第二个文件 compress.o没有添加额外的选项,它准备用来建立静态链接库。

  除了上面的两个文件之外,Libtool 还建立了一个文件 compress.lo,这个文件就是 Libtool 对象文件,实际上也就是一个文本文件,里面记录了建立动态链接库静态链接库分别所需要的真实文件名称,后面 Libtool 将使用这个文件而不是直接的使用 .libs/compress.o 和 compress.o。

(2)建立 Libtool 库

$ libtool --mode=link gcc -o libcompress.la【目标文件】 compress.lo【输入文件】 -rpath /tmp -lz

  注意这里使用 compress.lo 作为输入文件,并且告诉 Libtool 生成的目标文件为 libcompress.la,.la 是 Libtool 的库文件后缀。

  -rpath选项告诉 Libtool 这个库将被安装到什么地方,如果省略了 -rpath选项,那么不会生成动态链接库

  因为我们的库中使用了 libz 提供的 compress 函数,所以也提供了 -lz 选项,Libtool 会记住这个依赖关系,后续在使用我们的库时自动的将依赖的库链接进来。

gcc -shared  .libs/compress.o  -lz  -Wl,-soname -Wl,libcompress.so.0 
                                       -o .libs/libcompress.so.0.0.0 
 (cd .libs && rm -f libcompress.so.0 && 
 ln -s libcompress.so.0.0.0 libcompress.so.0) 
 (cd .libs && rm -f libcompress.so && 
 ln -s libcompress.so.0.0.0 libcompress.so) 
 ar cru .libs/libcompress.a  compress.o 
 ranlib .libs/libcompress.a 
 creating libcompress.la 
 (cd .libs && rm -f libcompress.la && 
 ln -s ../libcompress.la libcompress.la)

  可以看到,Libtool 自动的插入了建立动态链接库需要的编译选项 -shared。并且,它也建立了静态链接库 .libs/libcompress.a,后面我们将会介绍如何控制 Libtool 只建立需要的库

  你可能会奇怪为什么建立的动态链接库有 .0 和 .0.0.0 这样的后缀,这里先不用理会它,后面在介绍 Libtool 库版本信息时将会解释这点。

  值得注意的是,Libtool 希望后续使用 libcompress.la 文件而不是直接使用 libcompress.a 和 libcompress.so 文件,如果你这样做,虽然可以,但会破坏 Libtool 库的可移植性。

4. 安装 Libtool 库

  发布建立好的 Libtool 库,可以使用下面的命令安装它 :

$ libtool --mode=install install -c libcompress.la /tmp

  我们需要告诉 Libtool 使用的安装命令,Libtool 支持 install 和 cp,这里使用的是 install。

  虽然前面我们在建立库时,通过 -rpath 选项指定了库准备安装的路径 (/tmp),但是这里我们还得要提供安装路径。请确保它们一致。

  这个命令的输出如下 :

install .libs/libcompress.so.0.0.0 /tmp/libcompress.so.0.0.0 
(cd /tmp && { ln -s -f libcompress.so.0.0.0 libcompress.so.0 || 
{ rm -f libcompress.so.0 && 
ln -s libcompress.so.0.0.0 libcompress.so.0; }; }) 
(cd /tmp && { ln -s -f libcompress.so.0.0.0 libcompress.so || 
{ rm -f libcompress.so && ln -s libcompress.so.0.0.0 libcompress.so; }; }) 
install .libs/libcompress.lai /tmp/libcompress.la 
install .libs/libcompress.a /tmp/libcompress.a 
chmod 644 /tmp/libcompress.a 
ranlib /tmp/libcompress.a 
...

  可以看到它安装了真实的动态链接库和静态链接库,同时也安装了 Libtool 库文件 libcompress.la,这个文件可以被后续的 Libtool 命令使用。

  在安装完成之后,可能还需要做一些配置才能正确使用,Libtool 的 finish 模式可以在这方面给我们一些提示 :

$ libtool -n --mode=finish /tmp

  这个命令的输出有点长,所以不在这里列出,如果不能正常的使用安装好的库,请运行这个命令。

5.使用 Libtool 库

  要在应用程序中使用前面创建的 Libtool 库很简单,准备一个源文件 main.c,它将使用 libcompress.la 库中定义的函数,代码如下 :

  清单 2: main.c

#include <stdio.h> 
 
extern int compress_file (const char *filename); 
 
int main (int argc, char *argv[]) 
{ 
 if (argc < 2) 
   {   
     printf ("usage : %s file
", argv[0]); 
     return 1; 
   }   
 return compress_file (argv[1]); 
}

  我们还是要先为 main.c 建立 Libtool 对象文件,这和前面的方法一样 :

$ libtool --mode=compile gcc -c main.c

。。。。。

参考博客:

  https://www.ibm.com/developerworks/cn/aix/library/1007_wuxh_libtool/index.html

原文地址:https://www.cnblogs.com/icmzn/p/9704543.html