如果只是一个很小的程序,没有几个文件,那么,我们手工书写Makefile还是可以忍受的,如果是一个超大型的工程,谁能忍受的了,光一个Makefile文件就足以是Writer头疼了,但也没有人说一定要手工书写makefile文件。linux中提供了自动生成Makefile的工具,而且通过,这些工具生成的makefile还很符合GNU的习惯的。而且功能也很齐全。
自动生成makefile需要些工具来支持:autoscan,aclocal,autoconf,automake如果需要生成动态库还需要libtool。
举例子来说,假如我有test.c,p1.c,p2.c,myheader.h这几个文件,而且test.c需要调用在myheader.h中声明过的p1,p2函数,他们分别有p1.c,p2.c来实现。
首先,进入上面几个文件的目录,然后执行autoscan,autoscan用来扫描源代码目录并生成cinfigure.scan文件,和一个日志文件,但对我们来说有用的是前者:
$autoscan
之后就会生成autoscan.log configure.scan这两个文件,为了简单,我将所有的注释(以#开头)都删除了,打开configure.scan:
AC_PREREQ(2.59)
AC_INIT(FULL-PACKAGE-NAME, VERSION, BUG-REPORT-ADDRESS)
AC_CONFIG_SRCDIR([p1.c])
AC_CONFIG_HEADER([config.h])
AC_PROG_CC
AC_OUTPUT
然后修改该文件,为后来使用作准备:
AC_INIT(test.c)
AM_INIT_AUTOMAKE(test, 1.0)
AC_PROG_CC
AC_OUTPUT(Makefile)
后面在解释为什么要这么修改。做完上面的工作之后,我们要将configure.scan改名为configure.in让aclocal使用,他是一个perl脚本程序,他根据configure.in文件的内容生成aclocal.m4文件
$aclocal
然后就是生成configure脚本文件,
$autoconf
这样ls后就会有一个configure的可执行的文件。当然,工作到这还没有结束,我们还要写一个Makefile.am文件,用来生成Makefile.in文件。
AUTOMAKE_OPTIONS= foreign
bin_PROGRAMS= test
test_SOURCES= test.c p1.c p2.c myheader.h
这样就生成一个Makefile.in的文件,此时运行一下configure脚本就会得到Makefile,打开之后会发现,太丰富了!make一下可执行的test被编译好了。
上面的工作可以用下面的图来表示:
autuscan -> (configure.scan) -> (configure.in) | -> autoconf ->(configure) -->(Makefile)
| | -> aclocal -> (aclocal.m4) -v
---------------------------------------v
(Makefile.am) -> automake-> (Makefile.in) -^
下面来分别解释一下上面的文件和命令:
autoscan : 用来扫描源代码的目录生成configure.scan,如果不指定扫描目录,则他会扫描当前工作目录,
configure.scan : 包含了系统配置的基本选项,一般用来制作configure.in文件。
configure.in : 该文件的内容都是一些宏,其中的顺序没有硬性的规定,不过建议的顺序是:
AC_INIT
测试程序
测试函数
测试头文件
测试类型定义
测试结构
测试编译器
测试库函数
测试系统调用
AC_OUTPOUT
更详细的请参阅《GNU/Linux编程指南(第二版)》
aclocal: 根据configure.in的内容自动生成aclocal.m4文件,其定义为:alocal : creat alocal.m4 by scanning configure.ac
alocal.m4 : 在执行automake的时候还需要其他的一些宏,这就由alocal产生。当有了 configure.in和alocal.m4两个宏文件后,就可以用automake来产生Makefile.in文件了。
autoconf: 用来产生configure脚本程序。
configure: 他能根据不同的系统,产生不同的Makefile,从而是我们的程序具有可移植性。他还有一些参数:
--cache-file=FILE 测试系统的特性,并将结果放到FILE中
--help 输出帮助信息
--no-create 阻止其生成输出文件
--quiet 执行是不做输出
--silent 同上,若设置则不会有任何输出到屏幕
--version 输出automake的版本号
--prefix=PEWFIX 安装位置设置(常用)
--exec-prefix=EPREFIX 设置结构倚赖的文件的安装位置
--bindir=DIR 指定可执行文件的安装位置.
--sbindir=DIR 指定超级用户可执行的安装位置.
--libexecdir=DIR 指定可执行支持文件的安装位置.
--datadir=DIR 指定通用数据文件的安装位置.
--sysconfdir=DIR 指定只读数据的安装位置.
--sharedstatedir=DIR 指定共享的可写数据的安装位置.
--localstatedir=DIR 指定(非共享)可写数据的安装位置.
--libdir=DIR 指定库文件的安装位置.
--includedir=DIR 指定C头文件的安装位置.
--oldincludedir=DIR 指定为除GCC外编译器安装的C头文件的安装位置.
--infodir=DIR 指定Info格式文档的安装位置.
--mandir=DIR 指定手册页的安装位置.
--srcdir=DIR 源码的位置
--program-prefix=PREFIX 增加安装程序名字前缀.
--program-suffix=SUFFIX 增加安装程序名字后缀.
--program-transform-name=PROGRAM 产生安装名
--build=BUILD 指定软件包安装的系统平台.
--host=HOST 指定软件运行的系统平台.
--target=GARGET 指定软件面向的系统平台
--disable-FEATURE 提供为大型选项的编译时配置
--enable-FEATURE[=ARG] 提供了一些默认被禁止的特性
--enable-FEATURE=no 同--disable-FEATURE
--with-PACKAGE[=ARG] 使用已有软件包和库
--with-PACKAGE=no --without-PACKAGE同义
--without-PACKAGE 禁止软件包与系统已有的软件包交互
--x-includes=DIR --with-PACKAGE的一个特例
--x-libraries=DIR 向configure脚本指明包含X11库的目录
makefile.am : 根据他生成makefile.in文件。
AUTOMAKE_OPTIONS = foreign 设置automake的选项,设置成foreign表示按一般软件处理。
bin_PROGRAMS = filename1 [...] 产生的可执行文件名
filename1_SOURCES = f1.c f2.c 指明生成filename可执行文件的源码
[...]
filename_LDADD =
filename_LDFLAGS =
filename_DEPENDENCIES =
静态库lib_LIBRARIES = libfilename.a
filename_a_SOURCES =
filename_a_LDADD =
filename_a_LIBADD =
filename_a_LDFALGS =
头文件include_HEADERS = filename.h
数据文件data_DATA = data1 data2
(对于可执行文件和静态库类型如果只想编译,不想安装到系统中,可以用:noinst_PROGRAMS代替bin_PROGRAMS
和noinst_LIBRARIES代替lib_LIBRARIES)
automake : 使用automake --add-missing来生成Makefile.in文件,后面的选项使我们制作出来的Makfile符合GNU的习惯,
Makefile : GNU的习惯:
make 编译,连接,生成可执行文件;
make clean 清除编译产生的obj文件,以及可执行文件;
make install 安装到系统中,一般是/usr/local/bin
make dist 将源码打包,以便发布
make diskcheck 生成包,并对其进行检查,以确保其正确性;