Skip to content

Commit

Permalink
target/riscv: prevent the fence.i from triggering the exception
Browse files Browse the repository at this point in the history
On platforms that do not support Zifencei, executing fence.i will
trigger an exception, this patch eliminates the impact of the
exception

Change-Id: I459cff096374064fdb6e18a9d95170510ae1f32e
Signed-off-by: Xiang W <[email protected]>
  • Loading branch information
wxjstz committed Aug 1, 2023
1 parent c07d925 commit 370262b
Show file tree
Hide file tree
Showing 2 changed files with 31 additions and 2 deletions.
1 change: 1 addition & 0 deletions src/target/riscv/gdb_regs.h
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ enum gdb_regno {
GDB_REGNO_TDATA3 = CSR_TDATA3 + GDB_REGNO_CSR0,
GDB_REGNO_TINFO = CSR_TINFO + GDB_REGNO_CSR0,
GDB_REGNO_MISA = CSR_MISA + GDB_REGNO_CSR0,
GDB_REGNO_MTVEC = CSR_MTVEC + GDB_REGNO_CSR0,
GDB_REGNO_DPC = CSR_DPC + GDB_REGNO_CSR0,
GDB_REGNO_DCSR = CSR_DCSR + GDB_REGNO_CSR0,
GDB_REGNO_DSCRATCH0 = CSR_DSCRATCH0 + GDB_REGNO_CSR0,
Expand Down
32 changes: 30 additions & 2 deletions src/target/riscv/riscv-013.c
Original file line number Diff line number Diff line change
Expand Up @@ -2837,16 +2837,44 @@ static int deassert_reset(struct target *target)

static int execute_fence(struct target *target)
{
int result;
riscv_reg_t saved_mtvec;
struct riscv_program program;
riscv013_info_t *info = get_info(target);

if (dm013_select_target(target) != ERROR_OK)
return ERROR_FAIL;

/* FIXME: For non-coherent systems we need to flush the caches right
* here, but there's no ISA-defined way of doing that. */
struct riscv_program program;

/* backup mtvec */
result = riscv_get_register(target, &saved_mtvec, GDB_REGNO_MTVEC);
if (result != ERROR_OK)
return result;

/* set mtvec after fence.i */
result = riscv_set_register(target, GDB_REGNO_MTVEC, info->progbuf_address + 4);
if (result != ERROR_OK)
return result;

riscv_program_init(&program, target);

/* fence.i may cause exception, Because mtvec is set after fence.i,
* triggering an exception does not affect the execution flow */
riscv_program_fence_i(&program);
result = riscv_program_exec(&program, target);
if (result != ERROR_OK)
LOG_TARGET_DEBUG(target, "Unable to execute pre-fence");

/* restore mtvec */
result = riscv_set_register(target, GDB_REGNO_MTVEC, saved_mtvec);
if (result != ERROR_OK)
return result;

riscv_program_init(&program, target);
riscv_program_fence_rw_rw(&program);
int result = riscv_program_exec(&program, target);
result = riscv_program_exec(&program, target);
if (result != ERROR_OK)
LOG_TARGET_DEBUG(target, "Unable to execute pre-fence");

Expand Down

0 comments on commit 370262b

Please sign in to comment.