123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- /*
- * Copyright (C) 2003 by Erik Andersen
- *
- * This program is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Library General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or (at your
- * option) any later version.
- *
- * This program is distributed in the hope that it will be useful, but WITHOUT
- * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
- * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License
- * for more details.
- *
- * You should have received a copy of the GNU Library General Public License
- * along with this program; see the file COPYING.LIB. If not, see
- * <http://www.gnu.org/licenses/>.
- */
- #include <features.h>
- /* Integer registers. */
- #define r0 0
- #define r1 1
- #define r2 2
- #define r3 3
- #define r4 4
- #define r5 5
- #define r6 6
- #define r7 7
- #define r8 8
- #define r9 9
- #define r10 10
- #define r13 13
- #define r31 31
- .text
- .globl _start
- .type _start,%function
- .type _init,%function
- .type _fini,%function
- #ifndef __UCLIBC_CTOR_DTOR__
- .weak _init
- .weak _fini
- #endif
- #ifdef L_rcrt1
- .type reloc_static_pie,%function
- #endif
- .type main,%function
- .type __uClibc_main,%function
- _start:
- mr r9,r1 /* Save the stack pointer and pass it to __uClibc_main */
- clrrwi r1,r1,4 /* Align stack ptr to 16 bytes */
- #ifdef __PIC__
- # ifdef HAVE_ASM_PPC_REL16
- bcl 20,31,1f
- 1: mflr r31
- addis r31,r31,_GLOBAL_OFFSET_TABLE_-1b@ha
- addi r31,r31,_GLOBAL_OFFSET_TABLE_-1b@l
- # else
- bl _GLOBAL_OFFSET_TABLE_-4@local
- mflr r31
- # endif
- /* in PIC/PIE, plt stubs need r30 to point to the GOT if using secure-plt */
- # ifdef PPC_HAS_SECUREPLT
- mr 30,31
- # endif
- #ifdef L_rcrt1
- stwu r3, -4(r1) /* Save r3 */
- stwu r9, -16(r1) /* Save r9 */
- bcl 20,31,2f /* Jump to label 2 */
- 2: mflr r3 /* Load lr into r3 */
- addis r3, r3, _DYNAMIC-2b@ha /* Add high half of _DYNAMIC to r3 */
- addi r3,r3,_DYNAMIC-2b@l /* Add low half of _DYNAMIC */
- lwz r4, 0(r31) /* load _DYNAMIC from the GOT */
- subf r3, r4, r3 /* sub _DYNAMIC@got and it's actual address */
- bl reloc_static_pie /* Call reloc_static_pie */
- lwzu r9, 0(r1) /* restore r9 */
- addi r1, r1, 16 /* update stack pointer */
- lwzu r3, 0(r1) /* restore r3 */
- addi r1, r1, 4 /* update stack pointer */
- li r5, 0 /* zero r5 */
- #endif
- #endif
- /* Set up the small data pointer in r13. */
- #ifdef __PIC__
- lwz r13,_SDA_BASE_@got(r31)
- #else
- lis r13,_SDA_BASE_@ha
- addi r13,r13,_SDA_BASE_@l
- #endif
- /* Set up an initial stack frame, and clear the LR. */
- li r0,0
- stwu r1,-16(r1)
- mtlr r0
- stw r0,0(r1)
- /* find argc from the stack pointer */
- lwz r4,0(r9)
- /* find argv one word offset from the stack pointer */
- addi r5,r9,4
- mr r8,r3 /* Pass _dl_fini from ldso or NULL if statically linked */
- /* Note: PPC depends on the kernel to zero r3 before */
- /* handing over to user space, otherwise static apps */
- /* will SEGV during exit() */
- /* Ok, now run uClibc's main() -- shouldn't return */
- #ifdef __PIC__
- lwz r6,_init@got(r31)
- lwz r7,_fini@got(r31)
- lwz r3,main@got(r31)
- b __uClibc_main@plt
- #else
- lis r6,_init@ha # load top 16 bits
- addi r6,r6,_init@l # load bottom 16 bits
- lis r7,_fini@ha # load top 16 bits
- addi r7,r7,_fini@l # load bottom 16 bits
- lis r3,main@ha # load top 16 bits
- addi r3,r3,main@l # load bottom 16 bits
- b __uClibc_main
- #endif
- .size _start,.-_start
- /* Define a symbol for the first piece of initialized data. */
- .data
- .globl __data_start
- __data_start:
- .long 0
- .weak data_start
- data_start = __data_start
|