4通用Makefile編寫

来源:http://www.cnblogs.com/CZM-/archive/2016/04/01/5346528.html
-Advertisement-
Play Games

a.c #include #include "a.h" int main() { printf("hello world\n"); printf("A= %d\n",A); test_fun(); return 0; } a.h #define A 1 b.c #include int test_f... ...


 

a.c

#include<stdio.h>
#include "a.h"

int main()
{
    
    printf("hello world\n");
    printf("A= %d\n",A);
    test_fun();
    return 0;
}

a.h

#define A 1

b.c

#include <stdio.h>

int test_fun()
{ 
    printf("it is B\n");
    return 0;
}

 

編譯test_Makefile的方法:
a. gcc -o test a.c b.c
對於a.c: 預處理、編譯、彙編
對於b.c:預處理、編譯、彙編
最後鏈接
優點:命令簡單
缺點:如果文件很多,即使你只修改了一個文件,但是所有的文件文件都要重新"預處理、編譯、彙編"
      效率低

 

b. 寫Makefile
核心:規則

目標:依賴1 依賴2
    命令

命令執行的條件:
i. "依賴"文件 比 "目標"文件 新
ii.沒有"目標"這個文件
    這是一個文件的依賴關係,也就是說,target這一個或多個的目標文件依賴於dependencies中的文件,其生成規則定義在command中。dependencies 中如果有一個以上的文件時間要比target文件要新的話,command所定義的命令就會被執行。這就是 Makefile的規則。也就是Makefile中最核心的內容。在Makefile中的命令,必須要以[Tab]鍵開始。

2.2 變數定義

    Makefile中變數的定義一般有兩種: =和:=。 =符號定義的變數叫延時變數,只有在使用的時候才擴展開來; :=符號定義的變數為立即變數,一旦定義就擴展。 使用=定義的變數不能追加新值,使用:=定義的變數可以使用+=追加新值

2.3 文件指示

    在Makefile使用include關鍵字可以把別的Makefile包含進來,這很像C語言的#include,被包含的文件會原模原樣的放在當前文件的包含位置。include的語法是:include <filename> filename可以是當前操作系統Shell的文件模式(可以保含路徑和通配符)

2.4 偽目標

    偽目標並不是一個文件,只是一個標簽,由於偽目標不是文件,所以make無法生成它的依賴關係和決定 它是否要執行。我們只有通過顯示地指明這個目標才能讓其生效。當然,偽目標的取名不能和文件名重名,不然其就失去了偽目標的意義了。當然,為了避免和文件重名的這種情況,我們可以使用一個特殊的標記.PHONY來顯示地指明一個目標是偽目標,向make說明,不管是否有這個文件,這個目標就是偽目標。

2.5 自動化變數

    $<    第一個依賴文件的名稱

    $?    所有的依賴文件,以空格分開,這些依賴文件的修改日期比目標的創建日期晚

    $@    目標的完整名稱

    $^    所有的依賴文件,以空格分開,不包含重覆的依賴文件

Makefile_1 

test: a.c b.c a.h 
    gcc -o test a.c b.c

只要有任一文件變化,就執行gcc -o test a.c b.c效率低

Makefile_2

test : a.o b.o 
    gcc -o test a.o b.o
a.o : a.c
    gcc -c -o a.o a.c
b.o : b.c
    gcc -c -o b.o b.c

文件不會隨著某個文件變化而全部編譯

 

Makefile_3

test : a.o b.o 
    gcc -o test a.o b.o
    
a.o : a.c a.h

%.o : %.c
    gcc -c -o $@ $<

加通通配符,通配符%,$@表示目標值,$<表示第一個依賴,$^表示所有依賴

不能生成依賴文件

 

頂層Makefile 頂層Makefile.build 子目錄Makefile

編譯過程:

    從頂層開始遞歸進入子目錄,當進入到一個目錄的最底層時,開始使用GCC編譯,再將該層的所有.o文件打包成build-in.o,返回它的上一層目錄再遞歸進入子目錄,當編譯完所有的子目錄後,就開始編譯頂層的.c文件,最後將頂層的.o文件和頂層每個子目錄的build-in.o鏈接成我們的目標文件

 

電子書Makefile

 

CROSS_COMPILE = arm-linux-
AS        = $(CROSS_COMPILE)as
LD        = $(CROSS_COMPILE)ld
CC        = $(CROSS_COMPILE)gcc
CPP        = $(CC) -E
AR        = $(CROSS_COMPILE)ar
NM        = $(CROSS_COMPILE)nm

STRIP        = $(CROSS_COMPILE)strip
OBJCOPY        = $(CROSS_COMPILE)objcopy
OBJDUMP        = $(CROSS_COMPILE)objdump

export AS LD CC CPP AR NM 
export STRIP OBJCOPY OBJDUMP
編譯選項

CFLAGS := -Wall -O2 -g
CFLAGS += -I $(shell pwd)/include
鏈接選項
LDFLAGS := -lm -lfreetype

export CFLAGS LDFLAGS

假目標

PHONY := __build
__build:

obj-y :=
subdir-y :=
包含當前目錄下的Makefile
include Makefile

 

工程必備:

頂層Makefile 頂層Makefile.build 子目錄Makefile

編譯過程:

    從頂層開始遞歸進入子目錄,當進入到一個目錄的最底層時,開始使用GCC編譯,再將該層的所有.o文件打包成build-in.o,返回它的上一層目錄再遞歸進入子目錄,當編譯完所有的子目錄後,就開始編譯頂層的.c文件,最後將頂層的.o文件和頂層每個子目錄的build-in.o鏈接成我們的目標文件

頂層Makefile解析(隨工程而變):

#----------------------------------------------指定編譯工具鏈---------------------------------------------------

CROSS_COMPILE =                             #指定編譯器種類

AS        = $(CROSS_COMPILE)as         #

LD        = $(CROSS_COMPILE)ld          #鏈接工具

CC        = $(CROSS_COMPILE)gcc       #編譯工具

CPP        = $(CC) -E                             #

AR        = $(CROSS_COMPILE)ar         #打包工具

NM        = $(CROSS_COMPILE)nm       #

STRIP        = $(CROSS_COMPILE)strip              #優化工具

OBJCOPY        = $(CROSS_COMPILE)objcopy   #

OBJDUMP        = $(CROSS_COMPILE)objdump  #

export AS LD CC CPP AR NM                           #將定義的變數導出,方便其他makefile使用

export STRIP OBJCOPY OBJDUMP                   #將定義的變數導出,方便其他makefile使用

CFLAGS := -Wall -O2 -g                                    #編譯器參數

CFLAGS += -I $(shell pwd)/include                     #指定編譯器頭文件(根據實際項目手動修改)

LDFLAGS := -lm -lfreetype -lvga                         #指定編譯器鏈接庫(根據實際項目手動修改)

export CFLAGS LDFLAGS                                #將定義的變數導出,方便其他makefile使用

TOPDIR := $(shell pwd)                                     #獲得當前程式的頂層目錄

export TOPDIR                                                 #輸出頂層目錄

TARGET := show_file                                      #編譯後的程式名(根據實際項目手動修改)

#-------------------------頂層要生成的.o文件以及頂層文件夾(根據實際項目手動修改)------------------

obj-y += main.o

obj-y += display/

obj-y += draw/

obj-y += encoding/

obj-y += fonts/

#--------------------------------------------頂層的第一個規則(預設規則)-----------------------------------------

all :

    make -C ./ -f $(TOPDIR)/Makefile.build           #進入當前目錄,使用頂層的makefile.build進行編譯

    $(CC) $(LDFLAGS) -o $(TARGET) built-in.o    #將編譯好的built-in.o文件鏈接生成我們的目標文件

#------------------------------------------------頂層的清除規則-------------------------------------------------------

clean:

    rm -f $(shell find -name "*.o")                        #刪除所有的.o文件

    rm -f $(shell find -name "*.d")                        #刪除所有的.d文件

    rm -f $(TARGET)                                         #刪除目標文件

.PHONY:all clean

頂層Makefile.build解析(無需改動):

PHONY := __build                                        #定義一個PHONY變數

__build:                                                       #開頭說明__build偽目標,使其成為Makefile.build的第一個目標

obj-y :=                                                        #定義當前目錄的目標變數,初始值為空

subdir-y :=                                                   #定義當前目錄的子目錄變數,初始值為空

include Makefile                                          #將當前目錄的Makefile包含進來,初始化obj-y

                                                                   #obj-y:=a.o b.o c/ d/

__subdir-y := $(patsubst %/,%,$(filter %/, $(obj-y)))   #篩選出當前目錄的目標變數中的子目錄,並且去掉/

                                                                                   #$(filter %/, $(obj-y)):c/ d/

                                                                                   #__subdir-y:c d

subdir-y += $(__subdir-y)                                           #將開始定義的subdir-y賦值為__subdir-y

                                                                                   #subdir-y:c d

subdir_objs := $(foreach f,$(subdir-y),$(f)/built-in.o)  #對於subdir-y裡面的每一個值(目錄),增加一個相應的目錄/built-in.o的變數值

                                                                                   #subdir_objs:c/built-in.o d/built-in.o

cur_objs := $(filter-out %/, $(obj-y))                            #得到obj-y中的.o文件

                                                                                   #cur_objs:a.o b.o

dep_files := $(foreach f,$(cur_objs),.$(f).d)                #對於所有的.o文件,定義它的依賴文件名

                                                                                   #dep_files: .a.d .b.d

dep_files := $(wildcard $(dep_files))

ifneq ($(dep_files),)                                                    #根據依賴文件名,判斷依賴文件是否存在,存在就包含就來

    include $(dep_files)

endif

PHONY += $(subdir-y) #將$(subdir-y)也加入到變數PHONY中

--------------------------------------------Makefile. build的第一個規則--------------------------------------------------------------

__build : $(subdir-y) built-in.o                                    #第一個規則

$(subdir-y):                                                                #第一個規則的第一個依賴規則

    make -C $@ -f $(TOPDIR)/Makefile.build              #依次進入該子目錄變數裡面存儲的值,使用的Makefile.build進行編譯

built-in.o : $(cur_objs) $(subdir_objs)                       #第一個規則的第二個依賴規則

      $(LD) -r -o $@ $^                                                 #該規則的命令:將該目錄下的.o和$(subdir_obj)打包成built-in.o文件

dep_file = [email protected]                                              #

%.o : %.c                                                                   #第一個規則的第二個依賴規則的依賴規則 

$(CC) $(CFLAGS) -Wp,-MD,$(dep_file) -c -o $@ $<#用於將目錄下所有的.c文件編譯成.o文件

.PHONY : $(PHONY)                                                  #將PHONY聲明為偽目標

 

本程式的Makefile分為3類:
1. 頂層目錄的Makefile
2. 頂層目錄的Makefile.build
3. 各級子目錄的Makefile

一、各級子目錄的Makefile:
   它最簡單,形式如下:
obj-y += file.o
obj-y += subdir/
  
   "obj-y += file.o"表示把當前目錄下的file.c編進程式里,
   "obj-y += subdir/"表示要進入subdir這個子目錄下去尋找文件來編進程式里,是哪些文件由subdir目錄下的Makefile決定。

註意: "subdir/"中的斜杠"/"不可省略

二、頂層目錄的Makefile:
   它除了定義obj-y來指定根目錄下要編進程式去的文件、子目錄外,主要是定義工具鏈、編譯參數、鏈接參數──就是文件中用export導出的各變數。

三、頂層目錄的Makefile.build:
   這是最複雜的部分,它的功能就是把某個目錄及它的所有子目錄中、需要編進程式去的文件都編譯出來,打包為built-in.o
   詳細的講解請看視頻。

四、怎麼使用這套Makefile:
1.把頂層Makefile, Makefile.build放入程式的頂層目錄
2.修改頂層Makefile
2.1 修改工具鏈
2.2 修改編譯選項、鏈接選項
2.3 修改obj-y決定頂層目錄下哪些文件、哪些子目錄被編進程式
2.4 修改TARGET,這是用來指定編譯出來的程式的名字

3. 在各一個子目錄下都建一個Makefile,形式為:
obj-y += file1.o
obj-y += file2.o
obj-y += subdir1/
obj-y += subdir2/

4. 執行"make"來編譯,執行"make clean"來清除,執行"make distclean"來徹底清除


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、 什麼是死鎖 死鎖是指兩個或兩個以上的進程在執行過程中,因爭奪資源而造成的一種互相等待的現象,若無外力作用,它們都將無法推進下去.此時稱系統處於死鎖狀態或系統產生了死鎖,這些永遠在互相等的進程稱為死鎖進程. 二、 死鎖產生的四個必要條件 互斥條件:指進程對所分配到的資源進行排它性使用,即在一段時 ...
  • NOLOCK(不加鎖) 此選項被選中時,SQL Server 在讀取或修改數據時不加任何鎖。 在這種情況下,用戶有可能讀取到未完成事務(Uncommited Transaction)或回滾(Roll Back)中的數據, 即所謂的“臟數據”。 HOLDLOCK(保持鎖) 此選項被選中時,SQL Se ...
  • Too many open files解決辦法 1,修改/etc/security/limits.conf文件內容 使用sudo vim /etc/security/limits.conf命令進入/etc/security/limits.conf文件中。 在/etc/security/limits. ...
  • 今天突然想要在linux上播放音樂,但是打開之後發現缺少瞭解碼器 於是去網上找瞭解決的方法,說得天花亂墜,不過有個帖子給出了很簡單的方法。 以下內容轉載於網路 上面提示沒有安裝解碼器,這是因為版權問題軟體不自帶解碼器 解決的方法很簡單,就是安裝第三方源 32位的系統第三方源:rpm -ivh htt ...
  • 讓.NET應用程式在linux上運行,目前通用的做法就是在Linux上安裝mono,然後通過”mono your.exe“命令運行這個程式。 這種運行.net程式的辦法有兩個弱點,一個是需要客戶機安裝mono,二個是 ”mono xx.exe“ 這種命令行總讓人感到有點不太專業的味道。 那麼,有沒有 ...
  • 實驗4 外部中斷實驗 一、實驗目的 理解中斷的基本概念。 掌握STM32的中斷源及中斷優先順序。 掌握STM32外部中斷技術的基本使用方法。 掌握STM32中斷處理程式的編程方法。 二、實驗內容 硬體設計原理圖如下圖1所示:LED的連接在上一節已經介紹過了,在STM32開發板上的按鍵KEY0是接在PE ...
  • Office 2016 VOL版 http://blog.sina.com.cn/s/blog_470614a90102vtmc.html 專業版合集: magnet:?xt=urn:btih:772912C80F19B7DC145C88810DD8FCD8031D4478 標準版合集: magne ...
  • 一 命令解釋: dd:用指定大小的塊拷貝一個文件,併在拷貝的同時進行指定的轉換。 註意:指定數字的地方若以下列字元結尾,則乘以相應的數字:b=512;c=1;k=1024;w=2 參數註釋: ascii:轉換ebcdic為ascii ebcdic:轉換ascii為ebcdic ibm:轉換ascii ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...