以TCC88XX爲例,當在Android頂層源碼目錄使用make編譯完成後,會生成這樣一個目錄:linux
out/target/product/tcc8800,該目錄內部有咱們須要的boot.img和system.img,boot.mgshell
使用kernel和out/target/product/tcc8800/root目錄打包而成(廣義的ramdisk),也就是說,框架
boot.img是由kernel和ramdisk.img生成獲得,在本文中主要分析root目錄和ramdisk.img的生成,ide
在Android編譯框架中,把許多固定的、反覆用到的目錄路徑定義爲宏變量,而上述生成的目錄ui
out/target/product/tcc8800的宏即爲:PRODUCT_OUTspa
out/target/product/tcc8800/system的宏即爲:TARGET_OUT設計
而out/target/product/tcc8800/root的宏即爲:TARGET_ROOT_OUT,code
out/target/product/tcc8800/root主要是由system/core/rootdir目錄拷貝獲得的,orm
爲此我分析了system/core/rootdir目錄中的Android.mk文件,具體狀況是這樣的:字符串
copy_from := etc/dbus.conf etc/hosts
copy_from += etc/vold.fstab
以上內容將須要拷貝的文件添加到copy_from變量中,以便後續處理。
拷貝到那裏呢? 在看看copy_to的定義:
copy_to := $(addprefix $(TARGET_OUT)/,$(coby_from))
該語句即爲copy_from中每一個字符串片斷添加一個TARGET_OUT前綴(即system),這樣copy_to的
內容就很明瞭:
copy_to :=out/target/product/tcc8800/system/etc/dbus.conf ...之類,在此略掉。
以後,給copy_from添加路徑前綴:
copy_from := $(addprefix $(LOCAL_PATH)/, $(copy_from)
之因此要添加前綴的緣由是接下來立刻要設置的拷貝語句:
$(copy_to) : $(TARGET_OUT)/% : $(LOCAL_PATH)/% | $(ACP) $(transform-prebuilt-to-target)
上述語句會讓Android在構建img前,自動完成拷貝工做,其中使用到符號%進行匹配,這也是爲何要
給copy_from添加前綴的緣由。
隨後,腳本將copy_to變量添加進 ALL_PREBUILT全局宏中:
ALL_PREBUILT += $(copy_to)
最後,在build/core/Makefile中看到copy_to的內容被提取到了另一個全局宏 ,具體以下:
#build/core/Makefile INTERNAL_SYSTEMIMAGE_FILES := $(filter $(TARGET_OUT)/%,$(ALL_PREBUILT) ......
因爲上述4行內容設計到system.img的生成,在此不深究。
看來system/core/rootdir中的部份內容是拷貝到了out/target/product/tcc8800/system中的,並非
完徹底全拷貝到out/target/product/tcc8800/root目錄中去的。
咱們回頭繼續查看system/core/rootdir/Android.mk文件,該文件中剩下的內容纔是與root密切相關的。
file := $(TARGET_ROOT_OUT)/init.rc
而後也是經典的拷貝設置:
$(file) : $(LOCAL_PATH)/% | $(ACP) $(transform-prebuilt-to-target)
接下來的腳本的內容是爲生成boot.img而寫的。
ALL_PREBUILT +=$(file) $(INSTALLED_RAMDISK_TARGET):$(file)
看來原理也和上述system的拷貝相同,在build/core/Makefile中是由INTERNAL_RAMDISK_FILE提取的,
具體以下:INTERNAL_RAMDISK_FILES := $(filter $(TARGET_ROOT_OUT)/%, $(ALL_PREBUILT) ...
隨後有一段很關鍵的句子直接道破了ramdisk.img的生成:
INSTALLED_RAMDISK_TARGET=$(BUILT_RAMDISK_TARGET) $(INSTALLED_RAMDISK_TARGET):$(MKBOOTFS $(INTERNAL_RAMDISK_FILES | $(MINIGZIP) $(hide) $(MKBOOTFS) $(TARGET_ROOT_OUT) | $(MINIGZIP) > $@
如此多的宏,讓咱們一一列出它們的值:
BUILT_RAMDISK_TARGET = $(PRODUCT_OUT/ramdisk.img 這是咱們的目標
INSTALLED_RAMDISK_TARGET = BUILT_RAMDISK_TARGET 目標假裝了一下。
MKBOOTFS = mkbootfs 就是位於out/host/linux-x86/bin目錄下的mkbootfs,這東西天然也有後話。
INTERNAL_RAMDISK_FILES = 全部TARGET_ROOT_OUT中的文件
由此能夠看出root目錄先被打包生成了ramdisk.img,而後才合併進boot.img的。