我的Makefile文件入门

本文分享了初学者如何通过实战理解并编写Makefile,讲解了Makefile的基本概念、GCC编译器及其在Makefile中的作用,以及一个实际的Makefile实例解析。适合对Linux开发和Makefile不熟悉的读者快速上手。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景:

前端时间搞了个底板,用的是恩智浦的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

以上,等等,感谢网友们的无私奉献!

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值