Makefile的條件執(zhí)行
1 前言
條件語句可以根據(jù)一個(gè)變量的值來控制make執(zhí)行或者忽略Makefile 的特定部分。條件語句可以是兩個(gè)不同變量叮趴、或者變量和常量值的比較
注意:條件語句只能用于控制make實(shí)際執(zhí)行的makefile文件部分歉备,它不能控制規(guī)則的shell命令執(zhí)行過程
2 基本語法
- 一個(gè)簡單的不包含“else”分支的條件判斷語句的語法格式為
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
endif
表達(dá)式中“TEXT-IF-TRUE”可以是若干任何文本行,當(dāng)條件為真時(shí)它就將被make作為需要執(zhí)行的一部分。當(dāng)條件為假時(shí)紫谷,不作為需要執(zhí)行的一部分
- 包含“else”的語法格式
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”對于上邊的兩種格式都是同樣的
2.1 例子
......
foo: $(objects)
ifeq ($(CC),gcc)
$(CC) -o foo $(objects) $(libs_for_gcc)
else
$(CC) -o foo $(objects) $(normal_libs)
endif
......
- “ifeq”表示條件語句的開始握恳,并指定了一個(gè)比較條件(相等)瞒窒。之后是用圓括號括包圍的、使用逗號“,”分割的兩個(gè)參數(shù)乡洼,和關(guān)鍵字“ifeq”用空格分開崇裁。參數(shù)中的變量引用在進(jìn)行變量值比較時(shí)被展開∈牵“ifeq”之后就是當(dāng)條件滿足make 需要執(zhí)行的拔稳,條件不滿足時(shí)忽略。
- “else”之后就是當(dāng)條件不滿足時(shí)的執(zhí)行部分锹雏。不是所有的條件語句都需要此部分巴比。
- “endif”表示一個(gè)條件語句的結(jié)束,任何一個(gè)條件表達(dá)式都必須以“endif”結(jié)束
注:條件的解析是由make來完成的礁遵。make是在讀取并解析Makefile時(shí)根據(jù)條件表達(dá)式忽略條件表達(dá)式中的某一個(gè)文本行匿辩,解析完成后保留的 只有表達(dá)式滿足條件所需要執(zhí)行的文本行
3 關(guān)鍵字
3.1 “ifeq”
用來判斷參數(shù)是否相等,格式如下
ifeq (ARG1, ARG2)
ifeq 'ARG1' 'ARG2'
ifeq "ARG1" "ARG2"
ifeq "ARG1" 'ARG2'
ifeq 'ARG1' "ARG2"
展開“ARG1”和“ARG1”后對它們的值進(jìn)行比較
- 如果相同則(條件為 真)將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分
- 否則將“TEXT-IF-FALSE”作為make要執(zhí)行的一部分(上邊的第二種格式)
通常我們會使用它來判斷一個(gè)變量的值是否為空
因?yàn)樗膮?shù)值可能是通過引用變量或者函數(shù)得到的榛丢,因而在展開過程中可能造成參數(shù)值中包含空字符铲球,這時(shí)用make的“strip”函數(shù)處理
ifeq ($(strip $(foo)),)
TEXT-IF-EMPTY
endif
這樣就可以去掉首位空格
3.2 “ifneq”
用來判斷參數(shù)是否不相等,格式如下
ifneq (ARG1, ARG2)
ifneq 'ARG1' 'ARG2'
ifneq "ARG1" "ARG2"
ifneq "ARG1" 'ARG2'
ifneq 'ARG1' "ARG2"
展開“ARG1”和“ARG1”對它們的值進(jìn)行比較
- 如果不相同(條件為真)則將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分
- 否則將“TEXT-IF-FALSE”作為make要執(zhí)行的一部分
3.3 “ifdef”
判斷一個(gè)變量是否已經(jīng)定義晰赞。格式如下
ifdef VARIABLE-NAME
- 如果變量“VAEIABLE_NAME”的值非空稼病,那么表達(dá)式為真选侨,將“TEXT-IF-TRUE”作為make要執(zhí)行的一部分。
- 表達(dá)式為假然走,如果存在“TEXT-IF-FALSE”援制,就將它作為make要執(zhí)行一部分。當(dāng)一個(gè)變量沒有被定義時(shí)芍瑞,它的值為空晨仑。“VARIABLE-NAME”可以是變量或者函數(shù)的引用
注1:在 Makefile 中沒有定義的變量的值為空
注2:ifdef只是測試一個(gè)變量是否有值拆檬,不會對變量進(jìn)行替換展開來判斷變量的值是否為空洪己。對于變量“VARIABLE-NAME”,除了 “VARIABLE-NAME=”這種情況以外竟贯,使用其它方式對它的定義都會使“ifdef”返回真答捕。就是說,即使我們通過其它方式(比如屑那,定義它的值引用了其它的變量)給它賦了一個(gè)空值拱镐,“ifdef”也會返回真
例如
bar =
foo = $(bar)
ifdef foo
frobozz = yes
else
frobozz = no
endif
這個(gè)ifdef返回的是true
3.4 “ifndef”
判斷一個(gè)變量是否已經(jīng)定義。格式如下
ifdef VARIABLE-NAME
它的功能就是實(shí)現(xiàn)了和“ifdef”相反的條件判斷
3.5 說明
- 在“CONDITIONAL-DIRECTIVE”這一行上持际,可以以若干個(gè)空格開始沃琅,make處理時(shí)會被忽略這些空格
- 不能以[Tab]字符做為開始(不然就被認(rèn)為是命令)
- 條件判斷語句中,在除關(guān)鍵字(包括“endif”)之前蜘欲、條件表達(dá)式參數(shù)中之外的其他任何地方都可以使用多個(gè)空格或[Tab]字符益眉,它不會影響條件判斷語句的功能。同樣行尾也可以使用注釋(“#”開始直到一行的結(jié)束)
- “else”和“endif”也是條件判斷語句的一部分芒填。在書寫時(shí)它們都是沒有任何參數(shù)的,可以以多個(gè)空格開始(同樣不能以[Tab]字符開始)多個(gè)空格或[Tab]字符結(jié)束空繁。行尾同樣可以有注釋內(nèi)容殿衰。
注: 在make讀取makefile文件時(shí)計(jì)算表達(dá)式的值,并根據(jù)表達(dá)式的值決定判斷語句中那一部分被作為此Makefile所要執(zhí)行的內(nèi)容(選擇符合條件的語句) 因此在條件表達(dá)式中不能使用自動化變量盛泡,自動化變量在規(guī)則命令執(zhí)行時(shí)才有效 更不能將一個(gè)完整的條件判斷語句分寫在兩個(gè)不同的makefile文件中闷祥,在一個(gè)makefile文件使用指示符“include”包含另外一個(gè)makefile文件
4 標(biāo)記測試的條件語句
我們可以使用條件判斷語句、變量“MAKEFLAGS”和函數(shù)“findstring”傲诵,實(shí)現(xiàn)對make命令行選項(xiàng)的測試(變量“MAKEFLAGS”之前的文章-7.3變量和遞歸有說明凯砍,函數(shù)“findstring”以后再說)
例如
archive.a: ...
ifneq (,$(findstring t,$(MAKEFLAGS)))
+touch archive.a
+ranlib -t archive.a
else
ranlib archive.a
endif
語句是判斷make的命令行參數(shù)中是否包含“-t”(用來更新目標(biāo)文件的時(shí)間戳)。根據(jù)命令行參數(shù)情況完成對“archive.a”執(zhí)行不同的操作拴竹。命令行前的“+”的意思是告訴make悟衩,即使make使用了“-t”參數(shù),“+”之后的命令都需要被執(zhí)行