Makefile的条件执行
生活随笔
收集整理的這篇文章主要介紹了
Makefile的条件执行
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
1? ?? ?? ?Makefile的條件判斷
條件語句可以根據一個變量的值來控制對Makefile的執行,執行或者忽略Makefile的特定部分。條件語句可以是兩個不同變量、或者變量和常量值得比較。需要注意的是:條件語句只能用于控制make實際執行的makefile文件部分,它不能控制規則的shell命令執行過程。Makefile中使用條件控制可以做到處理的靈活性和高效性。
1.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
? ? ……
在上例中,條件語句中使用到了三個關鍵字:“ifeq”、“else”和“endif”。其中:
1.? ?? ?“ifeq”表示條件語句的開始,并指定了一個比較條件(相等)。之后是用圓括號括包圍的、使用逗號“,”分割的兩個參數,和關鍵字“ifeq”用空格分開。參數中的變量引用在進行變量值比較時被展開。“ifeq”之后就是當條件滿足make需要執行的,條件不滿足時忽略。
2.? ?? ?“else”之后就是當條件不滿足時的執行部分。不是所有的條件語句都需要此部分。
3.? ?? ?“endif”表示一個條件語句的結束,任何一個條件表達式都必須以“endif”結束。
通過上邊的例子我們可以了解到。Makefile中,條件表達式是工作文本級別的,條件的解析是由make程序來完成的。make是在解析Makefile時根據條件表達式忽略條件表達式中的某一個文本行,解析完成后保留的只有表達式滿足條件所需要執行的文本行(可參考
[url=#_make%E5%A6%82%E4%BD%95%E8%AF%BB%E5%8F%96makefile]2.9 make如何讀取Makefile[/url]
一節)。對于上例,make處理這個條件時:
當變量“CC”的值為“gcc”時,整個條件表達式等效于:
foo: $(objects)
? ?? ?$(CC) -o foo $(objects) $(libs_for_gcc)
當變量“CC”值不等于“gcc”時等效于:
foo: $(objects)
? ?? ?$(CC) -o foo $(objects) $(normal_libs)
上面的例子,一種更簡潔實現方式:
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)
1.2? ?? ???條件判斷的基本語法
一個簡單的不包含“else”分支的條件判斷語句的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
endif
表達式中“TEXT-IF-TRUE”可以是若干任何文本行,當條件為真時它就將被make程序作為需要執行的一部分。當條件為假時,不作為需要執行的一部分。
另外包含“else”的復雜一點的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
else
TEXT-IF-FALSE
endif
表示了如果條件為真,則將“TEXT-IF-TRUE”作為執行Makefile的一部分,否則將“TEXT-IF-FALSE”作為執行的Makefile的一部分。和“TEXT-IF-TRUE”一樣,“TEXT-IF-FALSE”可以是若干任何文本行。
條件判斷語句中“CONDITIONAL-DIRECTIVE”對于上邊的兩種格式都是同樣的。可以是以下四種用于測試不同條件的關鍵字。
1.2.1.1? ?? ? 關鍵字“ifeq”
此關鍵字用來判斷參數是否相等,格式如下:
? ?? ? `ifeq (ARG1, ARG2)'
`ifeq 'ARG1' 'ARG2''
`ifeq "ARG1" "ARG2"'
`ifeq "ARG1" 'ARG2''
`ifeq 'ARG1' "ARG2"'
替換并展開“ARG1”和“ARG1”之后,對它們的值進行比較。如果相同則(條件為真)將“TEXT-IF-TRUE”作為make要執行的一部分,否則將“TEXT-IF-FALSE”作為make要執行的一部分(上邊的第二種格式)。
通常我們會使用它來判斷一個變量的值是否為空(不是任何字符)。參數值可能是通過引用變量或者函數得到的,因而在展開過程中可能造成參數值中包含空字符(空格等)。一般在處理這種情況時我們使用make的“strip”函數(參考
[url=#_%E6%96%87%E6%9C%AC%E5%A4%84%E7%90%86%E5%87%BD%E6%95%B0]7.2 文本處理函數[/url]
一節)來對它進行處理,去掉變量值中的空字符。格式為:
ifeq ($(strip $(foo)),)
TEXT-IF-EMPTY
endif
這樣,即就是在“$(foo)”中存在若干前導和結尾空格,make在解析Makefile時“TEXT-IF-EMPTY”也會被作為執行的一部分。
1.2.1.2? ?? ? 關鍵字“ifneq”
次關鍵字是用來判斷參數是否不相等,格式為:
`ifneq (ARG1, ARG2)'
`ifneq 'ARG1' 'ARG2''
`ifneq "ARG1" "ARG2"'
`ifneq "ARG1" 'ARG2''
`ifneq 'ARG1' "ARG2"'
關鍵字“ifneq”實現的條件判斷語句和“ifeq”相反。首先替換并展開“ARG1”和“ARG1”,對它們的值進行比較。如果不相同(條件為真)則將“TEXT-IF-TRUE”作為make要執行的一部分,否則將“TEXT-IF-FALSE”作為make要執行的一部分。
1.2.1.3? ?? ? 關鍵字“ifdef”
關鍵字是“ifdef”用來判斷一個變量是否定義。格式為:
`ifdef VARIABLE-NAME'
如果變量“VAEIABLE_NAME”的值非空,那么表達式為真,將“TEXT-IF-TRUE”作為make要執行的一部分。否則,表達式為假,如果存在“TEXT-IF-FALSE”,就將它作為make要執行一部分。當一個變量沒有被定義時,它的值為空。“VARIABLE-NAME”可以是變量或者函數的引用。
對于“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中的結果是:“frobozz = yes”;而例2的結果是:“frobozz = no”。其原因就是在例1中,變量“foo”的定義是“foo = $(bar)”。雖然變量“bar”的值為空,但是“ifdef”判斷的結果是真。因此當我們需要判斷一個變量的值是否為空的情況時,需要使用“ifeq”(或者“ifneq”)而不是“ifdef”。可參考前兩個小節的內容。
1.2.1.4? ?? ? 關鍵字“ifndef”
關鍵字“ifndef”實現的功能和“ifdef”相反。格式為:
`ifdef VARIABLE-NAME'
這個就不詳細討論了,它的功能就是實現了和“ifdef”相反的條件判斷。
在“ CONDITIONAL-DIRECTIVE”這一行上,可以以若干個空格開始,make處理時會被忽略這些空格。但不能以[Tab]字符做為開始(不然就被認為是命令)。條件判斷語句中,在除關鍵字(包括“endif”)之前、和條件表達式的參數中之外,其他任何地方都可以使用多個空格或[Tab]字符,它不會影響條件判斷語句的功能。同樣行尾也可以使用注釋(“#”開始直到一行的結束)。
“else”和“endif”也是條件判斷語句的一部分。在書寫時它們都是沒有任何參數的,可以以多個空格開始(同樣不能以[Tab]字符開始)多個空格或[Tab]字符結束。行尾同樣可以有注釋內容。
在make讀取makefile文件時計算表達式的值,并根據表達式的值決定判斷語句中那一部分被作為此Makefile所要執行的內容(選擇符合條件的語句)。因此在條件表達式中不能使用自動化變量,自動化變量在規則命令執行時才有效。更不能將一個完整的條件判斷語句分寫在兩個不同的makefile文件中,其中在某一個makefile文件使用指示符“include”包含另外一個。
1.3? ?? ???標記測試的條件語句
我們可以使用條件判斷語句、并使用使用變量“MAKEFLAGS”(參考
[url=#_%E5%91%BD%E4%BB%A4%E8%A1%8C%E9%80%89%E9%A1%B9%E5%92%8C%E9%80%92%E5%BD%92]4.6.3 命令行選項和遞歸[/url]
一小節)和函數“findstring”(參考
[url=#_%E6%96%87%E6%9C%AC%E5%A4%84%E7%90%86%E5%87%BD%E6%95%B0]7.2 文本處理函數[/url]
一節),實現對make命令行選項的測試。看一個例子:
archive.a: ...
ifneq (,$(findstring t,$(MAKEFLAGS)))
? ?? ? +touch archive.a
? ?? ? +ranlib -t archive.a
else
? ?? ? ranlib archive.a
endif
這個條件語句判斷make的命令行參數是否包含“-t”(用來更新目標文件的時間戳)。根據命令行參數情況完成對“archive.a”執行不同的操作。命令行前的“+”的意思是告訴make,即使make使用了“-t”參數,“+”之后的命令都需要被執行
條件語句可以根據一個變量的值來控制對Makefile的執行,執行或者忽略Makefile的特定部分。條件語句可以是兩個不同變量、或者變量和常量值得比較。需要注意的是:條件語句只能用于控制make實際執行的makefile文件部分,它不能控制規則的shell命令執行過程。Makefile中使用條件控制可以做到處理的靈活性和高效性。
1.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
? ? ……
在上例中,條件語句中使用到了三個關鍵字:“ifeq”、“else”和“endif”。其中:
1.? ?? ?“ifeq”表示條件語句的開始,并指定了一個比較條件(相等)。之后是用圓括號括包圍的、使用逗號“,”分割的兩個參數,和關鍵字“ifeq”用空格分開。參數中的變量引用在進行變量值比較時被展開。“ifeq”之后就是當條件滿足make需要執行的,條件不滿足時忽略。
2.? ?? ?“else”之后就是當條件不滿足時的執行部分。不是所有的條件語句都需要此部分。
3.? ?? ?“endif”表示一個條件語句的結束,任何一個條件表達式都必須以“endif”結束。
通過上邊的例子我們可以了解到。Makefile中,條件表達式是工作文本級別的,條件的解析是由make程序來完成的。make是在解析Makefile時根據條件表達式忽略條件表達式中的某一個文本行,解析完成后保留的只有表達式滿足條件所需要執行的文本行(可參考
[url=#_make%E5%A6%82%E4%BD%95%E8%AF%BB%E5%8F%96makefile]2.9 make如何讀取Makefile[/url]
一節)。對于上例,make處理這個條件時:
當變量“CC”的值為“gcc”時,整個條件表達式等效于:
foo: $(objects)
? ?? ?$(CC) -o foo $(objects) $(libs_for_gcc)
當變量“CC”值不等于“gcc”時等效于:
foo: $(objects)
? ?? ?$(CC) -o foo $(objects) $(normal_libs)
上面的例子,一種更簡潔實現方式:
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)
1.2? ?? ???條件判斷的基本語法
一個簡單的不包含“else”分支的條件判斷語句的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
endif
表達式中“TEXT-IF-TRUE”可以是若干任何文本行,當條件為真時它就將被make程序作為需要執行的一部分。當條件為假時,不作為需要執行的一部分。
另外包含“else”的復雜一點的語法格式為:
CONDITIONAL-DIRECTIVE
TEXT-IF-TRUE
else
TEXT-IF-FALSE
endif
表示了如果條件為真,則將“TEXT-IF-TRUE”作為執行Makefile的一部分,否則將“TEXT-IF-FALSE”作為執行的Makefile的一部分。和“TEXT-IF-TRUE”一樣,“TEXT-IF-FALSE”可以是若干任何文本行。
條件判斷語句中“CONDITIONAL-DIRECTIVE”對于上邊的兩種格式都是同樣的。可以是以下四種用于測試不同條件的關鍵字。
1.2.1.1? ?? ? 關鍵字“ifeq”
此關鍵字用來判斷參數是否相等,格式如下:
? ?? ? `ifeq (ARG1, ARG2)'
`ifeq 'ARG1' 'ARG2''
`ifeq "ARG1" "ARG2"'
`ifeq "ARG1" 'ARG2''
`ifeq 'ARG1' "ARG2"'
替換并展開“ARG1”和“ARG1”之后,對它們的值進行比較。如果相同則(條件為真)將“TEXT-IF-TRUE”作為make要執行的一部分,否則將“TEXT-IF-FALSE”作為make要執行的一部分(上邊的第二種格式)。
通常我們會使用它來判斷一個變量的值是否為空(不是任何字符)。參數值可能是通過引用變量或者函數得到的,因而在展開過程中可能造成參數值中包含空字符(空格等)。一般在處理這種情況時我們使用make的“strip”函數(參考
[url=#_%E6%96%87%E6%9C%AC%E5%A4%84%E7%90%86%E5%87%BD%E6%95%B0]7.2 文本處理函數[/url]
一節)來對它進行處理,去掉變量值中的空字符。格式為:
ifeq ($(strip $(foo)),)
TEXT-IF-EMPTY
endif
這樣,即就是在“$(foo)”中存在若干前導和結尾空格,make在解析Makefile時“TEXT-IF-EMPTY”也會被作為執行的一部分。
1.2.1.2? ?? ? 關鍵字“ifneq”
次關鍵字是用來判斷參數是否不相等,格式為:
`ifneq (ARG1, ARG2)'
`ifneq 'ARG1' 'ARG2''
`ifneq "ARG1" "ARG2"'
`ifneq "ARG1" 'ARG2''
`ifneq 'ARG1' "ARG2"'
關鍵字“ifneq”實現的條件判斷語句和“ifeq”相反。首先替換并展開“ARG1”和“ARG1”,對它們的值進行比較。如果不相同(條件為真)則將“TEXT-IF-TRUE”作為make要執行的一部分,否則將“TEXT-IF-FALSE”作為make要執行的一部分。
1.2.1.3? ?? ? 關鍵字“ifdef”
關鍵字是“ifdef”用來判斷一個變量是否定義。格式為:
`ifdef VARIABLE-NAME'
如果變量“VAEIABLE_NAME”的值非空,那么表達式為真,將“TEXT-IF-TRUE”作為make要執行的一部分。否則,表達式為假,如果存在“TEXT-IF-FALSE”,就將它作為make要執行一部分。當一個變量沒有被定義時,它的值為空。“VARIABLE-NAME”可以是變量或者函數的引用。
對于“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中的結果是:“frobozz = yes”;而例2的結果是:“frobozz = no”。其原因就是在例1中,變量“foo”的定義是“foo = $(bar)”。雖然變量“bar”的值為空,但是“ifdef”判斷的結果是真。因此當我們需要判斷一個變量的值是否為空的情況時,需要使用“ifeq”(或者“ifneq”)而不是“ifdef”。可參考前兩個小節的內容。
1.2.1.4? ?? ? 關鍵字“ifndef”
關鍵字“ifndef”實現的功能和“ifdef”相反。格式為:
`ifdef VARIABLE-NAME'
這個就不詳細討論了,它的功能就是實現了和“ifdef”相反的條件判斷。
在“ CONDITIONAL-DIRECTIVE”這一行上,可以以若干個空格開始,make處理時會被忽略這些空格。但不能以[Tab]字符做為開始(不然就被認為是命令)。條件判斷語句中,在除關鍵字(包括“endif”)之前、和條件表達式的參數中之外,其他任何地方都可以使用多個空格或[Tab]字符,它不會影響條件判斷語句的功能。同樣行尾也可以使用注釋(“#”開始直到一行的結束)。
“else”和“endif”也是條件判斷語句的一部分。在書寫時它們都是沒有任何參數的,可以以多個空格開始(同樣不能以[Tab]字符開始)多個空格或[Tab]字符結束。行尾同樣可以有注釋內容。
在make讀取makefile文件時計算表達式的值,并根據表達式的值決定判斷語句中那一部分被作為此Makefile所要執行的內容(選擇符合條件的語句)。因此在條件表達式中不能使用自動化變量,自動化變量在規則命令執行時才有效。更不能將一個完整的條件判斷語句分寫在兩個不同的makefile文件中,其中在某一個makefile文件使用指示符“include”包含另外一個。
1.3? ?? ???標記測試的條件語句
我們可以使用條件判斷語句、并使用使用變量“MAKEFLAGS”(參考
[url=#_%E5%91%BD%E4%BB%A4%E8%A1%8C%E9%80%89%E9%A1%B9%E5%92%8C%E9%80%92%E5%BD%92]4.6.3 命令行選項和遞歸[/url]
一小節)和函數“findstring”(參考
[url=#_%E6%96%87%E6%9C%AC%E5%A4%84%E7%90%86%E5%87%BD%E6%95%B0]7.2 文本處理函數[/url]
一節),實現對make命令行選項的測試。看一個例子:
archive.a: ...
ifneq (,$(findstring t,$(MAKEFLAGS)))
? ?? ? +touch archive.a
? ?? ? +ranlib -t archive.a
else
? ?? ? ranlib archive.a
endif
這個條件語句判斷make的命令行參數是否包含“-t”(用來更新目標文件的時間戳)。根據命令行參數情況完成對“archive.a”執行不同的操作。命令行前的“+”的意思是告訴make,即使make使用了“-t”參數,“+”之后的命令都需要被執行
總結
以上是生活随笔為你收集整理的Makefile的条件执行的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Linux中cp和scp命令的使用方法
- 下一篇: C语言之详解#ifdef等宏