解决Linux内核生成uImage时Entry Point自动偏移64个字节

时间:2021-07-29 07:33:36

解决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参数,不能作为内核启动入口地址。

解决Linux内核生成uImage时Entry Point自动偏移64个字节


解决加载地址和入口地址相同的方法:

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)
# 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
将77行的$(obj)/uImage: STARTADDR=$(LOADADDR)修改为

$(obj)/uImage: STARTADDR=$(shell echo $(LOADADDR) | sed -e "s/..$$/40/")


其中,sed -e "s/..$$/40/" 的意思是把输出的字符串的最后两个字符删掉,并且用40来补充,也可以理解为,把字符串最后两个字符用40来替换。这样,原来的入口地址0x30008000就改成了0x30008040,多了0x40,即64个字节。


然后执行make uImage 生成uImage就可以了。

解决Linux内核生成uImage时Entry Point自动偏移64个字节