無論對於一個初學者還是一個資深的Linux程序員,編寫Makefile文件都是一件很麻煩的事;再者,開發人員應該把主要的精力放在程序代碼的編寫上,而在Makefile文件花費太多的精力顯然是不明智的;還有,對於不同的處理器架構,往往編譯器不同,環境不同,特別是一些嵌入式系統中的各種程序的編譯,於是移植問題也使Makefile文件編寫趨於複雜,也顯得這個問題很重要。對於這些問題Linux的高手們早已想到了,所以他們開發了一些能夠自動生成Makefile文件的工具。他們就是下面這些工具:
〉GNU Automake
〉GNU Autoconf
〉GNU m4
〉perl
〉GNU Libtool
因此您的操作系統必須安裝以上軟件,否則就不能夠自動生成Makefile文件或者在生成的過程中出現各種各樣的問題。用 autoconf/automake/autoheader工具來處理各種移植性的問題,用這些工具完成系統配置信息的收集,製作Makefile文件。然後在打算編譯源碼時只需要通過 "./configure; make"這樣簡單的命令就可以得到乾淨利落的編譯。
製作Makefile文件需要以下幾步:
1〉建立編譯目錄和源代碼目錄及源代碼文件(即要編譯的源文件)
[root@localhost leaf]#mkdir testmk
[root@localhost leaf]#cd testmk
[root@localhost testmk]#vi hello.c
編輯hello.c文件如下內容:
/*filename:hello.c*/
#include <stdio.h>
int main(int argc,char **argv)
{
printf("%s/n","Hello, World!")
return 0;
}
2〉利用autoscan工具產生一個configure.in文件的模板文件configure.scan文件:
[root@localhost testmk]#autoscan
[root@localhost testmk]#ls
configure.scan hello.c
3〉把configure.scan文件更名爲configure.in文件,並編譯其內容如下:
[root@localhost testmk]#mv configure.scan configure.in
[root@localhost testmk]#vi configure.in
dnl Process this file with autoconf to produce a configure script.
AC_INIT(hello.c)
dnl Add the file by leaf
AM_INIT_AUTOMAKE(hello,1.0)
dnl Checks for programs.
AC_PROG_CC
dnl Checks for libraries.
dnl Checks for header files.
dnl Checks for typedefs, structures, and compiler characteristics.
dnl Checks for library functions.
AC_OUTPUT(Makefile)
4〉執行aclocal,會產生aclocal.m4文件
[root@localhost testmk]#aclocal
[root@localhost testmk]#ls
aclocal.m4 configure.in hello.c
5〉執行autoconf,會產生confiure文件
[root@localhost testmk]#autoconf
[root@localhost testmk]#ls
aclocal.m4 [autom4te.cache] configure configure.in hello.c
6〉創建文件Makefile.am並編輯其內容如下:
AUTOMAKE_OPTIONS=foreign
bin_PROGRAMS=hello
hello_SOURCES=hello.c
其中,hello爲編譯後產生的可執行文件名稱,而第三行等號後面爲源文件列表
7〉執行automake程序,automake程序會根據Makefile.am產生一些文件,其中最重要的是Makefile.in文件:
[root@localhost testmk]#automake --add-missing
configure.in: installing `./install-sh'
configure.in: installing `./mkinstalldirs'
configure.in: installing `./missing'
Makefile.am: installing `./depcomp'
[root@localhost testmk]#ls
aclocal.m4 [autom4te.cache] configure configure.in depcomp
hello.c install-sh Makefile.am Makefile.in missing
mkinstalldirs
8〉執行configure腳本,生成我們需要的Makefile文件。
[root@localhost testmk]#./configure
checking for a BSD-compatible install... /usr/bin/install -c
checking whether build environment is sane... yes
checking for gawk... gawk
checking whether make sets $(MAKE)... yes
checking for gcc... gcc
checking for C compiler default output... a.out
checking whether the C compiler works... yes
checking whether we are cross compiling... no
checking for suffix of executables...
checking for suffix of object files... o
checking whether we are using the GNU C compiler... yes
checking whether gcc accepts -g... yes
checking for gcc option to accept ANSI C... none needed
checking for style of include used by make... GNU
checking dependency style of gcc... gcc3
configure: creating ./config.status
config.status: creating Makefile
config.status: executing depfiles commands
9〉最後只執行make就大功告成了:
[root@localhost testmk]#make
source='hello.c' object='hello.o' libtool=no /
depfile='.deps/hello.Po' tmpdepfile='.deps/hello.TPo' /
depmode=gcc3 /bin/sh ./depcomp /
gcc -DPACKAGE_NAME="" -DPACKAGE_TARNAME="" -DPACKAGE_VERSION="" -DPACKAGE_STRING="" -DPACKAGE_BUGREPORT="" -DPACKAGE="hello" -DVERSION="1.0" -I. -I. -g -O2 -c `test -f 'hello.c' || echo './'`hello.c
gcc -g -O2 -o hello hello.o
備註:
1.以上內容均在RedHat Linux 9.0環境下測試通過。
2.參考書目《Linux 程序設計權威指南》於明儉、陳向陽、方漢編著
3.其它國內外網站資料
4.RedHat 9.0下帶的程序文件及版本
autoconf-2.57-3.noarch.rpm
automake-1.6.3-5.noarch.rpm
gcc-3.2.2-5.i386.rpm
在介紹了簡單的如何產生Makefile文件之後,相信已經能夠編寫Makefile文件了,但那還遠
遠不夠,因爲編寫那麼簡單的Makefile文件還要如此繁瑣和興師動衆,那未免小題大做了。因此,
我們有必要進一步瞭解如何製作Makefile文件。在製作Makefile文件的過程中,編寫configure.in
文件是關鍵,因此這一部分將重點介紹configure.in文件的編寫的相關問題。對於Makefile.am文
件的編寫在編譯多文件和多目錄中要用到,所以下一篇將會介紹。
在打開生成的Makefile文件,才發現能夠編譯的目標(target)有很多,因此有必要介紹一下
其中比較重要而且常用的目標(target)的含義,也就是說,如何使用這個Makefile文件。
1〉make 或 make all
開始執行編譯過程,產生我們設定的目標。這時會開始編譯必要的源代碼文件,然後進
行連結,並且最終生成可執行文件或我們想要生成庫文件。 具體命令行如下:
[root@localhost testmk]#make
或
[root@localhost testmk]#make all
2〉make dist
將程序的源代碼和相關文檔打包成一個壓縮文件,以用來備份源代碼文件。命令完成後
會在目錄下會產生一個以 PACKAGE-VERSION.tar.gz 爲名稱的打包文件。PACKAGE 和 VERSION
這兩個變量是在configure.in文件中定義的。在我們的例子中定義如下:
AM_INIT_AUTOMAKE(hello,1.0)
因此會在目錄中生成一個名爲hello-1.0.tar.gz的文件。具體命令行如下:
[root@localhost testmk]#make dist
3〉make install
將正確編譯生成的可執行文件或庫文件安裝到系統中,通常是/usr/local/bin這個目錄。
如果沒有需要安裝的可執行文件或庫文件,將會自動執行make命令進行編譯,然後再進行安裝操
作。具體命令行如下:
[root@localhost testmk]#make install
4〉make clean
清除之前所編譯的可執行文件以及目標文件(Object Files, *.o)。 具體命令行如下:
[root@localhost testmk]#make clean
5〉make distclean
清除之前所編譯的可執行文件、目標文件(Object Files, *.o)以及由執行./configure
所產生的 Makefile文件。 具體命令行如下:
[root@localhost testmk]#make distclean
在弄清Makefile文件如何使用之後,我們來進一步瞭解生成Makefile文件的有關問題。先看
一下源文件的結構和內容:
/hello-1.0
/hello-1.0/pubfun.h
/hello-1.0/pubfun.c
/hello-1.0/hello.c
-------------------------------------------------
/*filename:pubfun.h */
#include <stdio.h>
void *printA(void *pdata);
-------------------------------------------------
/*filename:pubfun.c */
#include <stdio.h>
#include <math.h>
#include <pubfun.h>
void *printA(void *pdata)
{
printf("%f/n",sin(2));
printf("Hello,World -->%d/n",getpid());
}
-------------------------------------------------
/*filename:hello.c */
#include <stdio.h>
#include <pthread.h>
#include "pubfun.h"
int main(int argc,char **argv)
{
int ret;
pthread_t ptid;
int index;
// create a thread
ret = pthread_create(&ptid,NULL,printA,(void *)&index);
if(ret)
return -1;
printA(NULL);
return 0;
}
---------------------------------------------------
對於我們的源文件中用到了數學庫和線程庫,還有我們自己寫的頭文件,我們修改configure.in 文件如下:
01:dnl Process this file with autoconf to produce a configure script.
02:AC_INIT(hello.c)
03:
04:dnl Add the file by leaf
05:AM_INIT_AUTOMAKE(hello,1.0)
06:
07:dnl 檢查C編譯器.如果在環境中沒有設定CC,就查找gcc,如果沒有找到,就使用cc.
08:AC_PROG_CC
09:
10: dnl 爲C編譯器提供的調試和優化選項.
11: CFLAGS=" -O2"
12:
13: dnl 爲C預處理器和編譯器提供頭文件搜索目錄選項('-Idir')以及其他各種選項.
14: CPPFLAGS=" -I."
15:
16: dnl 自定義輸出的檢查信息
17:AC_MSG_CHECKING([for architecture type])
18:
19:dnl 輸出檢查結果
20: AC_MSG_RESULT([ok])
21:
22: dnl 傳遞給連接器的'-l'和'-L'選項.
23: LIBS=" -L."
24:
25:dnl Checks for libraries,Might abort.
26:AC_CHECK_LIB(m,sin,[LIBS="$LIBS -lm"],exit 1)
27: AC_CHECK_LIB(pthread,pthread_create,[LIBS="$LIBS -pthread"],exit 1)
28: dnl AC_CHECK_LIB(socket, connect)
29:
30:dnl Checks for header files.(檢查頭文件是否存在)
31:AC_CHECK_HEADERS(sys/socket.h)
32:
33:dnl Checks for typedefs, structures, and compiler characteristics.
34:
35:dnl Checks for library functions.
36:
37:AC_OUTPUT(Makefile)
這個configure.in文件中使用了幾個常用的宏,還有一些如AC_ARG_ENABLE宏、AC_MSG_ERROR
宏、AC_MSG_RESULT宏、AM_PATH_GTK宏等有用的宏,你可以在附帶的附件文件裏查找,以便使用。
這裏我要重點提一下第26到28行,對於程序中需要用到一些特定的庫,需要在編譯時進行指定,
否則會出現連結錯誤。例如第26行就對數學庫進行了檢查,如果沒有,將退出Makefile文件的生
成,因爲找不到數學庫生成也編譯不過去;同理第27行對線程庫進行了檢查,第28行對socket庫
進行了檢查(被註釋掉的原因是引用的例子中沒用到此庫)。這裏面用到了AC_CHECK_LIB宏。具體
用法如下:
AC_CHECK_LIB(庫名稱,需要庫中的函數,[如果找到,[如果沒找到]])
在這個宏中的[庫名稱]實在編譯時 -l 選項後面的名稱,如數學庫 -lm 就用 m 就行了。在例子
中,如果找到了庫,就在編譯選項中添加 -lm 選項。
還是如上一篇中所述的那樣,執行aclocal和autoconf命令,然後再創建並編輯Makefile.am
文件內容如下:
AUTOMAKE_OPTIONS=foreign
bin_PROGRAMS=hello
hello_SOURCES=hello.c pubfun.c pubfun.h
Makefile.am文件和上一篇中相比就增加了源代碼的數量,其他都沒有改變。
然後再執行automake --add-missing即可。信息如下:
[root@leaf hello-1.0]# ./configure
checking for a BSD-compatible install... /usr/bin/install -c
checking whether build environment is sane... yes
checking for gawk... gawk
checking whether make sets $(MAKE)... yes
checking for gcc... gcc
checking for C compiler default output... a.out
checking whether the C compiler works... yes
checking whether we are cross compiling... no
checking for suffix of executables...
checking for suffix of object files... o
checking whether we are using the GNU C compiler... yes
checking whether gcc accepts -g... yes
checking for gcc option to accept ANSI C... none needed
checking for style of include used by make... GNU
checking dependency style of gcc... gcc3
checking for sin in -lm... yes
checking for pthread_create in -lpthread... yes
configure: creating ./config.status
config.status: creating Makefile
config.status: executing depfiles commands
[root@leaf hello-1.0]# make
source='hello.c' object='hello.o' libtool=no /
depfile='.deps/hello.Po' tmpdepfile='.deps/hello.TPo' /
depmode=gcc3 /bin/sh ./depcomp /
gcc -DPACKAGE_NAME="" -DPACKAGE_TARNAME="" -DPACKAGE_VERSION="" -DPACKAGE_STRING=""
-DPACKAGE_BUGREPORT="" -DPACKAGE="hello" -DVERSION="1.0" -I.
-I. -I. -O2 -c `test -f 'hello.c' || echo './'`hello.c
source='pubfun.c' object='pubfun.o' libtool=no /
depfile='.deps/pubfun.Po' tmpdepfile='.deps/pubfun.TPo' /
depmode=gcc3 /bin/sh ./depcomp /
gcc -DPACKAGE_NAME="" -DPACKAGE_TARNAME="" -DPACKAGE_VERSION="" -DPACKAGE_STRING=""
-DPACKAGE_BUGREPORT="" -DPACKAGE="hello" -DVERSION="1.0" -I.
-I. -I. -O2 -c `test -f 'pubfun.c' || echo './'`pubfun.c
gcc -O2 -o hello hello.o pubfun.o -L. -lm -pthread
備註:
1.以上內容均在RedHat Linux 9.0環境下測試通過。
2.詳細的選項可參考由王立翻譯的文檔http://www.cngnu.org/technology/1657/297.html。
3.其它國內外網站資料
4.RedHat 9.0下帶的程序文件及版本
autoconf-2.57-3.noarch.rpm
automake-1.6.3-5.noarch.rpm
gcc-3.2.2-5.i386.rpm