close

輕輕鬆鬆產生 Makefile

許明彥

February 11, 1999

Abstract:

在 Unix 上寫程式的人大概都碰過 Makefile,尤其是用 C 來開發程式的人。用 make 來開發和編譯程式的確很方便,可是要寫出一個 Makefile 就不簡單了。偏偏介紹 Makefile 的文件不多,GNU Make 那份印出來要幾百頁的文件,光看完 Overview 就快陣亡了,難怪許多人聞 Unix 色變。本文將介紹如何利用 GNU Autoconf 及 Automake 這兩套軟體來協助我們『自動』產生 Makefile 檔,並且讓開發出來的軟體可以像 Apache, MySQL 和常見的 GNU 軟體一樣,只要會 ``./configure'', ``make'', ``make install'' 就可以把程式安裝到系統中。如果您有心開發 Open Source 的軟體,或只是想在 Unix 系統下寫寫程式。希望這份介紹文件能幫助您輕鬆地進入 Unix Programming 的殿堂。

 

 

 


 

1. 簡介

Makefile 基本上就是『目標』(target), 『關連』(dependencies) 和『動作』三者所組成的一連串規則。而 make 就會根據 Makefile 的規則來決定如何編譯 (compile) 和連結 (link) 程式。實際上,make 可做的不只是編譯和連結程式,例如 FreeBSD 的 port collection 中,Makefile 還可以做到自動下載原始程式套件,解壓縮 (extract) ,修補 (patch),設定,然後編譯,安裝至系統中。

Makefile 基本構造雖然簡單,但是妥善運用這些規則就也可以變出許多不同的花招。卻也因此,許多剛開始學習寫 Makefile 時會感到沒有規範可循,每個人寫出來的 Makefile 長得都不太一樣,不知道從何下手,而且常常會受限於自己的開發環境,只要環境變數不同或路徑改一下,可能 Makefile 就得跟著修改。雖然有 GNU Makefile Conventions (GNU Makefile 慣例) 訂出一些使用 GNU 程式設計時撰寫 Makefile 的一些標準和規範,但是內容很長又很複雜, 並且經常做些調整,為了減輕程式設計師維護 Makefile 的負擔,因此有了 Automake。

程式設計師只需寫一些預先定義好的巨集 (macro),交給 Automake 處理後會產生一個可供 Autoconf 使用的 Makefile.in 檔。再配合利用 Autoconf 產生的自動設定檔 configure 即可產生一份符合 GNU Makefile 慣例的 Makeifle 了。

 

2. 上路之前

在開始試著用 Automake 之前,請先確認你的系統已經安裝以下的軟體:

  1. GNU Automake
  2. GNU Autoconf
  3. GNU m4
  4. perl
  5. GNU Libtool (如果你需要產生 shared library)

 

我會建議你最好也使用 GNU C/C++ 編譯器 、GNU Make 以及其它 GNU 的工具程式來做為開發的環境,這些工具都是屬於 Open Source Software 不僅免費而且功能強大。如果你是使用 Red Hat Linux 可以找到所有上述軟體的 rpm 檔,FreeBSD 也有現成的 package 可以直接安裝,或著你也可以自行下載這些軟體的原始檔回來 DIY。以下的範例是在 Red Hat Linux 5.2 + CLE2 的環境下所完成的。

 

3. 一個簡單的例子

Automake 所產生的 Makefile 除了可以做到程式的編譯和連結,也已經把如何產生程式文件 (如 manual page, info 檔及 dvi 檔) 的動作,還有把原始程式包裝起來以供散佈的動作都考慮進去了,所以原始程式所存放的目錄架構最好符合 GNU 的標準慣例,接下來我拿 hello.c 來做為例子。

在工作目錄下建立一個新的子目錄 ``devel'',再在 devel 下建立一個 ``hello'' 的子目錄,這個目錄將作為我們存放 hello 這個程式及其相關檔案的地方:

% mkdir devel

% cd devel

% mkdir hello

% cd hello

 

用編輯器寫個 hello.c 檔,

#include <stdio.h>

int main(int argc, char** argv)

{

printf(``Hello, GNU!n'');

return 0;

}

 

接下來就要用 Autoconf 及 Automake 來幫我們產生 Makefile 檔了,

 

1.
用 autoscan 產生一個 configure.in 的雛型,執行 autoscan 後會產生一個configure.scan 的檔案,我們可以用它做為 configure.in 檔的藍本。
% autoscan

% ls

configure.scan hello.c

2.
編輯 configure.scan 檔,如下所示,並且把它的檔名改成 configure.in
dnl Process this file with autoconf to produce a configure script.

AC_INIT(hello.c)

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)

3.
執行 aclocal 和 autoconf ,分別會產生 aclocal.m4 及 configure 兩個檔案
% aclocal

% autoconf

% ls

aclocal.m4 configure configure.in hello.c

4.
編輯 Makefile.am 檔,內容如下 
AUTOMAKE_OPTIONS= foreign

bin_PROGRAMS= hello

hello_SOURCES= hello.c

5.
執行 automake --add-missing ,Automake 會根據 Makefile.am 檔產生一些檔案,包含最重要的 Makefile.in 
% automake --add-missing

automake: configure.in: installing `./install-sh'

automake: configure.in: installing `./mkinstalldirs'

automake: configure.in: installing `./missing'

6.
最後執行 ./configure ,
% ./configure

creating cache ./config.cache

checking for a BSD compatible install... /usr/bin/install -c

checking whether build environment is sane... yes

checking whether make sets ${MAKE}... yes

checking for working aclocal... found

checking for working autoconf... found

checking for working automake... found

checking for working autoheader... found

checking for working makeinfo... found

checking for gcc... gcc

checking whether the C compiler (gcc ) works... yes

checking whether the C compiler (gcc ) is a cross-compiler... no

checking whether we are using GNU C... yes

checking whether gcc accepts -g... yes

updating cache ./config.cache

creating ./config.status

creating Makefile

 

現在你的目錄下已經產生了一個 Makefile 檔,下個 ``make'' 指令就可以開始編譯 hello.c 成執行檔,執行 ./hello 和 GNU 打聲招呼吧!

% make

gcc -DPACKAGE="hello" -DVERSION="1.0" -I. -I. -g -O2 -c hello.c

gcc -g -O2 -o hello hello.o

% ./hello

Hello! GNU!

 

你還可以試試 ``make clean'',''make install'',''make dist'' 看看會有什麼結果。你也可以把產生出來的 Makefile 秀給你的老闆,讓他從此對你刮目相看 :-)

Subsections


4. 一探究竟

上述產生 Makefile 的過程和以往自行編寫的方式非常不一樣,捨棄傳統自行定義 make 的規則,使用 Automake 只需用到一些已經定義好的巨集即可。我們把巨集及目標 (target) 寫在 Makefile.am 檔內,Automake 讀入 Makefile.am 檔後會把這一串已經定義好的巨集展開並且產生對應的 Makefile.in 檔, 然後再由 configure 這個 shell script 根據 Makefile.in 產生適合的 Makefile。

 

[Figure 1:利用 autoconf 及 automake 產生 Makefile 的流程]

上圖中表示在上一節範例中所要用的檔案以及產生出來的檔案,有星號 (*) 者代表可執行檔。在此範例中可藉由 Autoconf 及 Automake 工具所產生的檔案有 configure.scan、aclocal.m4、configure、Makefile.in,需要我們加入設定者為 configure.in 及 Makefile.am。

 

4.1 編輯 configure.in 檔

Autoconf 是用來產生 'configure' 檔的工具。'configure' 是一個 shell script,它可以自動設定原始程式以符合各種不同平台上 Unix 系統的特性,並且根據系統參數及環境產生合適的 Makefile 檔或是C 的標頭檔 (header file),讓原始程式可以很方便地在這些不同的平台上被編譯出來。Autoconf 會讀取 configure.in 檔然後產生 'configure' 這個 shell script。

configure.in 檔的內容是一連串 GNU m4 的巨集,這些巨集經過 autoconf 處理後會變成檢查系統特徵的 shell script。configure.in 內巨集的順序並沒有特別的規定,但是每一個 configure.in 檔必須在所有巨集前加入 AC_INIT 巨集,然後在所有巨集的最後面加上 AC_OUTPUT 巨集。我們可先用 autoscan 掃瞄原始檔以產生一個 configure.scan 檔,再對 configure.scan 做些修改成 configure.in 檔。在範例中所用到的巨集如下:

 

dnl
這個巨集後面的字不會被處理,可視為註解。
AC_INIT(FILE)
這個巨集用來檢查原始碼所在的路徑,autoscan 會自動產生,我們不必修改它。
AM_INIT_AUTOMAKE(PACKAGE,VERSION)
這是使用 Automake 所必備的巨集,PACKAGE 是我們所要產生軟體套件的名稱,VERSION 是版本編號。
AC_PROG_CC
檢查系統可用的 C 編譯器,如果原始程式是用 C 寫的就需要這個巨集。
AC_OUTPUT(FILE)
設定 configure 所要產生的檔案,如果是 Makefile 的話,configure 便會把它檢查出來的結果帶入 Makefile.in 檔然後產生合適的 Makefile。

 

實際上,我們使用 Automake 時,還須要一些其它的巨集,這些額外的巨集我們用 aclocal 來幫我們產生。執行 aclocal 會產生 aclocal.m4 檔,如果沒有特別的用途,我們可以不必修改它,用 aclocal 所產生的巨集會告訴 Automake 怎麼做。

有了 configure.in 及 aclocal.m4 兩個檔案後,便可以執行 autoconf 來產生 configure 檔了。

 

4.2 編輯 Makefile.am 檔

接下來我們要編輯 Makefile.am 檔,Automake 會根據 configure.in 中的巨集把Makefile.am 轉成 Makefile.in 檔。Makefile.am 檔定義我們所要產的目標:

AUTOMAKE_OPTIONS
設定 automake 的選項。Automake 主要是幫助開發 GNU 軟體的人員維護軟體套件,所以在執行 automake 時,會檢查目錄下是否存在標準 GNU 軟體套件中應具備的文件檔案,例如 'NEWS'、'AUTHOR'、'ChangeLog' 等文件檔。設成 foreign 時,automake 會改用一般軟體套件的標準來檢查。
bin_PROGRAMS
定義我們所要產生的執行檔檔名。如果要產生多個執行檔,每個檔名用空白字元隔開。
hello_SOURCES
定義 'hello' 這個執行檔所需要的原始檔。如果 'hello' 這個程式是由多個原始檔所產生,必須把它所用到的原始檔都列出來,以空白字元隔開。假設 'hello' 這個程式需要 'hello.c'、'main.c'、'hello.h' 三個檔案的話,則定義
hello_SOURCES= hello.c main.c hello.h
如果我們定義多個執行檔,則對每個執行檔都要定義相對的 filename_SOURCES。

 

編輯好 Makefile.am 檔,就可以用 automake -add-missing 來產生 Makefile.in。加上 -add-missing 選項是告訴 automake 順便幫我們加入包裝一個軟體套件所必備的檔案。Automake 產生出來的 Makefile.in 檔是完全符合 GNU Makefile 的慣例,我們只要執行 configure 這個 shell script 便可以產生合適的 Makefile 檔了。

 

4.3 使用 Makefile

利用 configure 所產生的 Makefile 檔有幾個預設的目標可供使用,我們只拿其中幾個簡述如下:

make all
產生我們設定的目標,即此範例中的執行檔。只打 make 也可以,此時會開始編譯原始碼,然後連結,並且產生執行檔。
make clean
清除之前所編譯的執行檔及目的檔 (object file, *.o)。
make distclean
除了清除執行檔和目的檔外,也把 configure 所產生的 Makefile 也清除掉。
make install
將程式安裝至系統中。如果原始碼編譯無誤,且執行結果正確,便可以把程式安裝至系統預設的執行檔存放路徑。如果我們用 bin_PROGRAMS 巨集的話,程式會被安裝至 /usr/local/bin 這個目錄。
make dist
將程式和相關的檔案包裝成一個壓縮檔以供散播 (distribution) 。執行完在目錄下會產生一個以 PACKAGE-VERSION.tar.gz 為名稱的檔案。PACKAGE 和 VERSION 這兩個變數是根據 configure.in 檔中 AM_INIT_AUTOMAKE(PACKAGE, VERSION) 的定義。在此範例中會產生 'hello-1.0.tar.gz' 的檔案。
make distcheck
和 make dist 類似,但是加入檢查包裝後的壓縮檔是否正常。這個目標除了把程式和相關檔案包裝成 tar.gz 檔外,還會自動把這個壓縮檔解開,執行 configure,並且進行 make all 的動作,確認編譯無誤後,會顯示這個 tar.gz 檔已經準備好可供散播了。這個檢查非常有用,檢查過關的套件,基本上可以給任何一個具備 GNU 發展環境的人去重新編譯。就 hello-1.tar.gz 這個範例而言,除了在 Red Hat Linux 上,在 FreeBSD 2.2.x 版也可以正確地重新編譯。

 

要注意的是,利用 Autoconf 及 Automake 所產生出來的軟體套件是可以在沒有安裝 Autoconf 及 Automake 的環境上使用的,因為 configure 是一個 shell script,它己被設計可以在一般 Unix 的 sh 這個 shell 下執行。但是如果要修改 configure.in 及 Makefile.am 檔再產生新的 configure 及 Makefile.in 檔時就一定要有 Autoconf 及 Automake 了。

5. 相關訊息

Autoconf 和 Automake 功能十分強大,你可以從它們所附的 info 檔找到詳細的用法。你也可以從許多現存的 GNU 軟體或 Open Source 軟體中找到相關的 configure.in 或 Makefile.am 檔,它們是學習 Autoconf 及 Automake 更多技巧的最佳範例。

這篇簡介只用到了 Autoconf 及 Automake 的皮毛罷了,如果你有心加入 Open Source 軟體開發的行列,希望這篇文件能幫助你對產生 Makefile 有個簡單的依據。其它有關開發 GNU 程式或 C 程式設計及 Makefile 的詳細運用及技巧,我建議你從 GNU Coding Standards3 (GNU 編碼標準規定) 讀起,裡面包含了 GNU Makefile 慣例,還有發展 GNU 軟體套件的標準程序和慣例。這些 GNU 軟體的線上說明文件可以在 http://www.gnu.org/ 這個網站上找到。

6. 結語

經由 Autoconf 及 Automake 的輔助,產生一個 Makefile 似乎不再像以前那麼困難了,而使用 Autoconf 也使得我們在不同平台上或各家 Unix 之間散播及編譯程式變得簡單,這對於在 Unix 系統上開發程式的人員來說減輕了許多負擔。妥善運用這些 GNU 的工具軟體,可以幫助我們更容易去發展程式,而且更容易維護原始程式碼。

一九九八年是 Open Source 運動風起雲湧的一年,許多 Open Source 的軟體普遍受到網路上大眾的歡迎和使用。感謝所有為 Open Source 奉獻的人們,也希望藉由本文能吸引更多的人加入『自由』、『開放』的 Open Source 行列。


[Note]
What is the "make" utility in UNIX? Explain its purposes and show how it is used. Give at least two examples.

  make 是 UNIX 中的一個方便程式編譯的工具程式。
  當我們有多個 C 或 FORTRAN subroutines 分散在不同的檔案時,如果我們寫一個叫做 Makefile 的檔案,再使用 make這個指令,則 UNIX 系統會根據 Makefile 內容將所有檔案個別 compile,並且在最後連結成可執行檔。
  make與其他解釋語言不同,不是直接告訴make需要執行的命令,而是給定一些依賴規則,即在什麼條件下應該執行什麼處理,那麼make就自動分析 檔案的更新時間,完成剩下的工作。規定make規則的檔案一般命名為Makefile,這是一個make指令的集合,這個檔案中包括目標定義、執行命令、 宏定義和make 偽指令。下面為一個簡單的Makefile:
  使用 make 跟 Makefile 的另一個優點,是如果我們在稍後修改了其中一個檔案,再重新 make,則系統會「只」compile那個修改過的檔案,重新連結成新的可執行檔。
  在shell的提示符號下,若鍵入 "make",則它會到目前的目錄下找尋 Makefile這個檔案。然後依照Makefile中所記錄的步驟一步一步的來執行。在我們寫程式的時候,如果事先就把compiler程式所需要的步 驟先寫在Makefile中的話,想要compiler程式的時候就只要打入make的指令。只要程式無誤的話,就可以獲得所需要的結果了!

 

example1

一個比較簡單的例子,這個例子中首先定義了一個宏CC,然後定義一個執行目標hello,這個目標依賴於hello.c檔案,一旦hello.c更新,就需要執行下面的編譯指令。注意,位於定義目標之後的執行命令應該使用一個 “Tab” 製表符引導,而不是其他空白字符。執行命令中首先將宏替換為它的值,再執行egcc命令編譯程序。
  一個Makefile檔案中可以定義多個目標,如下面例子中的hello和clean,如果不使用任何命令行參數來啟動make,那麼預設使用第一個目標。為了應用其他的make目標,則必須使用make的命令行參數。

CC = /usr/local/bin/egcc
hello: hello.c
$(CC) -o hello hello.c
clean:
echo delete files!
rm hello

 

example2

 make使用的預設檔案名為目前目錄下的makefile或Makefile,如果使用其他檔案,必須使用命令行參數-f指定檔案名。

$ make clean
delete files!

 

exmple3

  GNU的make命令首先查看的檔案名為GNUmakefile。

$ make -f newmakefile

 

example4

假設我們有一個程式,共分為下面的部份:
menu.c 主要的程式碼部份
menu.h menu.c的include file
utils.c 提供menu.c呼叫的一些function calls
utils.h utils.c的include file
同時本程式亦叫用了ncurses的function calls。 而menu.c和utils.c皆放在/usr/src/menu下。 但menu.h和utils.h卻放在/usr/src/menu/include下。 而程式做完之後,執行檔名為menu且要放在/usr/bin下面。
------------------------------------ (Makefile begin:不含此行)
# This is the Makefile of menu
CC = gcc
CFLAGS = -DDEBUG -c
LIBS = -lncurses
INCLUDE = -I/usr/src/menu/include

#以上 CC, CFLAGS,LIBS,跟 INCLUDE 是四個我們自己定義的 macro

all: clean install
install: menu
chmod 750 menu
cp menu /usr/bin
menu: menu.o
$(CC) -o $@ $? $(LIBS)
menu.o:
$(CC) $(CFLAGS) -o $@ menu.c $(INCLUDE)
utils.o: $(CC) $(CFLAGS) -o $@ utils.c $(INCLUDE)
clean: -rm *.o -rm *~
------------------------------------ (Makefile end:不含此行)

注意:上例 Makefile 中,某些行前面有一大段空格, 那是使用 vi 編輯器「按一次」 「鍵」空出來的, 如果你用 ve ,因為 ve 將 當空格處理, make 時就會發生錯誤。

 

example5

以下是另一個 FORTRAN 程式使用 Makefile 的例子:
------------------------------ (Makefile begin:不含此行)
定義 SUBS 這個 macro (名字可以任取,習慣上用大寫),用來包含所有個別檔案的 object file,
例如 add.o 是 add.f 被編譯後的 object file name

SUBS = add.o adjusth.o augment.o bfgs.o compar.o con_eval.o conmin.o eq.o eval.o form.o cost.o geteps.o grad.o inith.o inverth.o invrt2.o

  你可以用 "make all", "make conmin", 或 "make" 指令來得到最後的可執行檔,以下這一行是說,"make all" 或 "make", make 程式就會先把以下 "conmin: $(SUBS)" 部分先完成

all: conmin

以下這個部分是說,要將可執行檔(-o conmin) conmin 編譯出來前, make 程式得先將上面定義的 "SUBS = ..." 的所有 *.o 檔先編譯出來然後才執行 f77 -o conmin $(SUBS) -llapack -lblas -lm # # "-llapack -lblas -lm" 表示在最後編譯階段連結 lapack, blas, 及 math library

conmin: $(SUBS) f77 -o conmin $(SUBS) -llapack -lblas -lm

you can type "make clean" to delete all *.o object files

clean: rm -f *.o

凡是碰到 .f 檔,就叫 make 用 f77 -c file_name.f,編譯產生 file_name.o Object file

.f.o: f77 -c $<
------------------------------------ (Makefile end:不含此行)
注意:上例 Makefile 中,某些行前面有一大段空格, 那是使用 vi 編輯器「按一次」 「鍵」空出來的, 如果你用 ve ,因為 ve 將 當空格處理, make 時就會發生錯誤。

 

example6

  Automake 所產生的 Makefile 除了可以做到程式的編譯和連結,也已經把如何產生程式文件 (如 manual page, info 檔及 dvi 檔) 的動作,還有把原始程式包裝起來以供散佈的動作都考慮進去了,所以原始程式所存放的目錄架構最好符合 GNU 的標準慣例,接下來我拿 hello.c 來做為例子。
在工作目錄下建立一個新的子目錄 ``devel'',再在 devel 下建立一個 ``hello'' 的子目錄,這個目錄將作為我們存放 hello 這個程式及其相關檔案的地方: 

% mkdir devel
% cd devel
% mkdir hello
% cd hello

用編輯器寫個 hello.c 檔,

#include <stdio.h>
int main(int argc, char** argv)
{
printf(``Hello, GNU!n'');
return 0;
}

接下來就要用 Autoconf 及 Automake 來幫我們產生 Makefile 檔了,

1.用 autoscan 產生一個 configure.in 的雛型,執行 autoscan 後會產生一個configure.scan 的檔案,我們可以用它做為 configure.in 檔的藍本。 

% autoscan
% ls
configure.scan hello.c
2.
編輯 configure.scan 檔,如下所示,並且把它的檔名改成 configure.in 
dnl Process this file with autoconf to produce a configure script.
AC_INIT(hello.c)
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)
3.
執行 aclocal 和 autoconf ,分別會產生 aclocal.m4 及 configure 兩個檔案 
% aclocal
% autoconf
% ls
aclocal.m4 configure configure.in hello.c
4.
編輯 Makefile.am 檔,內容如下
AUTOMAKE_OPTIONS= foreign
bin_PROGRAMS= hello
hello_SOURCES= hello.c
5.
執行 automake --add-missing ,Automake 會根據 Makefile.am 檔產生一些檔案,包含最重要的 Makefile.in
% automake --add-missing
automake: configure.in: installing `./install-sh'
automake: configure.in: installing `./mkinstalldirs'
automake: configure.in: installing `./missing'
6.
最後執行 ./configure , 
% ./configure
creating cache ./config.cache
checking for a BSD compatible install... /usr/bin/install -c
checking whether build environment is sane... yes
checking whether make sets ${MAKE}... yes
checking for working aclocal... found
checking for working autoconf... found
checking for working automake... found
checking for working autoheader... found
checking for working makeinfo... found
checking for gcc... gcc
checking whether the C compiler (gcc ) works... yes
checking whether the C compiler (gcc ) is a cross-compiler... no
checking whether we are using GNU C... yes
checking whether gcc accepts -g... yes
updating cache ./config.cache
creating ./config.status
creating Makefile

現在你的目錄下已經產生了一個 Makefile 檔,下個 ``make'' 指令就可以開始編譯 hello.c 成執行檔,執行 ./hello 和 GNU 打聲招呼吧!

% make
gcc -DPACKAGE="hello" -DVERSION="1.0" -I. -I. -g -O2 -c hello.c
gcc -g -O2 -o hello hello.o
% ./hello
Hello! GNU!

 

About this document ...

輕輕鬆鬆產生 Makefile1

This document was generated using the LaTeX2HTML translator Version 98.1p1 release (March 2nd, 1998)

Copyright © 1993, 1994, 1995, 1996, 1997, Nikos Drakos, Computer Based Learning Unit, University of Leeds.

The command line arguments were:
latex2html -split 4 -show_section_numbers automake-html.

The translation was initiated by Ming-Yen Hsu on 2000-08-15


arrow
arrow
    文章標籤
    Makefile autotools
    全站熱搜
    創作者介紹
    創作者 Bluelove1968 的頭像
    Bluelove1968

    藍色情懷

    Bluelove1968 發表在 痞客邦 留言(0) 人氣()