aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm/lapic.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kvm/lapic.c')
-rw-r--r--arch/x86/kvm/lapic.c24
1 files changed, 21 insertions, 3 deletions
diff --git a/arch/x86/kvm/lapic.c b/arch/x86/kvm/lapic.c
index b1694dc45729..3bde43c3789e 100644
--- a/arch/x86/kvm/lapic.c
+++ b/arch/x86/kvm/lapic.c
@@ -165,29 +165,46 @@ static int find_highest_vector(void *bitmap)
165 165
166static inline int apic_test_and_set_irr(int vec, struct kvm_lapic *apic) 166static inline int apic_test_and_set_irr(int vec, struct kvm_lapic *apic)
167{ 167{
168 apic->irr_pending = true;
168 return apic_test_and_set_vector(vec, apic->regs + APIC_IRR); 169 return apic_test_and_set_vector(vec, apic->regs + APIC_IRR);
169} 170}
170 171
171static inline void apic_clear_irr(int vec, struct kvm_lapic *apic) 172static inline int apic_search_irr(struct kvm_lapic *apic)
172{ 173{
173 apic_clear_vector(vec, apic->regs + APIC_IRR); 174 return find_highest_vector(apic->regs + APIC_IRR);
174} 175}
175 176
176static inline int apic_find_highest_irr(struct kvm_lapic *apic) 177static inline int apic_find_highest_irr(struct kvm_lapic *apic)
177{ 178{
178 int result; 179 int result;
179 180
180 result = find_highest_vector(apic->regs + APIC_IRR); 181 if (!apic->irr_pending)
182 return -1;
183
184 result = apic_search_irr(apic);
181 ASSERT(result == -1 || result >= 16); 185 ASSERT(result == -1 || result >= 16);
182 186
183 return result; 187 return result;
184} 188}
185 189
190static inline void apic_clear_irr(int vec, struct kvm_lapic *apic)
191{
192 apic->irr_pending = false;
193 apic_clear_vector(vec, apic->regs + APIC_IRR);
194 if (apic_search_irr(apic) != -1)
195 apic->irr_pending = true;
196}
197
186int kvm_lapic_find_highest_irr(struct kvm_vcpu *vcpu) 198int kvm_lapic_find_highest_irr(struct kvm_vcpu *vcpu)
187{ 199{
188 struct kvm_lapic *apic = vcpu->arch.apic; 200 struct kvm_lapic *apic = vcpu->arch.apic;
189 int highest_irr; 201 int highest_irr;
190 202
203 /* This may race with setting of irr in __apic_accept_irq() and
204 * value returned may be wrong, but kvm_vcpu_kick() in __apic_accept_irq
205 * will cause vmexit immediately and the value will be recalculated
206 * on the next vmentry.
207 */
191 if (!apic) 208 if (!apic)
192 return 0; 209 return 0;
193 highest_irr = apic_find_highest_irr(apic); 210 highest_irr = apic_find_highest_irr(apic);
@@ -843,6 +860,7 @@ void kvm_lapic_reset(struct kvm_vcpu *vcpu)
843 apic_set_reg(apic, APIC_ISR + 0x10 * i, 0); 860 apic_set_reg(apic, APIC_ISR + 0x10 * i, 0);
844 apic_set_reg(apic, APIC_TMR + 0x10 * i, 0); 861 apic_set_reg(apic, APIC_TMR + 0x10 * i, 0);
845 } 862 }
863 apic->irr_pending = false;
846 update_divide_count(apic); 864 update_divide_count(apic);
847 atomic_set(&apic->lapic_timer.pending, 0); 865 atomic_set(&apic->lapic_timer.pending, 0);
848 if (kvm_vcpu_is_bsp(vcpu)) 866 if (kvm_vcpu_is_bsp(vcpu))