實戰Makefile.am
Makefile.am是一種比Makefile更高層次的規則。只需指定要產生什麼目標,它由什麼源檔案產生,要安裝到什麼目錄等構成。
表一列出了可執行檔、靜態庫、標頭檔和資料檔案,四種書寫Makefile.am檔案個一般格式。
表 1Makefile.am一般格式
對於可執行檔和靜態庫類型,如果只想編譯,不想安裝到系統中,可以用noinst_PROGRAMS代替bin_PROGRAMS,noinst_LIBRARIES代替lib_LIBRARIES。
Makefile.am還提供了一些全域變數供所有的目標體使用:
表 2 Makefile.am中可用的全域變數
在Makefile.am中盡量使用相對路徑,系統預定義了兩個基本路徑:
表 3Makefile.am中可用的路徑變數
在上文中我們提到過安裝路徑,automake設定了預設的安裝路徑:
1)標準安裝路徑
預設安裝路徑為:$(prefix) = /usr/local,可以通過./configure --prefix=<new_path>的方法來覆蓋。
其它的預定義目錄還包括:bindir = $(prefix)/bin, libdir = $(prefix)/lib, datadir = $(prefix)/share, sysconfdir = $(prefix)/etc等等。
2) 定義一個新的安裝路徑
比如test, 可定義testdir = $(prefix)/test, 然後test_DATA =test1 test2,則test1,test2會作為資料檔案安裝到$(prefix)/ /test目錄下。
我們首先需要在工程頂層目錄下(即project/)建立一個Makefile.am來指明包含的子目錄:
SUBDIRS=src/lib src/ModuleA/apple/shell src/ModuleA/apple/core CURRENTPATH=$(shell /bin/pwd) INCLUDES=-I$(CURRENTPATH)/src/include -I$(CURRENTPATH)/src/ModuleA/apple/include export INCLUDES |
由於每個源檔案都會用到相同的標頭檔,所以我們在最頂層的Makefile.am中包含了編譯源檔案時所用到的標頭檔,並匯出,見藍色部分代碼。
我們將lib目錄下的swap.c檔案編譯成libswap.a檔案,被apple/shell/apple.c檔案調用,那麼lib目錄下的Makefile.am如下所示:
noinst_LIBRARIES=libswap.a libswap_a_SOURCES=swap.c INCLUDES=-I$(top_srcdir)/src/includ |
細心的讀者可能就會問:怎麼表1中給出的是bin_LIBRARIES,而這裡是noinst_LIBRARIES?這
是因為如果只想編譯,而不想安裝到系統中,就用noinst_LIBRARIES代替bin_LIBRARIES,對於可執行檔就用
noinst_PROGRAMS代替bin_PROGRAMS。對於安裝的情況,庫將會安裝到$(prefix)/lib目錄下,可執行檔將會安裝
到${prefix}/bin。如果想安裝該庫,則Makefile.am樣本如下:
bin_LIBRARIES=libswap.a libswap_a_SOURCES=swap.c INCLUDES=-I$(top_srcdir)/src/include swapincludedir=$(includedir)/swap swapinclude_HEADERS=$(top_srcdir)/src/include/swap.h |
最後兩行的意思是將swap.h安裝到${prefix}/include /swap目錄下。
接下來,對於可執行檔類型的情況,我們將討論如何寫Makefile.am?對於編譯apple/core目錄下的檔案,我們寫成的Makefile.am如下所示:
noinst_PROGRAMS=test test_SOURCES=test.c test_LDADD=$(top_srcdir)/src/ModuleA/apple/shell/apple.o $(top_srcdir)/src/lib/libswap.a test_LDFLAGS=-D_GNU_SOURCE DEFS+=-D_GNU_SOURCE #LIBS=-lpthread |
由於我們的test.c檔案在連結時,需要apple.o和libswap.a檔案,所以我們需要在
test_LDADD中包含這兩個檔案。對於Linux下的訊號量/讀寫鎖檔案進行編譯,需要在編譯選項中指明-D_GNU_SOURCE。所以在
test_LDFLAGS中指明。而test_LDFLAGS只是連結時的選項,編譯時間同樣需要指明該選項,所以需要DEFS來指明編譯選項,由於
DEFS已經有初始值,所以這裡用+=的形式指明。從這裡可以看出,Makefile.am中的文法與Makefile的文法一致,也可以採用條件表達
式。如果你的程式還包含其他的庫,除了用AC_CHECK_LIB宏來指明外,還可以用LIBS來指明。
如果你只想編譯某一個檔案,那麼Makefile.am如何寫呢?這個檔案也很簡單,寫法跟可執行檔的差不多,如下例所示:
noinst_PROGRAMS=apple apple_SOURCES=apple.c DEFS+=-D_GNU_SOURCE |
我們這裡只是欺騙automake,假裝要產生apple檔案,讓它為我們產生依賴關係和執行命令。所以當你運行完automake命令後,然後修改apple/shell/下的Makefile.in檔案,直接將LINK語句刪除,即:
…….
clean-noinstPROGRAMS:
-test -z "$(noinst_PROGRAMS)" || rm -f $(noinst_PROGRAMS)
apple$(EXEEXT): $(apple_OBJECTS) $(apple_DEPENDENCIES)
@rm -f apple$(EXEEXT)
#$(LINK) $(apple_LDFLAGS) $(apple_OBJECTS) $(apple_LDADD) $(LIBS)