2012-11-22 06:34:09 +04:00
/*
2014-06-26 23:11:34 +04:00
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
* KVM / MIPS : Deliver / Emulate exceptions to the guest kernel
*
* Copyright ( C ) 2012 MIPS Technologies , Inc . All rights reserved .
* Authors : Sanjay Lal < sanjayl @ kymasys . com >
*/
2012-11-22 06:34:09 +04:00
# include <linux/errno.h>
# include <linux/err.h>
# include <linux/module.h>
# include <linux/vmalloc.h>
# include <linux/kvm_host.h>
2014-06-26 23:11:38 +04:00
# include "opcode.h"
# include "interrupt.h"
2012-11-22 06:34:09 +04:00
static gpa_t kvm_trap_emul_gva_to_gpa_cb ( gva_t gva )
{
gpa_t gpa ;
uint32_t kseg = KSEGX ( gva ) ;
if ( ( kseg = = CKSEG0 ) | | ( kseg = = CKSEG1 ) )
gpa = CPHYSADDR ( gva ) ;
else {
2014-06-26 23:11:35 +04:00
kvm_err ( " %s: cannot find GPA for GVA: %#lx \n " , __func__ , gva ) ;
2012-11-22 06:34:09 +04:00
kvm_mips_dump_host_tlbs ( ) ;
gpa = KVM_INVALID_ADDR ;
}
kvm_debug ( " %s: gva %#lx, gpa: %#llx \n " , __func__ , gva , gpa ) ;
return gpa ;
}
static int kvm_trap_emul_handle_cop_unusable ( struct kvm_vcpu * vcpu )
{
2015-02-06 13:56:27 +03:00
struct mips_coproc * cop0 = vcpu - > arch . cop0 ;
2012-11-22 06:34:09 +04:00
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
2015-02-06 13:56:27 +03:00
if ( ( ( cause & CAUSEF_CE ) > > CAUSEB_CE ) = = 1 ) {
/* FPU Unusable */
if ( ! kvm_mips_guest_has_fpu ( & vcpu - > arch ) | |
( kvm_read_c0_guest_status ( cop0 ) & ST0_CU1 ) = = 0 ) {
/*
* Unusable / no FPU in guest :
* deliver guest COP1 Unusable Exception
*/
er = kvm_mips_emulate_fpu_exc ( cause , opc , run , vcpu ) ;
} else {
/* Restore FPU state */
kvm_own_fpu ( vcpu ) ;
er = EMULATE_DONE ;
}
} else {
2012-11-22 06:34:09 +04:00
er = kvm_mips_emulate_inst ( cause , opc , run , vcpu ) ;
2015-02-06 13:56:27 +03:00
}
2012-11-22 06:34:09 +04:00
switch ( er ) {
case EMULATE_DONE :
ret = RESUME_GUEST ;
break ;
case EMULATE_FAIL :
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
break ;
case EMULATE_WAIT :
run - > exit_reason = KVM_EXIT_INTR ;
ret = RESUME_HOST ;
break ;
default :
BUG ( ) ;
}
return ret ;
}
static int kvm_trap_emul_handle_tlb_mod ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long badvaddr = vcpu - > arch . host_cp0_badvaddr ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
if ( KVM_GUEST_KSEGX ( badvaddr ) < KVM_GUEST_KSEG0
| | KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG23 ) {
2014-06-26 23:11:34 +04:00
kvm_debug ( " USER/KSEG23 ADDR TLB MOD fault: cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
er = kvm_mips_handle_tlbmod ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else if ( KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG0 ) {
2014-06-26 23:11:34 +04:00
/*
* XXXKYMA : The guest kernel does not expect to get this fault
* when we are not using HIGHMEM . Need to address this in a
* HIGHMEM kernel
2012-11-22 06:34:09 +04:00
*/
2014-06-26 23:11:35 +04:00
kvm_err ( " TLB MOD fault not handled, cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
kvm_mips_dump_host_tlbs ( ) ;
kvm_arch_vcpu_dump_regs ( vcpu ) ;
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
} else {
2014-06-26 23:11:35 +04:00
kvm_err ( " Illegal TLB Mod fault address , cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
kvm_mips_dump_host_tlbs ( ) ;
kvm_arch_vcpu_dump_regs ( vcpu ) ;
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_tlb_st_miss ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long badvaddr = vcpu - > arch . host_cp0_badvaddr ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
if ( ( ( badvaddr & PAGE_MASK ) = = KVM_GUEST_COMMPAGE_ADDR )
& & KVM_GUEST_KERNEL_MODE ( vcpu ) ) {
if ( kvm_mips_handle_commpage_tlb_fault ( badvaddr , vcpu ) < 0 ) {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else if ( KVM_GUEST_KSEGX ( badvaddr ) < KVM_GUEST_KSEG0
| | KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG23 ) {
2014-06-26 23:11:34 +04:00
kvm_debug ( " USER ADDR TLB LD fault: cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
er = kvm_mips_handle_tlbmiss ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else if ( KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG0 ) {
2014-06-26 23:11:34 +04:00
/*
* All KSEG0 faults are handled by KVM , as the guest kernel does
* not expect to ever get them
2012-11-22 06:34:09 +04:00
*/
if ( kvm_mips_handle_kseg0_tlb_fault
( vcpu - > arch . host_cp0_badvaddr , vcpu ) < 0 ) {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else {
2014-06-26 23:11:34 +04:00
kvm_err ( " Illegal TLB LD fault address , cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
kvm_mips_dump_host_tlbs ( ) ;
kvm_arch_vcpu_dump_regs ( vcpu ) ;
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_tlb_ld_miss ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long badvaddr = vcpu - > arch . host_cp0_badvaddr ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
if ( ( ( badvaddr & PAGE_MASK ) = = KVM_GUEST_COMMPAGE_ADDR )
& & KVM_GUEST_KERNEL_MODE ( vcpu ) ) {
if ( kvm_mips_handle_commpage_tlb_fault ( badvaddr , vcpu ) < 0 ) {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else if ( KVM_GUEST_KSEGX ( badvaddr ) < KVM_GUEST_KSEG0
| | KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG23 ) {
kvm_debug ( " USER ADDR TLB ST fault: PC: %#lx, BadVaddr: %#lx \n " ,
vcpu - > arch . pc , badvaddr ) ;
2014-06-26 23:11:34 +04:00
/*
* User Address ( UA ) fault , this could happen if
* ( 1 ) TLB entry not present / valid in both Guest and shadow host
* TLBs , in this case we pass on the fault to the guest
* kernel and let it handle it .
* ( 2 ) TLB entry is present in the Guest TLB but not in the
* shadow , in this case we inject the TLB from the Guest TLB
* into the shadow host TLB
2012-11-22 06:34:09 +04:00
*/
er = kvm_mips_handle_tlbmiss ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else if ( KVM_GUEST_KSEGX ( badvaddr ) = = KVM_GUEST_KSEG0 ) {
if ( kvm_mips_handle_kseg0_tlb_fault
( vcpu - > arch . host_cp0_badvaddr , vcpu ) < 0 ) {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
} else {
2014-06-26 23:11:35 +04:00
kvm_err ( " Illegal TLB ST fault address , cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
kvm_mips_dump_host_tlbs ( ) ;
kvm_arch_vcpu_dump_regs ( vcpu ) ;
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_addr_err_st ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long badvaddr = vcpu - > arch . host_cp0_badvaddr ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
if ( KVM_GUEST_KERNEL_MODE ( vcpu )
& & ( KSEGX ( badvaddr ) = = CKSEG0 | | KSEGX ( badvaddr ) = = CKSEG1 ) ) {
kvm_debug ( " Emulate Store to MMIO space \n " ) ;
er = kvm_mips_emulate_inst ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_FAIL ) {
2014-06-26 23:11:35 +04:00
kvm_err ( " Emulate Store to MMIO space failed \n " ) ;
2012-11-22 06:34:09 +04:00
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
} else {
run - > exit_reason = KVM_EXIT_MMIO ;
ret = RESUME_HOST ;
}
} else {
2014-06-26 23:11:35 +04:00
kvm_err ( " Address Error (STORE): cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_addr_err_ld ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long badvaddr = vcpu - > arch . host_cp0_badvaddr ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
if ( KSEGX ( badvaddr ) = = CKSEG0 | | KSEGX ( badvaddr ) = = CKSEG1 ) {
kvm_debug ( " Emulate Load from MMIO space @ %#lx \n " , badvaddr ) ;
er = kvm_mips_emulate_inst ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_FAIL ) {
2014-06-26 23:11:35 +04:00
kvm_err ( " Emulate Load from MMIO space failed \n " ) ;
2012-11-22 06:34:09 +04:00
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
} else {
run - > exit_reason = KVM_EXIT_MMIO ;
ret = RESUME_HOST ;
}
} else {
2014-06-26 23:11:35 +04:00
kvm_err ( " Address Error (LOAD): cause %#lx, PC: %p, BadVaddr: %#lx \n " ,
cause , opc , badvaddr ) ;
2012-11-22 06:34:09 +04:00
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
er = EMULATE_FAIL ;
}
return ret ;
}
static int kvm_trap_emul_handle_syscall ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_emulate_syscall ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_res_inst ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_handle_ri ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
static int kvm_trap_emul_handle_break ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_emulate_bp_exc ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE )
ret = RESUME_GUEST ;
else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
2015-02-06 19:03:57 +03:00
static int kvm_trap_emul_handle_trap ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_emulate_trap_exc ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE ) {
ret = RESUME_GUEST ;
} else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
2015-02-06 13:56:27 +03:00
static int kvm_trap_emul_handle_msa_fpe ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_emulate_msafpe_exc ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE ) {
ret = RESUME_GUEST ;
} else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
2015-02-06 13:56:27 +03:00
static int kvm_trap_emul_handle_fpe ( struct kvm_vcpu * vcpu )
{
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
er = kvm_mips_emulate_fpe_exc ( cause , opc , run , vcpu ) ;
if ( er = = EMULATE_DONE ) {
ret = RESUME_GUEST ;
} else {
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
}
return ret ;
}
2015-02-06 13:56:27 +03:00
/**
* kvm_trap_emul_handle_msa_disabled ( ) - Guest used MSA while disabled in root .
* @ vcpu : Virtual CPU context .
*
* Handle when the guest attempts to use MSA when it is disabled .
*/
2015-02-06 14:11:56 +03:00
static int kvm_trap_emul_handle_msa_disabled ( struct kvm_vcpu * vcpu )
{
2015-02-06 13:56:27 +03:00
struct mips_coproc * cop0 = vcpu - > arch . cop0 ;
2015-02-06 14:11:56 +03:00
struct kvm_run * run = vcpu - > run ;
uint32_t __user * opc = ( uint32_t __user * ) vcpu - > arch . pc ;
unsigned long cause = vcpu - > arch . host_cp0_cause ;
enum emulation_result er = EMULATE_DONE ;
int ret = RESUME_GUEST ;
2015-02-06 13:56:27 +03:00
if ( ! kvm_mips_guest_has_msa ( & vcpu - > arch ) | |
( kvm_read_c0_guest_status ( cop0 ) & ( ST0_CU1 | ST0_FR ) ) = = ST0_CU1 ) {
/*
* No MSA in guest , or FPU enabled and not in FR = 1 mode ,
* guest reserved instruction exception
*/
er = kvm_mips_emulate_ri_exc ( cause , opc , run , vcpu ) ;
} else if ( ! ( kvm_read_c0_guest_config5 ( cop0 ) & MIPS_CONF5_MSAEN ) ) {
/* MSA disabled by guest, guest MSA disabled exception */
er = kvm_mips_emulate_msadis_exc ( cause , opc , run , vcpu ) ;
} else {
/* Restore MSA/FPU state */
kvm_own_msa ( vcpu ) ;
er = EMULATE_DONE ;
}
2015-02-06 14:11:56 +03:00
switch ( er ) {
case EMULATE_DONE :
ret = RESUME_GUEST ;
break ;
case EMULATE_FAIL :
run - > exit_reason = KVM_EXIT_INTERNAL_ERROR ;
ret = RESUME_HOST ;
break ;
default :
BUG ( ) ;
}
return ret ;
}
2012-11-22 06:34:09 +04:00
static int kvm_trap_emul_vm_init ( struct kvm * kvm )
{
return 0 ;
}
static int kvm_trap_emul_vcpu_init ( struct kvm_vcpu * vcpu )
{
return 0 ;
}
static int kvm_trap_emul_vcpu_setup ( struct kvm_vcpu * vcpu )
{
struct mips_coproc * cop0 = vcpu - > arch . cop0 ;
uint32_t config1 ;
int vcpu_id = vcpu - > vcpu_id ;
2014-06-26 23:11:34 +04:00
/*
* Arch specific stuff , set up config registers properly so that the
* guest will come up as expected , for now we simulate a MIPS 24 kc
2012-11-22 06:34:09 +04:00
*/
kvm_write_c0_guest_prid ( cop0 , 0x00019300 ) ;
2015-03-04 18:56:47 +03:00
/* Have config1, Cacheable, noncoherent, write-back, write allocate */
kvm_write_c0_guest_config ( cop0 , MIPS_CONF_M | ( 0x3 < < CP0C0_K0 ) |
( 0x1 < < CP0C0_AR ) |
2012-11-22 06:34:09 +04:00
( MMU_TYPE_R4000 < < CP0C0_MT ) ) ;
/* Read the cache characteristics from the host Config1 Register */
config1 = ( read_c0_config1 ( ) & ~ 0x7f ) ;
/* Set up MMU size */
config1 & = ~ ( 0x3f < < 25 ) ;
config1 | = ( ( KVM_MIPS_GUEST_TLB_SIZE - 1 ) < < 25 ) ;
/* We unset some bits that we aren't emulating */
config1 & =
~ ( ( 1 < < CP0C1_C2 ) | ( 1 < < CP0C1_MD ) | ( 1 < < CP0C1_PC ) |
( 1 < < CP0C1_WR ) | ( 1 < < CP0C1_CA ) ) ;
kvm_write_c0_guest_config1 ( cop0 , config1 ) ;
2015-03-04 18:56:47 +03:00
/* Have config3, no tertiary/secondary caches implemented */
kvm_write_c0_guest_config2 ( cop0 , MIPS_CONF_M ) ;
/* MIPS_CONF_M | (read_c0_config2() & 0xfff) */
2014-06-26 18:11:29 +04:00
/* Have config4, UserLocal */
kvm_write_c0_guest_config3 ( cop0 , MIPS_CONF_M | MIPS_CONF3_ULRI ) ;
/* Have config5 */
kvm_write_c0_guest_config4 ( cop0 , MIPS_CONF_M ) ;
/* No config6 */
kvm_write_c0_guest_config5 ( cop0 , 0 ) ;
2012-11-22 06:34:09 +04:00
/* Set Wait IE/IXMT Ignore in Config7, IAR, AR */
kvm_write_c0_guest_config7 ( cop0 , ( MIPS_CONF7_WII ) | ( 1 < < 10 ) ) ;
2014-06-26 23:11:34 +04:00
/*
* Setup IntCtl defaults , compatibilty mode for timer interrupts ( HW5 )
*/
2012-11-22 06:34:09 +04:00
kvm_write_c0_guest_intctl ( cop0 , 0xFC000000 ) ;
/* Put in vcpu id as CPUNum into Ebase Reg to handle SMP Guests */
kvm_write_c0_guest_ebase ( cop0 , KVM_GUEST_KSEG0 | ( vcpu_id & 0xFF ) ) ;
return 0 ;
}
2014-05-29 13:16:29 +04:00
static int kvm_trap_emul_get_one_reg ( struct kvm_vcpu * vcpu ,
const struct kvm_one_reg * reg ,
s64 * v )
{
switch ( reg - > id ) {
case KVM_REG_MIPS_CP0_COUNT :
2014-05-29 13:16:35 +04:00
* v = kvm_mips_read_count ( vcpu ) ;
2014-05-29 13:16:29 +04:00
break ;
2014-05-29 13:16:37 +04:00
case KVM_REG_MIPS_COUNT_CTL :
* v = vcpu - > arch . count_ctl ;
break ;
case KVM_REG_MIPS_COUNT_RESUME :
* v = ktime_to_ns ( vcpu - > arch . count_resume ) ;
break ;
2014-05-29 13:16:38 +04:00
case KVM_REG_MIPS_COUNT_HZ :
* v = vcpu - > arch . count_hz ;
break ;
2014-05-29 13:16:29 +04:00
default :
return - EINVAL ;
}
return 0 ;
}
static int kvm_trap_emul_set_one_reg ( struct kvm_vcpu * vcpu ,
const struct kvm_one_reg * reg ,
s64 v )
{
struct mips_coproc * cop0 = vcpu - > arch . cop0 ;
2014-05-29 13:16:37 +04:00
int ret = 0 ;
2014-06-26 18:11:29 +04:00
unsigned int cur , change ;
2014-05-29 13:16:29 +04:00
switch ( reg - > id ) {
case KVM_REG_MIPS_CP0_COUNT :
2014-05-29 13:16:35 +04:00
kvm_mips_write_count ( vcpu , v ) ;
2014-05-29 13:16:29 +04:00
break ;
case KVM_REG_MIPS_CP0_COMPARE :
2014-05-29 13:16:35 +04:00
kvm_mips_write_compare ( vcpu , v ) ;
break ;
case KVM_REG_MIPS_CP0_CAUSE :
/*
* If the timer is stopped or started ( DC bit ) it must look
* atomic with changes to the interrupt pending bits ( TI , IRQ5 ) .
* A timer interrupt should not happen in between .
*/
if ( ( kvm_read_c0_guest_cause ( cop0 ) ^ v ) & CAUSEF_DC ) {
if ( v & CAUSEF_DC ) {
/* disable timer first */
kvm_mips_count_disable_cause ( vcpu ) ;
kvm_change_c0_guest_cause ( cop0 , ~ CAUSEF_DC , v ) ;
} else {
/* enable timer last */
kvm_change_c0_guest_cause ( cop0 , ~ CAUSEF_DC , v ) ;
kvm_mips_count_enable_cause ( vcpu ) ;
}
} else {
kvm_write_c0_guest_cause ( cop0 , v ) ;
}
2014-05-29 13:16:29 +04:00
break ;
2014-06-26 18:11:29 +04:00
case KVM_REG_MIPS_CP0_CONFIG :
/* read-only for now */
break ;
case KVM_REG_MIPS_CP0_CONFIG1 :
cur = kvm_read_c0_guest_config1 ( cop0 ) ;
change = ( cur ^ v ) & kvm_mips_config1_wrmask ( vcpu ) ;
if ( change ) {
v = cur ^ change ;
kvm_write_c0_guest_config1 ( cop0 , v ) ;
}
break ;
case KVM_REG_MIPS_CP0_CONFIG2 :
/* read-only for now */
break ;
case KVM_REG_MIPS_CP0_CONFIG3 :
cur = kvm_read_c0_guest_config3 ( cop0 ) ;
change = ( cur ^ v ) & kvm_mips_config3_wrmask ( vcpu ) ;
if ( change ) {
v = cur ^ change ;
kvm_write_c0_guest_config3 ( cop0 , v ) ;
}
break ;
case KVM_REG_MIPS_CP0_CONFIG4 :
cur = kvm_read_c0_guest_config4 ( cop0 ) ;
change = ( cur ^ v ) & kvm_mips_config4_wrmask ( vcpu ) ;
if ( change ) {
v = cur ^ change ;
kvm_write_c0_guest_config4 ( cop0 , v ) ;
}
break ;
case KVM_REG_MIPS_CP0_CONFIG5 :
cur = kvm_read_c0_guest_config5 ( cop0 ) ;
change = ( cur ^ v ) & kvm_mips_config5_wrmask ( vcpu ) ;
if ( change ) {
v = cur ^ change ;
kvm_write_c0_guest_config5 ( cop0 , v ) ;
}
break ;
2014-05-29 13:16:37 +04:00
case KVM_REG_MIPS_COUNT_CTL :
ret = kvm_mips_set_count_ctl ( vcpu , v ) ;
break ;
case KVM_REG_MIPS_COUNT_RESUME :
ret = kvm_mips_set_count_resume ( vcpu , v ) ;
break ;
2014-05-29 13:16:38 +04:00
case KVM_REG_MIPS_COUNT_HZ :
ret = kvm_mips_set_count_hz ( vcpu , v ) ;
break ;
2014-05-29 13:16:29 +04:00
default :
return - EINVAL ;
}
2014-05-29 13:16:37 +04:00
return ret ;
2014-05-29 13:16:29 +04:00
}
2015-02-09 19:35:20 +03:00
static int kvm_trap_emul_vcpu_get_regs ( struct kvm_vcpu * vcpu )
{
2014-11-18 17:09:12 +03:00
kvm_lose_fpu ( vcpu ) ;
2015-02-09 19:35:20 +03:00
return 0 ;
}
static int kvm_trap_emul_vcpu_set_regs ( struct kvm_vcpu * vcpu )
{
return 0 ;
}
2012-11-22 06:34:09 +04:00
static struct kvm_mips_callbacks kvm_trap_emul_callbacks = {
/* exit handlers */
. handle_cop_unusable = kvm_trap_emul_handle_cop_unusable ,
. handle_tlb_mod = kvm_trap_emul_handle_tlb_mod ,
. handle_tlb_st_miss = kvm_trap_emul_handle_tlb_st_miss ,
. handle_tlb_ld_miss = kvm_trap_emul_handle_tlb_ld_miss ,
. handle_addr_err_st = kvm_trap_emul_handle_addr_err_st ,
. handle_addr_err_ld = kvm_trap_emul_handle_addr_err_ld ,
. handle_syscall = kvm_trap_emul_handle_syscall ,
. handle_res_inst = kvm_trap_emul_handle_res_inst ,
. handle_break = kvm_trap_emul_handle_break ,
2015-02-06 19:03:57 +03:00
. handle_trap = kvm_trap_emul_handle_trap ,
2015-02-06 13:56:27 +03:00
. handle_msa_fpe = kvm_trap_emul_handle_msa_fpe ,
2015-02-06 13:56:27 +03:00
. handle_fpe = kvm_trap_emul_handle_fpe ,
2015-02-06 14:11:56 +03:00
. handle_msa_disabled = kvm_trap_emul_handle_msa_disabled ,
2012-11-22 06:34:09 +04:00
. vm_init = kvm_trap_emul_vm_init ,
. vcpu_init = kvm_trap_emul_vcpu_init ,
. vcpu_setup = kvm_trap_emul_vcpu_setup ,
. gva_to_gpa = kvm_trap_emul_gva_to_gpa_cb ,
. queue_timer_int = kvm_mips_queue_timer_int_cb ,
. dequeue_timer_int = kvm_mips_dequeue_timer_int_cb ,
. queue_io_int = kvm_mips_queue_io_int_cb ,
. dequeue_io_int = kvm_mips_dequeue_io_int_cb ,
. irq_deliver = kvm_mips_irq_deliver_cb ,
. irq_clear = kvm_mips_irq_clear_cb ,
2014-05-29 13:16:29 +04:00
. get_one_reg = kvm_trap_emul_get_one_reg ,
. set_one_reg = kvm_trap_emul_set_one_reg ,
2015-02-09 19:35:20 +03:00
. vcpu_get_regs = kvm_trap_emul_vcpu_get_regs ,
. vcpu_set_regs = kvm_trap_emul_vcpu_set_regs ,
2012-11-22 06:34:09 +04:00
} ;
int kvm_mips_emulation_init ( struct kvm_mips_callbacks * * install_callbacks )
{
* install_callbacks = & kvm_trap_emul_callbacks ;
return 0 ;
}