diff options
Diffstat (limited to 'arch/x86/kvm/lapic.c')
-rw-r--r-- | arch/x86/kvm/lapic.c | 142 |
1 files changed, 113 insertions, 29 deletions
diff --git a/arch/x86/kvm/lapic.c b/arch/x86/kvm/lapic.c index 2fb20caae5d8..54abb40199d6 100644 --- a/arch/x86/kvm/lapic.c +++ b/arch/x86/kvm/lapic.c | |||
@@ -138,9 +138,23 @@ static inline int apic_lvt_vector(struct kvm_lapic *apic, int lvt_type) | |||
138 | return apic_get_reg(apic, lvt_type) & APIC_VECTOR_MASK; | 138 | return apic_get_reg(apic, lvt_type) & APIC_VECTOR_MASK; |
139 | } | 139 | } |
140 | 140 | ||
141 | static inline int apic_lvtt_oneshot(struct kvm_lapic *apic) | ||
142 | { | ||
143 | return ((apic_get_reg(apic, APIC_LVTT) & | ||
144 | apic->lapic_timer.timer_mode_mask) == APIC_LVT_TIMER_ONESHOT); | ||
145 | } | ||
146 | |||
141 | static inline int apic_lvtt_period(struct kvm_lapic *apic) | 147 | static inline int apic_lvtt_period(struct kvm_lapic *apic) |
142 | { | 148 | { |
143 | return apic_get_reg(apic, APIC_LVTT) & APIC_LVT_TIMER_PERIODIC; | 149 | return ((apic_get_reg(apic, APIC_LVTT) & |
150 | apic->lapic_timer.timer_mode_mask) == APIC_LVT_TIMER_PERIODIC); | ||
151 | } | ||
152 | |||
153 | static inline int apic_lvtt_tscdeadline(struct kvm_lapic *apic) | ||
154 | { | ||
155 | return ((apic_get_reg(apic, APIC_LVTT) & | ||
156 | apic->lapic_timer.timer_mode_mask) == | ||
157 | APIC_LVT_TIMER_TSCDEADLINE); | ||
144 | } | 158 | } |
145 | 159 | ||
146 | static inline int apic_lvt_nmi_mode(u32 lvt_val) | 160 | static inline int apic_lvt_nmi_mode(u32 lvt_val) |
@@ -169,7 +183,7 @@ static inline int apic_x2apic_mode(struct kvm_lapic *apic) | |||
169 | } | 183 | } |
170 | 184 | ||
171 | static unsigned int apic_lvt_mask[APIC_LVT_NUM] = { | 185 | static unsigned int apic_lvt_mask[APIC_LVT_NUM] = { |
172 | LVT_MASK | APIC_LVT_TIMER_PERIODIC, /* LVTT */ | 186 | LVT_MASK , /* part LVTT mask, timer mode mask added at runtime */ |
173 | LVT_MASK | APIC_MODE_MASK, /* LVTTHMR */ | 187 | LVT_MASK | APIC_MODE_MASK, /* LVTTHMR */ |
174 | LVT_MASK | APIC_MODE_MASK, /* LVTPC */ | 188 | LVT_MASK | APIC_MODE_MASK, /* LVTPC */ |
175 | LINT_MASK, LINT_MASK, /* LVT0-1 */ | 189 | LINT_MASK, LINT_MASK, /* LVT0-1 */ |
@@ -572,6 +586,9 @@ static u32 __apic_read(struct kvm_lapic *apic, unsigned int offset) | |||
572 | break; | 586 | break; |
573 | 587 | ||
574 | case APIC_TMCCT: /* Timer CCR */ | 588 | case APIC_TMCCT: /* Timer CCR */ |
589 | if (apic_lvtt_tscdeadline(apic)) | ||
590 | return 0; | ||
591 | |||
575 | val = apic_get_tmcct(apic); | 592 | val = apic_get_tmcct(apic); |
576 | break; | 593 | break; |
577 | 594 | ||
@@ -666,37 +683,40 @@ static void update_divide_count(struct kvm_lapic *apic) | |||
666 | 683 | ||
667 | static void start_apic_timer(struct kvm_lapic *apic) | 684 | static void start_apic_timer(struct kvm_lapic *apic) |
668 | { | 685 | { |
669 | ktime_t now = apic->lapic_timer.timer.base->get_time(); | 686 | ktime_t now; |
670 | |||
671 | apic->lapic_timer.period = (u64)apic_get_reg(apic, APIC_TMICT) * | ||
672 | APIC_BUS_CYCLE_NS * apic->divide_count; | ||
673 | atomic_set(&apic->lapic_timer.pending, 0); | 687 | atomic_set(&apic->lapic_timer.pending, 0); |
674 | 688 | ||
675 | if (!apic->lapic_timer.period) | 689 | if (apic_lvtt_period(apic) || apic_lvtt_oneshot(apic)) { |
676 | return; | 690 | /* lapic timer in oneshot or peroidic mode */ |
677 | /* | 691 | now = apic->lapic_timer.timer.base->get_time(); |
678 | * Do not allow the guest to program periodic timers with small | 692 | apic->lapic_timer.period = (u64)apic_get_reg(apic, APIC_TMICT) |
679 | * interval, since the hrtimers are not throttled by the host | 693 | * APIC_BUS_CYCLE_NS * apic->divide_count; |
680 | * scheduler. | 694 | |
681 | */ | 695 | if (!apic->lapic_timer.period) |
682 | if (apic_lvtt_period(apic)) { | 696 | return; |
683 | s64 min_period = min_timer_period_us * 1000LL; | 697 | /* |
684 | 698 | * Do not allow the guest to program periodic timers with small | |
685 | if (apic->lapic_timer.period < min_period) { | 699 | * interval, since the hrtimers are not throttled by the host |
686 | pr_info_ratelimited( | 700 | * scheduler. |
687 | "kvm: vcpu %i: requested %lld ns " | 701 | */ |
688 | "lapic timer period limited to %lld ns\n", | 702 | if (apic_lvtt_period(apic)) { |
689 | apic->vcpu->vcpu_id, apic->lapic_timer.period, | 703 | s64 min_period = min_timer_period_us * 1000LL; |
690 | min_period); | 704 | |
691 | apic->lapic_timer.period = min_period; | 705 | if (apic->lapic_timer.period < min_period) { |
706 | pr_info_ratelimited( | ||
707 | "kvm: vcpu %i: requested %lld ns " | ||
708 | "lapic timer period limited to %lld ns\n", | ||
709 | apic->vcpu->vcpu_id, | ||
710 | apic->lapic_timer.period, min_period); | ||
711 | apic->lapic_timer.period = min_period; | ||
712 | } | ||
692 | } | 713 | } |
693 | } | ||
694 | 714 | ||
695 | hrtimer_start(&apic->lapic_timer.timer, | 715 | hrtimer_start(&apic->lapic_timer.timer, |
696 | ktime_add_ns(now, apic->lapic_timer.period), | 716 | ktime_add_ns(now, apic->lapic_timer.period), |
697 | HRTIMER_MODE_ABS); | 717 | HRTIMER_MODE_ABS); |
698 | 718 | ||
699 | apic_debug("%s: bus cycle is %" PRId64 "ns, now 0x%016" | 719 | apic_debug("%s: bus cycle is %" PRId64 "ns, now 0x%016" |
700 | PRIx64 ", " | 720 | PRIx64 ", " |
701 | "timer initial count 0x%x, period %lldns, " | 721 | "timer initial count 0x%x, period %lldns, " |
702 | "expire @ 0x%016" PRIx64 ".\n", __func__, | 722 | "expire @ 0x%016" PRIx64 ".\n", __func__, |
@@ -705,6 +725,30 @@ static void start_apic_timer(struct kvm_lapic *apic) | |||
705 | apic->lapic_timer.period, | 725 | apic->lapic_timer.period, |
706 | ktime_to_ns(ktime_add_ns(now, | 726 | ktime_to_ns(ktime_add_ns(now, |
707 | apic->lapic_timer.period))); | 727 | apic->lapic_timer.period))); |
728 | } else if (apic_lvtt_tscdeadline(apic)) { | ||
729 | /* lapic timer in tsc deadline mode */ | ||
730 | u64 guest_tsc, tscdeadline = apic->lapic_timer.tscdeadline; | ||
731 | u64 ns = 0; | ||
732 | struct kvm_vcpu *vcpu = apic->vcpu; | ||
733 | unsigned long this_tsc_khz = vcpu_tsc_khz(vcpu); | ||
734 | unsigned long flags; | ||
735 | |||
736 | if (unlikely(!tscdeadline || !this_tsc_khz)) | ||
737 | return; | ||
738 | |||
739 | local_irq_save(flags); | ||
740 | |||
741 | now = apic->lapic_timer.timer.base->get_time(); | ||
742 | guest_tsc = kvm_x86_ops->read_l1_tsc(vcpu); | ||
743 | if (likely(tscdeadline > guest_tsc)) { | ||
744 | ns = (tscdeadline - guest_tsc) * 1000000ULL; | ||
745 | do_div(ns, this_tsc_khz); | ||
746 | } | ||
747 | hrtimer_start(&apic->lapic_timer.timer, | ||
748 | ktime_add_ns(now, ns), HRTIMER_MODE_ABS); | ||
749 | |||
750 | local_irq_restore(flags); | ||
751 | } | ||
708 | } | 752 | } |
709 | 753 | ||
710 | static void apic_manage_nmi_watchdog(struct kvm_lapic *apic, u32 lvt0_val) | 754 | static void apic_manage_nmi_watchdog(struct kvm_lapic *apic, u32 lvt0_val) |
@@ -792,7 +836,6 @@ static int apic_reg_write(struct kvm_lapic *apic, u32 reg, u32 val) | |||
792 | 836 | ||
793 | case APIC_LVT0: | 837 | case APIC_LVT0: |
794 | apic_manage_nmi_watchdog(apic, val); | 838 | apic_manage_nmi_watchdog(apic, val); |
795 | case APIC_LVTT: | ||
796 | case APIC_LVTTHMR: | 839 | case APIC_LVTTHMR: |
797 | case APIC_LVTPC: | 840 | case APIC_LVTPC: |
798 | case APIC_LVT1: | 841 | case APIC_LVT1: |
@@ -806,7 +849,22 @@ static int apic_reg_write(struct kvm_lapic *apic, u32 reg, u32 val) | |||
806 | 849 | ||
807 | break; | 850 | break; |
808 | 851 | ||
852 | case APIC_LVTT: | ||
853 | if ((apic_get_reg(apic, APIC_LVTT) & | ||
854 | apic->lapic_timer.timer_mode_mask) != | ||
855 | (val & apic->lapic_timer.timer_mode_mask)) | ||
856 | hrtimer_cancel(&apic->lapic_timer.timer); | ||
857 | |||
858 | if (!apic_sw_enabled(apic)) | ||
859 | val |= APIC_LVT_MASKED; | ||
860 | val &= (apic_lvt_mask[0] | apic->lapic_timer.timer_mode_mask); | ||
861 | apic_set_reg(apic, APIC_LVTT, val); | ||
862 | break; | ||
863 | |||
809 | case APIC_TMICT: | 864 | case APIC_TMICT: |
865 | if (apic_lvtt_tscdeadline(apic)) | ||
866 | break; | ||
867 | |||
810 | hrtimer_cancel(&apic->lapic_timer.timer); | 868 | hrtimer_cancel(&apic->lapic_timer.timer); |
811 | apic_set_reg(apic, APIC_TMICT, val); | 869 | apic_set_reg(apic, APIC_TMICT, val); |
812 | start_apic_timer(apic); | 870 | start_apic_timer(apic); |
@@ -902,6 +960,32 @@ void kvm_free_lapic(struct kvm_vcpu *vcpu) | |||
902 | *---------------------------------------------------------------------- | 960 | *---------------------------------------------------------------------- |
903 | */ | 961 | */ |
904 | 962 | ||
963 | u64 kvm_get_lapic_tscdeadline_msr(struct kvm_vcpu *vcpu) | ||
964 | { | ||
965 | struct kvm_lapic *apic = vcpu->arch.apic; | ||
966 | if (!apic) | ||
967 | return 0; | ||
968 | |||
969 | if (apic_lvtt_oneshot(apic) || apic_lvtt_period(apic)) | ||
970 | return 0; | ||
971 | |||
972 | return apic->lapic_timer.tscdeadline; | ||
973 | } | ||
974 | |||
975 | void kvm_set_lapic_tscdeadline_msr(struct kvm_vcpu *vcpu, u64 data) | ||
976 | { | ||
977 | struct kvm_lapic *apic = vcpu->arch.apic; | ||
978 | if (!apic) | ||
979 | return; | ||
980 | |||
981 | if (apic_lvtt_oneshot(apic) || apic_lvtt_period(apic)) | ||
982 | return; | ||
983 | |||
984 | hrtimer_cancel(&apic->lapic_timer.timer); | ||
985 | apic->lapic_timer.tscdeadline = data; | ||
986 | start_apic_timer(apic); | ||
987 | } | ||
988 | |||
905 | void kvm_lapic_set_tpr(struct kvm_vcpu *vcpu, unsigned long cr8) | 989 | void kvm_lapic_set_tpr(struct kvm_vcpu *vcpu, unsigned long cr8) |
906 | { | 990 | { |
907 | struct kvm_lapic *apic = vcpu->arch.apic; | 991 | struct kvm_lapic *apic = vcpu->arch.apic; |