|  | /* | 
|  | *  RISC-V signal definitions | 
|  | * | 
|  | *  Copyright (c) 2019 Mark Corbin | 
|  | * | 
|  | *  This program is free software; you can redistribute it and/or modify | 
|  | *  it under the terms of the GNU General Public License as published by | 
|  | *  the Free Software Foundation; either version 2 of the License, or | 
|  | *  (at your option) any later version. | 
|  | * | 
|  | *  This program is distributed in the hope that it will be useful, | 
|  | *  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | *  GNU General Public License for more details. | 
|  | * | 
|  | *  You should have received a copy of the GNU General Public License | 
|  | *  along with this program; if not, see <http://www.gnu.org/licenses/>. | 
|  | */ | 
|  | #include "qemu/osdep.h" | 
|  |  | 
|  | #include "qemu.h" | 
|  |  | 
|  | /* | 
|  | * Compare with sendsig() in riscv/riscv/exec_machdep.c | 
|  | * Assumes that target stack frame memory is locked. | 
|  | */ | 
|  | abi_long | 
|  | set_sigtramp_args(CPURISCVState *regs, int sig, struct target_sigframe *frame, | 
|  | abi_ulong frame_addr, struct target_sigaction *ka) | 
|  | { | 
|  | /* | 
|  | * Arguments to signal handler: | 
|  | *  a0 (10) = signal number | 
|  | *  a1 (11) = siginfo pointer | 
|  | *  a2 (12) = ucontext pointer | 
|  | *  pc      = signal pointer handler | 
|  | *  sp (2)  = sigframe pointer | 
|  | *  ra (1)  = sigtramp at base of user stack | 
|  | */ | 
|  |  | 
|  | regs->gpr[xA0] = sig; | 
|  | regs->gpr[xA1] = frame_addr + | 
|  | offsetof(struct target_sigframe, sf_si); | 
|  | regs->gpr[xA2] = frame_addr + | 
|  | offsetof(struct target_sigframe, sf_uc); | 
|  | regs->pc = ka->_sa_handler; | 
|  | regs->gpr[xSP] = frame_addr; | 
|  | regs->gpr[xRA] = TARGET_PS_STRINGS - TARGET_SZSIGCODE; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Compare to riscv/riscv/exec_machdep.c sendsig() | 
|  | * Assumes that the memory is locked if frame points to user memory. | 
|  | */ | 
|  | abi_long setup_sigframe_arch(CPURISCVState *env, abi_ulong frame_addr, | 
|  | struct target_sigframe *frame, int flags) | 
|  | { | 
|  | target_mcontext_t *mcp = &frame->sf_uc.uc_mcontext; | 
|  |  | 
|  | get_mcontext(env, mcp, flags); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Compare with get_mcontext() in riscv/riscv/machdep.c | 
|  | * Assumes that the memory is locked if mcp points to user memory. | 
|  | */ | 
|  | abi_long get_mcontext(CPURISCVState *regs, target_mcontext_t *mcp, | 
|  | int flags) | 
|  | { | 
|  |  | 
|  | mcp->mc_gpregs.gp_t[0] = tswap64(regs->gpr[5]); | 
|  | mcp->mc_gpregs.gp_t[1] = tswap64(regs->gpr[6]); | 
|  | mcp->mc_gpregs.gp_t[2] = tswap64(regs->gpr[7]); | 
|  | mcp->mc_gpregs.gp_t[3] = tswap64(regs->gpr[28]); | 
|  | mcp->mc_gpregs.gp_t[4] = tswap64(regs->gpr[29]); | 
|  | mcp->mc_gpregs.gp_t[5] = tswap64(regs->gpr[30]); | 
|  | mcp->mc_gpregs.gp_t[6] = tswap64(regs->gpr[31]); | 
|  |  | 
|  | mcp->mc_gpregs.gp_s[0] = tswap64(regs->gpr[8]); | 
|  | mcp->mc_gpregs.gp_s[1] = tswap64(regs->gpr[9]); | 
|  | mcp->mc_gpregs.gp_s[2] = tswap64(regs->gpr[18]); | 
|  | mcp->mc_gpregs.gp_s[3] = tswap64(regs->gpr[19]); | 
|  | mcp->mc_gpregs.gp_s[4] = tswap64(regs->gpr[20]); | 
|  | mcp->mc_gpregs.gp_s[5] = tswap64(regs->gpr[21]); | 
|  | mcp->mc_gpregs.gp_s[6] = tswap64(regs->gpr[22]); | 
|  | mcp->mc_gpregs.gp_s[7] = tswap64(regs->gpr[23]); | 
|  | mcp->mc_gpregs.gp_s[8] = tswap64(regs->gpr[24]); | 
|  | mcp->mc_gpregs.gp_s[9] = tswap64(regs->gpr[25]); | 
|  | mcp->mc_gpregs.gp_s[10] = tswap64(regs->gpr[26]); | 
|  | mcp->mc_gpregs.gp_s[11] = tswap64(regs->gpr[27]); | 
|  |  | 
|  | mcp->mc_gpregs.gp_a[0] = tswap64(regs->gpr[10]); | 
|  | mcp->mc_gpregs.gp_a[1] = tswap64(regs->gpr[11]); | 
|  | mcp->mc_gpregs.gp_a[2] = tswap64(regs->gpr[12]); | 
|  | mcp->mc_gpregs.gp_a[3] = tswap64(regs->gpr[13]); | 
|  | mcp->mc_gpregs.gp_a[4] = tswap64(regs->gpr[14]); | 
|  | mcp->mc_gpregs.gp_a[5] = tswap64(regs->gpr[15]); | 
|  | mcp->mc_gpregs.gp_a[6] = tswap64(regs->gpr[16]); | 
|  | mcp->mc_gpregs.gp_a[7] = tswap64(regs->gpr[17]); | 
|  |  | 
|  | if (flags & TARGET_MC_GET_CLEAR_RET) { | 
|  | mcp->mc_gpregs.gp_a[0] = 0; /* a0 */ | 
|  | mcp->mc_gpregs.gp_a[1] = 0; /* a1 */ | 
|  | mcp->mc_gpregs.gp_t[0] = 0; /* clear syscall error */ | 
|  | } | 
|  |  | 
|  | mcp->mc_gpregs.gp_ra = tswap64(regs->gpr[1]); | 
|  | mcp->mc_gpregs.gp_sp = tswap64(regs->gpr[2]); | 
|  | mcp->mc_gpregs.gp_gp = tswap64(regs->gpr[3]); | 
|  | mcp->mc_gpregs.gp_tp = tswap64(regs->gpr[4]); | 
|  | mcp->mc_gpregs.gp_sepc = tswap64(regs->pc); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Compare with set_mcontext() in riscv/riscv/exec_machdep.c */ | 
|  | abi_long set_mcontext(CPURISCVState *regs, target_mcontext_t *mcp, | 
|  | int srflag) | 
|  | { | 
|  |  | 
|  | regs->gpr[5] = tswap64(mcp->mc_gpregs.gp_t[0]); | 
|  | regs->gpr[6] = tswap64(mcp->mc_gpregs.gp_t[1]); | 
|  | regs->gpr[7] = tswap64(mcp->mc_gpregs.gp_t[2]); | 
|  | regs->gpr[28] = tswap64(mcp->mc_gpregs.gp_t[3]); | 
|  | regs->gpr[29] = tswap64(mcp->mc_gpregs.gp_t[4]); | 
|  | regs->gpr[30] = tswap64(mcp->mc_gpregs.gp_t[5]); | 
|  | regs->gpr[31] = tswap64(mcp->mc_gpregs.gp_t[6]); | 
|  |  | 
|  | regs->gpr[8] = tswap64(mcp->mc_gpregs.gp_s[0]); | 
|  | regs->gpr[9] = tswap64(mcp->mc_gpregs.gp_s[1]); | 
|  | regs->gpr[18] = tswap64(mcp->mc_gpregs.gp_s[2]); | 
|  | regs->gpr[19] = tswap64(mcp->mc_gpregs.gp_s[3]); | 
|  | regs->gpr[20] = tswap64(mcp->mc_gpregs.gp_s[4]); | 
|  | regs->gpr[21] = tswap64(mcp->mc_gpregs.gp_s[5]); | 
|  | regs->gpr[22] = tswap64(mcp->mc_gpregs.gp_s[6]); | 
|  | regs->gpr[23] = tswap64(mcp->mc_gpregs.gp_s[7]); | 
|  | regs->gpr[24] = tswap64(mcp->mc_gpregs.gp_s[8]); | 
|  | regs->gpr[25] = tswap64(mcp->mc_gpregs.gp_s[9]); | 
|  | regs->gpr[26] = tswap64(mcp->mc_gpregs.gp_s[10]); | 
|  | regs->gpr[27] = tswap64(mcp->mc_gpregs.gp_s[11]); | 
|  |  | 
|  | regs->gpr[10] = tswap64(mcp->mc_gpregs.gp_a[0]); | 
|  | regs->gpr[11] = tswap64(mcp->mc_gpregs.gp_a[1]); | 
|  | regs->gpr[12] = tswap64(mcp->mc_gpregs.gp_a[2]); | 
|  | regs->gpr[13] = tswap64(mcp->mc_gpregs.gp_a[3]); | 
|  | regs->gpr[14] = tswap64(mcp->mc_gpregs.gp_a[4]); | 
|  | regs->gpr[15] = tswap64(mcp->mc_gpregs.gp_a[5]); | 
|  | regs->gpr[16] = tswap64(mcp->mc_gpregs.gp_a[6]); | 
|  | regs->gpr[17] = tswap64(mcp->mc_gpregs.gp_a[7]); | 
|  |  | 
|  |  | 
|  | regs->gpr[1] = tswap64(mcp->mc_gpregs.gp_ra); | 
|  | regs->gpr[2] = tswap64(mcp->mc_gpregs.gp_sp); | 
|  | regs->gpr[3] = tswap64(mcp->mc_gpregs.gp_gp); | 
|  | regs->gpr[4] = tswap64(mcp->mc_gpregs.gp_tp); | 
|  | regs->pc = tswap64(mcp->mc_gpregs.gp_sepc); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Compare with sys_sigreturn() in riscv/riscv/machdep.c */ | 
|  | abi_long get_ucontext_sigreturn(CPURISCVState *regs, | 
|  | abi_ulong target_sf, abi_ulong *target_uc) | 
|  | { | 
|  |  | 
|  | *target_uc = target_sf; | 
|  | return 0; | 
|  | } |