mirror of
				https://github.com/riscv-software-src/opensbi
				synced 2025-11-03 21:48:45 +00:00 
			
		
		
		
	We have redundant semicolon at quite a few places so let's remove it. Signed-off-by: Xiang W <wxjstz@126.com> Reviewed-by: Anup Patel <anup@brainfault.org>
		
			
				
	
	
		
			170 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 *
 | 
						|
 * Copyright (c) 2019 Western Digital Corporation or its affiliates.
 | 
						|
 *
 | 
						|
 * Authors:
 | 
						|
 *   Anup Patel <anup.patel@wdc.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include <sbi/riscv_encoding.h>
 | 
						|
#include <sbi/sbi_bitops.h>
 | 
						|
#include <sbi/sbi_hart.h>
 | 
						|
#include <sbi/sbi_scratch.h>
 | 
						|
#include <sbi/sbi_trap.h>
 | 
						|
#include <sbi/sbi_unpriv.h>
 | 
						|
 | 
						|
/**
 | 
						|
 * a3 must a pointer to the sbi_trap_info and a4 is used as a temporary
 | 
						|
 * register in the trap handler. Make sure that compiler doesn't use a3 & a4.
 | 
						|
 */
 | 
						|
#define DEFINE_UNPRIVILEGED_LOAD_FUNCTION(type, insn)                         \
 | 
						|
	type sbi_load_##type(const type *addr,                                \
 | 
						|
			     struct sbi_trap_info *trap)                      \
 | 
						|
	{                                                                     \
 | 
						|
		register ulong tinfo asm("a3");                               \
 | 
						|
		register ulong mstatus = 0;                                   \
 | 
						|
		register ulong mtvec = sbi_hart_expected_trap_addr();         \
 | 
						|
		type ret = 0;                                                 \
 | 
						|
		trap->cause = 0;                                              \
 | 
						|
		asm volatile(                                                 \
 | 
						|
			"add %[tinfo], %[taddr], zero\n"                      \
 | 
						|
			"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n"      \
 | 
						|
			"csrrs %[mstatus], " STR(CSR_MSTATUS) ", %[mprv]\n"   \
 | 
						|
			".option push\n"                                      \
 | 
						|
			".option norvc\n"                                     \
 | 
						|
			#insn " %[ret], %[addr]\n"                            \
 | 
						|
			".option pop\n"                                       \
 | 
						|
			"csrw " STR(CSR_MSTATUS) ", %[mstatus]\n"             \
 | 
						|
			"csrw " STR(CSR_MTVEC) ", %[mtvec]"                   \
 | 
						|
		    : [mstatus] "+&r"(mstatus), [mtvec] "+&r"(mtvec),         \
 | 
						|
		      [tinfo] "+&r"(tinfo), [ret] "=&r"(ret)                  \
 | 
						|
		    : [addr] "m"(*addr), [mprv] "r"(MSTATUS_MPRV),            \
 | 
						|
		      [taddr] "r"((ulong)trap)                                \
 | 
						|
		    : "a4", "memory");                                        \
 | 
						|
		return ret;                                                   \
 | 
						|
	}
 | 
						|
 | 
						|
#define DEFINE_UNPRIVILEGED_STORE_FUNCTION(type, insn)                        \
 | 
						|
	void sbi_store_##type(type *addr, type val,                           \
 | 
						|
			      struct sbi_trap_info *trap)                     \
 | 
						|
	{                                                                     \
 | 
						|
		register ulong tinfo asm("a3") = (ulong)trap;                 \
 | 
						|
		register ulong mstatus = 0;                                   \
 | 
						|
		register ulong mtvec = sbi_hart_expected_trap_addr();         \
 | 
						|
		trap->cause = 0;                                              \
 | 
						|
		asm volatile(                                                 \
 | 
						|
			"add %[tinfo], %[taddr], zero\n"                      \
 | 
						|
			"csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n"      \
 | 
						|
			"csrrs %[mstatus], " STR(CSR_MSTATUS) ", %[mprv]\n"   \
 | 
						|
			".option push\n"                                      \
 | 
						|
			".option norvc\n"                                     \
 | 
						|
			#insn " %[val], %[addr]\n"                            \
 | 
						|
			".option pop\n"                                       \
 | 
						|
			"csrw " STR(CSR_MSTATUS) ", %[mstatus]\n"             \
 | 
						|
			"csrw " STR(CSR_MTVEC) ", %[mtvec]"                   \
 | 
						|
		    : [mstatus] "+&r"(mstatus), [mtvec] "+&r"(mtvec),         \
 | 
						|
		      [tinfo] "+&r"(tinfo)                                    \
 | 
						|
		    : [addr] "m"(*addr), [mprv] "r"(MSTATUS_MPRV),            \
 | 
						|
		      [val] "r"(val), [taddr] "r"((ulong)trap)                \
 | 
						|
		    : "a4", "memory");                                        \
 | 
						|
	}
 | 
						|
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(u8, lbu)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(u16, lhu)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(s8, lb)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(s16, lh)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(s32, lw)
 | 
						|
DEFINE_UNPRIVILEGED_STORE_FUNCTION(u8, sb)
 | 
						|
DEFINE_UNPRIVILEGED_STORE_FUNCTION(u16, sh)
 | 
						|
DEFINE_UNPRIVILEGED_STORE_FUNCTION(u32, sw)
 | 
						|
#if __riscv_xlen == 64
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(u32, lwu)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(u64, ld)
 | 
						|
DEFINE_UNPRIVILEGED_STORE_FUNCTION(u64, sd)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(ulong, ld)
 | 
						|
#elif __riscv_xlen == 32
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(u32, lw)
 | 
						|
DEFINE_UNPRIVILEGED_LOAD_FUNCTION(ulong, lw)
 | 
						|
 | 
						|
u64 sbi_load_u64(const u64 *addr,
 | 
						|
		 struct sbi_trap_info *trap)
 | 
						|
{
 | 
						|
	u64 ret = sbi_load_u32((u32 *)addr, trap);
 | 
						|
 | 
						|
	if (trap->cause)
 | 
						|
		return 0;
 | 
						|
	ret |= ((u64)sbi_load_u32((u32 *)addr + 1, trap) << 32);
 | 
						|
	if (trap->cause)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
void sbi_store_u64(u64 *addr, u64 val,
 | 
						|
		   struct sbi_trap_info *trap)
 | 
						|
{
 | 
						|
	sbi_store_u32((u32 *)addr, val, trap);
 | 
						|
	if (trap->cause)
 | 
						|
		return;
 | 
						|
 | 
						|
	sbi_store_u32((u32 *)addr + 1, val >> 32, trap);
 | 
						|
	if (trap->cause)
 | 
						|
		return;
 | 
						|
}
 | 
						|
#else
 | 
						|
# error "Unexpected __riscv_xlen"
 | 
						|
#endif
 | 
						|
 | 
						|
ulong sbi_get_insn(ulong mepc, struct sbi_trap_info *trap)
 | 
						|
{
 | 
						|
	register ulong tinfo asm("a3");
 | 
						|
	register ulong ttmp asm("a4");
 | 
						|
	register ulong mstatus = 0;
 | 
						|
	register ulong mtvec = sbi_hart_expected_trap_addr();
 | 
						|
	ulong insn = 0;
 | 
						|
 | 
						|
	trap->cause = 0;
 | 
						|
 | 
						|
	asm volatile(
 | 
						|
	    "add %[tinfo], %[taddr], zero\n"
 | 
						|
	    "csrrw %[mtvec], " STR(CSR_MTVEC) ", %[mtvec]\n"
 | 
						|
	    "csrrs %[mstatus], " STR(CSR_MSTATUS) ", %[mprv]\n"
 | 
						|
	    "lhu %[insn], (%[addr])\n"
 | 
						|
	    "andi %[ttmp], %[insn], 3\n"
 | 
						|
	    "addi %[ttmp], %[ttmp], -3\n"
 | 
						|
	    "bne %[ttmp], zero, 2f\n"
 | 
						|
	    "lhu %[ttmp], 2(%[addr])\n"
 | 
						|
	    "sll %[ttmp], %[ttmp], 16\n"
 | 
						|
	    "add %[insn], %[insn], %[ttmp]\n"
 | 
						|
	    "2: csrw " STR(CSR_MSTATUS) ", %[mstatus]\n"
 | 
						|
	    "csrw " STR(CSR_MTVEC) ", %[mtvec]"
 | 
						|
	    : [mstatus] "+&r"(mstatus), [mtvec] "+&r"(mtvec),
 | 
						|
	      [tinfo] "+&r"(tinfo), [ttmp] "+&r"(ttmp),
 | 
						|
	      [insn] "=&r"(insn)
 | 
						|
	    : [mprv] "r"(MSTATUS_MPRV | MSTATUS_MXR),
 | 
						|
	      [taddr] "r"((ulong)trap), [addr] "r"(mepc)
 | 
						|
	    : "memory");
 | 
						|
 | 
						|
	switch (trap->cause) {
 | 
						|
	case CAUSE_LOAD_ACCESS:
 | 
						|
		trap->cause = CAUSE_FETCH_ACCESS;
 | 
						|
		trap->tinst = 0UL;
 | 
						|
		break;
 | 
						|
	case CAUSE_LOAD_PAGE_FAULT:
 | 
						|
		trap->cause = CAUSE_FETCH_PAGE_FAULT;
 | 
						|
		trap->tinst = 0UL;
 | 
						|
		break;
 | 
						|
	case CAUSE_LOAD_GUEST_PAGE_FAULT:
 | 
						|
		trap->cause = CAUSE_FETCH_GUEST_PAGE_FAULT;
 | 
						|
		if (trap->tinst != INSN_PSEUDO_VS_LOAD &&
 | 
						|
		    trap->tinst != INSN_PSEUDO_VS_STORE)
 | 
						|
			trap->tinst = 0UL;
 | 
						|
		break;
 | 
						|
	default:
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	return insn;
 | 
						|
}
 |