如何自动生成Makefile

时间:2022-08-19 04:43:49
如何自动生成Makefile

如果只是一个很小的程序,没有几个文件,那么,我们手工书写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 生成包,并对其进行检查,以确保其正确性;