Commit 7801bbe1 authored by James Hogan's avatar James Hogan

KVM: MIPS/T&E: Implement CP0_EBase register

The CP0_EBase register is a standard feature of MIPS32r2, so we should
always have been implementing it properly. However the register value
was ignored and wasn't exposed to userland.

Fix the emulation of exceptions and interrupts to use the value stored
in guest CP0_EBase, and fix the masks so that the top 3 bits (rather
than the standard 2) are fixed, so that it is always in the guest KSeg0
segment.

Also add CP0_EBASE to the KVM one_reg interface so it can be accessed by
userland, also allowing the CPU number field to be written (which isn't
permitted by the guest).
Signed-off-by: default avatarJames Hogan <james.hogan@imgtec.com>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: "Radim Krčmář" <rkrcmar@redhat.com>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: linux-mips@linux-mips.org
Cc: kvm@vger.kernel.org
parent 654229a0
...@@ -2074,6 +2074,7 @@ registers, find a list below: ...@@ -2074,6 +2074,7 @@ registers, find a list below:
MIPS | KVM_REG_MIPS_CP0_CAUSE | 32 MIPS | KVM_REG_MIPS_CP0_CAUSE | 32
MIPS | KVM_REG_MIPS_CP0_EPC | 64 MIPS | KVM_REG_MIPS_CP0_EPC | 64
MIPS | KVM_REG_MIPS_CP0_PRID | 32 MIPS | KVM_REG_MIPS_CP0_PRID | 32
MIPS | KVM_REG_MIPS_CP0_EBASE | 64
MIPS | KVM_REG_MIPS_CP0_CONFIG | 32 MIPS | KVM_REG_MIPS_CP0_CONFIG | 32
MIPS | KVM_REG_MIPS_CP0_CONFIG1 | 32 MIPS | KVM_REG_MIPS_CP0_CONFIG1 | 32
MIPS | KVM_REG_MIPS_CP0_CONFIG2 | 32 MIPS | KVM_REG_MIPS_CP0_CONFIG2 | 32
......
...@@ -88,6 +88,7 @@ ...@@ -88,6 +88,7 @@
#define KVM_GUEST_KUSEG 0x00000000UL #define KVM_GUEST_KUSEG 0x00000000UL
#define KVM_GUEST_KSEG0 0x40000000UL #define KVM_GUEST_KSEG0 0x40000000UL
#define KVM_GUEST_KSEG1 0x40000000UL
#define KVM_GUEST_KSEG23 0x60000000UL #define KVM_GUEST_KSEG23 0x60000000UL
#define KVM_GUEST_KSEGX(a) ((_ACAST32_(a)) & 0xe0000000) #define KVM_GUEST_KSEGX(a) ((_ACAST32_(a)) & 0xe0000000)
#define KVM_GUEST_CPHYSADDR(a) ((_ACAST32_(a)) & 0x1fffffff) #define KVM_GUEST_CPHYSADDR(a) ((_ACAST32_(a)) & 0x1fffffff)
...@@ -713,6 +714,8 @@ extern enum emulation_result kvm_mips_emulate_inst(u32 cause, ...@@ -713,6 +714,8 @@ extern enum emulation_result kvm_mips_emulate_inst(u32 cause,
struct kvm_run *run, struct kvm_run *run,
struct kvm_vcpu *vcpu); struct kvm_vcpu *vcpu);
long kvm_mips_guest_exception_base(struct kvm_vcpu *vcpu);
extern enum emulation_result kvm_mips_emulate_syscall(u32 cause, extern enum emulation_result kvm_mips_emulate_syscall(u32 cause,
u32 *opc, u32 *opc,
struct kvm_run *run, struct kvm_run *run,
......
...@@ -1200,14 +1200,13 @@ enum emulation_result kvm_mips_emulate_CP0(union mips_instruction inst, ...@@ -1200,14 +1200,13 @@ enum emulation_result kvm_mips_emulate_CP0(union mips_instruction inst,
er = EMULATE_FAIL; er = EMULATE_FAIL;
break; break;
} }
#define C0_EBASE_CORE_MASK 0xff
if ((rd == MIPS_CP0_PRID) && (sel == 1)) { if ((rd == MIPS_CP0_PRID) && (sel == 1)) {
/* Preserve CORE number */ /*
kvm_change_c0_guest_ebase(cop0, * Preserve core number, and keep the exception
~(C0_EBASE_CORE_MASK), * base in guest KSeg0.
*/
kvm_change_c0_guest_ebase(cop0, 0x1ffff000,
vcpu->arch.gprs[rt]); vcpu->arch.gprs[rt]);
kvm_err("MTCz, cop0->reg[EBASE]: %#lx\n",
kvm_read_c0_guest_ebase(cop0));
} else if (rd == MIPS_CP0_TLB_HI && sel == 0) { } else if (rd == MIPS_CP0_TLB_HI && sel == 0) {
u32 nasid = u32 nasid =
vcpu->arch.gprs[rt] & KVM_ENTRYHI_ASID; vcpu->arch.gprs[rt] & KVM_ENTRYHI_ASID;
...@@ -1917,6 +1916,22 @@ enum emulation_result kvm_mips_emulate_inst(u32 cause, u32 *opc, ...@@ -1917,6 +1916,22 @@ enum emulation_result kvm_mips_emulate_inst(u32 cause, u32 *opc,
return er; return er;
} }
/**
* kvm_mips_guest_exception_base() - Find guest exception vector base address.
*
* Returns: The base address of the current guest exception vector, taking
* both Guest.CP0_Status.BEV and Guest.CP0_EBase into account.
*/
long kvm_mips_guest_exception_base(struct kvm_vcpu *vcpu)
{
struct mips_coproc *cop0 = vcpu->arch.cop0;
if (kvm_read_c0_guest_status(cop0) & ST0_BEV)
return KVM_GUEST_CKSEG1ADDR(0x1fc00200);
else
return kvm_read_c0_guest_ebase(cop0) & MIPS_EBASE_BASE;
}
enum emulation_result kvm_mips_emulate_syscall(u32 cause, enum emulation_result kvm_mips_emulate_syscall(u32 cause,
u32 *opc, u32 *opc,
struct kvm_run *run, struct kvm_run *run,
...@@ -1942,7 +1957,7 @@ enum emulation_result kvm_mips_emulate_syscall(u32 cause, ...@@ -1942,7 +1957,7 @@ enum emulation_result kvm_mips_emulate_syscall(u32 cause,
(EXCCODE_SYS << CAUSEB_EXCCODE)); (EXCCODE_SYS << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver SYSCALL when EXL is already set\n"); kvm_err("Trying to deliver SYSCALL when EXL is already set\n");
...@@ -1976,13 +1991,13 @@ enum emulation_result kvm_mips_emulate_tlbmiss_ld(u32 cause, ...@@ -1976,13 +1991,13 @@ enum emulation_result kvm_mips_emulate_tlbmiss_ld(u32 cause,
arch->pc); arch->pc);
/* set pc to the exception entry point */ /* set pc to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x0; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x0;
} else { } else {
kvm_debug("[EXL == 1] delivering TLB MISS @ pc %#lx\n", kvm_debug("[EXL == 1] delivering TLB MISS @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} }
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
...@@ -2019,16 +2034,14 @@ enum emulation_result kvm_mips_emulate_tlbinv_ld(u32 cause, ...@@ -2019,16 +2034,14 @@ enum emulation_result kvm_mips_emulate_tlbinv_ld(u32 cause,
kvm_debug("[EXL == 0] delivering TLB INV @ pc %#lx\n", kvm_debug("[EXL == 0] delivering TLB INV @ pc %#lx\n",
arch->pc); arch->pc);
/* set pc to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} else { } else {
kvm_debug("[EXL == 1] delivering TLB MISS @ pc %#lx\n", kvm_debug("[EXL == 1] delivering TLB MISS @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} }
/* set pc to the exception entry point */
arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
(EXCCODE_TLBL << CAUSEB_EXCCODE)); (EXCCODE_TLBL << CAUSEB_EXCCODE));
...@@ -2064,11 +2077,11 @@ enum emulation_result kvm_mips_emulate_tlbmiss_st(u32 cause, ...@@ -2064,11 +2077,11 @@ enum emulation_result kvm_mips_emulate_tlbmiss_st(u32 cause,
arch->pc); arch->pc);
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x0; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x0;
} else { } else {
kvm_debug("[EXL == 1] Delivering TLB MISS @ pc %#lx\n", kvm_debug("[EXL == 1] Delivering TLB MISS @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} }
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
...@@ -2104,15 +2117,14 @@ enum emulation_result kvm_mips_emulate_tlbinv_st(u32 cause, ...@@ -2104,15 +2117,14 @@ enum emulation_result kvm_mips_emulate_tlbinv_st(u32 cause,
kvm_debug("[EXL == 0] Delivering TLB MISS @ pc %#lx\n", kvm_debug("[EXL == 0] Delivering TLB MISS @ pc %#lx\n",
arch->pc); arch->pc);
/* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} else { } else {
kvm_debug("[EXL == 1] Delivering TLB MISS @ pc %#lx\n", kvm_debug("[EXL == 1] Delivering TLB MISS @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} }
/* Set PC to the exception entry point */
arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
(EXCCODE_TLBS << CAUSEB_EXCCODE)); (EXCCODE_TLBS << CAUSEB_EXCCODE));
...@@ -2146,14 +2158,13 @@ enum emulation_result kvm_mips_emulate_tlbmod(u32 cause, ...@@ -2146,14 +2158,13 @@ enum emulation_result kvm_mips_emulate_tlbmod(u32 cause,
kvm_debug("[EXL == 0] Delivering TLB MOD @ pc %#lx\n", kvm_debug("[EXL == 0] Delivering TLB MOD @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} else { } else {
kvm_debug("[EXL == 1] Delivering TLB MOD @ pc %#lx\n", kvm_debug("[EXL == 1] Delivering TLB MOD @ pc %#lx\n",
arch->pc); arch->pc);
arch->pc = KVM_GUEST_KSEG0 + 0x180;
} }
arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
(EXCCODE_MOD << CAUSEB_EXCCODE)); (EXCCODE_MOD << CAUSEB_EXCCODE));
...@@ -2185,7 +2196,7 @@ enum emulation_result kvm_mips_emulate_fpu_exc(u32 cause, ...@@ -2185,7 +2196,7 @@ enum emulation_result kvm_mips_emulate_fpu_exc(u32 cause,
} }
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
kvm_change_c0_guest_cause(cop0, (0xff), kvm_change_c0_guest_cause(cop0, (0xff),
(EXCCODE_CPU << CAUSEB_EXCCODE)); (EXCCODE_CPU << CAUSEB_EXCCODE));
...@@ -2219,7 +2230,7 @@ enum emulation_result kvm_mips_emulate_ri_exc(u32 cause, ...@@ -2219,7 +2230,7 @@ enum emulation_result kvm_mips_emulate_ri_exc(u32 cause,
(EXCCODE_RI << CAUSEB_EXCCODE)); (EXCCODE_RI << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver RI when EXL is already set\n"); kvm_err("Trying to deliver RI when EXL is already set\n");
...@@ -2254,7 +2265,7 @@ enum emulation_result kvm_mips_emulate_bp_exc(u32 cause, ...@@ -2254,7 +2265,7 @@ enum emulation_result kvm_mips_emulate_bp_exc(u32 cause,
(EXCCODE_BP << CAUSEB_EXCCODE)); (EXCCODE_BP << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver BP when EXL is already set\n"); kvm_err("Trying to deliver BP when EXL is already set\n");
...@@ -2289,7 +2300,7 @@ enum emulation_result kvm_mips_emulate_trap_exc(u32 cause, ...@@ -2289,7 +2300,7 @@ enum emulation_result kvm_mips_emulate_trap_exc(u32 cause,
(EXCCODE_TR << CAUSEB_EXCCODE)); (EXCCODE_TR << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver TRAP when EXL is already set\n"); kvm_err("Trying to deliver TRAP when EXL is already set\n");
...@@ -2324,7 +2335,7 @@ enum emulation_result kvm_mips_emulate_msafpe_exc(u32 cause, ...@@ -2324,7 +2335,7 @@ enum emulation_result kvm_mips_emulate_msafpe_exc(u32 cause,
(EXCCODE_MSAFPE << CAUSEB_EXCCODE)); (EXCCODE_MSAFPE << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver MSAFPE when EXL is already set\n"); kvm_err("Trying to deliver MSAFPE when EXL is already set\n");
...@@ -2359,7 +2370,7 @@ enum emulation_result kvm_mips_emulate_fpe_exc(u32 cause, ...@@ -2359,7 +2370,7 @@ enum emulation_result kvm_mips_emulate_fpe_exc(u32 cause,
(EXCCODE_FPE << CAUSEB_EXCCODE)); (EXCCODE_FPE << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver FPE when EXL is already set\n"); kvm_err("Trying to deliver FPE when EXL is already set\n");
...@@ -2394,7 +2405,7 @@ enum emulation_result kvm_mips_emulate_msadis_exc(u32 cause, ...@@ -2394,7 +2405,7 @@ enum emulation_result kvm_mips_emulate_msadis_exc(u32 cause,
(EXCCODE_MSADIS << CAUSEB_EXCCODE)); (EXCCODE_MSADIS << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
} else { } else {
kvm_err("Trying to deliver MSADIS when EXL is already set\n"); kvm_err("Trying to deliver MSADIS when EXL is already set\n");
...@@ -2560,7 +2571,7 @@ static enum emulation_result kvm_mips_emulate_exc(u32 cause, ...@@ -2560,7 +2571,7 @@ static enum emulation_result kvm_mips_emulate_exc(u32 cause,
(exccode << CAUSEB_EXCCODE)); (exccode << CAUSEB_EXCCODE));
/* Set PC to the exception entry point */ /* Set PC to the exception entry point */
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc = kvm_mips_guest_exception_base(vcpu) + 0x180;
kvm_write_c0_guest_badvaddr(cop0, vcpu->arch.host_cp0_badvaddr); kvm_write_c0_guest_badvaddr(cop0, vcpu->arch.host_cp0_badvaddr);
kvm_debug("Delivering EXC %d @ pc %#lx, badVaddr: %#lx\n", kvm_debug("Delivering EXC %d @ pc %#lx, badVaddr: %#lx\n",
......
...@@ -183,10 +183,11 @@ int kvm_mips_irq_deliver_cb(struct kvm_vcpu *vcpu, unsigned int priority, ...@@ -183,10 +183,11 @@ int kvm_mips_irq_deliver_cb(struct kvm_vcpu *vcpu, unsigned int priority,
(exccode << CAUSEB_EXCCODE)); (exccode << CAUSEB_EXCCODE));
/* XXXSL Set PC to the interrupt exception entry point */ /* XXXSL Set PC to the interrupt exception entry point */
arch->pc = kvm_mips_guest_exception_base(vcpu);
if (kvm_read_c0_guest_cause(cop0) & CAUSEF_IV) if (kvm_read_c0_guest_cause(cop0) & CAUSEF_IV)
arch->pc = KVM_GUEST_KSEG0 + 0x200; arch->pc += 0x200;
else else
arch->pc = KVM_GUEST_KSEG0 + 0x180; arch->pc += 0x180;
clear_bit(priority, &vcpu->arch.pending_exceptions); clear_bit(priority, &vcpu->arch.pending_exceptions);
} }
......
...@@ -653,6 +653,7 @@ static u64 kvm_trap_emul_get_one_regs[] = { ...@@ -653,6 +653,7 @@ static u64 kvm_trap_emul_get_one_regs[] = {
KVM_REG_MIPS_CP0_CAUSE, KVM_REG_MIPS_CP0_CAUSE,
KVM_REG_MIPS_CP0_EPC, KVM_REG_MIPS_CP0_EPC,
KVM_REG_MIPS_CP0_PRID, KVM_REG_MIPS_CP0_PRID,
KVM_REG_MIPS_CP0_EBASE,
KVM_REG_MIPS_CP0_CONFIG, KVM_REG_MIPS_CP0_CONFIG,
KVM_REG_MIPS_CP0_CONFIG1, KVM_REG_MIPS_CP0_CONFIG1,
KVM_REG_MIPS_CP0_CONFIG2, KVM_REG_MIPS_CP0_CONFIG2,
...@@ -735,6 +736,9 @@ static int kvm_trap_emul_get_one_reg(struct kvm_vcpu *vcpu, ...@@ -735,6 +736,9 @@ static int kvm_trap_emul_get_one_reg(struct kvm_vcpu *vcpu,
case KVM_REG_MIPS_CP0_PRID: case KVM_REG_MIPS_CP0_PRID:
*v = (long)kvm_read_c0_guest_prid(cop0); *v = (long)kvm_read_c0_guest_prid(cop0);
break; break;
case KVM_REG_MIPS_CP0_EBASE:
*v = (long)kvm_read_c0_guest_ebase(cop0);
break;
case KVM_REG_MIPS_CP0_CONFIG: case KVM_REG_MIPS_CP0_CONFIG:
*v = (long)kvm_read_c0_guest_config(cop0); *v = (long)kvm_read_c0_guest_config(cop0);
break; break;
...@@ -837,6 +841,14 @@ static int kvm_trap_emul_set_one_reg(struct kvm_vcpu *vcpu, ...@@ -837,6 +841,14 @@ static int kvm_trap_emul_set_one_reg(struct kvm_vcpu *vcpu,
case KVM_REG_MIPS_CP0_PRID: case KVM_REG_MIPS_CP0_PRID:
kvm_write_c0_guest_prid(cop0, v); kvm_write_c0_guest_prid(cop0, v);
break; break;
case KVM_REG_MIPS_CP0_EBASE:
/*
* Allow core number to be written, but the exception base must
* remain in guest KSeg0.
*/
kvm_change_c0_guest_ebase(cop0, 0x1ffff000 | MIPS_EBASE_CPUNUM,
v);
break;
case KVM_REG_MIPS_CP0_COUNT: case KVM_REG_MIPS_CP0_COUNT:
kvm_mips_write_count(vcpu, v); kvm_mips_write_count(vcpu, v);
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment