goto vmgexit_err;
break;
case SVM_VMGEXIT_AP_CREATION:
- if (!is_sev_snp_guest(vcpu))
- goto vmgexit_err;
if (lower_32_bits(control->exit_info_1) != SVM_VMGEXIT_AP_DESTROY)
if (!kvm_ghcb_rax_is_valid(svm))
goto vmgexit_err;
case SVM_VMGEXIT_TERM_REQUEST:
break;
case SVM_VMGEXIT_PSC:
- if (!is_sev_snp_guest(vcpu) || !kvm_ghcb_sw_scratch_is_valid(svm))
+ if (!kvm_ghcb_sw_scratch_is_valid(svm))
goto vmgexit_err;
break;
case SVM_VMGEXIT_GUEST_REQUEST:
case SVM_VMGEXIT_EXT_GUEST_REQUEST:
- if (!is_sev_snp_guest(vcpu) ||
- !PAGE_ALIGNED(control->exit_info_1) ||
+ if (!PAGE_ALIGNED(control->exit_info_1) ||
!PAGE_ALIGNED(control->exit_info_2) ||
control->exit_info_1 == control->exit_info_2)
goto vmgexit_err;
return 0;
}
+static bool is_snp_only_vmgexit(u64 exit_code)
+{
+ switch (exit_code) {
+ case SVM_VMGEXIT_AP_CREATION:
+ case SVM_VMGEXIT_GUEST_REQUEST:
+ case SVM_VMGEXIT_EXT_GUEST_REQUEST:
+ case SVM_VMGEXIT_PSC:
+ return true;
+ default:
+ return false;
+ }
+}
+
int sev_handle_vmgexit(struct kvm_vcpu *vcpu)
{
struct vcpu_svm *svm = to_svm(vcpu);
return 1;
}
+ if (is_snp_only_vmgexit(control->exit_code) && !is_sev_snp_guest(vcpu)) {
+ vcpu_unimpl(vcpu, "vmgexit: exit code %#llx is SNP-only\n",
+ control->exit_code);
+ svm_vmgexit_bad_input(svm, GHCB_ERR_INVALID_EVENT);
+ return 1;
+ }
+
ret = sev_es_validate_vmgexit(svm);
if (ret)
return ret;