1da8c70eaSSong Gao /* SPDX-License-Identifier: GPL-2.0-or-later */ 2da8c70eaSSong Gao /* 3da8c70eaSSong Gao * QEMU LoongArch user cpu_loop. 4da8c70eaSSong Gao * 5da8c70eaSSong Gao * Copyright (c) 2021 Loongson Technology Corporation Limited 6da8c70eaSSong Gao */ 7da8c70eaSSong Gao 8da8c70eaSSong Gao #include "qemu/osdep.h" 9da8c70eaSSong Gao #include "qemu.h" 10da8c70eaSSong Gao #include "user-internals.h" 11da8c70eaSSong Gao #include "cpu_loop-common.h" 12da8c70eaSSong Gao #include "signal-common.h" 13da8c70eaSSong Gao 14da8c70eaSSong Gao void cpu_loop(CPULoongArchState *env) 15da8c70eaSSong Gao { 16da8c70eaSSong Gao CPUState *cs = env_cpu(env); 17da8c70eaSSong Gao int trapnr, si_code; 18da8c70eaSSong Gao abi_long ret; 19da8c70eaSSong Gao 20da8c70eaSSong Gao for (;;) { 21da8c70eaSSong Gao cpu_exec_start(cs); 22da8c70eaSSong Gao trapnr = cpu_exec(cs); 23da8c70eaSSong Gao cpu_exec_end(cs); 24da8c70eaSSong Gao process_queued_cpu_work(cs); 25da8c70eaSSong Gao 26da8c70eaSSong Gao switch (trapnr) { 27da8c70eaSSong Gao case EXCP_INTERRUPT: 28da8c70eaSSong Gao /* just indicate that signals should be handled asap */ 29da8c70eaSSong Gao break; 30da8c70eaSSong Gao case EXCCODE_SYS: 31da8c70eaSSong Gao env->pc += 4; 32da8c70eaSSong Gao ret = do_syscall(env, env->gpr[11], 33da8c70eaSSong Gao env->gpr[4], env->gpr[5], 34da8c70eaSSong Gao env->gpr[6], env->gpr[7], 35da8c70eaSSong Gao env->gpr[8], env->gpr[9], 36da8c70eaSSong Gao -1, -1); 37da8c70eaSSong Gao if (ret == -QEMU_ERESTARTSYS) { 38da8c70eaSSong Gao env->pc -= 4; 39da8c70eaSSong Gao break; 40da8c70eaSSong Gao } 41da8c70eaSSong Gao if (ret == -QEMU_ESIGRETURN) { 42da8c70eaSSong Gao /* 43da8c70eaSSong Gao * Returning from a successful sigreturn syscall. 44da8c70eaSSong Gao * Avoid clobbering register state. 45da8c70eaSSong Gao */ 46da8c70eaSSong Gao break; 47da8c70eaSSong Gao } 48da8c70eaSSong Gao env->gpr[4] = ret; 49da8c70eaSSong Gao break; 50da8c70eaSSong Gao case EXCCODE_INE: 51da8c70eaSSong Gao force_sig_fault(TARGET_SIGILL, 0, env->pc); 52da8c70eaSSong Gao break; 53da8c70eaSSong Gao case EXCCODE_FPE: 54da8c70eaSSong Gao si_code = TARGET_FPE_FLTUNK; 55da8c70eaSSong Gao if (GET_FP_CAUSE(env->fcsr0) & FP_INVALID) { 56da8c70eaSSong Gao si_code = TARGET_FPE_FLTINV; 57da8c70eaSSong Gao } else if (GET_FP_CAUSE(env->fcsr0) & FP_DIV0) { 58da8c70eaSSong Gao si_code = TARGET_FPE_FLTDIV; 59da8c70eaSSong Gao } else if (GET_FP_CAUSE(env->fcsr0) & FP_OVERFLOW) { 60da8c70eaSSong Gao si_code = TARGET_FPE_FLTOVF; 61da8c70eaSSong Gao } else if (GET_FP_CAUSE(env->fcsr0) & FP_UNDERFLOW) { 62da8c70eaSSong Gao si_code = TARGET_FPE_FLTUND; 63da8c70eaSSong Gao } else if (GET_FP_CAUSE(env->fcsr0) & FP_INEXACT) { 64da8c70eaSSong Gao si_code = TARGET_FPE_FLTRES; 65da8c70eaSSong Gao } 66da8c70eaSSong Gao force_sig_fault(TARGET_SIGFPE, si_code, env->pc); 67da8c70eaSSong Gao break; 68da8c70eaSSong Gao case EXCP_DEBUG: 69da8c70eaSSong Gao case EXCCODE_BRK: 70da8c70eaSSong Gao force_sig_fault(TARGET_SIGTRAP, TARGET_TRAP_BRKPT, env->pc); 71da8c70eaSSong Gao break; 72da8c70eaSSong Gao case EXCCODE_BCE: 73da8c70eaSSong Gao force_sig_fault(TARGET_SIGSYS, TARGET_SI_KERNEL, env->pc); 74da8c70eaSSong Gao break; 75*91ffd93bSRichard Henderson 76*91ffd93bSRichard Henderson /* 77*91ffd93bSRichard Henderson * Begin with LSX and LASX disabled, then enable on the first trap. 78*91ffd93bSRichard Henderson * In this way we can tell if the unit is in use. This is used to 79*91ffd93bSRichard Henderson * choose the layout of any signal frame. 80*91ffd93bSRichard Henderson */ 81*91ffd93bSRichard Henderson case EXCCODE_SXD: 82*91ffd93bSRichard Henderson env->CSR_EUEN |= R_CSR_EUEN_SXE_MASK; 83*91ffd93bSRichard Henderson break; 84*91ffd93bSRichard Henderson case EXCCODE_ASXD: 85*91ffd93bSRichard Henderson env->CSR_EUEN |= R_CSR_EUEN_ASXE_MASK; 86*91ffd93bSRichard Henderson break; 87*91ffd93bSRichard Henderson 88da8c70eaSSong Gao case EXCP_ATOMIC: 89da8c70eaSSong Gao cpu_exec_step_atomic(cs); 90da8c70eaSSong Gao break; 91da8c70eaSSong Gao default: 92da8c70eaSSong Gao EXCP_DUMP(env, "qemu: unhandled CPU exception 0x%x - aborting\n", 93da8c70eaSSong Gao trapnr); 94da8c70eaSSong Gao exit(EXIT_FAILURE); 95da8c70eaSSong Gao } 96da8c70eaSSong Gao process_pending_signals(env); 97da8c70eaSSong Gao } 98da8c70eaSSong Gao } 99da8c70eaSSong Gao 100da8c70eaSSong Gao void target_cpu_copy_regs(CPUArchState *env, struct target_pt_regs *regs) 101da8c70eaSSong Gao { 102da8c70eaSSong Gao int i; 103da8c70eaSSong Gao 104da8c70eaSSong Gao for (i = 0; i < 32; i++) { 105da8c70eaSSong Gao env->gpr[i] = regs->regs[i]; 106da8c70eaSSong Gao } 107da8c70eaSSong Gao env->pc = regs->csr.era; 108da8c70eaSSong Gao 109da8c70eaSSong Gao } 110