URL
https://opencores.org/ocsvn/openrisc/openrisc/trunk
Subversion Repositories openrisc
[/] [openrisc/] [trunk/] [gnu-old/] [newlib-1.17.0/] [libgloss/] [arm/] [crt0.S] - Rev 816
Compare with Previous | Blame | View Log
#include "newlib.h"#include "swi.h"/* ANSI concatenation macros. */#define CONCAT(a, b) CONCAT2(a, b)#define CONCAT2(a, b) a ## b#ifdef __USER_LABEL_PREFIX__#define FUNCTION( name ) CONCAT (__USER_LABEL_PREFIX__, name)#else#error __USER_LABEL_PREFIX is not defined#endif#ifdef HAVE_INITFINI_ARRAY#define _init __libc_init_array#define _fini __libc_fini_array#endif/* .text is used instead of .section .text so it works with arm-aout too. */.text#if defined(__thumb2__).syntax unified.thumb.macro FUNC_START name.global \name.thumb_func\name:.endm#else.code 32.macro FUNC_START name.global \name\name:.endm#endif.align 0FUNC_START _mainCRTStartupFUNC_START _startFUNC_START start#if defined(__ELF__) && !defined(__USING_SJLJ_EXCEPTIONS__)/* Annotation for EABI unwinding tables. */.fnstart#endif/* Start by setting up a stack */#ifdef ARM_RDP_MONITOR/* Issue Demon SWI to read stack info */swi SWI_GetEnv /* Returns command line in r0 */mov sp,r1 /* and the highest memory address in r1 */ldr sl, .LC2 /* stack limit is at end of data */add sl, sl, #256 /* allow slop for stack overflow handling *//* and small frames */#else#ifdef ARM_RDI_MONITOR/* Issue Angel SWI to read stack info */mov r0, #AngelSWI_Reason_HeapInfoadr r1, .LC0 /* point at ptr to 4 words to receive data */#if defined(__thumb2__)bkpt AngelSWI#else/* We are always in ARM mode for startup */AngelSWIAsm AngelSWI_ARM#endifldr r0, .LC0 /* point at values read */ldr sp, [r0, #8]ldr sl, [r0, #12]add sl, sl, #256 /* allow slop for stack overflow handling *//* and small frames */#else/* Set up the stack pointer to a fixed value *//* Changes by toralf:- Allow linker script to provide stack via __stack symbol - seedefintion of .Lstack- Provide "hooks" that may be used by the application to addcustom init code - see .Lhwinit and .Lswinit- Go through all execution modes and set up stack for each of them.Loosely based on init.s from ARM/Motorola example code.Note: Mode switch via CPSR is not allowed once in non-privilegedmode, so we take care not to enter "User" to set up its sp,and also skip most operations if already in that mode. */ldr r3, .Lstackcmp r3, #0#ifdef __thumb2__it eq#endifldreq r3, .LC0/* Note: This 'mov' is essential when starting in User, and ensures wealways get *some* sp value for the initial mode, even if wehave somehow missed it below (in which case it gets the samevalue as FIQ - not ideal, but better than nothing.) */mov sp, r3#ifdef __thumb2__/* XXX Fill in stack assignments for interrupt modes. */#elsemrs r2, CPSRtst r2, #0x0F /* Test mode bits - in User of all are 0 */beq .LC23 /* "eq" means r2 AND #0x0F is 0 */msr CPSR_c, #0xD1 /* FIRQ mode, interrupts disabled */mov sp, r3sub sl, sp, #0x1000 /* This mode also has its own sl (see below) */mov r3, slmsr CPSR_c, #0xD7 /* Abort mode, interrupts disabled */mov sp, r3sub r3, r3, #0x1000msr CPSR_c, #0xDB /* Undefined mode, interrupts disabled */mov sp, r3sub r3, r3, #0x1000msr CPSR_c, #0xD2 /* IRQ mode, interrupts disabled */mov sp, r3sub r3, r3, #0x2000msr CPSR_c, #0xD3 /* Supervisory mode, interrupts disabled */mov sp, r3sub r3, r3, #0x8000 /* Min size 32k */bic r3, r3, #0x00FF /* Align with current 64k block */bic r3, r3, #0xFF00str r3, [r3, #-4] /* Move value into user mode sp without */ldmdb r3, {sp}^ /* changing modes, via '^' form of ldm */orr r2, r2, #0xC0 /* Back to original mode, presumably SVC, */msr CPSR_c, r2 /* with FIQ/IRQ disable bits forced to 1 */#endif.LC23:/* Setup a default stack-limit in-case the code has beencompiled with "-mapcs-stack-check". Hard-wiring this valueis not ideal, since there is currently no support forchecking that the heap and stack have not collided, or thatthis default 64k is enough for the program being executed.However, it ensures that this simple crt0 world will notimmediately cause an overflow event: */sub sl, r3, #64 << 10 /* Still assumes 256bytes below sl */#endif#endif/* Zero the memory in the .bss section. */mov a2, #0 /* Second arg: fill value */mov fp, a2 /* Null frame pointer */mov r7, a2 /* Null frame pointer for Thumb */ldr a1, .LC1 /* First arg: start of memory block */ldr a3, .LC2sub a3, a3, a1 /* Third arg: length of block */#if defined(__thumb__) && !defined(__thumb2__)/* Enter Thumb mode.... */add a4, pc, #1 /* Get the address of the Thumb block */bx a4 /* Go there and start Thumb decoding */.code 16.global __change_mode.thumb_func__change_mode:#endifbl FUNCTION (memset)#if !defined (ARM_RDP_MONITOR) && !defined (ARM_RDI_MONITOR)/* Changes by toralf: Taken from libgloss/m68k/crt0.S* initialize target specific stuff. Only execute these* functions it they exist.*/ldr r3, .Lhwinitcmp r3, #0beq .LC24#if defined(__thumb__) || defined(__thumb2__)blx r3#elsemov lr, pcmov pc, r3#endif.LC24:ldr r3, .Lswinitcmp r3, #0beq .LC25#if defined(__thumb__) || defined(__thumb2__)blx r3#elsemov lr, pcmov pc, r3#endif.LC25:mov r0, #0 /* no arguments */mov r1, #0 /* no argv either */#else/* Need to set up standard file handles */bl FUNCTION (initialise_monitor_handles)#ifdef ARM_RDP_MONITORswi SWI_GetEnv /* sets r0 to point to the command line */mov r1, r0#elsemov r0, #AngelSWI_Reason_GetCmdLineadr r1, .LC30 /* Space for command line */AngelSWIAsm AngelSWIldr r1, .LC30#endif/* Parse string at r1 */mov r0, #0 /* count of arguments so far *//* Push a NULL argument onto the end of the list. */#ifdef __thumb__push {r0}#elsestmfd sp!, {r0}#endif.LC10:/* Skip leading blanks */#ifdef __thumb__ldrb r3, [r1]add r1, #1#elseldrb r3, [r1], #1#endifcmp r3, #0beq .LC12cmp r3, #' 'beq .LC10/* See whether we are scanning a string */cmp r3, #'"'#ifdef __thumb__beq .LC20cmp r3, #'\''bne .LC21.LC20:mov r2, r3b .LC22.LC21:mov r2, #' ' /* terminator type */sub r1, r1, #1 /* adjust back to point at start char */.LC22:#elsecmpne r3, #'\''moveq r2, r3movne r2, #' ' /* terminator type */subne r1, r1, #1 /* adjust back to point at start char */#endif/* Stack a pointer to the current argument */#ifdef __thumb__push {r1}#elsestmfd sp!, {r1}#endifadd r0, r0, #1.LC11:#ifdef __thumb__ldrb r3, [r1]add r1, #1#elseldrb r3, [r1], #1#endifcmp r3, #0beq .LC12cmp r2, r3 /* reached terminator? */bne .LC11mov r2, #0sub r3, r1, #1strb r2, [r3] /* terminate the arg string */b .LC10.LC12:mov r1, sp /* point at stacked arg pointers *//* We've now got the stacked args in order reverse the */#ifdef __thumb__mov r2, r0lsl r2, #2add r2, spmov r3, sp.LC15: cmp r2, r3bls .LC14sub r2, #4ldr r4, [r2]ldr r5, [r3]str r5, [r2]str r4, [r3]add r3, #4b .LC15.LC14:/* Ensure doubleword stack alignment. */mov r4, spmov r5, #7bic r4, r5mov sp, r4#elseadd r2, sp, r0, LSL #2 /* End of args */mov r3, sp /* Start of args */.LC13: cmp r2, r3ldrhi r4,[r2, #-4] /* Reverse ends of list */ldrhi r5, [r3]strhi r5, [r2, #-4]!strhi r4, [r3], #4bhi .LC13/* Ensure doubleword stack alignment. */bic sp, sp, #7#endif#endif#ifdef __USES_INITFINI__/* Some arm/elf targets use the .init and .fini sectionsto create constructors and destructors, and for thesetargets we need to call the _init function and arrangefor _fini to be called at program exit. */mov r4, r0mov r5, r1ldr r0, .Lfinibl FUNCTION (atexit)bl FUNCTION (_init)mov r0, r4mov r1, r5#endifbl FUNCTION (main)bl FUNCTION (exit) /* Should not return. */#if defined(__thumb__) && !defined(__thumb2__)/* Come out of Thumb mode. This code should be redundant. */mov a4, pcbx a4.code 32.global change_backchange_back:/* Halt the execution. This code should never be executed. *//* With no debug monitor, this probably aborts (eventually).With a Demon debug monitor, this halts cleanly.With an Angel debug monitor, this will report 'Unknown SWI'. */swi SWI_Exit#endif/* For Thumb, constants must be after the code since onlypositive offsets are supported for PC relative addresses. */.align 0.LC0:#ifdef ARM_RDI_MONITOR.word HeapBase#else#ifndef ARM_RDP_MONITOR/* Changes by toralf: Provide alternative "stack" variable whose valuemay be defined externally; .Lstack will be used instead of .LC0 ifit points to a non-0 value. Also set up references to "hooks" thatmay be used by the application to provide additional init code. */#ifdef __pe__.word 0x800000#else.word 0x80000 /* Top of RAM on the PIE board. */#endif.Lstack:.word __stack.Lhwinit:.word FUNCTION (hardware_init_hook).Lswinit:.word FUNCTION (software_init_hook)/* Set up defaults for the above variables in the form of weak symbols- so that application will link correctly, and get value 0 inruntime (meaning "ignore setting") for the variables, when the userdoes not provide the symbols. (The linker uses a weak symbol if,and only if, a normal version of the same symbol isn't providede.g. by a linker script or another object file.) */.weak __stack.weak FUNCTION (hardware_init_hook).weak FUNCTION (software_init_hook)#endif#endif#if defined(__ELF__) && !defined(__USING_SJLJ_EXCEPTIONS__)/* Protect against unhandled exceptions. */.cantunwind.fnend#endif.LC1:.word __bss_start__.LC2:.word __bss_end__#ifdef __USES_INITFINI__.Lfini:.word FUNCTION(_fini)#endif#ifdef ARM_RDI_MONITOR.LC30:.word CommandLine.word 255/* Workspace for Angel calls. */.data/* Data returned by monitor SWI. */.global __stack_base__HeapBase: .word 0HeapLimit: .word 0__stack_base__: .word 0StackLimit: .word 0CommandLine: .space 256,0 /* Maximum length of 255 chars handled. */#endif#ifdef __pe__.section .idata$3.long 0,0,0,0,0,0,0,0#endif
