forked from mirrors/linux
		
	x86/kvm: Support Hyper-V reenlightenment
When running nested KVM on Hyper-V guests its required to update masterclocks for all guests when L1 migrates to a host with different TSC frequency. Implement the procedure in the following way: - Pause all guests. - Tell the host (Hyper-V) to stop emulating TSC accesses. - Update the gtod copy, recompute clocks. - Unpause all guests. This is somewhat similar to cpufreq but there are two important differences: - TSC emulation can only be disabled globally (on all CPUs) - The new TSC frequency is not known until emulation is turned off so there is no way to 'prepare' for the event upfront. Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com> Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Acked-by: Paolo Bonzini <pbonzini@redhat.com> Cc: Stephen Hemminger <sthemmin@microsoft.com> Cc: kvm@vger.kernel.org Cc: Radim Krčmář <rkrcmar@redhat.com> Cc: Haiyang Zhang <haiyangz@microsoft.com> Cc: "Michael Kelley (EOSG)" <Michael.H.Kelley@microsoft.com> Cc: Roman Kagan <rkagan@virtuozzo.com> Cc: Andy Lutomirski <luto@kernel.org> Cc: devel@linuxdriverproject.org Cc: "K. Y. Srinivasan" <kys@microsoft.com> Cc: Cathy Avery <cavery@redhat.com> Cc: Mohammed Gamal <mmorsy@redhat.com> Link: https://lkml.kernel.org/r/20180124132337.30138-8-vkuznets@redhat.com
This commit is contained in:
		
							parent
							
								
									b0c39dc68e
								
							
						
					
					
						commit
						0092e4346f
					
				
					 1 changed files with 45 additions and 0 deletions
				
			
		| 
						 | 
					@ -68,6 +68,7 @@
 | 
				
			||||||
#include <asm/div64.h>
 | 
					#include <asm/div64.h>
 | 
				
			||||||
#include <asm/irq_remapping.h>
 | 
					#include <asm/irq_remapping.h>
 | 
				
			||||||
#include <asm/mshyperv.h>
 | 
					#include <asm/mshyperv.h>
 | 
				
			||||||
 | 
					#include <asm/hypervisor.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define CREATE_TRACE_POINTS
 | 
					#define CREATE_TRACE_POINTS
 | 
				
			||||||
#include "trace.h"
 | 
					#include "trace.h"
 | 
				
			||||||
| 
						 | 
					@ -5932,6 +5933,43 @@ static void tsc_khz_changed(void *data)
 | 
				
			||||||
	__this_cpu_write(cpu_tsc_khz, khz);
 | 
						__this_cpu_write(cpu_tsc_khz, khz);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void kvm_hyperv_tsc_notifier(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					#ifdef CONFIG_X86_64
 | 
				
			||||||
 | 
						struct kvm *kvm;
 | 
				
			||||||
 | 
						struct kvm_vcpu *vcpu;
 | 
				
			||||||
 | 
						int cpu;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						spin_lock(&kvm_lock);
 | 
				
			||||||
 | 
						list_for_each_entry(kvm, &vm_list, vm_list)
 | 
				
			||||||
 | 
							kvm_make_mclock_inprogress_request(kvm);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						hyperv_stop_tsc_emulation();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* TSC frequency always matches when on Hyper-V */
 | 
				
			||||||
 | 
						for_each_present_cpu(cpu)
 | 
				
			||||||
 | 
							per_cpu(cpu_tsc_khz, cpu) = tsc_khz;
 | 
				
			||||||
 | 
						kvm_max_guest_tsc_khz = tsc_khz;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						list_for_each_entry(kvm, &vm_list, vm_list) {
 | 
				
			||||||
 | 
							struct kvm_arch *ka = &kvm->arch;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							spin_lock(&ka->pvclock_gtod_sync_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							pvclock_update_vm_gtod_copy(kvm);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							kvm_for_each_vcpu(cpu, vcpu, kvm)
 | 
				
			||||||
 | 
								kvm_make_request(KVM_REQ_CLOCK_UPDATE, vcpu);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							kvm_for_each_vcpu(cpu, vcpu, kvm)
 | 
				
			||||||
 | 
								kvm_clear_request(KVM_REQ_MCLOCK_INPROGRESS, vcpu);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							spin_unlock(&ka->pvclock_gtod_sync_lock);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						spin_unlock(&kvm_lock);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int kvmclock_cpufreq_notifier(struct notifier_block *nb, unsigned long val,
 | 
					static int kvmclock_cpufreq_notifier(struct notifier_block *nb, unsigned long val,
 | 
				
			||||||
				     void *data)
 | 
									     void *data)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
| 
						 | 
					@ -6217,6 +6255,9 @@ int kvm_arch_init(void *opaque)
 | 
				
			||||||
	kvm_lapic_init();
 | 
						kvm_lapic_init();
 | 
				
			||||||
#ifdef CONFIG_X86_64
 | 
					#ifdef CONFIG_X86_64
 | 
				
			||||||
	pvclock_gtod_register_notifier(&pvclock_gtod_notifier);
 | 
						pvclock_gtod_register_notifier(&pvclock_gtod_notifier);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (x86_hyper_type == X86_HYPER_MS_HYPERV)
 | 
				
			||||||
 | 
							set_hv_tscchange_cb(kvm_hyperv_tsc_notifier);
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
| 
						 | 
					@ -6229,6 +6270,10 @@ int kvm_arch_init(void *opaque)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void kvm_arch_exit(void)
 | 
					void kvm_arch_exit(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
 | 
					#ifdef CONFIG_X86_64
 | 
				
			||||||
 | 
						if (x86_hyper_type == X86_HYPER_MS_HYPERV)
 | 
				
			||||||
 | 
							clear_hv_tscchange_cb();
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
	kvm_lapic_exit();
 | 
						kvm_lapic_exit();
 | 
				
			||||||
	perf_unregister_guest_info_callbacks(&kvm_guest_cbs);
 | 
						perf_unregister_guest_info_callbacks(&kvm_guest_cbs);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue