GNU make中文手冊-第七章:Makefile的條件執(zhí)行
作者: hew 發(fā)布日期:
2006-3-21
查看數(shù):
124
出自: http://www.
第七章:Makefile的條件執(zhí)行
--------------------------------------------------------------------------------
條件語句可以根據(jù)一個變量的值來控制對Makefile的執(zhí)行,執(zhí)行或者忽略Makefile的特定部分。條件語句可以是兩個不同變量、或者變量和常量值
得比較。需要注意的是:條件語句只能用于控制make實際執(zhí)行的makefile文件部分,它不能控制規(guī)則的shell命令執(zhí)行過程。Makefile中
使用條件控制可以做到處理的靈活性和高效性。
7.1 一個例子
首先我們先看一個使用條件語句的Makefile例子;對變量“CC”進行判斷,值如果是“gcc”那么在進行程序連接時使用庫“libgnu.so”或者“libgnu.a”,否則不鏈接任何庫。我們Makefile中的條件判斷部分如下:
……
libs_for_gcc = -lgnu
normal_libs =
……
foo: $(objects)
ifeq ($(CC),gcc)
$(CC) -o foo $(objects) $(libs_for_gcc)
else
$(CC) -o foo $(objects) $(normal_libs)
endif
……
在上例中,條件語句中使用到了三個關(guān)鍵字:“ifeq”、“else”和“endif”。其中:
1.
“ifeq”表示條件語句的開始,并指定了一個比較條件(相等)。之后是用圓括號括包圍的、使用逗號“,”分割的兩個參數(shù),和關(guān)鍵字“ifeq”用空格分
開。參數(shù)中的變量引用在進行變量值比較時被展開。“ifeq”之后就是當(dāng)條件滿足make需要執(zhí)行的,條件不滿足時忽略。
2. “else”之后就是當(dāng)條件不滿足時的執(zhí)行部分。不是所有的條件語句都需要此部分。
3. “endif”表示一個條件語句的結(jié)束,任何一個條件表達式都必須以“endif”結(jié)束。
通過上邊的例子我們可以了解到。Makefile中,條件表達式是工作文本級別的,條件的解析是由make程序來完成的。make是在解析
Makefile時根據(jù)條件表達式忽略條件表達式中的某一個文本行,解析完成后保留的只有表達式滿足條件所需要執(zhí)行的文本行。對于上例,make處理這個
條件時:
當(dāng)變量“CC”的值為“gcc”時,整個條件表達式等效于:
foo: $(objects)
$(CC) -o foo $(objects) $(libs_for_gcc)
當(dāng)變量“CC”值不等于“gcc”時等效于:
foo: $(objects)
$(CC) -o foo $(objects) $(normal_libs)
上面的例子,一種更簡潔實現(xiàn)方式:
libs_for_gcc = -lgnu
normal_libs =
ifeq ($(CC),gcc)
libs=$(libs_for_gcc)
else
libs=$(normal_libs)
endif
foo: $(objects)
$(CC) -o foo $(objects) $(libs)
7.2 條件判斷的基本語法
一個簡單的不包含“else”分支的條件判斷語句的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
endif
表達式中“TEXT-IF-TRUE”可以是若干任何文本行,當(dāng)條件為真時它就將被make程序作為需要執(zhí)行的一部分。當(dāng)條件為假時,不作為需要執(zhí)行的一部分。
另外包含“else”的復(fù)雜一點的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
else
TEXT-IF-FALSE
endif
表示了如果條件為真,則將“TEXT-IF-TRUE”作為執(zhí)行Makefile的一部分,否則將“TEXT-IF-FALSE”作為執(zhí)行的Makefile的一部分。和“TEXT-IF-TRUE”一樣,“TEXT-IF-FALSE”可以是若干任何文本行。
條件判斷語句中“CONDITIONAL-DIRECTIVE”對于上邊的兩種格式都是同樣的??梢允且韵滤姆N用于測試不同條件的關(guān)鍵字。
7.2.1 關(guān)鍵字“ifeq”
此關(guān)鍵字用來判斷參數(shù)是否相等,格式如下:
`ifeq (ARG1, ARG2)‘
`ifeq ‘ARG1‘ ‘ARG2‘‘
`ifeq "ARG1" "ARG2"‘
`ifeq "ARG1" ‘ARG2‘‘
`ifeq ‘ARG1‘ "ARG2"‘
替換并展開“ARG1”和“ARG1”之后,對它們的值進行比較。如果相同則(條件為真)將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分,否則將“TEXT-IF-FALSE”作為make要執(zhí)行的一部分(上邊的第二種格式)。
通常我們會使用它來判斷一個變量的值是否為空(不是任何字符)。參數(shù)值可能是通過引用變量或者函數(shù)得到的,因而在展開過程中可能造成參數(shù)值中包含空字符
(空格等)。一般在處理這種情況時我們使用make的“strip”函數(shù)(參考文本處理函數(shù)一節(jié))來對它進行處理,去掉變量值中的空字符。格式為:
ifeq ($(strip $(foo)),)
TEXT-IF-EMPTY
endif
這樣,即就是在“$(foo)”中存在若干前導(dǎo)和結(jié)尾空格,make在解析Makefile時“TEXT-IF-EMPTY”也會被作為執(zhí)行的一部分。
7.2.2 關(guān)鍵字“ifneq”
次關(guān)鍵字是用來判斷參數(shù)是否不相等,格式為:
`ifneq (ARG1, ARG2)‘
`ifneq ‘ARG1‘ ‘ARG2‘‘
`ifneq "ARG1" "ARG2"‘
`ifneq "ARG1" ‘ARG2‘‘
`ifneq ‘ARG1‘ "ARG2"‘
關(guān)鍵字“ifneq”實現(xiàn)的條件判斷語句和“ifeq”相反。首先替換并展開“ARG1”和“ARG1”,對它們的值進行比較。如果不相同(條件為真)則
將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分,否則將“TEXT-IF-FALSE”作為make要執(zhí)行的一部分。
7.2.3 關(guān)鍵字“ifdef”
關(guān)鍵字是“ifdef”用來判斷一個變量是否定義。格式為:
`ifdef VARIABLE-NAME‘
如果變量“VAEIABLE_NAME”的值非空,那么表達式為真,將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分。否則,表達式為假,如果
存在“TEXT-IF-FALSE”,就將它作為make要執(zhí)行一部分。當(dāng)一個變量沒有被定義時,它的值為空。“VARIABLE-NAME”可以是變量
或者函數(shù)的引用。
對于“ifdef”需要說明的是:ifdef只是測試一個變量是否有值,其并不會對變量進行替換擴展來判斷變量的值是否為空。對于變量“VARIABLE
-NAME”,除了“VARIABLE-NAME=”這種情況以外,使用其它方式對它的定義都會使“ifdef”返回真。就是說,即使我們通過其它方式
(比如,定義它的值引用了其它的變量)給它富了一個空值,“ifdef”也會返回真。我們來看一個例子:
例1:
bar =
foo = $(bar)
ifdef foo
frobozz = yes
else
frobozz = no
endif
例2:
foo =
ifdef foo
frobozz = yes
else
frobozz = no
endif
例1中的結(jié)果是:“frobozz = yes”;而例2的結(jié)果是:“frobozz =
no”。其原因就是在例1中,變量“foo”的定義是“foo =
$(bar)”。雖然變量“bar”的值為空,但是“ifdef”判斷的結(jié)果是真。因此當(dāng)我們需要判斷一個變量的值是否為空的情況時,需要使用
“ifeq”(或者“ifneq”)而不是“ifdef”。可參考前兩個小節(jié)的內(nèi)容。
7.2.4 關(guān)鍵字“ifndef”
關(guān)鍵字“ifndef”實現(xiàn)的功能和“ifdef”相反。格式為:
`ifdef VARIABLE-NAME‘
這個就不詳細(xì)討論了,它的功能就是實現(xiàn)了和“ifdef”相反的條件判斷。
在“CONDITIONAL-DIRECTIVE”這一行上,可以以若干個空格開始,make處理時會被忽略這些空格。但不能以[Tab]字符做為開始
(不然就被認(rèn)為是命令)。條件判斷語句中,在除關(guān)鍵字(包括“endif”)之前、和條件表達式的參數(shù)中之外,其他任何地方都可以使用多個空格或
[Tab]字符,它不會影響條件判斷語句的功能。同樣行尾也可以使用注釋(“#”開始直到一行的結(jié)束)。
“else”和“endif”也是條件判斷語句的一部分。在書寫時它們都是沒有任何參數(shù)的,可以以多個空格開始(同樣不能以[Tab]字符開始)多個空格或[Tab]字符結(jié)束。行尾同樣可以有注釋內(nèi)容。
在make讀取makefile文件時計算表達式的值,并根據(jù)表達式的值決定判斷語句中那一部分被作為此Makefile所要執(zhí)行的內(nèi)容(選擇符合條件的
語句)。因此在條件表達式中不能使用自動化變量,自動化變量在規(guī)則命令執(zhí)行時才有效。更不能將一個完整的條件判斷語句分寫在兩個不同的makefile文
件中,其中在某一個makefile文件使用指示符“include”包含另外一個。
7.3 標(biāo)記測試的條件語句
我們可以使用條件判斷語句、并使用使用變量“MAKEFLAGS”和函數(shù)“findstring”,實現(xiàn)對make命令行選項的測試。看一個例子:
archive.a: ...
ifneq (,$(findstring t,$(MAKEFLAGS)))
+touch archive.a
+ranlib -t archive.a
else
ranlib archive.a
endif
這個條件語句判斷make的命令行參數(shù)是否包含“-t”(用來更新目標(biāo)文件的時間戳)。根據(jù)命令行參數(shù)情況完成對“archive.a”執(zhí)行不同的操作。命令行前的“+”的意思是告訴make,即使make使用了“-t”參數(shù),“+”之后的命令都需要被執(zhí)行。 |
|