Linux 環境下 Makefile 檔案製作淺談(二)(轉)

post0發表於2007-08-11
Linux 環境下 Makefile 檔案製作淺談(二)(轉)[@more@]

編寫:Leaf Zhou

EMAIL:leaf_zhou_8@hotmail.com

可自由複製但禁止刪改

2003-10-15

在介紹了簡單的如何產生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

void *printA(void *pdata);

-------------------------------------------------

/*filename:pubfun.c */

#include

#include

#include

void *printA(void *pdata)

{

printf("%f ",sin(2));

printf("Hello,World --&gt%d ",getpid());

}

-------------------------------------------------

/*filename:hello.c */

#include

#include

#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.詳細的選項可參考由王立翻譯的文件。

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

5.附件原始檔用如下命令開啟即可:

tar xvzf hello-1.0.tar.gz

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/8225414/viewspace-944615/,如需轉載,請註明出處,否則將追究法律責任。

相關文章