|  | /* | 
|  | * Copyright (C) Paul Mackerras 1997. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License | 
|  | * as published by the Free Software Foundation; either version | 
|  | * 2 of the License, or (at your option) any later version. | 
|  | * | 
|  | * NOTE: this code runs in 32 bit mode and is packaged as ELF32. | 
|  | */ | 
|  |  | 
|  | #include "ppc_asm.h" | 
|  |  | 
|  | .text | 
|  | /* a procedure descriptor used when booting this as a COFF file */ | 
|  | .globl	_zimage_start_opd | 
|  | _zimage_start_opd: | 
|  | .long	_zimage_start, 0, 0, 0 | 
|  |  | 
|  | .weak	_zimage_start | 
|  | .globl	_zimage_start | 
|  | _zimage_start: | 
|  | .globl	_zimage_start_lib | 
|  | _zimage_start_lib: | 
|  | /* Work out the offset between the address we were linked at | 
|  | and the address where we're running. */ | 
|  | bl	1f | 
|  | 1:	mflr	r0 | 
|  | lis	r9,1b@ha | 
|  | addi	r9,r9,1b@l | 
|  | subf.	r0,r9,r0 | 
|  | beq	3f		/* if running at same address as linked */ | 
|  |  | 
|  | /* The .got2 section contains a list of addresses, so add | 
|  | the address offset onto each entry. */ | 
|  | lis	r9,__got2_start@ha | 
|  | addi	r9,r9,__got2_start@l | 
|  | lis	r8,__got2_end@ha | 
|  | addi	r8,r8,__got2_end@l | 
|  | subf.	r8,r9,r8 | 
|  | beq	3f | 
|  | srwi.	r8,r8,2 | 
|  | mtctr	r8 | 
|  | add	r9,r0,r9 | 
|  | 2:	lwz	r8,0(r9) | 
|  | add	r8,r8,r0 | 
|  | stw	r8,0(r9) | 
|  | addi	r9,r9,4 | 
|  | bdnz	2b | 
|  |  | 
|  | /* Do a cache flush for our text, in case the loader didn't */ | 
|  | 3:	lis	r9,_start@ha | 
|  | addi	r9,r9,_start@l | 
|  | add	r9,r0,r9 | 
|  | lis	r8,_etext@ha | 
|  | addi	r8,r8,_etext@l | 
|  | add	r8,r0,r8 | 
|  | 4:	dcbf	r0,r9 | 
|  | icbi	r0,r9 | 
|  | addi	r9,r9,0x20 | 
|  | cmplw	cr0,r9,r8 | 
|  | blt	4b | 
|  | sync | 
|  | isync | 
|  |  | 
|  | /* Clear the BSS */ | 
|  | lis	r9,__bss_start@ha | 
|  | addi	r9,r9,__bss_start@l | 
|  | add	r9,r0,r9 | 
|  | lis	r8,_end@ha | 
|  | addi	r8,r8,_end@l | 
|  | add	r8,r0,r8 | 
|  | li	r10,0 | 
|  | 5:	stw	r10,0(r9) | 
|  | addi	r9,r9,4 | 
|  | cmplw	cr0,r9,r8 | 
|  | blt	5b | 
|  |  | 
|  | /* Possibly set up a custom stack */ | 
|  | .weak	_platform_stack_top | 
|  | lis	r8,_platform_stack_top@ha | 
|  | addi	r8,r8,_platform_stack_top@l | 
|  | cmpwi	r8,0 | 
|  | beq	6f | 
|  | add	r8,r0,r8 | 
|  | lwz	r1,0(r8) | 
|  | add	r1,r0,r1 | 
|  | li	r0,0 | 
|  | stwu	r0,-16(r1)	/* establish a stack frame */ | 
|  | 6: | 
|  |  | 
|  | /* Call platform_init() */ | 
|  | bl	platform_init | 
|  |  | 
|  | /* Call start */ | 
|  | b	start |