blob: ea4187a7356996b3e15f40db944636f579cdcef5 [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
/**
* Flash header.
*
* Contains the address of the entry point.
*/
.section .flash_header, "a", @progbits
.4byte _start
/**
* Flash executable runtime initialization code.
*/
// NOTE: The "ax" flag below is necessary to ensure that this section
// is allocated space in ROM by the linker.
.section .crt, "ax", @progbits
.extern abort
.extern main
.extern crt_interrupt_vector
.extern crt_section_clear
.extern crt_section_copy
/**
* Callable entry point for flash.
*
* This sets up the stack, zeroes `.bss`, and sets up `.data`.
* It then jumps into main.
*/
_start:
.globl _start
// Set up the stack. We have no expectation that the rom that
// jumps here will have the correct stack start linked in.
la sp, _stack_end
// Set up the global pointer. This requires that we disable linker relaxations
// (or it will be relaxed to `mv gp, gp`).
.option push
.option norelax
la gp, __global_pointer$
.option pop
// Set up the new interrupt vector.
la t0, (crt_interrupt_vector + 1)
csrw mtvec, t0
// Zero out the `.bss` segment.
la a0, _bss_start
la a1, _bss_end
call crt_section_clear
// Zero out the stack
la a0, _stack_start
la a1, _stack_end
call crt_section_clear
// Initialize the `.data` segment from the `.idata` segment.
la a0, _data_start
la a1, _data_end
la a2, _data_init_start
call crt_section_copy
// Call the functions in the `.init_array` section.
//
// This section is typically empty except for executables built with LLVM
// coverage enabled. When coverage is enabled, the compiler emits pointers to
// the functions that initialize the profile buffer in this section. These
// functions must be called before the instrumented functions in the program.
//
// We use `s0` and `s1` to represent the start and end pointers of
// `.init_array`, respectively, and `t0` to store the addresses of the
// functions to be called.
la s0, _init_array_start
la s1, _init_array_end
bgeu s0, s1, init_array_loop_end
init_array_loop:
lw t0, 0(s0)
jalr t0
addi s0, s0, 0x4
bltu s0, s1, init_array_loop
init_array_loop_end:
// Jump into the C program entry point. This is your standard
// C `main()`, so we need to pass dummy values for `argc` and `argv`.
li a0, 0x0 // argc = 0
li a1, 0x0 // argv = NULL
call main
// Halt the core (wfi loop)
tail abort