背景:
前端时间搞了个底板,用的是恩智浦的imx8平台,安装了Ubuntu系统,各种驱动安装涉及linux开发,我这方面太弱势,抱大腿学了点皮毛。想写个功能demo,发现Makefile都不会,太悲剧。
用window开发通过ide不需要过多关注底层原理,基本都是ide给你自动生成连接关系。这次我得打破舒适区,拿下Makefile就从这次做起。
基础知识:
GCC(GNU Compiler Collection,GNU编译器套件),是由GNU开发的编程语言编译器。它是以GPL许可证所发行的自由软件,也是 GNU计划的关键部分。我在linux下用的make命令就是去执行了gcc编译编译命令(unix下是cc,当然你也可以是其他编译器的),当你输入make指令的时候,它会自动查找当前目录下的Makefile或者makefile(注意大小写,首字母大小写都可以),然后执行里面的内容。其实本质上Makefile就是个脚本,通过make命令来运行它。但是这个脚本真的很关键呀,兄弟们儿!它能告诉系统我这个文件依赖哪些文件,我要输出的名字是什么,头文件去哪里找,有没有指定的库,甚至于如何克服不同路径带来的安装问题等等。我目前也就学了7788,也不敢乱说,不过就我现在的理解,它真的很强大,我后悔没有早点搞明白这个东西,导致自己对底层的编译过程一头雾水。
学习过程:
我的学习过程很简单,去网上找教程发现大多数都罗七八嗦的,太费时间了,我需要快速上手,所以就大概看了一下(5分钟),然后就找了个阿里的sdk,看其中的Makefile文件,把这个Makefile文件看懂了,我至少能把用法学个大概,可以解决我现在的困难(自己写工程的Makefile)。废话不多说,下面是我今天整理出来的Makefile文件加备注版本,一个字一个字抠出来的,有问题可以留言,咱们可以交流,向大家学习。
Q := @ #此种形式都是定义,给变量赋值,后面的值赋给前面的
TOP_DIR := $(shell pwd) #pwd获取当前目录,然后赋值
DEMOS_DIR := demos
OUT_DIR := output
CC := gcc #定义编译器,GCC(GNU Compiler Collection,GNU编译器套件),是由GNU开发的编程语言编译器。它是以GPL许可证所发行的自由软件,也是 GNU计划的关键部分。
BLD_LDFLAGS := $(LDFLAGS) -lpthread
#LDFLAGS可以用来link动态库或者静态库的,用在gcc选项上,后面可以加-L指定库目录,加-l指定库名称,这里的-lpthread,告诉编译器去连接libpthread.so动态库,如果gcc选项里指定了-static,则代表连接静态库。
SRC_DIR := $(shell find . -type d \( ! -name demos \)) #这里的d类型是目录(directory),但是删去了demos这个目录
HDR_DIR := $(SRC_DIR)
BLD_CFLAGS += $(addprefix -I,$(HDR_DIR))#这里给HDR_DIR加了一个-I前缀,然后再添加到BLD_LDFLAGS的后面,用在gcc选项上,作用是指定头文件目录
SRC_FILES := $(shell find $(SRC_DIR) -not -path "*/demos/*.c" -name "*.c")#这里是按照SRC_DIR中的路径去搜索名字中带.c的文件,但是要去掉SRC_DIR中带*/demos/*.c条目
OBJ_FILES := $(SRC_FILES:.c=.o)#把SRC_FILES中的.c换成.o,然后赋值给OBJ_FILES
OBJ_FILES := $(addprefix $(OUT_DIR)/,$(OBJ_FILES))#在OBJ_FILES中加output/前缀
PROG_TARGET := $(subst _,-,$(patsubst %.c,%,$(wildcard demos/*_demo.c)))#wildcard以demos/*_demo.c为模板输出,*代表任意,patsubst是筛选替换,把第一次的输出中包含.c结尾的去掉.c,subst是替换,把第二次输出条目中的_替换为-
all: prepare $(PROG_TARGET)#all是目标,它依赖于:后面的两个项目,这两个项目在下面都做了定义
prepare:
$(Q)mkdir -p output #prepare没有依赖,它的内容是依次创建output目录
$(PROG_TARGET): $(OBJ_FILES) #PROG_TARGET的依赖是OBJ_FILES,上文都有定义
$(Q)echo "+ Linking $(OUT_DIR)/$(notdir $@) ..."
#echo是打印输出提示信息或者结果,这里输出编译最终的文件,其中$@代表目标(aim),也就是PROG_TARGET
$(Q)mkdir -p $(dir $@)
#这里是创建路径,也就是demos
$(Q)$(CC) -o $@ \ #编译命令,后面是编译参数,-o代表指定输出的名字,名字包括$@和后面的patsubst生成的.o文件
$(patsubst $(OUT_DIR)/%,%,$(addsuffix .c,$(subst $(notdir $@),$(subst -,_,$(notdir $@)),$@))) \
#上面这句中addsuffix第二个参数其实就把-换回了_,addsuffix加了.c的尾缀,patsubst是去掉(OUT_DIR)/
$(BLD_CFLAGS) $^ $(BLD_LDFLAGS)
#上面这句加了头文件,依赖,最后还添加了链接库
$(Q)mv $@ $(OUT_DIR)
#上面这句是移动输出文件到output文件夹里
#下面的这个定义并没有执行,因为Makefile文件中多个gcc命令的时候,只执行第一个。
$(OUT_DIR)/%.o: %.c
$(Q)echo ": Compiling $< ..."
$(Q)mkdir -p $(OUT_DIR)/$(dir $<)
$(Q)$(CC) -o $@ -c $< $(BLD_CFLAGS)
#下面的clean可以通过make clean执行,清理生成的文件
clean:
$(Q)rm -rf $(OUT_DIR)
为了看懂上面的Makefile,我看了好多博客,有几个比较详细推荐的,可以给大家参考一下:
linux编程入门(六)-编写Makefile文件 :https://www.jianshu.com/p/442e71755643
Makefile选项CFLAGS,LDFLAGS,LIBS:https://www.cnblogs.com/fire909090/p/11160219.html
一文入门makefile:https://zhuanlan.zhihu.com/p/56489231
以上,等等,感谢网友们的无私奉献!