MTK編譯過程

MTK 軟件工程的配置說明 請下載我的資源《MTK軟件工程和配置簡介.doc》

 

本文詳解MTK的編譯過程,並會根據我的理解來更新。閱讀上文對本文的一些文件和術語會有很好的幫助。

導讀:
  Comp.mak 完成模塊編譯連接,生成bin
  Gsm2.mak 主編譯文件,完成clean,remake,new等工作
  XXX_GPRS.mak 客戶私有的配置,根據客戶的不同,而設定不同的配置文件
  Option.mak 工程中的基本配置及宏定義文件
  Verno_XXX.bld 版本文件
  Custom.bld 要保證在客戶版本中都使用相同的配置的地方要寫在這裏,這個裏面的一些文件不能被改變
  第一步:
  1.Make.bat命令首先調用ChgFileMode.bat,將編譯過程中需要用到的文件的只讀屬性修改爲可讀寫,再調用make2.pl。
  2.Make2.pl的主要工作就是匹配gsm2.mak的參數,然後通過命令
  system("${makeCmd} -f${makeFolder}${myMF} -r -R CUSTOMER=$custom PROJECT=$project $action");來調用gsm2.mak。
  具體解析出來就是
  make -f make/gsm2.mak -r –R CUSTOMER=工程名 PROJECT=gprs new|update|remake
  第二步:整個過程如圖
  Gsm2.mak
  Option.mak
  XXX_GPRS.mak
  REL_CR_MMI_GPRS.mak
  USER_SPECIFIC.mak
  
  1.在Gsm2.mak文件的開始處調用了option.mak文件。
  2.Option.mak又將make/$(strip $(CUSTOMER))_$(strip$(PROJECT)).mak文件包含進來,這個模式匹配結果爲:make目錄下,工程名稱_GPRS.mak,如K500GSD_GPRS.mak。
  3.在$(strip $(CUSTOMER))_$(strip$(PROJECT)).mak文件中又調用了REL_CR_MMI_$(strip$(PROJECT)).mak文件,這個文件模式具體被解析爲make目錄下的REL_CR_MMI_GPRS.mak文件。
  REL_CR_MMI_GPRS.mak這個文件定義了哪些文件需要加入到編譯目錄中。
  其中定義了MMI部分的編譯目錄爲MMIDIR = plutommi。
  在變量CUS_REL_BASE_COMP中定義了資源的編譯目錄:
  CUS_REL_BASE_COMP += $(strip $(MMIDIR))/mmi $(strip $(MMIDIR))/mtkapp $(strip$(MMIDIR))/tool $(MMIDIR)/WIN32FS
  CUS_REL_BASE_COMP += $(strip$(MMIDIR))/Customer/CustomerInc /
  $(strip $(MMIDIR))/Customer/Customize /
  $(strip $(MMIDIR))/Customer/CustResource/$(strip $(MMI_VERSION))/
  $(strip $(MMIDIR))/Customer/debug /
  $(strip $(MMIDIR))/Customer/Images/GameImages /
  $(strip $(MMIDIR))/Customer/Images/decoder /
  $(strip $(MMIDIR))/Customer/Res_MMI /
  $(strip $(MMIDIR))/Customer/ResGenerator /
  $(strip $(MMIDIR))/Customer/ResourceDLL /
  $(strip $(MMIDIR))/Customer/Resources /
  $(strip $(MMIDIR))/Customer/Audio
  客戶的資源需要編譯如下文件:CUS_REL_SRC_COMP += mmiresource mtkapp gdi_arm plutommi vendorapp
  圖片名稱爲:CUS_REL_BASE_COMP += $(strip$(MMIDIR))/Customer/Images/$(strip $(MMI_PROJ))$(strip$(MAIN_LCD_SIZE)),可解析爲如:
  plutommi/customer/images/K500GSD176X220
  
  REL_CR_MMI_$(strip $(PROJECT)).mak
  其中CUS_REL_OBJ_LIST這個變量存儲了連接過程中所需要的文件名稱
  
  4.make/$(strip $(CUSTOMER))_$(strip$(PROJECT)).mak文件的功能。
  CUSTOM_OPTION定義了所有需要編譯進去的功能模塊的宏。
  COMPLIST變量的功能:
  ifeq ($(strip $(RTOS)),NUCLEUS)
  COMPLIST = nucleus nucleus_int nucleus_ctrl_code nucleus_critical_data
  Endif
  如果操作系統爲NUCLEUS,那麼COMPLIST爲後面的值,COMPLIST爲所要編譯的文件列表,裏面存放了很多*.inc文件,展開後就變爲
  config/include
  stacklib/include
  adaptation/include
  kal/include等等
  
  5. option.mak文件的在後面調用了make/USER_SPECIFIC.mak
  如果使用CPU爲ARM,則編譯工具目錄爲如下定義
  ifeq ($(strip $(COMPILER)),ADS)
  DIR_ARM = c:/progra~1/arm/adsv1_2
  DIR_ARM := $(strip $(DIR_ARM))
  DIR_TOOL = $(DIR_ARM)/bin
  DIR_ARMLIB = $(DIR_ARM)/lib
  DIR_ARMINC = $(DIR_ARM)/include
  Endif
  連接打包工具等爲如下幾個:
  DIR_TOOL := $(strip $(DIR_TOOL))
  LINK = $(DIR_TOOL)/armlink.exe # Linker
  ASM = $(DIR_TOOL)/armasm.exe # ARM assembler
  LIB = $(DIR_TOOL)/armar.exe # Library tool
  BIN_CREATE = $(DIR_TOOL)/fromelf.exe # Binary tool
  下面這段代碼告訴我們需要用什麼編譯工具來編譯
  ifeq ($(strip $(COMPILER)),ADS)
  ifeq ($(strip $(COMPILE_MODE)),INST16)
  CC = $(DIR_TOOL)/tcc.exe # Thumb Mode(16bits), use tcc
  CC32 = $(DIR_TOOL)/armcc.exe # ARM Mode(32bits), use armcc
  CPPC = $(DIR_TOOL)/tcpp.exe # Thumb Mode(16bits), use tcc
  CPPC32 = $(DIR_TOOL)/armcpp.exe # ARM Mode(32bits), use armcc
  else
  ifeq ($(strip $(COMPILE_MODE)),INST32)
  CC = $(DIR_TOOL)/armcc.exe # ARM Mode(32bits), use armcc
  CPPC = $(DIR_TOOL)/armcpp.exe # ARM Mode(32bits), use armcc
  else
  CC = $(DIR_TOOL)/tcc.exe # Default tcc
  CC32 = $(DIR_TOOL)/armcc.exe # ARM Mode(32bits), use armcc
  CPPC = $(DIR_TOOL)/tcpp.exe # Thumb Mode(16bits), use tcc
  CPPC32 = $(DIR_TOOL)/armcpp.exe # ARM Mode(32bits), use armcc
  endif
  endif
  endif
  
  ifeq ($(strip $(PLATFORM)),MT6223P)
  AFLAGS := -g -littleend -cpu ARM7EJ-S
  Endif
  
  COMMINCDIRS變量在先包含了基本功能模塊的inc目錄後再包含如下文件
  COMMINCDIRS += $(DIR_ARMINC) $(CUSTOM_COMMINC),
  
  6.Option.mak中定義了一些附加功能模塊的編譯模式(是否被編譯,被編譯成什麼樣的結果),比如說藍牙,UART3,WIFI,USB,WAP等等,還有一些編譯器的設置。
  我們的版本號和Scat文件被定義在這個變量中:5056L
  SCATTERFILE = custom/system/$(strip$(BOARD_VER))/scat$(strip $(PLATFORM)).txt
  VERNOFILE = make/Verno_$(CUSTOMER).bld
  .bin文件的名稱設置在這個變量TARGNAME = $(CUSTOMER)_$(strip$(SUB_BOARD_VER))_$(PROJECT)_$(strip $(PLATFORM))_$(strip$(CHIP_VER))
  TST_DB := $(strip $(TSTDIR))/database_classb
  然後被包含到Option.mak文件中來,include $(strip$(VERNOFILE))。
  
  7.Gsm2.mak文件中new執行的指令
  new : cleanall cmmgen mmi_feature_check asngen codegen asnregen operator_check_lite update
  
  update所執行的指令
  update : cleanlog cleanbin mcddll_update codegen resgen cksysdrv remake
  
  remake所執行的指令
  remake : mcp_check cleanlog cleanbin genverno libs $(BIN_FILE) done
  
  resgen用來編譯資源文件:Res_XXX.c
  
  (echo CUSTOM_OPTION = $(foreach def,$(BOARD_VER) $(PLATFORM) $(LCD_MODULE) $(EXT_CAM_MODULE)$(CMOS_SENSOR),-D "$(def)") />$(MMIDIR)/customer/resGenerator/custom_option.txt)
  首先將CUSTOM_OPTION重定向到custom_option.txt中
  
  (type make/~cus_opt.tmp >>$(MMIDIR)/customer/resGenerator/custom_option.txt)
  使用tools/strcmpex.exe生成~cus_opt.tmp, ~tgt_opt.tmp,~inc.tmp三個文件
  將~cus_opt.tmp 重定向到custom_option.txt中
  
  (@del $(MMIDIR)/mmi/TargetOption.txt) &/
  (copy /Y make/~tgt_opt.tmp $(MMIDIR)/MMI/TargetOption.txt)
  再將TargetOption.txt更新一下,將~tgt_opt.tmp重定向到TargetOption.txt中
  
  (type make/~inc.tmp >$(MMIDIR)/customer/resGenerator/custom_include.tmp)
  再將~inc.tmp重定向到custom_include.tmp中,
  
  執行replace_project_name.pl文件,編譯Res_XXX.c
  再執行ResGenerator_HW.bat,編譯資源文件。
  
  8.Remake中的編譯過程是在libs中完成的
  目標依賴:libs: cleanlib startbuildlibs $(COMPLIBLIST)
  Cleanlib在這個依賴中做的動作是清掉上次生成的.bin,.elf,.lis等文件,接着清掉需要重新來生成的.lib文件。$(COMPLIBLIST)這個依賴就是我們需要重新生成的.lib文件。
  .lib文件的依賴關係:%.lib:
  在這個依賴關係中所做的動作是設置編譯器,鏈接器以及這個過程的參數等其他信息,然後將這些信息輸出到:~compbld.tmp臨時文件中。
  
  (tools/make.exe -fmake/comp.mak -k -r -R $(strip $(CMD_ARGU)) COMPONENT=$* >$(strip $(COMPLOGDIR))/$*.log) /
  調用了make命令來執行comp.mak文件,並且可以看到> $(strip$(COMPLOGDIR))/$*.log
  該語句將執行comp.mak文件時產生的信息存放在當前編譯部分的.log文件中,通常看編譯信息的比如custom.log等就是該中類型的文件。
  
  依賴關係:update_lib: $(TARGLIB)
  $(TARGLIB) : $(COBJS) $(CPPOBJS) $(AOBJS) $(ARMOBJS)
  $(TARGLIB)依賴很多.c和.obj
  .c.obj:
  @echo Compiling $< ...
  @tools/strcmpex.exe $(ACTION) remake e $(*F).via $(CINTWORK) -c $(CFLAGS) $(CDEFS) $(CINCDIRS) -o $(COMPOBJS_DIR)/$@$<
  @tools/strcmpex.exe $(ACTION) remake n $(*F).via $(CINTWORK) -c $(CFLAGS) $(CDEFS) $(CINCDIRS) $(MD) -o$(COMPOBJS_DIR)/$@ $<
  @if exist $(*F).via tools/warp.exe $(*F).via
  @if exist $(*F).via $(CMPLR) $(VIA) $(*F).via
  @if not $(ACTION)==remake if exist $(FIXPATH)/$(*F).d perl./tools/pack_dep.pl $(FIXPATH)/$(*F).d >$(RULESDIR)/$(COMPONENT)_dep/$(*F).det
  @if not $(ACTION)==remake if exist $(FIXPATH)/$(*F).d del /f /q$(FIXPATH)/$(*F).d >nul
  @if exist $(*F).via del /f /q $(*F).via
  要生成.o依賴於.c文件,如果我的.c文件經過更新,那麼該.o就需要重新生成,該.c就需要重新來編譯。
  @tools/strcmpex.exe $(ACTION) remake e$(*F).via $(CINTWORK) -c $(CFLAGS) $(CDEFS)$(CINCDIRS) -o $(COMPOBJS_DIR)/$@ $<
  @tools/strcmpex.exe $(ACTION) remake n $(*F).via $(CINTWORK) -c $(CFLAGS) $(CDEFS) $(CINCDIRS) $(MD) -o$(COMPOBJS_DIR)/$@ $<
  有兩個編譯環節,針對不同的編譯動作,兩者的不同點在於中間有個$(MD)編譯選項,在有該選項的時間編譯的時候就會生成依賴關係文件.d。
  @if not $(ACTION)==remake if exist $(FIXPATH)/$(*F).d perl./tools/pack_dep.pl $(FIXPATH)/$(*F).d >$(RULESDIR)/$(COMPONENT)_dep/$(*F).det
  在這個的語句中有$(FIXPATH)/$(*F).d >$(RULESDIR)/$ (COMPONENT)_dep/$(*F).det該動作又將.d文件作爲.pl文件的參數,執行該perl文件後將結果輸出放在.det文件中。
  @if not $(ACTION)==remake if exist $(FIXPATH)/$(*F).d del /f /q$(FIXPATH)/$(*F).d >nul
  然後又執行這個語句,將.d文件刪除。
  在make文件中可以看到-include$(RULESDIR)/$(COMPONENT).dep
  在查看各個.o的依賴關係的時間要查看該.dep文件來決定是否需要來重新編譯生成該.o。
  如果新添加了.h文件又沒有來更新.dep文件就可能導致該.o查找依賴時沒有依賴該.h。不去更新該.o。
  @if exist $(FIXPATH)/$(CUS_MTK_LIB)/$(COMPONENT).lib /
  (copy /z $(FIXPATH)/$(CUS_MTK_LIB)/$(COMPONENT).lib $(subst/,/,$(TARGLIB))) &/
  ($(LIB) -r $(TARGLIB) $(COMPOBJS_DIR)/*.obj) /
  else /
  ($(LIB) -create $(TARGLIB) $(COMPOBJS_DIR)/*.obj)
  在生成了.o文件後在.lib的依賴中可以看到($(LIB) -r$(TARGLIB) $(COMPOBJS_DIR)/*.obj)
  該語句將生成的.o文件打包成.lib庫文件。
  
  連接過程:$(BIN_FILE):
  最後完成編譯:done:
  # -----------------------------
  # Clean temporary files in make directory
  # -----------------------------
  @echo Cleaning make/~*.tmp files ...
  @if exist make/~*.tmp /
  del make/~*.tmp
  @echo Done.
  @perl tools/time.pl -n
  清掉臨時文件,打出done信息,最後打出時間標籤。


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/dragoniye/archive/2008/12/04/3445134.aspx

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章