Received: by 10.192.165.156 with SMTP id m28csp583040imm; Fri, 13 Apr 2018 04:30:21 -0700 (PDT) X-Google-Smtp-Source: AIpwx48zJxF4cVYjqPPT3MWyPjyUQh6Qdak0AQQvyBY7wlfe/G1oF1d7z4gV+i7rIPM1sCgbzcXV X-Received: by 2002:a17:902:d807:: with SMTP id a7-v6mr4984946plz.314.1523619021035; Fri, 13 Apr 2018 04:30:21 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1523619020; cv=none; d=google.com; s=arc-20160816; b=zWxJ5bYwgQeHSH1AMs9x/t3wgmb4NfrhO/gM2+5M+gGjoOIitP/PPN4xFhBg/Di0Qd NqAP2ZqEZtQb/cJtRhUox00okQCj5hgqsiXe8hCxq9UYJo5N50UDrzNAv5XBCkK1f6l3 vW+fuxu3oMIL8Medpy3g6ULRdG17QcybCemT8fuRl6cmvINdGu0wT0oE23Cx5k9og47e Lb3WqJRHhXpC5o6kRijcXoge4c53i8naqmuENtAo/V8jlJk+xfAfLplwpjsfmezAhax5 QWO05cSO0UncD9EZLAlGk3fsRkTHjISk43grbsls6q6ZA/9Mm4GoKs1kKslXancWRJJb Q+zw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature:arc-authentication-results; bh=mQQyJx9UF6CzKXSGYvLjlizTU8B8DJShDwYd0LYpeCE=; b=BnI1MmHCynqHP8EovCAqBTfMGVnGidDPQDuzdi5zylRt9lZGOYfnlRABHN4oMG7Sap iBq4QkWkQNNlE7lx1+UQnuT/N32LTyCoNBgARNbzw18++XYxJ7ml/d1Q/bnKSJXluCoZ KXGH4W6Fup0lqghWkeVQqD+sRyc8gg2VCxN3jGqn132992ZWAYJeDcQIrl2XppOcCQDB /CVjrvv4TzAbJYt7ErJrmO0qYz3OWl2pBt7QpHwonObc/mlaRNrraBcHZ4ZiqN+4Wuzf htUxfnPzmd9OAMyy5zS4YEWV4gofqaSXfBmho2cGjryUgnN/u1ubjSFy0kvbLpk1bpTW L8Kg== ARC-Authentication-Results: i=1; mx.google.com; dkim=fail header.i=@gmail.com header.s=20161025 header.b=WqA9lbMe; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=redhat.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id b2si493944pgf.13.2018.04.13.04.30.06; Fri, 13 Apr 2018 04:30:20 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=fail header.i=@gmail.com header.s=20161025 header.b=WqA9lbMe; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=redhat.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1754636AbeDML1I (ORCPT + 99 others); Fri, 13 Apr 2018 07:27:08 -0400 Received: from mail-wm0-f66.google.com ([74.125.82.66]:55409 "EHLO mail-wm0-f66.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1754482AbeDMLXg (ORCPT ); Fri, 13 Apr 2018 07:23:36 -0400 Received: by mail-wm0-f66.google.com with SMTP id b127so4470392wmf.5; Fri, 13 Apr 2018 04:23:35 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=sender:from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=mQQyJx9UF6CzKXSGYvLjlizTU8B8DJShDwYd0LYpeCE=; b=WqA9lbMeaQ2xKQioKEb1iZUc61HiNLUgXcNjRgXdX5V3yBbCDCp6EMem/StUj1NwT0 vymxn7XbyfM4zewSKZ566Vfz/gb6KfAdBdbMY5SmGcHBqj1G9D1p0fIKOEofRdndv1Fw Gr5/62X7U2KocOihYPXrWFjuiPqXV2CIFm1fZ0SGU3E6SFhFEa6bFR00iYRJF6lQJcBu ZTiJsbZ3M39EZ+wiXFdpybunkoTdL4UGPs1FjER2V3fTblTowlXWcB5CJBrBC5r+5um+ WNxc0QgKI0q4P+SWql981z1XQUiafLci5zBm/Z/0OBSV+qJO2rwB2aqZv0F12HW+queW VnZA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:sender:from:to:cc:subject:date:message-id :in-reply-to:references:mime-version:content-transfer-encoding; bh=mQQyJx9UF6CzKXSGYvLjlizTU8B8DJShDwYd0LYpeCE=; b=iHK7GQVXAPAA1XomKWNrO39Epd9uR6tTlMNKCtktko9f9+9PMP9yf2NYNelRibze1p tTaIE4ir8+FiynZ/lPF5v1KUQjx+f/0VP8IbimbWulDKR7lc2d59MaZ44FItoIGYFFgY lwn/cAc2lIhwsvTkIs+bPxEhKQIdCzkdzIrO9Jwhy0OZJrxBxqUhprQsKeUH+VTnZgRI hlZnkUTATUapC4FWLtNtHqzVF7Btw6Acy9ycvcn4cB0xX9qmyseQaG5YGsEejv0rVBm8 h/GfzBmC02C5qDAUqZk4huKFgr3/CmBdgcAc8iaOPlW+CFuVhA9bCILUCH3LuBY38riE Nt7w== X-Gm-Message-State: ALQs6tDF1X5ki9FTzZzFFa1NRgeZD3mXJTj/kNIcDpC5I45cKzXZDkED sOELzSMXSzTAzPNwIJvJgSJ0JTWB X-Received: by 10.28.114.15 with SMTP id n15mr3629181wmc.88.1523618614529; Fri, 13 Apr 2018 04:23:34 -0700 (PDT) Received: from 640k.lan ([82.84.121.178]) by smtp.gmail.com with ESMTPSA id n49sm9372900wrn.50.2018.04.13.04.23.32 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Fri, 13 Apr 2018 04:23:33 -0700 (PDT) From: Paolo Bonzini To: linux-kernel@vger.kernel.org, kvm@vger.kernel.org Cc: karahmed@amazon.de, jmattson@google.com, rkrcmar@redhat.com Subject: [PATCH 1/2] X86/KVM: Properly update 'tsc_offset' to represent the running guest Date: Fri, 13 Apr 2018 13:23:26 +0200 Message-Id: <1523618608-30574-2-git-send-email-pbonzini@redhat.com> X-Mailer: git-send-email 1.8.3.1 In-Reply-To: <1523618608-30574-1-git-send-email-pbonzini@redhat.com> References: <1523618608-30574-1-git-send-email-pbonzini@redhat.com> MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: KarimAllah Ahmed Update 'tsc_offset' on vmenty/vmexit of L2 guests to ensure that it always captures the TSC_OFFSET of the running guest whether it is the L1 or L2 guest. Cc: Jim Mattson Cc: Paolo Bonzini Cc: Radim Krčmář Cc: kvm@vger.kernel.org Cc: linux-kernel@vger.kernel.org Suggested-by: Paolo Bonzini Signed-off-by: KarimAllah Ahmed [AMD changes, fix update_ia32_tsc_adjust_msr. - Paolo] Signed-off-by: Paolo Bonzini --- arch/x86/include/asm/kvm_host.h | 1 + arch/x86/kvm/svm.c | 17 ++++++++++++++++- arch/x86/kvm/vmx.c | 25 ++++++++++++++++++++----- arch/x86/kvm/x86.c | 6 ++++-- 4 files changed, 41 insertions(+), 8 deletions(-) diff --git a/arch/x86/include/asm/kvm_host.h b/arch/x86/include/asm/kvm_host.h index 949c977bc4c9..c25775fad4ed 100644 --- a/arch/x86/include/asm/kvm_host.h +++ b/arch/x86/include/asm/kvm_host.h @@ -1013,6 +1013,7 @@ struct kvm_x86_ops { bool (*has_wbinvd_exit)(void); + u64 (*read_l1_tsc_offset)(struct kvm_vcpu *vcpu); void (*write_tsc_offset)(struct kvm_vcpu *vcpu, u64 offset); void (*get_exit_info)(struct kvm_vcpu *vcpu, u64 *info1, u64 *info2); diff --git a/arch/x86/kvm/svm.c b/arch/x86/kvm/svm.c index b3ebc8ad6891..ea7c6d29aca5 100644 --- a/arch/x86/kvm/svm.c +++ b/arch/x86/kvm/svm.c @@ -1424,12 +1424,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; @@ -3323,6 +3334,7 @@ static int nested_svm_vmexit(struct vcpu_svm *svm) /* Restore the original control entries */ copy_vmcb_control_area(vmcb, hsave); + vcpu->arch.tsc_offset = svm->vmcb->control.tsc_offset; kvm_clear_exception_queue(&svm->vcpu); kvm_clear_interrupt_queue(&svm->vcpu); @@ -3483,10 +3495,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); + vcpu->arch.tsc_offset += nested_vmcb->control.tsc_offset; + svm->vmcb->control.tsc_offset = 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; @@ -7102,6 +7116,7 @@ static int svm_unregister_enc_region(struct kvm *kvm, .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, diff --git a/arch/x86/kvm/vmx.c b/arch/x86/kvm/vmx.c index a13c603bdefb..6553419202ee 100644 --- a/arch/x86/kvm/vmx.c +++ b/arch/x86/kvm/vmx.c @@ -2874,6 +2874,17 @@ static void setup_msrs(struct vcpu_vmx *vmx) vmx_update_msr_bitmap(&vmx->vcpu); } +static u64 vmx_read_l1_tsc_offset(struct kvm_vcpu *vcpu) +{ + struct vmcs12 *vmcs12 = get_vmcs12(vcpu); + + if (is_guest_mode(vcpu) && + (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING)) + return vcpu->arch.tsc_offset - vmcs12->tsc_offset; + + return vcpu->arch.tsc_offset; +} + /* * reads and returns guest's timestamp counter "register" * guest_tsc = (host_tsc * tsc multiplier) >> 48 + tsc_offset @@ -11175,11 +11186,8 @@ static int prepare_vmcs02(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12, vmcs_write64(GUEST_IA32_PAT, vmx->vcpu.arch.pat); } - if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING) - vmcs_write64(TSC_OFFSET, - vcpu->arch.tsc_offset + vmcs12->tsc_offset); - else - vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset); + vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset); + if (kvm_has_tsc_control) decache_tsc_multiplier(vmx); @@ -11489,6 +11497,9 @@ static int nested_vmx_run(struct kvm_vcpu *vcpu, bool launch) if (enable_shadow_vmcs) copy_shadow_to_vmcs12(vmx); + if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING) + vcpu->arch.tsc_offset += vmcs12->tsc_offset; + /* * The nested entry process starts with enforcing various prerequisites * on vmcs12 as required by the Intel SDM, and act appropriately when @@ -12035,6 +12046,9 @@ static void nested_vmx_vmexit(struct kvm_vcpu *vcpu, u32 exit_reason, leave_guest_mode(vcpu); + if (vmcs12->cpu_based_vm_exec_control & CPU_BASED_USE_TSC_OFFSETING) + vcpu->arch.tsc_offset -= vmcs12->tsc_offset; + if (likely(!vmx->fail)) { if (exit_reason == -1) sync_vmcs12(vcpu, vmcs12); @@ -12725,6 +12739,7 @@ static int enable_smi_window(struct kvm_vcpu *vcpu) .has_wbinvd_exit = cpu_has_vmx_wbinvd_exit, + .read_l1_tsc_offset = vmx_read_l1_tsc_offset, .write_tsc_offset = vmx_write_tsc_offset, .set_tdp_cr3 = vmx_set_cr3, diff --git a/arch/x86/kvm/x86.c b/arch/x86/kvm/x86.c index 0334b250e102..3f3fba58c960 100644 --- a/arch/x86/kvm/x86.c +++ b/arch/x86/kvm/x86.c @@ -1490,7 +1490,7 @@ static void kvm_track_tsc_matching(struct kvm_vcpu *vcpu) static void update_ia32_tsc_adjust_msr(struct kvm_vcpu *vcpu, s64 offset) { - u64 curr_offset = vcpu->arch.tsc_offset; + u64 curr_offset = kvm_x86_ops->read_l1_tsc_offset(vcpu); vcpu->arch.ia32_tsc_adjust_msr += offset - curr_offset; } @@ -1532,7 +1532,9 @@ static u64 kvm_compute_tsc_offset(struct kvm_vcpu *vcpu, u64 target_tsc) u64 kvm_read_l1_tsc(struct kvm_vcpu *vcpu, u64 host_tsc) { - return vcpu->arch.tsc_offset + kvm_scale_tsc(vcpu, host_tsc); + u64 tsc_offset = kvm_x86_ops->read_l1_tsc_offset(vcpu); + + return tsc_offset + kvm_scale_tsc(vcpu, host_tsc); } EXPORT_SYMBOL_GPL(kvm_read_l1_tsc); -- 1.8.3.1