u-boot Makefile Source Test
一、概述
笔者已经写了一篇实现目标文件与源码分开的makefile测试实验,但是觉得不够完美,没有更多的体现u-boot Makefile的工作原理和特点。所以,决定重新修订,使之更加充分的接近u-boot Makfefile的原貌。
至于,u-boot Makfefile的整体论述和特点,笔者在u-boot Makefile整体解析文中已经详细论述,不在多言。
二、测试源码构成
整个测试源码,由四个文件夹构成,三个是源文件夹,分别包含源程序(.c)和Makefile。顶层目录中包含config.mk和rules.mk文件。
三、测试源码注释
1、顶层Makfile
ifneq ($(BUILD_DIR),) #此变量在命令行选择定义,定义了就是目标文件和源码文件分开,否则是同一个
saved-output := $(BUILD_DIR)
# Attempt to create a output directory.
$(shell [ -d ${BUILD_DIR} ] || mkdir -p ${BUILD_DIR}) #创建目录BUILD_DIR
# Verify if it was successful.
BUILD_DIR := $(shell cd $(BUILD_DIR) && /bin/pwd)
$(if $(BUILD_DIR),,$(error output directory "$(saved-output)" does not exist))
endif # ifneq ($(BUILD_DIR),)
OBJTREE := $(if $(BUILD_DIR),$(BUILD_DIR),$(CURDIR)) #目标文件生成的根目录,可以认为是一个常量,\
区别于obj
SRCTREE := $(CURDIR) #CURDIR是当前目录的路径,目录改变,它随之改变
TOPDIR := $(SRCTREE)
LNDIR := $(OBJTREE)
export SRCTREE OBJTREE TOPDIR ifneq ((OBJTREE),(SRCTREE))
REMOTE_BUILD :=
export REMOTE_BUILD #此变量会被config.mk所调用
endif include $(TOPDIR)/config.mk #config.mk中定义了obj、CC等变量,下边的内容需要调用 CROSS_COMPILE =
export CROSS_COMPILE OBJS = add/add.o sub/sub.o test/test.o #需要生成的目标文件
OBJS := $(addprefix $(obj),$(OBJS)) #需要生成目标文件的完整路径
.PHONY : $(OBJS) #确保源文件修改了,目标文件能够自动重建 ALL = $(obj)program #在program前边也加上了前缀obj
all: $(ALL) $(obj)program: $(OBJS)
$(CC) $^ -o $@ $(OBJS):
make -C $(dir $(subst $(obj),,$@)) #依次执行各个子目录中的makefile SUBDIRS = add/ sub/ test/
.PHONY: clean distclean
clean:
for f in $(SUBDIRS); do make -C $$f clean; done //依次在各个子目录中删除
rm -f $(ALL) distclean:
for f in $(SUBDIRS); do make -C $$f distclean; done
rm -f $(ALL)
2、config.mk
ifneq ($(OBJTREE),$(SRCTREE)) #判断是否选择目标文件生成目录和源码目录分开
ifeq ($(CURDIR),$(SRCTREE)) #以下内容是目标文件生成目录和源码目录分开的情况
dir :=
else
dir := $(subst $(SRCTREE)/,,$(CURDIR))
endif obj := $(if $(dir),$(OBJTREE)/$(dir)/,$(OBJTREE)/) #确定要生成目标文件的完整路径,
#注意这是个变量,在不同的子目录中,执行makefile,obj的值相应的发生变化
src := $(if $(dir),$(SRCTREE)/$(dir)/,$(SRCTREE)/) #确定当前源文件的目录 $(shell mkdir -p $(obj)) #创建目标文件的完整路径,注意gcc命令并不能自动生成缺少的路径,所以需要事先创建
else #以下内容是目标文件生成目录和源码目录同一的情况
obj := #目标文件与源文件目录相同时,obj没有必要赋值
src :=
endif #以下的规则将用来编译子目录中的源文件
# Include the make variables (CC, etc...)
#
AS = $(CROSS_COMPILE)as
LD = $(CROSS_COMPILE)ld
CC = $(CROSS_COMPILE)gcc
CPP = $(CC) -E
AR = $(CROSS_COMPILE)ar
OBJCOPY = $(CROSS_COMPILE)objcopy
OBJDUMP = $(CROSS_COMPILE)objdump ######################################################################### ifndef REMOTE_BUILD %.s: %.S
$(CPP) $(AFLAGS) -o $@ $<
%.o: %.S
$(CC) $(AFLAGS) -c -o $@ $<
%.o: %.c
$(CC) $(CFLAGS) -c -o $@ $< else $(obj)%.s: %.S
$(CPP) $(AFLAGS) -o $@ $<
$(obj)%.o: %.S
$(CC) $(AFLAGS) -c -o $@ $<
$(obj)%.o: %.c
$(CC) $(CFLAGS) -c -o $@ $<
endif #########################################################################
3、rules.mk
#########################################################################
$(obj).depend: $(src)Makefile $(TOPDIR)/config.mk $(SRCS) #SRCS在子目录Makefile中定义
@rm -f $@
@for f in $(SRCS); do \
g=`basename $$f | sed -e 's/\(.*\)\.\w/\1.o/'`; \
$(CC) -M $(HOST_CFLAGS) $(CPPFLAGS) -MQ $(obj)$$g $$f >> $@ ; \
done
#########################################################################
4、子目录Makefile
include $(TOPDIR)/config.mk #包含config.mk COBJS := add.o #.c文件生成的目标文件
SOBJS := #.S文件生成的目标文件 SRCS := $(SOBJS:.o=.S) $(COBJS:.o=.c) #所有参与编译的源文件(.c、.S)
OBJS := $(addprefix $(obj),$(COBJS))#添加目录obj
SOBJS := $(addprefix $(obj),$(SOBJS)) all: $(obj).depend $(OBJS) #先生成依赖文件,然后才能编译源程序 clean:
rm -f $(SOBJS) $(OBJS) distclean: clean
rm -f $(LIB) $(obj).depend ######################################################################### # defines $(obj).depend target
include $(SRCTREE)/rules.mk #包含rules.mk sinclude $(obj).depend #包含.depend文件 #########################################################################
四、编译方法
1、目标文件与源码混在一起
直接在顶层目录中执行:#make
2、目标文件与源码分开
方法一:
先声明个环境变量:#exprort BUILD_DIR=/tmp/build //这里以/tmp/build为例
然后在顶层目录中执行:#make
方法二:
直接在顶层目录中执行:#make BUILD_DIR=/tmp/build
u-boot Makefile Source Test的更多相关文章
- linux内核的makefile.txt讲解
linux内核的linux-3.6.5\Documentation\kbuild\makefiles.txt Linux Kernel Makefiles This document describe ...
- Linux内核Makefile文件(翻译自内核手册)
--译自Linux3.9.5 Kernel Makefiles(内核目录documention/kbuild/makefiles.txt) kbuild(kernel build) 内核编译器 Thi ...
- Linux Kernel的Makefile与Kconfig文件的语法
https://www.kernel.org/doc/Documentation/kbuild/kconfig-language.txt Introduction ------------ The c ...
- kernel Makefile Kconfig说明
实际文档位置:Documentation/kbuild/makefiles.txt,此为翻译稿. *************************************************** ...
- Linux 2.6内核Makefile浅析
1 概述 Makefile由五个部分组成: Makefile:根目录Makefile,它读取.config文件,并负责创建vmlinux(内核镜像)和modules(模块文件). .config:内核 ...
- 【linux】 Makefile之make menuconfig /uImage
欢迎转载,转载时请保留作者信息,谢谢. 邮箱:tangzhongp@163.com 博客园地址:http://www.cnblogs.com/embedded-tzp Csdn博客地址:http: ...
- Makefile常用万能模板(包括静态链接库、动态链接库、可执行文件)
本文把makefile 分成了三份:生成可执行文件的makefile,生成静态链接库的makefile,生成动态链接库的makefile. 这些makefile都很简单,一般都是一看就会用,用法也很容 ...
- Spring Boot Reference Guide
Spring Boot Reference Guide Authors Phillip Webb, Dave Syer, Josh Long, Stéphane Nicoll, Rob Winch, ...
- makefile 模板 (template)
本文把makefile 分成了三份:生成可执行文件的makefile,生成静态链接库的makefile,生成动态链接库的makefile. 这些makefile都很简单,一般都是一看就会用,用法也很容 ...
随机推荐
- Eclipse快捷键调试
Eclipse中有如下一些和运行调试相关的快捷键Ctrl+Shift+B:在当前行设置断点或取消设置的断点 F11:调试最后一次执行的程序 Ctrl+F11:运行最后一次执行的程序F5:跟踪到 ...
- onInterceptTouchEvent和onTouchEvent调用时序
onInterceptTouchEvent()是ViewGroup的一个方法,目的是在系统向该ViewGroup及其各个childView触发onTouchEvent()之前对相关事件进行一次拦截,A ...
- WPF组件开发之组件的基类
之前在网上看到很多关于组件开发的资料,但真正可以用到框架内的却很少.今天贴出自己做的组件,并适合大部分框架的代码. 组件开发需要先做出组件的基类,然后由其他的各类组件去继承这个基类,下面是组件基类的代 ...
- iOS内存泄漏自动检测工具PLeakSniffer
新款objective-C内存泄漏自动检测工具 PLeakSniffer , GitHub地址 (https://github.com/music4kid/PLeakSniffer). 背景 前些天读 ...
- IT行业智力测试
1.有10筐苹果,其中有1筐是次品,正品苹果每个10两,次品苹果每个9两,现有一称,问怎么一次称出次品是哪筐? 2.有甲.乙.丙.丁四个人,要在夜里过一座桥.他们通过这座桥分别需要耗时1.2.5.10 ...
- WebService学习笔记系列(三)
网上有一些提供webservice服务的网站,我们要怎么调用呢? 今天来看个如何调用手机归属地查询服务.这个网站上提供了许多webservice服务,其中包括手机归属地查询服务,我们今天就用wsimp ...
- Android开发之UI更新交互机制与实例解析
android开发过程中,经常需要更新UI的状态和文案等.这是就需要对UI进行 更新.在android中更新UI一般有三种方法,handler机制.RunOnUiThread方法以及AsyncTask ...
- Android ListView实现仿iPhone实现左滑删除按钮
需要自定义ListView.这里就交FloatDelListView吧. 复写onTouchEvent方法.如下: @Override public boolean onTouchEvent(Moti ...
- css3之@font-face---再也不用被迫使用web安全字体了
1,@font-face 的出现在没有css3之前,我们给网页设计字体只能设置web安全字体,使得我们的网页表现看上去好像都是那个样子,那么如果我们想给字体设置web设计者提供的字体呢?没有问题,cs ...
- Java SE (5)之 线程使用
JAVA有两种线程的方法Thread 和Runnable 能够使用,这是为了弥补不能多继承的缺陷 首先是 Thread package com.sunzhiyan03; /* * 演示线程 * */ ...