mirror of
				https://github.com/riscv-software-src/opensbi
				synced 2025-10-31 20:18:23 +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;
 | |
| }
 |