diff options
Diffstat (limited to 'arch/arm/lib')
| -rw-r--r-- | arch/arm/lib/Makefile | 2 | ||||
| -rw-r--r-- | arch/arm/lib/bootm.c | 27 | ||||
| -rw-r--r-- | arch/arm/lib/relocate.S | 61 | ||||
| -rw-r--r-- | arch/arm/lib/sections.c (renamed from arch/arm/lib/bss.c) | 8 | 
4 files changed, 40 insertions, 58 deletions
diff --git a/arch/arm/lib/Makefile b/arch/arm/lib/Makefile index 8ad9f66a5..9ecafb272 100644 --- a/arch/arm/lib/Makefile +++ b/arch/arm/lib/Makefile @@ -43,7 +43,7 @@ SOBJS-y += relocate.o  ifndef CONFIG_SYS_GENERIC_BOARD  COBJS-y	+= board.o  endif -COBJS-y += bss.o +COBJS-y += sections.o  COBJS-y	+= bootm.o  COBJS-$(CONFIG_OF_LIBFDT) += bootm-fdt.o diff --git a/arch/arm/lib/bootm.c b/arch/arm/lib/bootm.c index 1b6e0ace4..b22fbc998 100644 --- a/arch/arm/lib/bootm.c +++ b/arch/arm/lib/bootm.c @@ -68,12 +68,19 @@ void arch_lmb_reserve(struct lmb *lmb)  		    gd->bd->bi_dram[0].start + gd->bd->bi_dram[0].size - sp);  } -static void announce_and_cleanup(void) +/** + * announce_and_cleanup() - Print message and prepare for kernel boot + * + * @fake: non-zero to do everything except actually boot + */ +static void announce_and_cleanup(int fake)  { -	printf("\nStarting kernel ...\n\n"); +	printf("\nStarting kernel ...%s\n\n", fake ? +		"(fake run for tracing)" : "");  	bootstage_mark_name(BOOTSTAGE_ID_BOOTM_HANDOFF, "start_kernel");  #ifdef CONFIG_BOOTSTAGE_FDT -	bootstage_fdt_add_report(); +	if (flag == BOOTM_STATE_OS_FAKE_GO) +		bootstage_fdt_add_report();  #endif  #ifdef CONFIG_BOOTSTAGE_REPORT  	bootstage_report(); @@ -225,12 +232,13 @@ static void boot_prep_linux(bootm_headers_t *images)  }  /* Subcommand: GO */ -static void boot_jump_linux(bootm_headers_t *images) +static void boot_jump_linux(bootm_headers_t *images, int flag)  {  	unsigned long machid = gd->bd->bi_arch_number;  	char *s;  	void (*kernel_entry)(int zero, int arch, uint params);  	unsigned long r2; +	int fake = (flag & BOOTM_STATE_OS_FAKE_GO);  	kernel_entry = (void (*)(int, int, uint))images->ep; @@ -243,14 +251,15 @@ static void boot_jump_linux(bootm_headers_t *images)  	debug("## Transferring control to Linux (at address %08lx)" \  		"...\n", (ulong) kernel_entry);  	bootstage_mark(BOOTSTAGE_ID_RUN_OS); -	announce_and_cleanup(); +	announce_and_cleanup(fake);  	if (IMAGE_ENABLE_OF_LIBFDT && images->ft_len)  		r2 = (unsigned long)images->ft_addr;  	else  		r2 = gd->bd->bi_boot_params; -	kernel_entry(0, machid, r2); +	if (!fake) +		kernel_entry(0, machid, r2);  }  /* Main Entry point for arm bootm implementation @@ -270,13 +279,13 @@ int do_bootm_linux(int flag, int argc, char *argv[], bootm_headers_t *images)  		return 0;  	} -	if (flag & BOOTM_STATE_OS_GO) { -		boot_jump_linux(images); +	if (flag & (BOOTM_STATE_OS_GO | BOOTM_STATE_OS_FAKE_GO)) { +		boot_jump_linux(images, flag);  		return 0;  	}  	boot_prep_linux(images); -	boot_jump_linux(images); +	boot_jump_linux(images, flag);  	return 0;  } diff --git a/arch/arm/lib/relocate.S b/arch/arm/lib/relocate.S index 4446da94c..949b9e802 100644 --- a/arch/arm/lib/relocate.S +++ b/arch/arm/lib/relocate.S @@ -37,56 +37,34 @@   */  ENTRY(relocate_code) -	mov	r6, r0	/* save addr of destination */ - -	ldr	r0, =_start		/* r0 <- SRC &_start */ -	subs	r9, r6, r0		/* r9 <- relocation offset */ +	ldr	r1, =__image_copy_start	/* r1 <- SRC &__image_copy_start */ +	subs	r9, r0, r1		/* r9 <- relocation offset */  	beq	relocate_done		/* skip relocation */ -	mov	r1, r6			/* r1 <- scratch for copy loop */ -	adr	r7, relocate_code	/* r7 <- SRC &relocate_code */ -	ldr	r3, _image_copy_end_ofs	/* r3 <- __image_copy_end local ofs */ -	add	r2, r7, r3		/* r2 <- SRC &__image_copy_end */ +	ldr	r2, =__image_copy_end	/* r2 <- SRC &__image_copy_end */  copy_loop: -	ldmia	r0!, {r10-r11}		/* copy from source address [r0]    */ -	stmia	r1!, {r10-r11}		/* copy to   target address [r1]    */ -	cmp	r0, r2			/* until source end address [r2]    */ +	ldmia	r1!, {r10-r11}		/* copy from source address [r1]    */ +	stmia	r0!, {r10-r11}		/* copy to   target address [r0]    */ +	cmp	r1, r2			/* until source end address [r2]    */  	blo	copy_loop  	/*  	 * fix .rel.dyn relocations  	 */ -	ldr	r10, _dynsym_start_ofs	/* r10 <- __dynsym_start local ofs */ -	add	r10, r10, r7		/* r10 <- SRC &__dynsym_start */ -	ldr	r2, _rel_dyn_start_ofs	/* r2 <- __rel_dyn_start local ofs */ -	add	r2, r2, r7		/* r2 <- SRC &__rel_dyn_start */ -	ldr	r3, _rel_dyn_end_ofs	/* r3 <- __rel_dyn_end local ofs */ -	add	r3, r3, r7		/* r3 <- SRC &__rel_dyn_end */ +	ldr	r2, =__rel_dyn_start	/* r2 <- SRC &__rel_dyn_start */ +	ldr	r3, =__rel_dyn_end	/* r3 <- SRC &__rel_dyn_end */  fixloop: -	ldr	r0, [r2]		/* r0 <- SRC location to fix up */ -	add	r0, r0, r9		/* r0 <- DST location to fix up */ -	ldr	r1, [r2, #4] -	and	r7, r1, #0xff -	cmp	r7, #23			/* relative fixup? */ -	beq	fixrel -	cmp	r7, #2			/* absolute fixup? */ -	beq	fixabs -	/* ignore unknown type of fixup */ -	b	fixnext -fixabs: -	/* absolute fix: set location to (offset) symbol value */ -	mov	r1, r1, LSR #4		/* r1 <- symbol index in .dynsym */ -	add	r1, r10, r1		/* r1 <- address of symbol in table */ -	ldr	r1, [r1, #4]		/* r1 <- symbol value */ -	add	r1, r1, r9		/* r1 <- relocated sym addr */ -	b	fixnext -fixrel: +	ldmia	r2!, {r0-r1}		/* (r0,r1) <- (SRC location,fixup) */ +	and	r1, r1, #0xff +	cmp	r1, #23			/* relative fixup? */ +	bne	fixnext +  	/* relative fix: increase location by offset */ +	add	r0, r0, r9  	ldr	r1, [r0]  	add	r1, r1, r9 -fixnext:  	str	r1, [r0] -	add	r2, r2, #8		/* each rel.dyn entry is 8 bytes */ +fixnext:  	cmp	r2, r3  	blo	fixloop @@ -100,13 +78,4 @@ relocate_done:          bx        lr  #endif -_image_copy_end_ofs: -	.word __image_copy_end - relocate_code -_rel_dyn_start_ofs: -	.word __rel_dyn_start - relocate_code -_rel_dyn_end_ofs: -	.word __rel_dyn_end - relocate_code -_dynsym_start_ofs: -	.word __dynsym_start - relocate_code -  ENDPROC(relocate_code) diff --git a/arch/arm/lib/bss.c b/arch/arm/lib/sections.c index 99eda5913..5921dd8d6 100644 --- a/arch/arm/lib/bss.c +++ b/arch/arm/lib/sections.c @@ -35,5 +35,9 @@   * aliasing warnings.   */ -char __bss_start[0] __attribute__((used, section(".__bss_start"))); -char __bss_end[0] __attribute__((used, section(".__bss_end"))); +char __bss_start[0] __attribute__((section(".__bss_start"))); +char __bss_end[0] __attribute__((section(".__bss_end"))); +char __image_copy_start[0] __attribute__((section(".__image_copy_start"))); +char __image_copy_end[0] __attribute__((section(".__image_copy_end"))); +char __rel_dyn_start[0] __attribute__((section(".__rel_dyn_start"))); +char __rel_dyn_end[0] __attribute__((section(".__rel_dyn_end")));  |