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