mirror of
https://github.com/riscv-software-src/opensbi.git
synced 2025-08-24 23:41:23 +01:00

When OpenSBI is compiled as fPIE mode, the assembler will translate "la" to GOT reference pattern. It will cause to cost an additional load instruction when obtaining the symbol address. However, if the symbol locates within the positive or negative 2GB region, we can use "lla" instead of "la" to avoid unneeded GOT references. This patch assumes that the OpenSBI image excluding the payload does not exceed 2GB. Based on this assumption, all "la" instructions are replaced by "lla" to avoid performance degradation when compiling as fPIE mode. Signed-off-by: Vincent Chen <vincent.chen@sifive.com> Reviewed-by: Anup Patel <anup.patel@wdc.com>
152 lines
3.3 KiB
ArmAsm
152 lines
3.3 KiB
ArmAsm
/*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*
|
|
* Copyright (c) 2019 Western Digital Corporation or its affiliates.
|
|
*
|
|
* Authors:
|
|
* Anup Patel <anup.patel@wdc.com>
|
|
*/
|
|
|
|
#include <sbi/fw_dynamic.h>
|
|
|
|
#include "fw_base.S"
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
_bad_dynamic_info:
|
|
wfi
|
|
j _bad_dynamic_info
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_boot_hart
|
|
/*
|
|
* This function is called very early even before
|
|
* fw_save_info() is called.
|
|
* We can only use a0, a1, and a2 registers here.
|
|
* The boot HART id should be returned in 'a0'.
|
|
*/
|
|
fw_boot_hart:
|
|
/* Sanity checks */
|
|
li a1, FW_DYNAMIC_INFO_MAGIC_VALUE
|
|
REG_L a0, FW_DYNAMIC_INFO_MAGIC_OFFSET(a2)
|
|
bne a0, a1, _bad_dynamic_info
|
|
li a1, FW_DYNAMIC_INFO_VERSION_MAX
|
|
REG_L a0, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
|
bgt a0, a1, _bad_dynamic_info
|
|
|
|
/* Read boot HART id */
|
|
li a1, 0x2
|
|
blt a0, a1, 2f
|
|
REG_L a0, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
|
ret
|
|
2: li a0, -1
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_save_info
|
|
/*
|
|
* We can only use a0, a1, a2, a3, and a4 registers here.
|
|
* The a0, a1, and a2 registers will be same as passed by
|
|
* previous booting stage.
|
|
* Nothing to be returned here.
|
|
*/
|
|
fw_save_info:
|
|
/* Save next arg1 in 'a1' */
|
|
lla a4, _dynamic_next_arg1
|
|
REG_S a1, (a4)
|
|
|
|
/* Sanity checks */
|
|
li a4, FW_DYNAMIC_INFO_MAGIC_VALUE
|
|
REG_L a3, FW_DYNAMIC_INFO_MAGIC_OFFSET(a2)
|
|
bne a3, a4, _bad_dynamic_info
|
|
li a4, FW_DYNAMIC_INFO_VERSION_MAX
|
|
REG_L a3, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
|
bgt a3, a4, _bad_dynamic_info
|
|
|
|
/* Save version == 0x1 fields */
|
|
lla a4, _dynamic_next_addr
|
|
REG_L a3, FW_DYNAMIC_INFO_NEXT_ADDR_OFFSET(a2)
|
|
REG_S a3, (a4)
|
|
lla a4, _dynamic_next_mode
|
|
REG_L a3, FW_DYNAMIC_INFO_NEXT_MODE_OFFSET(a2)
|
|
REG_S a3, (a4)
|
|
lla a4, _dynamic_options
|
|
REG_L a3, FW_DYNAMIC_INFO_OPTIONS_OFFSET(a2)
|
|
REG_S a3, (a4)
|
|
|
|
/* Save version == 0x2 fields */
|
|
li a4, 0x2
|
|
REG_L a3, FW_DYNAMIC_INFO_VERSION_OFFSET(a2)
|
|
blt a3, a4, 2f
|
|
lla a4, _dynamic_boot_hart
|
|
REG_L a3, FW_DYNAMIC_INFO_BOOT_HART_OFFSET(a2)
|
|
REG_S a3, (a4)
|
|
2:
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_next_arg1
|
|
/*
|
|
* We can only use a0, a1, and a2 registers here.
|
|
* The a0, a1, and a2 registers will be same as passed by
|
|
* previous booting stage.
|
|
* The next arg1 should be returned in 'a0'.
|
|
*/
|
|
fw_next_arg1:
|
|
lla a0, _dynamic_next_arg1
|
|
REG_L a0, (a0)
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_next_addr
|
|
/*
|
|
* We can only use a0, a1, and a2 registers here.
|
|
* The next address should be returned in 'a0'.
|
|
*/
|
|
fw_next_addr:
|
|
lla a0, _dynamic_next_addr
|
|
REG_L a0, (a0)
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_next_mode
|
|
/*
|
|
* We can only use a0, a1, and a2 registers here.
|
|
* The next address should be returned in 'a0'
|
|
*/
|
|
fw_next_mode:
|
|
lla a0, _dynamic_next_mode
|
|
REG_L a0, (a0)
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
.global fw_options
|
|
/*
|
|
* We can only use a0, a1, and a2 registers here.
|
|
* The 'a4' register will have default options.
|
|
* The next address should be returned in 'a0'.
|
|
*/
|
|
fw_options:
|
|
lla a0, _dynamic_options
|
|
REG_L a0, (a0)
|
|
ret
|
|
|
|
.section .entry, "ax", %progbits
|
|
.align 3
|
|
_dynamic_next_arg1:
|
|
RISCV_PTR 0x0
|
|
_dynamic_next_addr:
|
|
RISCV_PTR 0x0
|
|
_dynamic_next_mode:
|
|
RISCV_PTR PRV_S
|
|
_dynamic_options:
|
|
RISCV_PTR 0x0
|
|
_dynamic_boot_hart:
|
|
RISCV_PTR -1
|