arch/x86/kvm/lapic.c | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-)
When the guest uses the APIC periodic timer, if the delay exceeds the
period, the delta will be negative. nsec_to_cycles() may then convert this
delta into an absolute value larger than guest_l1_tsc, resulting in a
negative tscdeadline. Since the hv timer supports a maximum bit width of
cpu_preemption_timer_multi + 32, this causes the hv timer setup to fail and
switch to the sw timer.
Moreover, due to the commit 98c25ead5eda ("KVM: VMX: Move preemption timer
<=> hrtimer dance to common x86"), if the guest is using the sw timer
before blocking, it will continue to use the sw timer after being woken up,
and will not switch back to the hv timer until the relevant APIC timer
register is reprogrammed. Since the periodic timer does not require
frequent APIC timer register programming, the guest may continue to use the
software timer for an extended period.
The reproduction steps and patch verification results at link [1].
[1]: https://github.com/cai-fuqiang/kernel_test/tree/master/period_timer_test
Fixes: 98c25ead5eda ("KVM: VMX: Move preemption timer <=> hrtimer dance to common x86")
Signed-off-by: fuqiang wang <fuqiang.wng@gmail.com>
---
arch/x86/kvm/lapic.c | 8 +++++++-
1 file changed, 7 insertions(+), 1 deletion(-)
diff --git a/arch/x86/kvm/lapic.c b/arch/x86/kvm/lapic.c
index 5fc437341e03..afd349f4d933 100644
--- a/arch/x86/kvm/lapic.c
+++ b/arch/x86/kvm/lapic.c
@@ -2036,6 +2036,9 @@ static void advance_periodic_target_expiration(struct kvm_lapic *apic)
u64 tscl = rdtsc();
ktime_t delta;
+ u64 delta_cycles_u;
+ u64 delta_cycles_s;
+
/*
* Synchronize both deadlines to the same time source or
* differences in the periods (caused by differences in the
@@ -2047,8 +2050,11 @@ static void advance_periodic_target_expiration(struct kvm_lapic *apic)
ktime_add_ns(apic->lapic_timer.target_expiration,
apic->lapic_timer.period);
delta = ktime_sub(apic->lapic_timer.target_expiration, now);
+ delta_cycles_u = nsec_to_cycles(apic->vcpu, abs(delta));
+ delta_cycles_s = delta > 0 ? delta_cycles_u : -delta_cycles_u;
+
apic->lapic_timer.tscdeadline = kvm_read_l1_tsc(apic->vcpu, tscl) +
- nsec_to_cycles(apic->vcpu, delta);
+ delta_cycles_s;
}
static void start_sw_period(struct kvm_lapic *apic)
--
2.47.0
© 2016 - 2025 Red Hat, Inc.