123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- /*
- * Copyright (C) Paul Mackerras 1997.
- *
- * Adapted for 64 bit LE PowerPC by Andrew Tauferner
- *
- * 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.
- *
- */
- #include "ppc_asm.h"
- RELA = 7
- RELACOUNT = 0x6ffffff9
- .data
- /* A procedure descriptor used when booting this as a COFF file.
- * When making COFF, this comes first in the link and we're
- * linked at 0x500000.
- */
- .globl _zimage_start_opd
- _zimage_start_opd:
- .long 0x500000, 0, 0, 0
- .text
- b _zimage_start
- #ifdef __powerpc64__
- .balign 8
- p_start: .llong _start
- p_etext: .llong _etext
- p_bss_start: .llong __bss_start
- p_end: .llong _end
- p_toc: .llong __toc_start + 0x8000 - p_base
- p_dyn: .llong __dynamic_start - p_base
- p_rela: .llong __rela_dyn_start - p_base
- p_prom: .llong 0
- .weak _platform_stack_top
- p_pstack: .llong _platform_stack_top
- #else
- p_start: .long _start
- p_etext: .long _etext
- p_bss_start: .long __bss_start
- p_end: .long _end
- .weak _platform_stack_top
- p_pstack: .long _platform_stack_top
- #endif
- .globl _zimage_start
- /* Clang appears to require the .weak directive to be after the symbol
- * is defined. See https://bugs.llvm.org/show_bug.cgi?id=38921 */
- .weak _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 .+4
- p_base: mflr r10 /* r10 now points to runtime addr of p_base */
- #ifndef __powerpc64__
- /* grab the link address of the dynamic section in r11 */
- addis r11,r10,(_GLOBAL_OFFSET_TABLE_-p_base)@ha
- lwz r11,(_GLOBAL_OFFSET_TABLE_-p_base)@l(r11)
- cmpwi r11,0
- beq 3f /* if not linked -pie */
- /* get the runtime address of the dynamic section in r12 */
- .weak __dynamic_start
- addis r12,r10,(__dynamic_start-p_base)@ha
- addi r12,r12,(__dynamic_start-p_base)@l
- subf r11,r11,r12 /* runtime - linktime offset */
- /* The dynamic section contains a series of tagged entries.
- * We need the RELA and RELACOUNT entries. */
- li r9,0
- li r0,0
- 9: lwz r8,0(r12) /* get tag */
- cmpwi r8,0
- beq 10f /* end of list */
- cmpwi r8,RELA
- bne 11f
- lwz r9,4(r12) /* get RELA pointer in r9 */
- b 12f
- 11: addis r8,r8,(-RELACOUNT)@ha
- cmpwi r8,RELACOUNT@l
- bne 12f
- lwz r0,4(r12) /* get RELACOUNT value in r0 */
- 12: addi r12,r12,8
- b 9b
- /* The relocation section contains a list of relocations.
- * We now do the R_PPC_RELATIVE ones, which point to words
- * which need to be initialized with addend + offset.
- * The R_PPC_RELATIVE ones come first and there are RELACOUNT
- * of them. */
- 10: /* skip relocation if we don't have both */
- cmpwi r0,0
- beq 3f
- cmpwi r9,0
- beq 3f
- add r9,r9,r11 /* Relocate RELA pointer */
- mtctr r0
- 2: lbz r0,4+3(r9) /* ELF32_R_INFO(reloc->r_info) */
- cmpwi r0,22 /* R_PPC_RELATIVE */
- bne 3f
- lwz r12,0(r9) /* reloc->r_offset */
- lwz r0,8(r9) /* reloc->r_addend */
- add r0,r0,r11
- stwx r0,r11,r12
- addi r9,r9,12
- bdnz 2b
- /* Do a cache flush for our text, in case the loader didn't */
- 3: lwz r9,p_start-p_base(r10) /* note: these are relocated now */
- lwz r8,p_etext-p_base(r10)
- 4: dcbf r0,r9
- icbi r0,r9
- addi r9,r9,0x20
- cmplw cr0,r9,r8
- blt 4b
- sync
- isync
- /* Clear the BSS */
- lwz r9,p_bss_start-p_base(r10)
- lwz r8,p_end-p_base(r10)
- li r0,0
- 5: stw r0,0(r9)
- addi r9,r9,4
- cmplw cr0,r9,r8
- blt 5b
- /* Possibly set up a custom stack */
- lwz r8,p_pstack-p_base(r10)
- cmpwi r8,0
- beq 6f
- lwz r1,0(r8)
- li r0,0
- stwu r0,-16(r1) /* establish a stack frame */
- 6:
- #else /* __powerpc64__ */
- /* Save the prom pointer at p_prom. */
- std r5,(p_prom-p_base)(r10)
- /* Set r2 to the TOC. */
- ld r2,(p_toc-p_base)(r10)
- add r2,r2,r10
- /* Grab the link address of the dynamic section in r11. */
- ld r11,-32768(r2)
- cmpwi r11,0
- beq 3f /* if not linked -pie then no dynamic section */
- ld r11,(p_dyn-p_base)(r10)
- add r11,r11,r10
- ld r9,(p_rela-p_base)(r10)
- add r9,r9,r10
- li r13,0
- li r8,0
- 9: ld r12,0(r11) /* get tag */
- cmpdi r12,0
- beq 12f /* end of list */
- cmpdi r12,RELA
- bne 10f
- ld r13,8(r11) /* get RELA pointer in r13 */
- b 11f
- 10: addis r12,r12,(-RELACOUNT)@ha
- cmpdi r12,RELACOUNT@l
- bne 11f
- ld r8,8(r11) /* get RELACOUNT value in r8 */
- 11: addi r11,r11,16
- b 9b
- 12:
- cmpdi r13,0 /* check we have both RELA and RELACOUNT */
- cmpdi cr1,r8,0
- beq 3f
- beq cr1,3f
- /* Calcuate the runtime offset. */
- subf r13,r13,r9
- /* Run through the list of relocations and process the
- * R_PPC64_RELATIVE ones. */
- mtctr r8
- 13: ld r0,8(r9) /* ELF64_R_TYPE(reloc->r_info) */
- cmpdi r0,22 /* R_PPC64_RELATIVE */
- bne 3f
- ld r12,0(r9) /* reloc->r_offset */
- ld r0,16(r9) /* reloc->r_addend */
- add r0,r0,r13
- stdx r0,r13,r12
- addi r9,r9,24
- bdnz 13b
- /* Do a cache flush for our text, in case the loader didn't */
- 3: ld r9,p_start-p_base(r10) /* note: these are relocated now */
- ld r8,p_etext-p_base(r10)
- 4: dcbf r0,r9
- icbi r0,r9
- addi r9,r9,0x20
- cmpld cr0,r9,r8
- blt 4b
- sync
- isync
- /* Clear the BSS */
- ld r9,p_bss_start-p_base(r10)
- ld r8,p_end-p_base(r10)
- li r0,0
- 5: std r0,0(r9)
- addi r9,r9,8
- cmpld cr0,r9,r8
- blt 5b
- /* Possibly set up a custom stack */
- ld r8,p_pstack-p_base(r10)
- cmpdi r8,0
- beq 6f
- ld r1,0(r8)
- li r0,0
- stdu r0,-112(r1) /* establish a stack frame */
- 6:
- #endif /* __powerpc64__ */
- /* Call platform_init() */
- bl platform_init
- /* Call start */
- b start
- #ifdef __powerpc64__
- #define PROM_FRAME_SIZE 512
- #define SAVE_GPR(n, base) std n,8*(n)(base)
- #define REST_GPR(n, base) ld n,8*(n)(base)
- #define SAVE_2GPRS(n, base) SAVE_GPR(n, base); SAVE_GPR(n+1, base)
- #define SAVE_4GPRS(n, base) SAVE_2GPRS(n, base); SAVE_2GPRS(n+2, base)
- #define SAVE_8GPRS(n, base) SAVE_4GPRS(n, base); SAVE_4GPRS(n+4, base)
- #define SAVE_10GPRS(n, base) SAVE_8GPRS(n, base); SAVE_2GPRS(n+8, base)
- #define REST_2GPRS(n, base) REST_GPR(n, base); REST_GPR(n+1, base)
- #define REST_4GPRS(n, base) REST_2GPRS(n, base); REST_2GPRS(n+2, base)
- #define REST_8GPRS(n, base) REST_4GPRS(n, base); REST_4GPRS(n+4, base)
- #define REST_10GPRS(n, base) REST_8GPRS(n, base); REST_2GPRS(n+8, base)
- /* prom handles the jump into and return from firmware. The prom args pointer
- is loaded in r3. */
- .globl prom
- prom:
- mflr r0
- std r0,16(r1)
- stdu r1,-PROM_FRAME_SIZE(r1) /* Save SP and create stack space */
- SAVE_GPR(2, r1)
- SAVE_GPR(13, r1)
- SAVE_8GPRS(14, r1)
- SAVE_10GPRS(22, r1)
- mfcr r10
- std r10,8*32(r1)
- mfmsr r10
- std r10,8*33(r1)
- /* remove MSR_LE from msr but keep MSR_SF */
- mfmsr r10
- rldicr r10,r10,0,62
- mtsrr1 r10
- /* Load FW address, set LR to label 1, and jump to FW */
- bl 0f
- 0: mflr r10
- addi r11,r10,(1f-0b)
- mtlr r11
- ld r10,(p_prom-0b)(r10)
- mtsrr0 r10
- rfid
- 1: /* Return from OF */
- FIXUP_ENDIAN
- /* Restore registers and return. */
- rldicl r1,r1,0,32
- /* Restore the MSR (back to 64 bits) */
- ld r10,8*(33)(r1)
- mtmsr r10
- isync
- /* Restore other registers */
- REST_GPR(2, r1)
- REST_GPR(13, r1)
- REST_8GPRS(14, r1)
- REST_10GPRS(22, r1)
- ld r10,8*32(r1)
- mtcr r10
- addi r1,r1,PROM_FRAME_SIZE
- ld r0,16(r1)
- mtlr r0
- blr
- #endif
|