Linux实用工具-GNU Auto Tools基本使用

Auto Tools是基于GNU的自动为你的软件源代码生成Makefile以及编译并发布软件软件包等的一套软件包管理工具集合。

这里,用一个具体的例子,来演示一下Autotools工具相对之前的"Auto Tools使用流程"例子(hello-1.0)比较"传统"的基本使用方法。假设程序名字为hello-2.0,通过这个例子,我们可以知道使用GNU Autotools 需要自己写哪些文件,如何编写这些文件,以及如何设置程序链接的静态库,如何指定库的安装与不安装,如何设置编译选项等等。

内容比较多,所以首先给出一个总体的使用方法,之后对每一个步骤进行详细的讲解。

主要内容:

一、总体步骤
二、详细解释
三、关于configure配置
四、文件详细信息
五、其它

一、总体步骤

1,编写hello-2.0程序

查看最初代码目录结构如下:

#pwd
/root/test/hello-2.0
#ls
lib src my_doc

这里大致内容如下:

  • src/中的文件是程序的主文件。
  • lib/中的文件是程序需要链接的库。
  • lib/iface1中的库只是作为链接使用,并不安装在系统中。
  • lib/ifacex中的库不仅要链接,而且安装的时候要安装到系统中。

2,为程序hello-2.0添加Autotools模板文件

为了让Autotools可以自动生成符合标准GNU风格的Makefile,需要添加一些文件。实际上,我们必须写并且要向源代码中添加的文件就两种:

  • configure.ac,
  • Makefile.am。

3,生成配置文件

# autoreconf --install

运行这个命令之后会生成了许多其他的文件,其中核心的程序是configure脚本,通过它以及生成的 Makefile.in 文件,最终会生成需要的Makefile.

4,配置并生成Makefile

#mkdir hello_build
#cd hello_build
#../configure CONFIG_SITE=$(pwd)/../config.site --prefix=$(pwd)/target

这里,会生成Makefile,生成Makefile之后,才能构进行后面的编译工作。

5,编译并安装软件

#make
#make install

运行 make 就是编译,运行 make install 会将生成的将要安装的文件复制到之前指定的prefix目录(这里就是target)中。

6,发布软件包

#make dist

这样就会对软件包进行发布。发布的内容被打包到一个tar包中,其中包含的内容没有编译中间文件,而是运行 autoreconf --install 命令之后的一些文件。

二、详细解释

这里,对以上的步骤进一步解释,想要知道其中涉及到的文件的具体内容,可以参考后面的"文件详细信息"

1、编写hello-2.0程序

编写之后,使用find命令查看程序的源代码目录结构如下:

#cd hello-2.0
#find .
.
./my_doc
./src
./src/main.c
./src/myprint.c
./src/myprint.h
./lib
./lib/iface1
./lib/iface1/print1.c
./lib/iface1/print1.h
./lib/ifacex
./lib/ifacex/print2.c
./lib/ifacex/print3.c
./lib/ifacex/printx.h

这里,为了方便阅读,我调整了输出的顺序。其中my_doc文件是一个自己随便写的说明文件,记录本程序使用的方法供学习使用。根据输出,我们可以了解到,

  • src/中的文件是程序的主文件。
  • lib/中的文件是程序需要链接的库。
  • lib/iface1中的库只是作为链接使用,并不安装在系统中。
  • lib/ifacex中的库不仅要链接,而且安装的时候要安装到系统中。

2、为程序hello-2.0添加Autotools模板文件

添加之后,使用find命令查看源代码目录结构如下:

#cd hello-2.0
#find .
.
./my_doc
./src
./src/main.c
./src/myprint.c
./src/myprint.h
./lib
./lib/iface1
./lib/iface1/print1.c
./lib/iface1/print1.h
./lib/ifacex
./lib/ifacex/print2.c
./lib/ifacex/print3.c
./lib/ifacex/printx.h
./configure.ac
./config.site
./Makefile.am
./src/Makefile.am
./lib/Makefile.am
./lib/iface1/Makefile.am
./lib/ifacex/Makefile.am

这里,为了方便阅读,我调整了输出顺序。根据输出我们可以知道,为了让Autotools可以自动生成符合标准GNU风格的Makefile,我们添加了不少的文件。下面分别对添加的文件的功能进行介绍。

./configure.ac

这个文件是Autoconf工具需要读取的模板文件,通过这个文件指定软件包一些全局的信息。

./config.site

这个文件是可选的,是将要生成configure脚本的配置文件,configure可以通过这个配置文件设置一些选项。

./Makefile.am, ./src/Makefile.am, ./lib/Makefile.am, ./lib/iface1/Makefile.am, ./lib/ifacex/Makefile.am

这些Makefile.am文件都是是生成Makefile文件所需的模板文件,包含源代码文件以及一些编译选项信息。

通过一个顶层的 Makefile.am 指定各个子 Makefile.am 位置,实际可以把所有文件内容集中到一个Makefile.am中,不过现在这样写,是为了了解关于Makefile.am更多的知识。

总结

更多这些文件具体的信息,通过后面给出的文件详细内容可以了解。实际上,通过以上的介绍,我们可知,为了让Autotools真正开始工作,我们必须写并且要向源代码中添加的文件就两种:

  • configure.ac,
  • Makefile.am。

3、生成配置文件

# autoreconf --install
configure.ac: installing `build-aux/install-sh'
configure.ac: installing `build-aux/missing'
lib/iface1/Makefile.am: installing `build-aux/depcomp'

运行之后,查看代码目录如下:

#find .
.
./my_doc
./src
./src/main.c
./src/myprint.c
./src/myprint.h
./lib
./lib/iface1
./lib/iface1/print1.c
./lib/iface1/print1.h
./lib/ifacex
./lib/ifacex/print2.c
./lib/ifacex/print3.c
./lib/ifacex/printx.h

./configure.ac
./config.site
./Makefile.am
./src/Makefile.am
./lib/Makefile.am
./lib/iface1/Makefile.am
./lib/ifacex/Makefile.am

./configure
./Makefile.in
./aclocal.m4
./config.h.in
./lib/Makefile.in
./lib/iface1/Makefile.in
./lib/ifacex/Makefile.in
./src/Makefile.in
./build-aux
./build-aux/missing
./build-aux/install-sh
./build-aux/depcomp
./autom4te.cache
./autom4te.cache/output.0
./autom4te.cache/requests
./autom4te.cache/traces.1
./autom4te.cache/output.1
./autom4te.cache/traces.0

这里,为了便于阅读,我调整了输出的顺序。

可见,除了之前我们添加的文件之外,还生成了许多其他的文件,其中核心的程序是configure脚本,通过它以及生成的Makefile.in文件,最终会生成需要的Makefile.

至此,这些文件也就是我们发布软件时候的文件了。下面我们看看编译的过程。

4、配置并生成Makefile

#mkdir hello_build
#cd hello_build
#../configure CONFIG_SITE=$(pwd)/../config.site --prefix=$(pwd)/target

运行之后,查看代码目录如下:

# pwd
/root/test/hello-2.0/hello_build
# find
.
./src
./src/.deps
./src/.deps/myprint.Po
./src/.deps/main.Po
./src/Makefile
./config.h
./lib
./lib/iface1
./lib/iface1/.deps
./lib/iface1/.deps/print1.Po
./lib/iface1/Makefile
./lib/ifacex
./lib/ifacex/.deps
./lib/ifacex/.deps/print2.Po
./lib/ifacex/.deps/print3.Po
./lib/ifacex/Makefile
./lib/Makefile
./config.log
./stamp-h1
./config.status
./Makefile

这里,生成Makefile之后,才能构进行后面的编译工作。

  1. 我们创建hello_build目录的目的是想要将编译的中间文件集中到一个目录中而不会影响到原来的源代码的目录结构,因为configure程序的特点就是,在那里运行它,那么就会将哪里做为编译的初始目录;
  2. 我们使用CONFIG_SITE环境变量设置configure读取的配置文件(config.site)的位置,这里要使用绝对路径,如果没有设置这个环境变量,那么就会寻找prefix/share/config.site文件,如果还没有就使用prefix/etc/config.site文件;
  3. prefix就是使用–prefix指定的软件安装目录,后面做"make install"安装软件的时候,会将软件安装到这个目录下。

另外,如果我们在config.site指定用交叉编译(比如arm),那么运行的命令类似如下

#../configure CONFIG_SITE=$(pwd)/../config.site --host=i686 --prefix=$(pwd)/target

这里使用–host指定编译主机。

5、编译并安装软件

#make
#make install

运行make就是为了编译软件,运行make install会将生成的将要安装的文件复制到之前指定的prefix目录(这里就是target)中。

查看运行之后的目录结构如下:

# pwd
/root/test/hello-2.0/hello_build
# find .
.
./src
./src/myprint.o
./src/.deps
./src/.deps/myprint.Po
./src/.deps/main.Po
./src/main.o
./src/hello
./src/Makefile
./config.h
./target
./target/bin
./target/bin/hello
./target/lib
./target/lib/libifacex.a
./target/include
./target/include/printx.h
./target/include/myprint.h
./lib
./lib/iface1
./lib/iface1/libiface1.a
./lib/iface1/print1.o
./lib/iface1/.deps
./lib/iface1/.deps/print1.Po
./lib/iface1/Makefile
./lib/ifacex
./lib/ifacex/print3.o
./lib/ifacex/libifacex.a
./lib/ifacex/print2.o
./lib/ifacex/.deps
./lib/ifacex/.deps/print2.Po
./lib/ifacex/.deps/print3.Po
./lib/ifacex/Makefile
./lib/Makefile
./config.log
./stamp-h1
./config.status
./Makefile

这里,我们就完成了软件的编译和安装。

6、发布软件包

#make dist

这样就会对软件包进行发布。查看这个命令之后的目录结构如下:

#ls -p
config.h config.log config.status hello-2.0.tar.gz lib/ Makefile src/ stamp-h1 target/

从这里我们发现,实际就是生成了一个hello-2.0.tar.gz,也就是软件的发布包。查看其中的内容如下:

#tar -tzvf hello-2.0.tar.gz
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/
-rwxrwxrwx 500/500 20 2011-06-17 11:19:57 hello-2.0/Makefile.am
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/src/
-rwxr-xr-x 0/0 228 2011-06-17 11:25:40 hello-2.0/src/Makefile.am
-rwxrwxrwx 500/500 292 2011-06-17 13:47:46 hello-2.0/src/main.c
-rwxrwxrwx 500/500 141 2011-06-16 18:15:09 hello-2.0/src/myprint.c
-rwxrwxrwx 500/500 63 2011-01-31 15:32:46 hello-2.0/src/myprint.h
-rw-r--r-- 0/0 13581 2011-06-17 14:28:58 hello-2.0/src/Makefile.in
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/build-aux/
-rwxr-xr-x 0/0 11014 2011-06-17 14:28:58 hello-2.0/build-aux/missing
-rwxr-xr-x 0/0 9233 2011-06-17 14:28:58 hello-2.0/build-aux/install-sh
-rwxr-xr-x 0/0 15936 2011-06-17 14:28:58 hello-2.0/build-aux/depcomp
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/lib/
-rwxr-xr-x 0/0 24 2011-06-16 18:09:59 hello-2.0/lib/Makefile.am
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/lib/iface1/
-rwxr-xr-x 0/0 119 2011-06-16 18:13:32 hello-2.0/lib/iface1/Makefile.am
-rwxrwxrwx 500/500 78 2011-02-01 14:42:55 hello-2.0/lib/iface1/print1.c
-rwxrwxrwx 500/500 60 2010-10-19 11:58:30 hello-2.0/lib/iface1/print1.h
-rw-r--r-- 0/0 11388 2011-06-17 14:28:58 hello-2.0/lib/iface1/Makefile.in
-rw-r--r-- 0/0 13019 2011-06-17 14:28:58 hello-2.0/lib/Makefile.in
drwxr-xr-x 0/0 0 2011-06-17 14:49:23 hello-2.0/lib/ifacex/
-rwxrwxrwx 500/500 82 2011-06-16 18:08:28 hello-2.0/lib/ifacex/print2.c
-rwxr-xr-x 0/0 235 2011-06-17 10:34:32 hello-2.0/lib/ifacex/Makefile.am
-rwxr-xr-x 0/0 82 2011-06-16 18:08:39 hello-2.0/lib/ifacex/print3.c
-rwxrwxrwx 500/500 75 2011-06-16 18:07:56 hello-2.0/lib/ifacex/printx.h
-rw-r--r-- 0/0 13795 2011-06-17 14:28:58 hello-2.0/lib/ifacex/Makefile.in
-rw-r--r-- 0/0 31120 2011-06-17 14:28:56 hello-2.0/aclocal.m4
-rwxrwxrwx 500/500 878 2011-06-17 11:19:04 hello-2.0/configure.ac
-rwxr-xr-x 0/0 132936 2011-06-17 14:28:57 hello-2.0/configure
-rw-r--r-- 0/0 18068 2011-06-17 14:28:58 hello-2.0/Makefile.in
-rw-r--r-- 0/0 557 2011-06-17 14:28:57 hello-2.0/config.h.in

这里,实际我们发布软件包的时候,发布的文件有一些是通过我们最初建立的那几个Makefile.am指定的,我们发布的文件都是运行

$autoreconf --install

生成configure之后的、但是生成Makefile之前的那些文件。(注意,我们可以看到其中的config.site和自己建立的文档my_doc并没有被包含进来。)

三、关于configure配置

对于configure配置,生成的Makefile,以及编译安装之后的程序,我们需要了解如下一些知识。

1,一些常用的make目标

实际上,在前面我们还可以运行 make clean, make distclean 等命令,之前的讲解已经对这些做了实践,这里就不演示了。

这里对一些常见的Makefile目标进行一点总结,如下:

  • "make all" 编译程序,库,以及文档等等(和"make"一样)。
  • "make install" 安装可执行程序。
  • "make install-strip" 和"make install"一样, 然后去掉调试信息。
  • "make uninstall" 和"make install"功能一样。
  • "make clean" 清除编译的中间文件(和"make all"功能相反)
  • "make distclean" 除了前面的"make clean"之外,还清除所有"./configure"创建的中间文件。
  • "make check" 如果有测试包的话,则运行之。
  • "make installcheck" 如果支持的化,就检查已经安装的程序或者库。
  • "make dist" 创建PACKAGE-VERSION.tar.gz发布包。

2,configure常用路径变量

./configure和安装路径相关的常用变量:

变量名称 默认值
prefix "/usr/local"
exec-prefix prefix
bindir exec-prefix"/bin"
libdir exec-prefix"/lib"
...
includedir prefix"/include"
datarootdir prefix"/share"
datadir datarootdir
mandir datarootdir"/man"
infodir datarootdir"/info"

这些变量可以自行指定,例如指定prefix之后再进行安装,例如:

$./configure --prefix ~/usr
$make
$make install

这里,–prefix后面可以有"="也可以没有。

3,configure常用配置选项变量:

./configure和配置相关的常用变量:

CC C编译器命令
CFLAGS C编译选项
CXX C++编译器命令
CXXFLAGS C++编译选项
LDFLAGS 链接选项
CPPFLAGS C/C++预处理选项

更多的选项可以参见 ./configure --help

使用举例:

$./configure --prefix ~/usr CC=gcc-3 CPPFLAGS=-I$HOME/usr/include LDFLAGS=-L$HOME/usr/lib

四、文件详细信息

这里,列出了最初那些需要我们写的文件的内容,也注明了其中需要注意的内容。

src/main.c文件:

#include <stdio.h>
#include "myprint.h"
int main(int argc, char *argv[])
{
#ifdef MYDEBUG1
printf("Define MYDEBUG1\n");
#else
printf("Havn't define MYDEBUG1\n");
#endif
#ifdef MYDEBUG2
printf("Define MYDEBUG2\n");
#else
printf("Havn't define MYDEBUG2\n");
#endif
myprint();
return 0;
}

src/myprint.h文件:

#ifndef __MYPRINT_H
#define __MYPRINT_H
void myprint();
#endif

src/myprint.c文件:

#include "myprint.h"
#include "../lib/iface1/print1.h"
#include "../lib/ifacex/printx.h"
void myprint()
{
print1();
print2();
print3();
}

lib/iface1/print1.h

#ifndef __PRINT1_H
#define __PRINT1_H
void print1();
#endif

lib/iface1/print1.c

#include <stdio.h>
#include "print1.h"
void print1()
{
printf("print1\n");
}

lib/ifacex/printx.h

#ifndef __PRINTX_H
#define __PRINTX_H
void print2();
void print3();
#endif

lib/ifacex/print2.c

#include <stdio.h>
#include "printx.h"
void print2(void)
{
printf("print2\n");
}

lib/ifacex/print3.c

#include <stdio.h>
#include "printx.h"
void print3(void)
{
printf("print3\n");
}

至此,列出的都是源代码,后面将列出Autotools需要的文件。

./configure.ac

#This file is edited by QuietHeart.

#*Package name, version, and bug report address.
AC_INIT([hello], [2.0], [quiet_heart000@126.com])

#A safe check make sure 'configure' is not run from outer space.
AC_CONFIG_SRCDIR([src/main.c])

#Auxiliary scripts such as install-sh and depcomp should be in this directory.
AC_CONFIG_AUX_DIR([build-aux])

#Turn automake warnings and report them as errors. This is a foreign package.
AM_INIT_AUTOMAKE([-Wall -Werror foreign])

#Check for C compile.
AC_PROG_CC

#For libuse in Makefile.am
AC_PROG_RANLIB
#AC_PROG_LIBTOOL

#Declare config.h as output header.
AC_CONFIG_HEADERS([config.h])

#XXX Declare Makefile as output file
#!!!Note:The order should be the same as Makefile.am, indicate the build order.
AC_CONFIG_FILES([Makefile
lib/Makefile
lib/iface1/Makefile
lib/ifacex/Makefile
src/Makefile
])

#Output all declared files.
AC_OUTPUT

这个文件,需要注意的一个地方就是使用AC_CONFIG_FILES指定的Makefile次序要和Makefile.am中指定的目录递归次序一致,并且这个次序也表明了编译的次序。

./config.site

###### Compile tools.
#CC=arm-sony-linux-gnueabi-gcc
CC=gcc
#LD=arm-sony-linux-gnueabi-ld
#AR=arm-sony-linux-gnueabi-ar
#LDFLAGS+="-lsonypthread -lsonydl -lsonyrt -lsonyc -lsonygcc_s "

###### Global options.
#!!!Note: You'd better put the variables value in quote,and add ' ' after param value.
CFLAGS+="-g "
CFLAGS+="-D MYDEBUG1 "
CFLAGS+="-D MYDEBUG2 "

###### Miscellaneous
#prefix=
#test -z "$CC" && CC=gcc-3
#test -z "$CPPFLAGS" && CPPFLAGS=-I$HOME/usr/include
#test -z "$LDFLAGS" && LDFLAGS=-L$HOME/usr/lib

这个文件需要注意的就是,编译选项的值要用""括起来,并且最后要有一个空格。

./Makefile.am

SUBDIRS = . lib src

src/Makefile.am

#AUTOMAKE_OPTIONS=foreign
bin_PROGRAMS=hello
hello_SOURCES=
include_HEADERS=

hello_SOURCES+=main.c \
myprint.c

include_HEADERS+=myprint.h

#!!!Note the path.
hello_LDADD=../lib/iface1/libiface1.a ../lib/ifacex/libifacex.a

这里需要注意的是,使用hello_LDADD可以指定hello需要链接的库,路径要使用相对当前Makefile.am的路径。如果需要专门指定hello这个程序的编译选项可以设置hello_CFLAGS等,具体参见手册。

lib/Makefile.am

SUBDIRS = iface1 ifacex

lib/iface1/Makefile.am

#lib not installed only used for building.
noinst_LIBRARIES=libiface1.a

libiface1_a_SOURCES= print1.c \
print1.h

lib/ifacex/Makefile.am

#lib tobe installed
lib_LIBRARIES=libifacex.a

#source code files.
#note: you can use 'find -name *.c' if has too much source code.
libifacex_a_SOURCES= print2.c \
print3.c

#header files to be installed.
include_HEADERS=printx.h

这里有个技巧,就是如果代码文件太多了的话Makefile.am也会很多,我们可以用类似"find -name *.c"的命令来输出相应的文件,用"+="的赋值的方式统一处理添加。

五、其它

参考资料:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343