| /* | 
 |  *  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; | 
 | } |