Android内核——Make脚本备忘

Make脚本的基本语法:
目标(target) : 条件(prerequest) (Tab键) 命令

  • target ~任意一个字符串名称/具体文件的名称。
  • prerequest ~其他目标的名称/具体文件的名称。

执行Make脚本时,Make解释器会检查target和prerequest中包含的文件时间戳是否相同,如果不同的话,解释器就会执行Tab键后面的“命令”,命令可以是任意可执行程序。


一个简单的Makefile文件

#Filename Makefile
#this file is used for show how to use makefile
$(info start working)
hello: hello.c
    echo hello.c -o hello.bin

hello.bin: hello.c
    @echo "now make hello.bin"
    gcc hello.c -o hello.bin

.PHONY:he 
he: hello.c
    @echo "now make he"
    gcc hello.c -o hello.bin
  • # ~ 注释符
  • $ ~ 函数调用符号,info是一个函数名称,作用是输出一段信息。
  • 目标定义前不能加任何空格,而命令行前必须以Tab键开始。
  • .PHONY ~ 声明一个目标,将总是执行其指定的命令。
  • @ ~ 不显示被执行的命令。
  • 对于hello.bin目标,该目标本身就是一个文件,其依赖的文件是hello.c文件。
    运行:
    $make -f Makefile hello
  • -f ~ 指定要执行的脚本文件的名称

变量的定义与赋值

  • := ~ 简单展开型
  • = ~ 递归展开型
  • ?= ~ 条件赋值
  • += ~附加赋值

Make解释器执行脚本的过程:

  1. 装载Makefile及Makefile中include的其他Makefile。
  1. 根据用户指定的target找出该target的全部依赖关系,并判断依赖条件中的时间戳。

条件控制语句

  • 在解析器解析脚本文件时处理
  • 在执行脚本文件时处理
    语法模型1:
    if-condition
    text if the condition is true
    endif
    语法模型2:
    if-condition
    text if the condition is true
    else
    text if the condition is false
    endif
    if-condition只能进行两种判断:
  1. 判断表达式是否相等 ifeq test/ifneq test
  2. 判断表达式是否被定义 ifdef var/ifndef var

宏(函数)定义

Make脚本中函数,按被调用的方式可分为三类:

  1. 内置函数 ~ Make解释器内部定义好的函数
    调用:$(fname, param...)

  2. 用户定义的、带参数的函数,使用define关键字进行调用
    调用:$(call fname, param)

  3. 用户定义的、不带参数的函数(宏)
    调用:$ (fname)
    用户函数的定义方式:
    define fname
    各种具体的命令
    endef
    举例:
    define showFirstName
    @echo $(1)
    endef

    .PHONY: name 
    name:
        $(call showFirstName, Nancymi, Yang)
    

执行结果:

$ make name
Nancymi


内置符号和变量

内置符号

  • $@ ~ target的名称
  • $* ~ 不包含target后缀的target名称
  • $^~所有的先决条件的名称,不包含重复的
  • $? ~ 有更新的先决条件列表
  • $+ ~ 所有的先决条件,包含了重复的
  • $< ~ 第一个先决条件的名称
    内置变量
  • MAKE_VERSION ~ make版本
  • CURDIR ~ 执行make的目录
  • MAKEFILE_LIST ~ 本次make命令执行时,所有被包含的makefile列表
  • VARIABLE ~ 所有的变量列表
  • CC ~ C编译器(默认gcc)
  • CXX ~ C++ 编译器(默认g++)
  • CXXFLAGS ~ C++的编译选项
  • CPPFLAGS ~ 仅为.cpp文件的编译选项
  • TARGET_ARCH ~ 目标主机架构
  • LDLIBS ~ 连接器库选项

模板目标(Pattern target)

使用一种“模板”来定义目标。
源脚本文件:

.PHONY: test
test: f21.o f2.o main.o
    gcc -o main.bin f1.o f2.o main.o

f1.o: f1.c
    gcc f1.c f1.o

f2.o: f2.c
    gcc f2.c f2.o

main.o: main.c
    gcc main.c -c main.o

使用模板目标后:

OBJ = f1.o f2.o main.o
.PHONY: test
test: $(OBJ)
    gcc $(OBJ) -o main.bin

%.o: %.c
    gcc -c -o $@ $<
  • %.o ~ 模板目标
  • % ~ 模板通配符,所有.o文件

目标特定的变量赋值(Target-specific variable)

CFLAGS = -c 
.PHONY: tar1
tar1:
    gcc $(CFLAGS) main.c

在以上脚本文件中,CFLAGS被赋为"-c",且对整个脚本文件有效。所以在执行Makefile tar1时,会执行gcc -c main.c.
如果在该脚本文件中另一目标想用自己特定的CFLAGS变量,可进行局部变量赋值:

tar2:CFLAGS = -c 
tar2:
    gcc $(CFLAGS) main.c

如果用户想在命令行中对变量进行赋值:$ make tar2 0e CFLAGS="-c -g".
如果Makefile文件不允许用户自己进行赋值,也就是必须强制使用Makefile文件中的赋值:

tar2: override CFLAGS = -c 
tar2:
    gcc $(CFLAGS) main.c

常用选项

不多说了,直接man make查看。

推荐阅读更多精彩内容

  • 本文发布于 Mlin Blog、简书,作者 @木林(Mlin) ,转载请保留原文链接。 上一篇文章 《快速搭建个人...
    mik1th0n阅读 5,116评论 0 28
  • 来自陈浩的一片老文,但绝对营养。 示例工程:3 个头文件*.h,和 8 个 C 文件*.c。 初 编译过程,源文件...
    周筱鲁阅读 3,968评论 0 17
  • @(linux 编程)[开发技能, 工具使用] What is GNU Make Make 是控制工程中通过源码生...
    orientlu阅读 10,245评论 0 26
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 121,948评论 17 134
  • 前言Android Build 系统是 Android 源码的一部分。关于如何获取 Android 源码,请参照 ...
    加菲猫Jack阅读 1,303评论 0 5