KVM: arm/arm64: timer: Add active state caching
Programming the active state in the (re)distributor can be an expensive operation so it makes some sense to try and reduce the number of accesses as much as possible. So far, we program the active state on each VM entry, but there is some opportunity to do less. An obvious solution is to cache the active state in memory, and only program it in the HW when conditions change. But because the HW can also change things under our feet (the active state can transition from 1 to 0 when the guest does an EOI), some precautions have to be taken, which amount to only caching an "inactive" state, and always programing it otherwise. With this in place, we observe a reduction of around 700 cycles on a 2GHz GICv2 platform for a NULL hypercall. Reviewed-by: Christoffer Dall <christoffer.dall@linaro.org> Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
This commit is contained in:
parent
d06a5440a0
commit
9b4a300443
|
@ -322,6 +322,7 @@ void kvm_arch_vcpu_put(struct kvm_vcpu *vcpu)
|
||||||
vcpu->cpu = -1;
|
vcpu->cpu = -1;
|
||||||
|
|
||||||
kvm_arm_set_running_vcpu(NULL);
|
kvm_arm_set_running_vcpu(NULL);
|
||||||
|
kvm_timer_vcpu_put(vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
int kvm_arch_vcpu_ioctl_get_mpstate(struct kvm_vcpu *vcpu,
|
int kvm_arch_vcpu_ioctl_get_mpstate(struct kvm_vcpu *vcpu,
|
||||||
|
|
|
@ -55,6 +55,9 @@ struct arch_timer_cpu {
|
||||||
|
|
||||||
/* VGIC mapping */
|
/* VGIC mapping */
|
||||||
struct irq_phys_map *map;
|
struct irq_phys_map *map;
|
||||||
|
|
||||||
|
/* Active IRQ state caching */
|
||||||
|
bool active_cleared_last;
|
||||||
};
|
};
|
||||||
|
|
||||||
int kvm_timer_hyp_init(void);
|
int kvm_timer_hyp_init(void);
|
||||||
|
@ -74,4 +77,6 @@ bool kvm_timer_should_fire(struct kvm_vcpu *vcpu);
|
||||||
void kvm_timer_schedule(struct kvm_vcpu *vcpu);
|
void kvm_timer_schedule(struct kvm_vcpu *vcpu);
|
||||||
void kvm_timer_unschedule(struct kvm_vcpu *vcpu);
|
void kvm_timer_unschedule(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
|
void kvm_timer_vcpu_put(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -34,6 +34,11 @@ static struct timecounter *timecounter;
|
||||||
static struct workqueue_struct *wqueue;
|
static struct workqueue_struct *wqueue;
|
||||||
static unsigned int host_vtimer_irq;
|
static unsigned int host_vtimer_irq;
|
||||||
|
|
||||||
|
void kvm_timer_vcpu_put(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
vcpu->arch.timer_cpu.active_cleared_last = false;
|
||||||
|
}
|
||||||
|
|
||||||
static cycle_t kvm_phys_timer_read(void)
|
static cycle_t kvm_phys_timer_read(void)
|
||||||
{
|
{
|
||||||
return timecounter->cc->read(timecounter->cc);
|
return timecounter->cc->read(timecounter->cc);
|
||||||
|
@ -130,6 +135,7 @@ static void kvm_timer_update_irq(struct kvm_vcpu *vcpu, bool new_level)
|
||||||
|
|
||||||
BUG_ON(!vgic_initialized(vcpu->kvm));
|
BUG_ON(!vgic_initialized(vcpu->kvm));
|
||||||
|
|
||||||
|
timer->active_cleared_last = false;
|
||||||
timer->irq.level = new_level;
|
timer->irq.level = new_level;
|
||||||
trace_kvm_timer_update_irq(vcpu->vcpu_id, timer->map->virt_irq,
|
trace_kvm_timer_update_irq(vcpu->vcpu_id, timer->map->virt_irq,
|
||||||
timer->irq.level);
|
timer->irq.level);
|
||||||
|
@ -245,10 +251,35 @@ void kvm_timer_flush_hwstate(struct kvm_vcpu *vcpu)
|
||||||
else
|
else
|
||||||
phys_active = false;
|
phys_active = false;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We want to avoid hitting the (re)distributor as much as
|
||||||
|
* possible, as this is a potentially expensive MMIO access
|
||||||
|
* (not to mention locks in the irq layer), and a solution for
|
||||||
|
* this is to cache the "active" state in memory.
|
||||||
|
*
|
||||||
|
* Things to consider: we cannot cache an "active set" state,
|
||||||
|
* because the HW can change this behind our back (it becomes
|
||||||
|
* "clear" in the HW). We must then restrict the caching to
|
||||||
|
* the "clear" state.
|
||||||
|
*
|
||||||
|
* The cache is invalidated on:
|
||||||
|
* - vcpu put, indicating that the HW cannot be trusted to be
|
||||||
|
* in a sane state on the next vcpu load,
|
||||||
|
* - any change in the interrupt state
|
||||||
|
*
|
||||||
|
* Usage conditions:
|
||||||
|
* - cached value is "active clear"
|
||||||
|
* - value to be programmed is "active clear"
|
||||||
|
*/
|
||||||
|
if (timer->active_cleared_last && !phys_active)
|
||||||
|
return;
|
||||||
|
|
||||||
ret = irq_set_irqchip_state(timer->map->irq,
|
ret = irq_set_irqchip_state(timer->map->irq,
|
||||||
IRQCHIP_STATE_ACTIVE,
|
IRQCHIP_STATE_ACTIVE,
|
||||||
phys_active);
|
phys_active);
|
||||||
WARN_ON(ret);
|
WARN_ON(ret);
|
||||||
|
|
||||||
|
timer->active_cleared_last = !phys_active;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in New Issue