mirror of https://gitee.com/openkylin/linux.git
KVM: SVM: Handle nested selective_cr0 intercept correctly
If we have the following situation with nested svm: 1. Host KVM intercepts cr0 writes 2. Guest hypervisor intercepts only selective cr0 writes Then we get an cr0 write intercept which is handled on the host. But that intercepts may actually be a selective cr0 intercept for the guest. This patch checks for this condition and injects a selective cr0 intercept if needed. Signed-off-by: Joerg Roedel <joerg.roedel@amd.com> Signed-off-by: Avi Kivity <avi@redhat.com>
This commit is contained in:
parent
b44ea385d8
commit
7f5d8b5600
|
@ -1043,6 +1043,27 @@ static void svm_set_cr0(struct kvm_vcpu *vcpu, unsigned long cr0)
|
||||||
{
|
{
|
||||||
struct vcpu_svm *svm = to_svm(vcpu);
|
struct vcpu_svm *svm = to_svm(vcpu);
|
||||||
|
|
||||||
|
if (is_nested(svm)) {
|
||||||
|
/*
|
||||||
|
* We are here because we run in nested mode, the host kvm
|
||||||
|
* intercepts cr0 writes but the l1 hypervisor does not.
|
||||||
|
* But the L1 hypervisor may intercept selective cr0 writes.
|
||||||
|
* This needs to be checked here.
|
||||||
|
*/
|
||||||
|
unsigned long old, new;
|
||||||
|
|
||||||
|
/* Remove bits that would trigger a real cr0 write intercept */
|
||||||
|
old = vcpu->arch.cr0 & SVM_CR0_SELECTIVE_MASK;
|
||||||
|
new = cr0 & SVM_CR0_SELECTIVE_MASK;
|
||||||
|
|
||||||
|
if (old == new) {
|
||||||
|
/* cr0 write with ts and mp unchanged */
|
||||||
|
svm->vmcb->control.exit_code = SVM_EXIT_CR0_SEL_WRITE;
|
||||||
|
if (nested_svm_exit_handled(svm) == NESTED_EXIT_DONE)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_X86_64
|
#ifdef CONFIG_X86_64
|
||||||
if (vcpu->arch.efer & EFER_LME) {
|
if (vcpu->arch.efer & EFER_LME) {
|
||||||
if (!is_paging(vcpu) && (cr0 & X86_CR0_PG)) {
|
if (!is_paging(vcpu) && (cr0 & X86_CR0_PG)) {
|
||||||
|
|
Loading…
Reference in New Issue