Makefile.am文件的实例讲解

Makefile.am是一种比Makefile更高层次的编译规则,可以和configure.in文件一起通过调用automake命令,生成Makefile.in文件,再调用./configure的时候,就将Makefile.in文件自动生成Makefile文件了。所以Makefile.am文件是比Makefile文件更高的抽象。

下面我根据自己的工作中的一些应用,来讨论Makefile.am的编写。我觉得主要是要注意的问题是将编译什么文件?这个文件会不会安装?这个文件被安装到什么目录下?可以将文件编译成可执行文件来安装,也可以编译成静态库文件安装,常见的文件编译类型有下面几种:

  1. PROGRAMS。表示可执行文件
  2. LIBRARIES。表示库文件
  3. LTLIBRARIES。这也是表示库文件,前面的LT表示libtool。
  4. HEADERS。头文件。
  5. SCRIPTS。脚本文件,这个可以被用于执行。如:example_SCRIPTS,如果用这样的话,需要我们自己定义安装目录下的example目录,很容易的,往下看。
  6. DATA。数据文件,不能执行。

一,可执行文件

先看一个实例:

[cpp]  view plain  copy
  1. bin_PROGRAMS = client  
  2.   
  3. client_SOURCES = key.c connect.c client.c main.c session.c hash.c  
  4. client_CPPFLAGS = -DCONFIG_DIR=\"$(sysconfdir)\" -DLIBRARY_DIR=\"$(pkglibdir)\"  
  5. client_LDFLAGS = -export-dynamic -lmemcached  
  6. noinst_HEADERS = client.h  
  7.   
  8. INCLUDES = -I/usr/local/libmemcached/include/  
  9.   
  10. client_LDADD = $(top_builddir)/sx/libsession.la \  
  11.             $(top_builddir)/util/libutil.la  
上面就是一个全部的Makefile.am文件,这个文件用于生成client可执行应用程序,引用了两个静态库和MC等动态库的连接。分析一下:

bin_PROGRAMS:表示指定要生成的可执行应用程序文件,这表示可执行文件在安装时需要被安装到系统中,如果只是想编译。不想被安装到系统中,可以用noinst_PROGRAMS来代替。

一个简单的问题是:bin_PROGRAMS=client 这一行表示什么意思?解释如下:

  1. PROGRAMS知道这是一个可执行文件。
  2. client表示编译的目标文件。
  3. bin表示目录文件被安装到系统的目录。
后面的包括头文件,静态库的定义都是这种形式,如lib_LIBRARIES=util,表示将util库安装到lib目录下。继续解释上面的文件

client_SOURCES:表示生成可执行应用程序所用的源文件,这里注意,client_是由前面的bin_PROGRAMS指定的,如果前面是生成example,那么这里就是example_SOURCES,其它的类似标识也是一样。

client_CPPFLAGS:这和Makefile文件中一样,表示C语言预处理器参数,这里指定了DCONFIG_DIR,以后在程序中,就可以直接使用CONFIG_DIR,不要把这个和另一个CFLAGS混淆,后者表示编译器参数。

client_LDFLAGS:这个表示在连接时所需要的库文件选项标识。这个也就是对应一些如-l,-shared等选项。

noinst_HEADERS:这个表示该头文件只是参加可执行文件的编译,而不用安装到安装目录下。如果需要安装到系统中,可以用include_HEADERS来代替。

INCLUDES:连接时所需要的头文件。

client_LDADD:连接时所需要的库文件,这里表示需要两个库文件的支持,下面会看到这个库文件又是怎么用Makefile.am文件后成的。

再谈谈关于上文中的全局变量引用,可能有人注意到$(top_builddir)等全局变量(因为这个文件之前没有定义),其实这个变量是Makefile.am系统定义的一个基本路径变量,表示生成目标文件的最上层目录,如果这个Makefile.am文件被其它的Makefile.am文件,这个会表示其它的目录,而不是这个当前目录。还可以使用$(top_srcdir),这个表示工程的最顶层目录,其实也是第一个Makefile.am的入口目录,因为Makefile.am文件可以被递归性的调用。

下面再说一下上文中出现的$(sysconfdir),在系统安装时,我们都记得先配置安装路径,如./configure --prefix=/install/apache 其实在调用这个之后,就定义了一个变量$(prefix),表示安装的路径,如果没有指定安装的路径,会被安装到默认的路径,一般都是/usr/local。在定义$(prefix),还有一些预定义好的目录,其实这一些定义都可以在顶层的Makefile文件中可以看到,如下面一些值:

bindir = $(prefix)/bin。

libdir = $(prefix)/lib。

datadir=$(prefix)/share。

sysconfdir=$(prefix)/etc。

includedir=$(prefix)/include。

这些量还可以用于定义其它目录,例如我想将client.h安装到include/client目录下,这样写Makefile.am文件:

[html]  view plain  copy
  1. clientincludedir=$(includedir)/client  
  2. clientinclude_HEADERS=$(top_srcdir)/client/client.h  
这就达到了我的目的,相当于定义了一个安装类型,这种安装类型是将文件安装到include/client目录下。

我们自己也可以定义新的安装目录下的路径,如我在应用中简单定义的:

[cpp]  view plain  copy
  1. devicedir = ${prefix}/device  
  2. device_DATA = package  
这样的话,package文件会作为数据文件安装到device目录之下,这样一个可执行文件就定义好了。注意,这也相当于定义了一种安装类型:devicedir,所以你想怎么安装就怎么安装,后面的XXXXXdir,dir是固定不变的。

二,静态库文件

编译静态库和编译动态库是不一样的,我们先看静态库的例子,这个比较简单。直接指定 XXXX_LTLIBRARIES或者XXXX_LIBRARIES就可以了。如果不需要安装到系统,将XXXX换成noinst就可以。还是再罗嗦一下:

  • 一般推荐使用libtool库编译目标,因为automake包含libtool,这对于跨平台可移植的库来说,肯定是一个福音。

看例子如下:

[cpp]  view plain  copy
  1. noinst_LTLIBRARIES = libutil.la  
  2.   
  3. noinst_HEADERS = inaddr.h util.h compat.h pool.h xhash.h url.h device.h   
  4.   
  5. libutil_la_SOURCES = access.c config.c datetime.c hex.c inaddr.c log.c device.c pool.c rate.c sha1.c stanza.c str.c xhash.c  
  6.   
  7. libutil_la_LIBADD = @LDFLAGS@  
第一行的noinst_LTLIBRARIES,这里要注意的是LTLIBRARIES,另外还有LIBRARIES,两个都表示库文件。前者表示libtool库,用法上基本是一样的。如果需要安装到系统中的话,用lib_LTLIBRARIES。

注意:静态库编译连接时需要其它的库的话,采用XXXX_LIBADD选项,而不是前面的XXXX_LDADD。编译静态库是比较简单的,因为直接可以指定其类型。

三,动态库文件

想要编译XXX.so文件,需要用_PROGRAMS类型,这里一个关于安装路径要注意的问题是,我们一般希望将动态库安装到lib目录下,按照前面所讨论的,只需要写成lib_PROGRAMS就可以了,因为前面的lib表示安装路径,但是automake不允许这么直接定义,可以采用下面的办法,也是将动态库安装到lib目录下

[cpp]  view plain  copy
  1. projectlibdir=$(libdir) //新建一个目录,就是该目录就是lib目录  
  2. projectlib_PROGRAMS=project.so  
  3. project_so_SOURCES=xxx.C  
  4. project_so_LDFLAGS=-shared -fpic //GCC编译动态库的选项  
这个动态库我没有测试,在网上找的一些资料,然后整理的。

四,SUBDIRS的用法

这是一个很重要的关键词,我们前面生成了一个一个的目标文件,但是一个大型的工程项目是由许多个可执行文件和库文件组成,也就是包含多个目录,每个目录下都有用于生成该目录下的目标文件的Makefile.am文件,但顶层目录是如何调用,才能使下面各个目录分别生成自己的目标文件呢?就是SUBDIRS关键词的用法了。

看一下我的工程项目,这是顶层的Makefile.am文件

[cpp]  view plain  copy
  1. EXTRA_DIST = Doxyfile.in README.win32 README.protocol contrib UPGRADE  
  2.   
  3. devicedir = ${prefix}/device  
  4. device_DATA = package  
  5.   
  6. SUBDIRS = etc man  
  7. if USE_LIBSUBST  
  8. SUBDIRS += subst  
  9. endif  
  10. SUBDIRS += tools io sessions util client dispatch server hash storage sms  
SUBDIRS表示在处理目录之前,要递归处理哪些子目录,这里还要注意处理的顺序。比如我的client对sessions和utils这两上目标文件有依赖,就在client之前需要处理这两个目标文件。

EXTRA_DIST:将哪些文件一起打包。

五,关于打包

Automake会自动的打包,自动打包的内容如下:

  1. 所有源文件。
  2. 所有的Makefile.am文件。
  3. configure读取的文件。
  4. Makefile.am中包含的文件。
  5. EXTRA_DIST指定的文件。
  6. 采用dist及nodist指定的文件,如可以将某一源文件指定为不打包:
    nodist_client_SOURCES = client.c
六,完成之前
基本上我介绍的都是我实际工作中所用的实例,GNU automake工具包含的东西,完不止这些。我介绍的过程也是我所认识的和参考的一些我所理解的,欢迎大家指出不足。
Makefile.am文件是GNU Automake工具使用的模板文件,它描述了软件包的编译规则和安装规则。Automake是一个从Makefile.in模板生成Makefile.in文件的工具,这些Makefile.in文件可以被GNU Autoconf生成的配置脚本用来生成最终的Makefile。 使用Makefile.am文件的步骤大致如下: 1. **创建Makefile.am文件**:在项目的相应目录中创建Makefile.am文件,该文件通常包含一些宏定义、源文件列表和编译安装规则等信息。 2. **编写Makefile.am内容**:Makefile.am文件中会定义一些宏,例如: - `bin_PROGRAMS`:定义可执行文件。 - `libRARIES`:定义库文件。 - `HEADERS`:定义头文件。 - `MAN`:定义手册页。 - `DATA`:定义数据文件。 在这些宏下面,会列出具体需要构建的文件,例如: ```makefile bin_PROGRAMS = myprog myprog_SOURCES = main.c utils.c myprog_CFLAGS = -Wall -g ``` 这个例子定义了一个名为myprog的可执行文件,其源文件main.c和utils.c,并指定了编译时的额外标志为-Wall -g。 3. **生成Makefile.in文件**:运行Automake工具,并传入Makefile.am文件所在的目录,Automake会根据Makefile.am的内容生成Makefile.in文件。 4. **配置软件包**:使用Autoconf生成的`configure`脚本来配置软件包。运行`./configure`将会读取Makefile.in并生成最终的Makefile。 5. **编译和安装**:使用生成的Makefile进行编译(`make`命令)和安装(`make install`命令)。 Makefile.am文件的编写需要遵循Automake的语法规则和约定,从而确保最终生成的Makefile能够正确地构建和安装软件包。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值