移植U-Boot
顶层Makefile的主要任务就是组织整个u-boot工程的编译,概括可以分为一下几个步骤:
1、首先通过执行make *_config传入$(@:_config=), ARCH, CPU, BOARD, VENDOR, SOC参数(一共六个参数但不一定同时存在),给mkconfig。
2、mkconfig接收到传递过来的参数后,将include头文件夹相应的头文件夹链接好,生成config.h。
3、然后执行make分别调用各个子目录的makefile文件,以生成所有的obj文件(包括start.o)和obj库文件*.a。
4、最后,通过链接器把所有目标文件链接起来,生成uboot镜像。不同格式的镜像都是调用相应工具,经由elf镜像间接或者直接的生成的。
UBoot1.1.6Makefile分析:
VERSION = 1
PATCHLEVEL = 1
SUBLEVEL = 6
EXTRAVERSION =
U_BOOT_VERSION = $(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION)
VERSION_FILE = $(obj)include/version_autogenerated.h // $(obj)为空
说明:?=条件赋值,只有变量在没有赋值的情况下才能赋值, :=表示变量的赋值是立即展开,即在定义时就给变量赋值,而=方式是延时展开,即在使用的时候才展开变量
HOSTARCH := $(shell uname -m | \
sed -e s/i.86/i386/ \
-e s/sun4u/sparc64/ \
-e s/arm.*/arm/ \
-e s/sa110/arm/ \
-e s/powerpc/ppc/ \
-e s/macppc/ppc/)
uname -m,就是得到machine hardware name,中文翻译成机器硬件名,sed -e的意思,就是表示后面跟的是一串命令脚本,s/abc/def/的命令表达式,就是表示要从标准输入中,查找到内容为abc的,然后替换成def。这样执行这一套程序下来,就知道了机器的硬件体系了。这里使用了make的shell函数来在shell环境中执行uname -m和sed -e命令。shell中()中的命令作为一个子shell来执行
HOSTOS := $(shell uname -s | tr '[:upper:]' '[:lower:]' | \
sed -e 's/\(cygwin\).*/cygwin/')
uname -s,得到kernel name,内核的名字。tr '[:upper:]' '[:lower:]',表示从标准输入里,把大字都换成小写,然后输出到标准输出。然后后面再跟了一串用来特别处理cygwin环境下编译的环境变量的配置的。
export HOSTARCH HOSTOS
export 命令将会使得被 export 的变量在运行的脚本(或 shell)的所有的子进程中都可用
# Deal with colliding definitions from tcsh etc.
#处理来自tcsh的互相冲突的定义等等。 一般来说,shell可以分成两类。第一类是由Bourne shell衍生出来的包括#sh,ksh,bash,与zsh。第二类是#由C shell衍生出来的,包括csh与 tcsh。除此之外还有一个rc,有人认为该自成一类,有人认为该归类在Bourne shell。
VENDOR=
##########################################################################
# U-boot build supports producing a object files to the separate external directory.
#Two use cases are supported:
#u-boot的编译程序支持将产生的对象文件放到一个独立的外部目录中,有下面两种使用方式:
# 1) Add O= to the make command line 在make命令行中使用变量并赋值O=
# 'make O=/tmp/build all'
# 2) Set environement variable BUILD_DIR to point to the desired location 设置环境变量指向希望存放的位置
# 'export BUILD_DIR=/tmp/build'
# 'make'
# The second approach can also be used with a MAKEALL script
# 'export BUILD_DIR=/tmp/build'
# './MAKEALL'
# Command line 'O=' setting overrides BUILD_DIR environent variable. 命令行的变量设置会重载 BUILD_DIR环境变量
# When none of the above methods is used the local build is performed and 当本地编译执行的时候,上面两种方法都
# the object files are placed in the source directory. 没有使用的时候,编译产生的目标文件被放到源码目录。
定义源码及生成的目标文件存放的目录,目标文件存放目录BUILD_DIR可以通过make O=dir 指定。如果没有指定,则设定为源码顶层目录。一般编译的时候不指定输出目录,则BUILD_DIR为空。
ifdef O
ifeq ("$(origin O)", "command line")
BUILD_DIR := $(O)
endif
endif
使用一个条件判断,判断O这个变量有没有被定义,如果定义了,则进一步判断,这个O的定义是从哪里来的,$(origin O)就是来得到O的定义的来源,如果是命令行,则采用,这里说明了命令行O变量会重载环境变量中对BUILD_DIR的赋值。
ifneq ($(BUILD_DIR),)
saved-output := $(BUILD_DIR)
# Attempt to create a output directory.
$(shell [ -d ${BUILD_DIR} ] || mkdir -p ${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),)
如果BUILD_DIR变量不为空,则给另外一个变量saved-output赋值。执行shell的时候,用了一个||符号,这个符号的功能与C语言里的||符号功能很相似,如果左边的值为真,则不执行后面的操作。所以这里就是先用-d选项判断${BUILD_DIR}这个目录存在与否,如果存在,就不执行后面的建目录的命令,如果不存在,则建目录。[ ] 部分是判断表达式,-d 表示判断是否是目录(directory)。
接下来这一段就是执行cd命令,进入到新建的目录里,然后执行pwd命令来得到当前目录的真实位置。为什么需要这样做呢,因为前面的创建目录工作可能不成功,所以导致后面的cd命令也没有进去,所以需要后面的pwd命令来确认一下。接下来的if用了一个三段式的形式,#if(a,b,c)这样的形式,执行步骤为,先判断a的真假,如果为真,则执行b,如果为假,则执行c。所以这里的意思就是判断目录建成没有建成,如果建成,则什么也不干,没建成,就使用error,输出错误信息且退出。
OBJTREE := $(if $(BUILD_DIR),$(BUILD_DIR),$(CURDIR))
SRCTREE := $(CURDIR)
TOPDIR := $(SRCTREE)
LNDIR := $(OBJTREE)
普通的变量赋值,OBJTREE和LNDIR为存放生成文件的目录,TOPDIR与SRCTREE为源码所在目录
export TOPDIR SRCTREE OBJTREE
MKCONFIG := $(SRCTREE)/mkconfig #定义变量MKCONFIG:这个变量指向一个脚本,即顶层目录的mkconfig
export MKCONFIG
ifneq ($(OBJTREE),$(SRCTREE)) #如果输出目录与源码目录不等,则设定REMOTE_BUILD项,并导出
REMOTE_BUILD := 1
export REMOTE_BUILD
endif
# $(obj) and (src) are defined in config.mk but here in main Makefile we also need them before config.mk is included #which is the case for some targets like unconfig, clean, clobber, distclean, etc.
ifneq ($(OBJTREE),$(SRCTREE))
obj := $(OBJTREE)/
src := $(SRCTREE)/
else
obj :=
src :=
endif
export obj src
如果输出目录OBJTREE与SRCTREE即当前的Uboot目录不等的话,对obj和src进行赋值,否则则obj,src为空。
obj src会在主目录中的config.mk定义,但在主makefile包含config.mk之前也需要,譬如unconfig, clean, clobber, distclean
#########################################################################
# $(OBJTREE)/include/config.mk文件是在make <board_name>_config的时候创建的
ifeq ($(OBJTREE)/include/config.mk,$(wildcard $(OBJTREE)/include/config.mk))
判断$(OBJTREE)/include/config.mk与$(wildcard $(OBJTREE)/include/config.mk)得到的值是否一样。$(wildcard PATTERN)的使用就是查找到与PATTERN相符合的,并且是存在的,以空格分开的文件列表。说白了,这句话就是判断$(OBJTREE)/include/config.mk文件是否存在。
# load ARCH, BOARD, and CPU configuration
include $(OBJTREE)/include/config.mk
如果include/config.mk文件已存在,就从里面获取目标板构架信息否则就要执行者make xxx_config来生成该文件。开头一句中的include,它是把后面跟的文件内容给加到它出现的地方,这里就是把$(OBJTREE)/include/config.mk文件的内容加入到了这里,所以就得到了ARCH,BOARD与CPU的值,如果VENDOR与SOC也定义了,也会得到它们的值。
export ARCH CPU BOARD VENDOR SOC
ifndef CROSS_COMPILE
ifeq ($(HOSTARCH),ppc)
CROSS_COMPILE =
else
ifeq ($(ARCH),ppc)
CROSS_COMPILE = powerpc-linux-
endif
ifeq ($(ARCH),arm)
CROSS_COMPILE = /opt/EmbedSky/crosstools_3.4.5_softfloat/gcc-3.4.5-glibc-2.3.6/arm-linux/bin/arm-linux-
endif
endif
endif
#上面的一大段,就是在判断没有定义CROSS_COMPILE变量的情况下,根据各种情况对这个变量的值进行判断。这个值就是交叉编译工具的开头部分
#的名称。建议在编译的时候,还是把这个变量的值给置上,置上当前系统里交叉编译工具相应的值,因为很难指望这些个判断就能把你现在系统里的值#给判断对了。置变量的方法有很多,其中一个,就是在make命令后跟上变量的赋值,例如:make CROSS_COMPILE=arm-linux-gnueabi-,或是把#CROSS_COMPILE设成一个系统变量,然后直接make CROSS_COMPILE=${CROSS_COMPILE}-这样来设置。
export CROSS_COMPILE
# load other configuration
include $(TOPDIR)/config.mk
#这里把$(TOPDIR)/config.mk这个文件的内容包含进来了,即包含顶层目录下的config.mk,这个文件里面主要定义了交叉编译器及选项和编译规则,这#样其中的大部分变量都在编译过程中有固定的值了(sinclude,它也是把后面所跟的#文件给包含进来。它与include的区别就是,include后面跟的文件,一定是存在的,如果不存在,执行就会出错误;#那么sinclude后面所跟的文件,可以不存在,如果文件存在则包含进来,如果不存在,则什么也不做。)
#########################################################################
# U-Boot objects....order is important (i.e. start must be first)
OBJS = cpu/$(CPU)/start.o
ifeq ($(CPU),i386)
OBJS += cpu/$(CPU)/start16.o
OBJS += cpu/$(CPU)/reset.o
endif
ifeq ($(CPU),ppc4xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc83xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc85xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc86xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),bf533)
OBJS += cpu/$(CPU)/start1.o cpu/$(CPU)/interrupt.o cpu/$(CPU)/cache.o
OBJS += cpu/$(CPU)/cplbhdlr.o cpu/$(CPU)/cplbmgr.o cpu/$(CPU)/flush.o
endif
#U-boot需要的目标文件,顺序很重要,start.o必须放第一位,针对不同架构,安排目标文件的布局,对于不同的CPU追加相应的目标文件。这段代码,#其实也是定义变量,根据不同的条件,给变量定义不同的值。这些变量都是在后面编译的时候,写到运行命令参数里面的,这些变量的意义,还有各种#参数的意义,可以查看相应的工具的参数手册来得到,跟移植没有什么太大的关系。需要注意一下各个变量值所添加的顺序,第一个添加的,编译的时###候,变量进行替换,就会被放在前面,然后就会被首先编译,连接,以此类推。可以看到,最先执行的就是start.o这个文件里的内容,这个文件是由#start.S来生成的。
OBJS := $(addprefix $(obj),$(OBJS))
#为“ OBJS…”中的每一个文件名添加前缀“ obj”。参数“ OBJS…”是空格分割的文件名序列,将“ obj”添加到此序列的每一个文件名之前。
LIBS = lib_generic/libgeneric.a
LIBS += board/$(BOARDDIR)/lib$(BOARD).a
LIBS += cpu/$(CPU)/lib$(CPU).a
ifdef SOC
LIBS += cpu/$(CPU)/$(SOC)/lib$(SOC).a
endif
LIBS += lib_$(ARCH)/lib$(ARCH).a
LIBS += fs/cramfs/libcramfs.a fs/jffs2/libjffs2.a
LIBS += net/libnet.a
LIBS += rtc/librtc.a
LIBS += drivers/libdrivers.a
LIBS += drivers/nand/libnand.a
LIBS += drivers/nand_legacy/libnand_legacy.a
LIBS += drivers/lcd/liblcd.a
LIBS += modules/usb.module
#LIBS += modules/wince.module
LIBS += common/libcommon.a
LIBS += $(BOARDLIBS)
##所依赖的库文件
LIBS := $(addprefix $(obj),$(LIBS)) ##给所有库文件增加前缀obj
#将一个目标声明为伪目标的方法是将它作为特殊目标.PHONY的依赖。这样目标“$(LIBS)”就被声明为一个伪目标,无论在当前目录下是否存 #在“ $(LIBS)”这个文件。我们输入“ make $(LIBS)”之后。对应的命令都会被执行。而且,当一个目标被声明为伪目标后, make 在执行此规则时不 #会试图去查找隐含规则来创建它。这样也提高了 make 的执行效率,同时也不用担心由于目标和文件名重名而使我们的期望失败。
.PHONY : $(LIBS)
# Add GCC lib
PLATFORM_LIBS += -L $(shell dirname `$(CC) $(CFLAGS) -print-libgcc-file-name`) -lgcc
#加入GCC的库,CC和CFLAGS都是make的隐含变量,CC表示C编译器,CFLAGS表示执行CC时的命令行参数。dirname从带路径的文件名中去掉文件##名, 只打印出路径信息
# The "tools" are needed early, so put this first
# Don't include stuff already done in $(LIBS)
SUBDIRS = tools \
examples \
post \
post/cpu
.PHONY : $(SUBDIRS)
#根据顶层makefile文件变量CONFIG_NAND_U_BOOT如果有定义则在include/config.mk文件中定义
ifeq ($(CONFIG_NAND_U_BOOT),y)
NAND_SPL = nand_spl
U_BOOT_NAND = $(obj)u-boot-nand.bin
endif
__OBJS := $(subst $(obj),,$(OBJS))
__LIBS := $(subst $(obj),,$(LIBS))
#字符替换函数$(subst FROM,TO,TEXT)把字串“ TEXT”中的“ FROM”字符替换为“ TO”。返回值:替换后的新字符串
#########################################################################
#########################################################################
#这里就是定义了各种目标的target。我们最终想要的目标就是$(obj)u-boot,然后看后面的,它都依赖了好多其它的目标,然后这些个目标,也是在此##处定义的,然后有相应的$(MAKE)来执行相应的操作,所以这样就实现了一个Makefile文件,套很多个其它的Makefile文件来编译整个工程的情况。#arm-linux-objcopy被用来复制一个目标文件的内容到另一个文件中,可用于不同源文件之间的格式转换OBJCOPY = $(CROSS_COMPILE)objcopy
ALL = $(obj)u-boot.srec $(obj)u-boot.bin $(obj)System.map $(U_BOOT_NAND)
all: $(ALL)
$(obj)u-boot.hex: $(obj)u-boot
$(OBJCOPY) ${OBJCFLAGS} -O ihex $< $@ #自动化变量“ $<”代表规则的依赖,“ $@”代表规则的目标。
$(obj)u-boot.srec: $(obj)u-boot
$(OBJCOPY) ${OBJCFLAGS} -O srec $< $@
$(obj)u-boot.bin: $(obj)u-boot
$(OBJCOPY) ${OBJCFLAGS} -O binary $< $@
$(obj)u-boot.img: $(obj)u-boot.bin
./tools/mkimage -A $(ARCH) -T firmware -C none \
-a $(TEXT_BASE) -e 0 \
-n $(shell sed -n -e 's/.*U_BOOT_VERSION//p' $(VERSION_FILE) | \
sed -e 's/"[ ]*$$/ for $(BOARD) board"/') \
-d $< $@
$(obj)u-boot.dis: $(obj)u-boot
$(OBJDUMP) -d $< > $@
$(obj)u-boot: depend version $(SUBDIRS) $(OBJS) $(LIBS) $(LDSCRIPT)
UNDEF_SYM=`$(OBJDUMP) -x $(LIBS) |sed -n -e 's/.*\(__u_boot_cmd_.*\)/-u\1/p'|sort|uniq`;\
cd $(LNDIR) && $(LD) $(LDFLAGS) $$UNDEF_SYM $(__OBJS) \
--start-group $(__LIBS) --end-group $(PLATFORM_LIBS) \
-Map u-boot.map -o u-boot
#此处生成的是uboot的ELF文件镜像从cd执行链接命令,把start.o和各个子目录makefile生成的库文件按照LDFLAGS链接在一起,生成ELF文件u-boot和链接时内存分配图文件u-boot.map。而对于各个子目录的makefile文件,主要生成*.o文件然后执行AR生成对应的库文件。
$(OBJS):
echo $(OBJS)
$(MAKE) -C cpu/$(CPU) $(if $(REMOTE_BUILD),$@,$(notdir $@))
$(LIBS):
$(MAKE) -C $(dir $(subst $(obj),,$@))
#依赖目标$(LIBS),目标很多,都是每个子目录的库文件*.a,通过执行相应子目录下的make来完成
#$(dir names...)
#抽取‘names’中每一个文件名的路径部分,文件名的路径部分包括从文件名的开始到最后一个斜杠(含斜杠)
#之前的一切字符。如果文件名中没有斜杠,路径部分是‘./’。如:
#$(dir src/foo.c hacks)
#产生的结果为 ‘src/ ./’。
#这个目标太多,都是每个子目录的库文件*.a ,通过执行相应子目录下的make来完成
lcd:
$(MAKE) -C drivers/lcd
#执行 tools ,examples ,post,post\cpu 子目录下面的 make文件
$(SUBDIRS):
$(MAKE) -C $@ all
$(NAND_SPL): version
$(MAKE) -C nand_spl/board/$(BOARDDIR) all
$(U_BOOT_NAND): $(NAND_SPL) $(obj)u-boot.bin
cat $(obj)nand_spl/u-boot-spl-16k.bin $(obj)u-boot.bin > $(obj)u-boot-nand.bin
version:
@echo -n "#define U_BOOT_VERSION \"U-Boot " > $(VERSION_FILE); \
echo -n "$(U_BOOT_VERSION)" >> $(VERSION_FILE); \
echo -n $(shell $(CONFIG_SHELL) $(TOPDIR)/tools/setlocalversion \
$(TOPDIR)) >> $(VERSION_FILE); \
echo "\"" >> $(VERSION_FILE)
#生成版本信息到版本文件VERSION_FILE中
gdbtools:
$(MAKE) -C tools/gdb all || exit 1
updater:
$(MAKE) -C tools/updater all || exit 1
env:
$(MAKE) -C tools/env all || exit 1
depend dep:
for dir in $(SUBDIRS) ; do $(MAKE) -C $$dir _depend ; done
#生成各个子目录的.depend文件,.depend列出每个目标文件的依赖文件。生成方法,调用每个子目录的 make _depend。
tags ctags:
ctags -w -o $(OBJTREE)/ctags `find $(SUBDIRS) include \
lib_generic board/$(BOARDDIR) cpu/$(CPU) lib_$(ARCH) \
fs/cramfs fs/jffs2 \
net rtc drivers wince common \
\( -name CVS -prune \) -o \( -name '*.[ch]' -print \)`
etags:
etags -a -o $(OBJTREE)/etags `find $(SUBDIRS) include \
lib_generic board/$(BOARDDIR) cpu/$(CPU) lib_$(ARCH) \
fs/cramfs fs/jffs2 \
net rtc drivers wince common \
\( -name CVS -prune \) -o \( -name '*.[ch]' -print \)`
$(obj)System.map: $(obj)u-boot
@$(NM) $< | \
grep -v '\(compiled\)\|\(\.o$$\)\|\( [aUw] \)\|\(\.\.ng$$\)\|\(LASH[RL]DI\)' | \
sort > $(obj)System.map
#########################################################################
#这里就明说了,如果$(OBJTREE)/include/config.mk文件不存在,则编译不能进行下去了。
else
all $(obj)u-boot.hex $(obj)u-boot.srec $(obj)u-boot.bin \
$(obj)u-boot.img $(obj)u-boot.dis $(obj)u-boot \
$(SUBDIRS) version gdbtools updater env depend \
dep tags ctags etags $(obj)System.map:
@echo "System not configured - see README" >&2
@ exit 1
endif
.PHONY : CHANGELOG
CHANGELOG:
git log --no-merges U-Boot-1_1_5.. | \
unexpand -a | sed -e 's/\s\s*$$//' > $@
#########################################################################
unconfig:
@rm -f $(obj)include/config.h $(obj)include/config.mk \
$(obj)board/*/config.tmp $(obj)board/*/*/config.tmp
#删掉了包括$(obj)include/config.mk文件在内的,几个在编译的时候生成的配置文件,这
#个$(obj)include/config.mk与$(OBJTREE)/include/config.mk是同一个文件。
#.................................................................................................................................................
#......................................................................省略其它不相关的体系结构配置..........................
#...................................................................................................................................................
sbc2410x_config: unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t sbc2410x NULL s3c24x0
scb9328_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t scb9328 NULL imx
smdk2400_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t smdk2400 NULL s3c24x0
smdk2410_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t smdk2410 NULL s3c24x0
EmbedSky_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t EmbedSky NULL s3c24x0
SX1_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm925t sx1
#$(a:patternA=patternB),这样的语法表示把a变量里的形式为patternA的换成为patternB,然后输出。@$(MKCONFIG) $(@:_config=) arm arm920t smdk2410 NULL s3c24x0根据前面的分析就是执行./mkconfig smdk2410 arm arm920t smdk2410 NULL s3c24x0
clean:
find $(OBJTREE) -type f \
\( -name 'core' -o -name '*.bak' -o -name '*~' \
-o -name '*.o' -o -name '*.a' \) -print \
| xargs rm -f
rm -f $(obj)examples/hello_world $(obj)examples/timer \
$(obj)examples/eepro100_eeprom $(obj)examples/sched \
$(obj)examples/mem_to_mem_idma2intr $(obj)examples/82559_eeprom \
$(obj)examples/smc91111_eeprom $(obj)examples/interrupt \
$(obj)examples/test_burst
rm -f $(obj)tools/img2srec $(obj)tools/mkimage $(obj)tools/envcrc \
$(obj)tools/gen_eth_addr
rm -f $(obj)tools/mpc86x_clk $(obj)tools/ncb
rm -f $(obj)tools/easylogo/easylogo $(obj)tools/bmp_logo
rm -f $(obj)tools/gdb/astest $(obj)tools/gdb/gdbcont $(obj)tools/gdb/gdbsend
rm -f $(obj)tools/env/fw_printenv $(obj)tools/env/fw_setenv
rm -f $(obj)board/cray/L1/bootscript.c $(obj)board/cray/L1/bootscript.image
rm -f $(obj)board/netstar/eeprom $(obj)board/netstar/crcek $(obj)board/netstar/crcit
rm -f $(obj)board/netstar/*.srec $(obj)board/netstar/*.bin
rm -f $(obj)board/trab/trab_fkt $(obj)board/voiceblue/eeprom
rm -f $(obj)board/integratorap/u-boot.lds $(obj)board/integratorcp/u-boot.lds
rm -f $(obj)include/bmp_logo.h
rm -f $(obj)nand_spl/u-boot-spl $(obj)nand_spl/u-boot-spl.map
clobber: clean
find $(OBJTREE) -type f \( -name .depend \
-o -name '*.srec' -o -name '*.bin' -o -name u-boot.dis -o -name u-boot.img \) \
-print0 \
| xargs -0 rm -f
rm -f $(OBJS) $(obj)*.bak $(obj)ctags $(obj)etags $(obj)TAGS $(obj)include/version_autogenerated.h
rm -fr $(obj)*.*~
rm -f $(obj)u-boot $(obj)u-boot.map $(obj)u-boot.hex $(ALL)
rm -f $(obj)tools/crc32.c $(obj)tools/environment.c $(obj)tools/env/crc32.c
rm -f $(obj)tools/inca-swap-bytes $(obj)cpu/mpc824x/bedbug_603e.c
rm -f $(obj)include/asm/proc $(obj)include/asm/arch $(obj)include/asm
[ ! -d $(OBJTREE)/nand_spl ] || find $(obj)nand_spl -lname "*" -print | xargs rm -f
ifeq ($(OBJTREE),$(SRCTREE))
mrproper \
distclean: clobber unconfig
else
mrproper \
distclean: clobber unconfig
rm -rf $(OBJTREE)/*
endif
backup:
F=`basename $(TOPDIR)` ; cd .. ; \
gtar --force-local -zcvf `date "+$$F-%Y-%m-%d-%T.tar.gz"` $$F
#########################################################################