diff options
Diffstat (limited to 'arch/x86/kvm/svm.c')
| -rw-r--r-- | arch/x86/kvm/svm.c | 31 | 
1 files changed, 18 insertions, 13 deletions
diff --git a/arch/x86/kvm/svm.c b/arch/x86/kvm/svm.c index b58787daf9f8..1fc05e428aba 100644 --- a/arch/x86/kvm/svm.c +++ b/arch/x86/kvm/svm.c @@ -1423,12 +1423,23 @@ static void init_sys_seg(struct vmcb_seg *seg, uint32_t type)  	seg->base = 0;  } +static u64 svm_read_l1_tsc_offset(struct kvm_vcpu *vcpu) +{ +	struct vcpu_svm *svm = to_svm(vcpu); + +	if (is_guest_mode(vcpu)) +		return svm->nested.hsave->control.tsc_offset; + +	return vcpu->arch.tsc_offset; +} +  static void svm_write_tsc_offset(struct kvm_vcpu *vcpu, u64 offset)  {  	struct vcpu_svm *svm = to_svm(vcpu);  	u64 g_tsc_offset = 0;  	if (is_guest_mode(vcpu)) { +		/* Write L1's TSC offset.  */  		g_tsc_offset = svm->vmcb->control.tsc_offset -  			       svm->nested.hsave->control.tsc_offset;  		svm->nested.hsave->control.tsc_offset = offset; @@ -3322,6 +3333,7 @@ static int nested_svm_vmexit(struct vcpu_svm *svm)  	/* Restore the original control entries */  	copy_vmcb_control_area(vmcb, hsave); +	svm->vcpu.arch.tsc_offset = svm->vmcb->control.tsc_offset;  	kvm_clear_exception_queue(&svm->vcpu);  	kvm_clear_interrupt_queue(&svm->vcpu); @@ -3482,10 +3494,12 @@ static void enter_svm_guest_mode(struct vcpu_svm *svm, u64 vmcb_gpa,  	/* We don't want to see VMMCALLs from a nested guest */  	clr_intercept(svm, INTERCEPT_VMMCALL); +	svm->vcpu.arch.tsc_offset += nested_vmcb->control.tsc_offset; +	svm->vmcb->control.tsc_offset = svm->vcpu.arch.tsc_offset; +  	svm->vmcb->control.virt_ext = nested_vmcb->control.virt_ext;  	svm->vmcb->control.int_vector = nested_vmcb->control.int_vector;  	svm->vmcb->control.int_state = nested_vmcb->control.int_state; -	svm->vmcb->control.tsc_offset += nested_vmcb->control.tsc_offset;  	svm->vmcb->control.event_inj = nested_vmcb->control.event_inj;  	svm->vmcb->control.event_inj_err = nested_vmcb->control.event_inj_err; @@ -4035,12 +4049,6 @@ static int svm_get_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)  	struct vcpu_svm *svm = to_svm(vcpu);  	switch (msr_info->index) { -	case MSR_IA32_TSC: { -		msr_info->data = svm->vmcb->control.tsc_offset + -			kvm_scale_tsc(vcpu, rdtsc()); - -		break; -	}  	case MSR_STAR:  		msr_info->data = svm->vmcb->save.star;  		break; @@ -4193,9 +4201,6 @@ static int svm_set_msr(struct kvm_vcpu *vcpu, struct msr_data *msr)  		svm->vmcb->save.g_pat = data;  		mark_dirty(svm->vmcb, VMCB_NPT);  		break; -	case MSR_IA32_TSC: -		kvm_write_tsc(vcpu, msr); -		break;  	case MSR_IA32_SPEC_CTRL:  		if (!msr->host_initiated &&  		    !guest_cpuid_has(vcpu, X86_FEATURE_IBRS)) @@ -5265,9 +5270,8 @@ static int svm_update_pi_irte(struct kvm *kvm, unsigned int host_irq,  		}  		if (!ret && svm) { -			trace_kvm_pi_irte_update(svm->vcpu.vcpu_id, -						 host_irq, e->gsi, -						 vcpu_info.vector, +			trace_kvm_pi_irte_update(host_irq, svm->vcpu.vcpu_id, +						 e->gsi, vcpu_info.vector,  						 vcpu_info.pi_desc_addr, set);  		} @@ -7102,6 +7106,7 @@ static struct kvm_x86_ops svm_x86_ops __ro_after_init = {  	.has_wbinvd_exit = svm_has_wbinvd_exit, +	.read_l1_tsc_offset = svm_read_l1_tsc_offset,  	.write_tsc_offset = svm_write_tsc_offset,  	.set_tdp_cr3 = set_tdp_cr3,  | 
