我們也說說Android.mk(1)
從函數(shù)說起
大家都習(xí)慣看從頭,從構(gòu)建目標(biāo)講起的,導(dǎo)致每篇文檔熟的都是前面的部分。很多教程也都是想辦法能夠觀其大略,從整體上給大家一個(gè)思路。比如《深入理解Android內(nèi)核設(shè)計(jì)思想》的第4章,比如《Android內(nèi)核剖析》的第18章,比如《深入解析Android 5.0系統(tǒng)》的第2章。
于是我打算反其道而行之,先從調(diào)用函數(shù)開始講。
最后一招:shell函數(shù)
我們最先把最后看家的絕招列出來吧,shell函數(shù),可以用來執(zhí)行shell命令。一切用之后講到的函數(shù)解決不了的問題,都可以靠shell函數(shù)調(diào)用外部功能來解決。
不過,調(diào)用shell需要啟動(dòng)新進(jìn)程,影響性能,只要有其它方法建議就不要使用它。
比如,我們想要列出當(dāng)前目錄下有哪些cpp文件,makefile可以這樣寫:
files := $(shell ls *.cpp)
all :
@echo -n "The files is:"
@echo $(files)
.PHONY : all
拼接字符串
在Makefile中用途相當(dāng)廣的就是拼接字符串,join函數(shù)就是干這事兒的。
格式:$(join 串1 , 串2)
也可以是多個(gè)串一起拼:$(join 串1 串2 串3, 串4 串5 串6)
多個(gè)串就是1和4拼,2和5拼。。??傊嵌禾?hào)前的和逗號(hào)后的拼。
去空格
有事兒沒事兒,去去空格總不是壞事兒。
$(strip 字符串)
文件路徑操作函數(shù)
在實(shí)際的Makefile開發(fā)中,經(jīng)常遇到要處理變量中存儲(chǔ)的路徑名。
先來個(gè)提綱:
- 有一個(gè)不知道哪里來的文件名,想取它的路徑,用dir
- 有一個(gè)不知道哪里來的文件名,想只取它的文件名,用notdir
- 如果只想要擴(kuò)展名,判斷是什么類型的,用suffix
- 如果想不要擴(kuò)展名,取出來拼個(gè)別的擴(kuò)展名上去,比如xxx.c,只要xxx,將來拼個(gè)xxx.o或者就是xxx之類的,用basename
- 想給文件添加個(gè)擴(kuò)展名,用addsuffix
- 想要把文件搞到另一個(gè)路徑上去,拼個(gè)目錄名上去,用addprefix
取一個(gè)文件的目錄路徑函數(shù)dir
例:
.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result := $(dir $(oatfile))
all2 :
@echo -n "The result is: "
@echo $(result)
輸出:The result is: out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/
取文件名函數(shù)
例:
.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_dir := $(dir $(oatfile))
result_notdir := $(notdir $(oatfile))
all2 :
@echo "The result is: "
@echo $(result_dir)
@echo $(result_notdir)
輸出:
The result is:
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/
package.odex
取文件擴(kuò)展名
例:
.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_dir := $(dir $(oatfile))
result_notdir := $(notdir $(oatfile))
result_suffix := $(suffix $(oatfile))
all2 :
@echo "The result is: "
@echo $(result_dir)
@echo $(result_notdir)
@echo $(result_suffix)
輸出:
The result is:
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/
package.odex
.odex
取文件基本名
如果只用basename函數(shù),是連路徑都有的,只去除掉了擴(kuò)展名的名字。不過我們剛學(xué)過notdir,一起用下就是了。
例:
.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_suffix := $(suffix $(oatfile))
result_basename := $(basename $(oatfile))
result_basename2 := $(basename $(notdir $(oatfile)))
all2 :
@echo "The result is: "
@echo $(result_suffix)
@echo $(result_basename)
@echo $(result_basename2)
輸出:
The result is:
.odex
out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package
package
給名字加后綴
使用addsuffix函數(shù),$(addsuffix 文件名,擴(kuò)展名)
我們舉個(gè)例子說明:現(xiàn)在想把oat文件壓縮成.tar.gz:
.PHONY : all2
oatfile := out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
result_basename := $(basename $(oatfile))
compress_oat := tar cfvz $(addsuffix .tar.gz , $(result_basename)) $(oatfile)
all2 :
@echo "The result is: "
@echo $(compress_oat)
輸出如下:
The result is:
tar cfvz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.tar.gz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
給文件名加前綴
這個(gè)用于換路徑, addprefix。一個(gè)前綴可以后邊都加上。
例:
.PHONY : all3
oatfile2 := system.odex music.odex Contacts.odex
result_oatfile2 = $(addprefix /data/dalvik-cache/, $(oatfile2))
all3:
@echo $(result_oatfile2)
輸出:
/data/dalvik-cache/system.odex /data/dalvik-cache/music.odex /data/dalvik-cache/Contacts.odex
直接做字符串替換
如果我們不想拆開加去的這么麻煩,有一個(gè)簡(jiǎn)易的方法是直接做字符串替換。
subst函數(shù)的定義如下:$(subst 源串,目標(biāo)串,要做替換的字符串)
例,我們把剛才將.odex擴(kuò)展名換成.tag.gz擴(kuò)展名,并拼成一個(gè)tar命令的makefile重寫一下:
.PHONY : all4
oatfile_targz := $(subst $(suffix $(oatfile)),.tar.gz,$(oatfile))
compress_oat := tar cfvz $(oatfile_targz) $(oatfile)
all4 :
@echo "The result is: "
@echo $(compress_oat)
輸出如下:
The result is:
tar cfvz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.tar.gz out/target/product/ali6753_65t_m0/obj/APPS/MusicFX_intermediates/oat/arm64/package.odex
過濾函數(shù)
過濾函數(shù)有兩種:
- filter是符合條件的留下
- filter-out是符合條件的去除掉
我們看一個(gè)例子:
all8:
@echo $(filter-out default interpreter jit optimizing,xoc)
@echo $(filter-out default interpreter jit optimizing,default)
輸出結(jié)果:
$ make all8
xoc
filter-out default interpreter jit optimizing,default這一句,因?yàn)閐efault在列表中,所以被過濾掉了,變成一個(gè)空串。
filter-out default interpreter jit optimizing,xoc:這句因?yàn)閤oc不在過濾列表之中,所以留下了。
單詞的處理
文件列表,參數(shù)列表等等可以看成對(duì)單詞的處理
比如下面的例子,我們想要查MAKEFILE_LIST中的最后一個(gè)文件,可以這樣寫:
my-dir = $(call parent-dir,$(lastword $(MAKEFILE_LIST)))
針對(duì)單詞處理,有下面的常用函數(shù):
- firstword:取第一個(gè)單詞
- lastword:取最后一個(gè)單詞
- words:統(tǒng)計(jì)一共有多少個(gè)單詞
- word:取第n個(gè)單詞
- wordlist:取單詞的子集
word的取值是從1開始,不能取0。
我們看一個(gè)例子:
SETTINGS_ART_DST := out/target/product/6753_doov_l5_64_m/system/priv-app/Settings/oat/arm64/Settings.odex
.PHONY : all10
all10:
@echo "Install: $@"
$(eval SETTINGS_ART_DST_LIST := $(subst /, ,$(SETTINGS_ART_DST)))
@echo $(words $(SETTINGS_ART_DST_LIST))
@echo $(word 1,$(SETTINGS_ART_DST_LIST))
@echo $(word 2,$(SETTINGS_ART_DST_LIST))
@echo $(wordlist 5,10,$(SETTINGS_ART_DST_LIST))
輸出如下:
$ make all10
Install: all10
10
out
target
system priv-app Settings oat arm64 Settings.odex