2000-03-17 14:48:54 -08:00
|
|
|
/*
|
|
|
|
* crt0.S -- startup file for MIPS.
|
|
|
|
*
|
2001-09-11 07:24:35 -07:00
|
|
|
* Copyright (c) 1995, 1996, 1997, 2001 Cygnus Support
|
2000-03-17 14:48:54 -08:00
|
|
|
*
|
|
|
|
* The authors hereby grant permission to use, copy, modify, distribute,
|
|
|
|
* and license this software and its documentation for any purpose, provided
|
|
|
|
* that existing copyright notices are retained in all copies and that this
|
|
|
|
* notice is included verbatim in any distributions. No written agreement,
|
|
|
|
* license, or royalty fee is required for any of the authorized uses.
|
|
|
|
* Modifications to this software may be copyrighted by their authors
|
|
|
|
* and need not follow the licensing terms described here, provided that
|
|
|
|
* the new terms are clearly indicated on the first page of each file where
|
|
|
|
* they apply.
|
|
|
|
*/
|
|
|
|
|
2014-11-28 00:51:03 -08:00
|
|
|
/* This file does not use any floating-point ABI. */
|
|
|
|
.gnu_attribute 4,0
|
|
|
|
|
2000-03-17 14:48:54 -08:00
|
|
|
#ifdef __mips16
|
|
|
|
/* This file contains 32 bit assembly code. */
|
|
|
|
.set nomips16
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "regs.S"
|
2014-11-28 00:51:03 -08:00
|
|
|
#include "abiflags.S"
|
2000-03-17 14:48:54 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up some room for a stack. We just grab a chunk of memory.
|
|
|
|
*/
|
|
|
|
#define STACK_SIZE 0x4000
|
|
|
|
#define GLOBAL_SIZE 0x2000
|
|
|
|
|
|
|
|
#define STARTUP_STACK_SIZE 0x0100
|
|
|
|
|
|
|
|
/* This is for referencing addresses that are not in the .sdata or
|
|
|
|
.sbss section under embedded-pic, or before we've set up gp. */
|
|
|
|
#ifdef __mips_embedded_pic
|
|
|
|
# ifdef __mips64
|
|
|
|
# define LA(t,x) la t,x-PICBASE ; daddu t,s0,t
|
|
|
|
# else
|
|
|
|
# define LA(t,x) la t,x-PICBASE ; addu t,s0,t
|
|
|
|
# endif
|
2002-11-12 15:14:24 -08:00
|
|
|
#else /* __mips_embedded_pic */
|
2000-03-17 14:48:54 -08:00
|
|
|
# define LA(t,x) la t,x
|
2002-11-12 15:14:24 -08:00
|
|
|
#endif /* __mips_embedded_pic */
|
2000-03-17 14:48:54 -08:00
|
|
|
|
|
|
|
.comm __memsize, 12
|
|
|
|
.comm __lstack, STARTUP_STACK_SIZE
|
|
|
|
|
|
|
|
.text
|
|
|
|
.align 2
|
|
|
|
|
|
|
|
/* Without the following nop, GDB thinks _start is a data variable.
|
|
|
|
* This is probably a bug in GDB in handling a symbol that is at the
|
|
|
|
* start of the .text section.
|
|
|
|
*/
|
|
|
|
nop
|
|
|
|
|
2002-07-12 08:37:04 -07:00
|
|
|
.globl hardware_hazard_hook .text
|
2000-03-17 14:48:54 -08:00
|
|
|
.globl _start
|
|
|
|
.ent _start
|
|
|
|
_start:
|
|
|
|
#ifdef __mips_embedded_pic
|
2002-07-12 08:37:04 -07:00
|
|
|
#define PICBASE start_PICBASE
|
2014-11-26 16:28:09 -08:00
|
|
|
.set noreorder
|
2000-03-17 14:48:54 -08:00
|
|
|
PICBASE = .+8
|
|
|
|
bal PICBASE
|
|
|
|
nop
|
|
|
|
move s0,$31
|
2014-11-26 16:28:09 -08:00
|
|
|
.set reorder
|
2000-03-17 14:48:54 -08:00
|
|
|
#endif
|
2006-11-24 10:19:18 -08:00
|
|
|
#if __mips<3
|
|
|
|
# define STATUS_MASK (SR_CU1|SR_PE)
|
2000-03-17 14:48:54 -08:00
|
|
|
#else
|
2006-11-24 10:19:18 -08:00
|
|
|
/* Post-mips2 has no SR_PE bit. */
|
|
|
|
# ifdef __mips64
|
|
|
|
/* Turn on 64-bit addressing and additional float regs. */
|
|
|
|
# define STATUS_MASK (SR_CU1|SR_FR|SR_KX|SR_SX|SR_UX)
|
|
|
|
# else
|
2006-11-27 08:12:51 -08:00
|
|
|
# if __mips_fpr==32
|
2006-11-24 10:19:18 -08:00
|
|
|
# define STATUS_MASK (SR_CU1)
|
|
|
|
# else
|
|
|
|
/* Turn on additional float regs. */
|
|
|
|
# define STATUS_MASK (SR_CU1|SR_FR)
|
|
|
|
# endif
|
|
|
|
# endif
|
2000-03-17 14:48:54 -08:00
|
|
|
#endif
|
2014-11-28 00:51:03 -08:00
|
|
|
|
|
|
|
/* Clear Cause register. */
|
|
|
|
mtc0 zero,C0_CAUSE
|
2000-03-17 14:48:54 -08:00
|
|
|
nop
|
|
|
|
|
2014-11-28 00:51:03 -08:00
|
|
|
/* Read MIPS_abiflags structure and set status/config registers
|
|
|
|
accordingly. */
|
|
|
|
.weak __MIPS_abiflags_start
|
|
|
|
.weak __MIPS_abiflags_end
|
|
|
|
LA (t0,__MIPS_abiflags_start)
|
|
|
|
LA (t1,__MIPS_abiflags_end)
|
|
|
|
addiu t1,t1,-24
|
|
|
|
move v0,zero /* Mask for C0_SR. */
|
2002-07-12 08:37:04 -07:00
|
|
|
|
2014-11-28 00:51:03 -08:00
|
|
|
/* Branch to 1f is the .MIPS.abiflags section is not 24 bytes. This
|
|
|
|
indicates it is either missing or corrupt. */
|
|
|
|
bne t0,t1,1f
|
|
|
|
|
|
|
|
/* Check isa_level. */
|
|
|
|
lbu t1,ABIFlags_isa_level(t0)
|
|
|
|
sltu v1,t1,3 /* Is MIPS < 3? */
|
|
|
|
xori t1,t1,64 /* Is MIPS64? */
|
|
|
|
beq v1,zero,4f
|
|
|
|
li v1,SR_PE
|
|
|
|
or v0,v0,v1 /* Enable soft reset. */
|
|
|
|
4:
|
|
|
|
li v1,(SR_KX|SR_SX|SR_UX)
|
|
|
|
bne t1,zero,5f
|
|
|
|
or v0,v0,v1 /* Enable extended addressing. */
|
|
|
|
5:
|
|
|
|
/* Check fp_abi. */
|
|
|
|
lbu t1,ABIFlags_fp_abi(t0)
|
|
|
|
xori t1,t1,Val_GNU_MIPS_ABI_FP_SOFT
|
|
|
|
li v1,SR_CU1
|
|
|
|
beq t1,zero,2f /* Skip MSA and cpr1_size checks. */
|
|
|
|
or v0,v0,v1 /* Enable co-processor 1. */
|
|
|
|
|
|
|
|
/* Check cpr1_size. */
|
|
|
|
lbu t1,ABIFlags_cpr1_size(t0)
|
|
|
|
xori t1,t1,AFL_REG_64
|
|
|
|
li v1,SR_FR
|
|
|
|
bne t1,zero,3f
|
|
|
|
or v0,v0,v1 /* Enable 64-bit FPU registers. */
|
|
|
|
3:
|
|
|
|
/* Check ases. */
|
|
|
|
lw t1,ABIFlags_ases(t0)
|
|
|
|
andi t1,t1,AFL_ASE_MSA
|
|
|
|
li v1,SR_FR
|
|
|
|
beq t1,zero,2f
|
|
|
|
or v0,v0,v1 /* Enable 64-bit FPU registers. */
|
|
|
|
li v1,SR_MSA
|
|
|
|
.set push
|
|
|
|
.set mips32
|
|
|
|
mtc0 v1,C0_CONFIG,5 /* Enable MSA. */
|
|
|
|
.set pop
|
|
|
|
b 2f
|
2000-03-17 14:48:54 -08:00
|
|
|
|
|
|
|
1:
|
2014-11-28 00:51:03 -08:00
|
|
|
/* MIPS_abiflags structure is not available. Set status/config
|
|
|
|
registers based on flags defined by compiler. */
|
|
|
|
#ifdef __mips_soft_float
|
|
|
|
li v0,(STATUS_MASK-(STATUS_MASK & SR_CU1))
|
|
|
|
#else
|
|
|
|
li v0,STATUS_MASK
|
|
|
|
#endif
|
|
|
|
|
|
|
|
2:
|
|
|
|
/* Set C0_SR, */
|
|
|
|
mtc0 v0,C0_SR
|
2000-03-17 14:48:54 -08:00
|
|
|
nop
|
2014-11-28 00:51:03 -08:00
|
|
|
|
|
|
|
/* Avoid hazard from C0_SR changes. */
|
|
|
|
LA (t0, hardware_hazard_hook)
|
2002-07-12 08:37:04 -07:00
|
|
|
beq t0,zero,2f
|
2014-11-26 16:28:09 -08:00
|
|
|
jalr t0
|
2000-03-17 14:48:54 -08:00
|
|
|
2:
|
|
|
|
|
|
|
|
|
2014-11-28 00:51:03 -08:00
|
|
|
/* Fix high bits, if any, of the PC so that exception handling doesn't get
|
|
|
|
confused. */
|
2000-03-17 14:48:54 -08:00
|
|
|
LA (v0, 3f)
|
|
|
|
jr v0
|
|
|
|
3:
|
|
|
|
LA (gp, _gp) # set the global data pointer
|
|
|
|
.end _start
|
|
|
|
|
|
|
|
/*
|
|
|
|
* zero out the bss section.
|
|
|
|
*/
|
|
|
|
.globl __memsize
|
|
|
|
.globl get_mem_info .text
|
|
|
|
.globl __stack
|
|
|
|
.globl __global
|
|
|
|
.ent zerobss
|
|
|
|
zerobss:
|
|
|
|
LA (v0, _fbss)
|
|
|
|
LA (v1, _end)
|
2014-11-26 16:28:09 -08:00
|
|
|
beq v0,v1,2f
|
|
|
|
1:
|
|
|
|
addiu v0,v0,4
|
|
|
|
sw zero,-4(v0)
|
|
|
|
bne v0,v1,1b
|
|
|
|
2:
|
2000-03-17 14:48:54 -08:00
|
|
|
la t0, __lstack # make a small stack so we
|
|
|
|
addiu sp, t0, STARTUP_STACK_SIZE # can run some C code
|
|
|
|
la a0, __memsize # get the usable memory size
|
|
|
|
jal get_mem_info
|
|
|
|
|
|
|
|
/* setup the stack pointer */
|
|
|
|
LA (t0, __stack) # is __stack set ?
|
|
|
|
bne t0,zero,4f
|
|
|
|
|
|
|
|
/* NOTE: a0[0] contains the amount of memory available, and
|
|
|
|
not the last memory address. */
|
2003-02-18 11:32:24 -08:00
|
|
|
la a0, __memsize
|
2000-03-17 14:48:54 -08:00
|
|
|
lw t0,0(a0) # last address of memory available
|
|
|
|
la t1,K0BASE # cached kernel memory
|
|
|
|
addu t0,t0,t1 # get the end of memory address
|
2003-01-08 04:54:29 -08:00
|
|
|
/* Allocate 32 bytes for the register parameters. Allocate 16
|
|
|
|
bytes for a null argv and envp. Round the result up to 64
|
|
|
|
bytes to preserve alignment. */
|
|
|
|
subu t0,t0,64
|
2000-03-17 14:48:54 -08:00
|
|
|
4:
|
|
|
|
move sp,t0 # set stack pointer
|
|
|
|
.end zerobss
|
|
|
|
|
|
|
|
/*
|
|
|
|
* initialize target specific stuff. Only execute these
|
|
|
|
* functions it they exist.
|
|
|
|
*/
|
|
|
|
.globl hardware_init_hook .text
|
|
|
|
.globl software_init_hook .text
|
2001-08-31 13:18:49 -07:00
|
|
|
.type _fini,@function
|
|
|
|
.type _init,@function
|
2000-03-17 14:48:54 -08:00
|
|
|
.globl atexit .text
|
|
|
|
.globl exit .text
|
|
|
|
.ent init
|
|
|
|
init:
|
|
|
|
LA (t9, hardware_init_hook) # init the hardware if needed
|
|
|
|
beq t9,zero,6f
|
2014-11-26 16:28:09 -08:00
|
|
|
jalr t9
|
2000-03-17 14:48:54 -08:00
|
|
|
6:
|
|
|
|
LA (t9, software_init_hook) # init the hardware if needed
|
|
|
|
beq t9,zero,7f
|
2014-11-26 16:28:09 -08:00
|
|
|
jalr t9
|
2000-03-17 14:48:54 -08:00
|
|
|
7:
|
2001-08-31 13:18:49 -07:00
|
|
|
LA (a0, _fini)
|
2000-03-17 14:48:54 -08:00
|
|
|
jal atexit
|
|
|
|
|
|
|
|
#ifdef GCRT0
|
|
|
|
.globl _ftext
|
|
|
|
.globl _extext
|
|
|
|
LA (a0, _ftext)
|
|
|
|
LA (a1, _etext)
|
|
|
|
jal monstartup
|
|
|
|
#endif
|
|
|
|
|
2002-11-12 15:14:24 -08:00
|
|
|
|
2001-08-31 13:18:49 -07:00
|
|
|
jal _init # run global constructors
|
|
|
|
|
2003-01-08 04:54:29 -08:00
|
|
|
addiu a1,sp,32 # argv = sp + 32
|
|
|
|
addiu a2,sp,40 # envp = sp + 40
|
|
|
|
#if __mips64
|
|
|
|
sd zero,(a1) # argv[argc] = 0
|
|
|
|
sd zero,(a2) # envp[0] = 0
|
|
|
|
#else
|
|
|
|
sw zero,(a1)
|
|
|
|
sw zero,(a2)
|
|
|
|
#endif
|
|
|
|
move a0,zero # set argc to 0
|
2014-11-26 16:28:09 -08:00
|
|
|
jal main # call the program start function
|
2000-03-17 14:48:54 -08:00
|
|
|
|
|
|
|
# fall through to the "exit" routine
|
2014-11-26 16:28:09 -08:00
|
|
|
move a0,v0 # pass through the exit code
|
2000-03-17 14:48:54 -08:00
|
|
|
jal exit # call libc exit to run the G++
|
|
|
|
# destructors
|
|
|
|
.end init
|
2002-07-12 08:37:04 -07:00
|
|
|
|
2002-11-12 15:14:24 -08:00
|
|
|
|
2002-07-12 08:37:04 -07:00
|
|
|
/* Assume the PICBASE set up above is no longer valid below here. */
|
|
|
|
#ifdef __mips_embedded_pic
|
|
|
|
#undef PICBASE
|
|
|
|
#endif
|
2000-03-17 14:48:54 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* _exit -- Exit from the application. Normally we cause a user trap
|
|
|
|
* to return to the ROM monitor for another run. NOTE: This is
|
|
|
|
* the only other routine we provide in the crt0.o object, since
|
|
|
|
* it may be tied to the "_start" routine. It also allows
|
|
|
|
* executables that contain a complete world to be linked with
|
|
|
|
* just the crt0.o object.
|
|
|
|
*/
|
2002-07-12 08:37:04 -07:00
|
|
|
.globl hardware_exit_hook .text
|
2000-03-17 14:48:54 -08:00
|
|
|
.globl _exit
|
|
|
|
.ent _exit
|
|
|
|
_exit:
|
|
|
|
7:
|
2002-07-12 08:37:04 -07:00
|
|
|
#ifdef __mips_embedded_pic
|
|
|
|
/* Need to reinit PICBASE, since we might be called via exit()
|
|
|
|
rather than via a return path which would restore old s0. */
|
|
|
|
#define PICBASE exit_PICBASE
|
2014-11-26 16:28:09 -08:00
|
|
|
.set noreorder
|
2002-07-12 08:37:04 -07:00
|
|
|
PICBASE = .+8
|
|
|
|
bal PICBASE
|
|
|
|
nop
|
|
|
|
move s0,$31
|
2014-11-26 16:28:09 -08:00
|
|
|
.set reorder
|
2002-07-12 08:37:04 -07:00
|
|
|
#endif
|
2000-03-17 14:48:54 -08:00
|
|
|
#ifdef GCRT0
|
2002-07-12 08:37:04 -07:00
|
|
|
LA (t0, _mcleanup)
|
2014-11-26 16:28:09 -08:00
|
|
|
jalr t0
|
2000-03-17 14:48:54 -08:00
|
|
|
#endif
|
2002-07-12 08:37:04 -07:00
|
|
|
LA (t0, hardware_exit_hook)
|
|
|
|
beq t0,zero,1f
|
2014-11-26 16:28:09 -08:00
|
|
|
jalr t0
|
2002-07-12 08:37:04 -07:00
|
|
|
1:
|
|
|
|
|
2000-03-17 14:48:54 -08:00
|
|
|
# break instruction can cope with 0xfffff, but GAS limits the range:
|
|
|
|
break 1023
|
|
|
|
b 7b # but loop back just in-case
|
|
|
|
.end _exit
|
2002-11-12 15:14:24 -08:00
|
|
|
|
2002-07-12 08:37:04 -07:00
|
|
|
/* Assume the PICBASE set up above is no longer valid below here. */
|
|
|
|
#ifdef __mips_embedded_pic
|
|
|
|
#undef PICBASE
|
|
|
|
#endif
|
|
|
|
|
2000-03-17 14:48:54 -08:00
|
|
|
/* EOF crt0.S */
|