解决Linux内核进行make uImage编译时,生成uImage Entry Point 自动偏移64个字节
make uImage 编译内核自动生成uImage 时,Entry Point 入口地址和Load Address 加载地址总是相同,都是Load Address:30008000、Entry Point:30008000。这样一来,内核解压完后,bootm之后就会出错,因为uImage 前面64个字节是u-boot参数,不能作为内核启动入口地址。
解决加载地址和入口地址相同的方法:
1、手动使用mkimage命令生成uImage。
mkimage -n 'linux-2.6.32' -A arm -O linux -T kernel -C none -a 0x30008000 -e 0x30008040 -d zImage uImage注意:要在arch/arm/boot 的目录下使用命令!
2、修改内核,自动编译成Entry Point 为0x30008040。
打开arch/arm/boot/Makefile 文件,可以发现62~65行就是利用mkimage生成uImage的相关命令。
quiet_cmd_uimage = UIMAGE $@
cmd_uimage = $(CONFIG_SHELL) $(MKIMAGE) -A arm -O linux -T kernel \
-C none -a $(LOADADDR) -e $(STARTADDR) \
-n 'Linux-$(KERNELRELEASE)' -d $< $@
其中LOADADDR 为加载地址,STARTADDR 为入口地址。
73~78行是对Entry Point入口地址的赋值 -- STARTADDR
ifeq ($(CONFIG_THUMB2_KERNEL),y)将77行的$(obj)/uImage: STARTADDR=$(LOADADDR)修改为
# Set bit 0 to 1 so that "mov pc, rx" switches to Thumb-2 mode
$(obj)/uImage: STARTADDR=$(shell echo $(LOADADDR) | sed -e "s/.$$/1/")
else
$(obj)/uImage: STARTADDR=$(LOADADDR)
endif
$(obj)/uImage: STARTADDR=$(shell echo $(LOADADDR) | sed -e "s/..$$/40/")
然后执行make uImage 生成uImage就可以了。