原文:
Guide for pkg-config
http://people.freedesktop.org/~dbn/pkg-config-guide.html
Dan Nicholson
Translated By Bob
2011-11-13
Email:[email protected]
Blog:http://blog.csdn.net/exbob
------------------------------------------------------
概述
为什么?
一些概念
写pkg-config文件
使用pkg-config文件
常见问题
-----------------------------------------------
概述
这个文档的目的是从用户和开发者的角度给一个pkg-config工具的使用概述。本文复习一些pkg-config背后的概念,怎样写pkg-config文件来支持你的项目,以及怎样用pkg-config集成第三方项目。
关于pkg-config的更多信息可以在web站点和pkg-config的man手册中找到。
本文档假的pkg-config在类UNIX操作系统中使用,例如Linux。其他平台可能在一些细节上的存在差别。
为什么?
现代计算机系统使用了很多分层组件为用户提供应用。其中一个困难就是如何正确的整合这些组件。pkg-config会收集系统中安装的库的数据,然后提供给用户。
如果没有pkg-config这样的数据系统,定位计算机提供的服务和获取它们的细节会很困难。对于开发者,安装软件包的pkg-config文件极大的简化了对API的获取。
一些概念
使用pkg-config的初级阶段是为编译和链接程序时提供必要的细节。数据存储在pkg-config文件中。这些文件有一个.pc的后缀,放在一个特定的、pkg-config工具所知道的位置。我们会在后面描述更多的细节。
这个文件的格式包括预定义的关键字和自由形式的变量。例如:
prefix=/usr/local
exec_prefix=${prefix}
includedir=${prefix}/include
libdir=${exec_prefix}/lib
Name: foo
Description: The foo library
Version: 1.0.0
Cflags: -I${includedir}/foo
Libs: -L${libdir} -lfoo
以预定义关键字Name:为例,以关键字开头,后面跟一个冒号和一个值。变量是一个字符串和一个值,例如prefix=,用等号分开。关键字是由pkg-config定义和输出的。变量不是必须的,但可以被关键字用来定位和存储pkg-config没有覆盖的数据。
这里只是简单的描述一下关键字。更深入的描述和怎样有效的使用它们将在“写pkg-config文件”段中给出。
Name:一个人们可读的链接库或软件包的名称,这不影响pkg-config的使用,它用的是.pc文件的名称。
Description:关于软件包的简单描述。
URL:一个URL,可以在那里获得更多的信息,并且下载这个软件包。
Version:软件包的版本。
Requires:这个软件包所需的包的列表。这些包的版本可能用一写运算符来指定:=、>、<、>=、<=。
Requires.private:这个软件包所需的私有包的列表,不会暴露给应用。版本的指定规则与Requires相同。
Conflicts:可选,描述了会与这个软件包产生冲突的包。版本的指定规则与Requires相同。这个域会提供同一个包的多个实例,例如:Conflicts: bar < 1.2.3, bar >= 1.3.0。
Cflags:为这个软件包指定编译器选项,以及pkg-config不支持的必要的库。如果所需的库支持pkg-config,应该将它们添加到Requires和Requires.private。
Libs:为这个软件包指定的链接选项,以及pkg-config不支持的必要的库。与Cflags的规则相同。
Libs.private:这个软件包所需的私有库的链接选项,不会暴露给应用。规则与Cflags相同。
写pkg-config文件
为一个软件包创建pkg-config时,首先要确定怎样描述它。一个文件最好只用于描述一个库,所以,每个软件包至少需要像它所需的链接库那么多的pkg-config文件。
软件包的名字是由pkg-config数据文件的名字确定的。就是文件名去掉.pc后缀的那一部分。通常都用库的名字命名.pc文件。例如,一个安装libfoo.so的包会有一个相应的libfoo.c文件来包含pkg-config数据。这不是必须的,.pc文件仅仅是一个对你的库的唯一标识符。所以,foo.pc或foolib.pc也能正常工作。
Name、Description和URL的值是纯粹的信息,容易填写。Version比较棘手,它要确保这个包可以被用户使用。pkg-config使用RPM算法来进行版本比较。Version最好是用点分开的十进制数字,例如1.2.3,因为字母可能引起意外的结果。数字应该是单调递增的,并且要竟可能具体的描述这个库。通常使用包的版本号即可,这样可以方便使用者跟踪。
在描述更多的有用的关键字之前,有必要展示一下变量的定义。最常见的用法是定义安装路径,这样就不会使其他字段显得杂乱。因为变量是扩大递归的,在结合autoconf派生路径时,这会很有用。
prefix=/usr/local
includedir=${prefix}/include
Cflags: -I${includedir}/foo
最重要的pkg-config数据字段是Requires,Requires.private,Cflags,Libs 和 Libs.private。它们定义的数据被外部项目用来编译和链接库。
Requires和Requires.private定义了库所需的其他模块。通常首选Requires.private,以便避免程序链接到一些不必要的库。如果一个程序不使用所需库的符号,它就不应该直接链接到这个库。可以在overlinking的讨论中看到更多详细的解释。
由于pkg-config通常会公开Requires库的链接标识,这些模块会变成程序的直接依赖。另外,Requires.private中的库只有在静态链接是才会被包含。正因如此,pkg-config通常只会适当的从Requires中的同一个包中添加模块。
Libs包含了使用库是所必须的链接标识。此外,Libs和Libs.private还包含了pkg-config不支持的库的链接标识。与Requires类似,首选将外部库的链接标识添加到Libs.private,这样,程序就不会获得额外的直接依赖。
最后,Cflags包含了所用的库的编译标识。与Libs不同,Cflags没有私有变种。这是因为,数据类型和宏定义在任何链接情况下都是需要的。
使用pkg-config文件
假设系统中已经安装了.pc文件,pkg-config工具就被用来提取其中的数据。执行pkg-config --help命令可以看到一些关于命令选项的简单描述。深入的描述可以在pkg-config(1)的man手册页中找到。本地将对一些常见的用法进行简单的描述。
假设系统中已经有了两个模块:foo和bar。它们的.pc文件可能像下面这样:
foo.pc:
prefix=/usr
exec_prefix=${prefix}
includedir=${prefix}/include
libdir=${exec_prefix}/lib
Name: foo
Description: The foo library
Version: 1.0.0
Cflags: -I${includedir}/foo
Libs: -L${libdir} -lfoo
bar.pc:
prefix=/usr
exec_prefix=${prefix}
includedir=${prefix}/include
libdir=${exec_prefix}/lib
Name: bar
Description: The bar library
Version: 2.1.2
Requires.private: foo >= 0.7
Cflags: -I${includedir}
Libs: -L${libdir} -lbar
模块的版本可以用--modversion选项获得。
$ pkg-config --modversion foo
1.0.0
$ pkg-config --modversion bar
2.1.2
要打印模块的链接标识,就用--libs选项。
$ pkg-config --libs foo
-lfoo
$ pkg-config --libs bar
-lbar
还有就是,虽然foo是bar所需要的,但是没有输出foo的链接标识。这是因为,只使用bar库的应用并不直接需要foo。对应静态链接bar的应用,我们需要两个链接标识:
$ pkg-config --libs --static bar
-lbar -lfoo
这种情况下,pkg-config就要输出两个链接标识,这样才能保证静态链接的应用可以找到所有必须的符号。另一方面,它会输出所有的Cflags字段。
$ pkg-config --cflags bar
-I/usr/include/foo
$ pkg-config --cflags --static bar
-I/usr/include/foo
还有一个有用的选项,--exists,可以用来测试模块的可用性。
$ pkg-config --exists foo
$ echo $?
0
最值得注意的pkg-config特性是它所提供的版本检测,可以用来确定某个版本是否可用。
$ pkg-config --exists foo
$ echo $?
0
有些命令在结合--print-errors选项使用时可以输出更详细的信息。
$ pkg-config --exists --print-errors xoxo
Package xoxo was not found in the pkg-config search path.
Perhaps you should add the directory containing `xoxo.pc'
to the PKG_CONFIG_PATH environment variable
No package 'xoxo' found
上面的信息出现了PKG_CONFIG_PATH环境变量。这个变量用来配置pkg-config的搜索路径。在类Unix操作系统中,会搜索/usr/lib/pkconfig和/usr/share/pkgconfig目录。这通常已经覆盖了系统已经安装的模块。但是,有些本地模块可能安装在了其他路径,例如/usr/local。这种情况下,需要指定搜索路径,以便pkg-config可以定位.pc文件。
$ pkg-config --modversion hello
Package hello was not found in the pkg-config search path.
Perhaps you should add the directory containing `hello.pc'
to the PKG_CONFIG_PATH environment variable
No package 'hello' found
$ export PKG_CONFIG_PATH=/usr/local/lib/pkgconfig
$ pkg-config --modversion hello
1.0.0
autoconf也提供了一些宏,可以将pkg-config集成到项目中。
* PKG_PROG_PKG_CONFIG([MIN-VERSION]):定位系统中的pkg-config工具,并检测版本兼容性。
* PKG_CHECK_EXISTS(MODULES,[ACTION-IF-FOUND], [ACTION-IF-NOT-FOUND]):检测指定的模块是否存在。
* PKG_CHECK_MODULES(VARIABLE-PREFIX,MODULES, [ACTION-IF-FOUND], [ACTION-IF-NOT-FOUND])::检测指定的模块是否存在,如果存在,就根据 pkg-config --cflags和pkg-config --libs的输出设置
常见问题
1. 我的程序使用了x库,我该怎么做?
pkg-config的输出可以在编译命令中使用,假设x库已经有了一个叫做x.pc的pkg-config文件:
cc `pkg-config --cflags --libs x` -o myapp myapp.c
将pkg-config集成到autoconf和automake中使用会更强大。但是,用PKG_CONFIG_PATH宏可以很容易的在建立过程中访问元数据。
configure.ac:
PKG_CHECK_MODULES([X], [x])
Makefile.am:
myapp_CFLAGS = $(X_CFLAGS)
myapp_LDADD = $(X_LIBS)
如果找到了x模块,宏会填充和替代X_CFLAGS和X_LIBS变量。如果没有找到,会产生错误。配置PKG_CHECK_MODULES的第3、4个参数,可以控制没有找到模块时的动作。
2. 我的z库安装了保护libx头的头文件。我应该在z.pc中添加什么?
如果x库支持pkg-config,将它添加到Requires.private字段。如果不支持,就配置Cflags字段,添加一些使用libx头时所需的编译器标识。在这两种情况下,无论是否使用了--static,pkg-config都会输出编译器标识。
3. 我的z库内部使用了libx,但是不能再公开API中暴露libx的数据类型。我应该在z.pc中添加什么?
同样的,如果x支持pkg-config,就把它添加到Requires.private。这种情况下,就没必要发出编译器标识,但是在今天链接时要确保有链接器标识。如果libx不支持pkg-config,就将必要的链接器标识添加到Libs.private。
——————————————————————
Dan Nicholson
Copyright (C) 2010 Dan Nicholson.
This document is licensed under the GNU General Public License, Version 2 or any later version.