[PATCH v6 15/24] KVM: arm64: vgic-its: Read config and pending bit in add_lpi()

Marc Zyngier marc.zyngier at arm.com
Fri May 5 05:50:18 PDT 2017


On 05/05/17 10:57, Christoffer Dall wrote:
> On Thu, May 04, 2017 at 01:44:35PM +0200, Eric Auger wrote:
>> When creating the lpi we now ask the redistributor what is the state
>> of the LPI (priority, enabled, pending).
>>
>> Signed-off-by: Eric Auger <eric.auger at redhat.com>
>>
>> ---
>>
>> v6: creation
>> ---
>>  virt/kvm/arm/vgic/vgic-its.c | 35 ++++++++++++++++++++++++-----------
>>  1 file changed, 24 insertions(+), 11 deletions(-)
>>
>> diff --git a/virt/kvm/arm/vgic/vgic-its.c b/virt/kvm/arm/vgic/vgic-its.c
>> index f43ea30c..3ad615a 100644
>> --- a/virt/kvm/arm/vgic/vgic-its.c
>> +++ b/virt/kvm/arm/vgic/vgic-its.c
>> @@ -38,6 +38,8 @@
>>  
>>  static int vgic_its_set_abi(struct vgic_its *its, int rev);
>>  static const struct vgic_its_abi *vgic_its_get_abi(struct vgic_its *its);
>> +static int update_lpi_config(struct kvm *kvm, struct vgic_irq *irq,
>> +			     struct kvm_vcpu *filter_vcpu);
>>  
>>  /*
>>   * Creates a new (reference to a) struct vgic_irq for a given LPI.
>> @@ -46,10 +48,12 @@ static const struct vgic_its_abi *vgic_its_get_abi(struct vgic_its *its);
>>   * If this is a "new" LPI, we allocate and initialize a new struct vgic_irq.
>>   * This function returns a pointer to the _unlocked_ structure.
>>   */
>> -static struct vgic_irq *vgic_add_lpi(struct kvm *kvm, u32 intid)
>> +static struct vgic_irq *vgic_add_lpi(struct kvm *kvm, u32 intid,
>> +				     struct kvm_vcpu *vcpu)
>>  {
>>  	struct vgic_dist *dist = &kvm->arch.vgic;
>>  	struct vgic_irq *irq = vgic_get_irq(kvm, NULL, intid), *oldirq;
>> +	int ret;
>>  
>>  	/* In this case there is no put, since we keep the reference. */
>>  	if (irq)
>> @@ -66,6 +70,7 @@ static struct vgic_irq *vgic_add_lpi(struct kvm *kvm, u32 intid)
>>  	irq->config = VGIC_CONFIG_EDGE;
>>  	kref_init(&irq->refcount);
>>  	irq->intid = intid;
>> +	irq->target_vcpu = vcpu;
>>  
>>  	spin_lock(&dist->lpi_list_lock);
>>  
>> @@ -97,6 +102,19 @@ static struct vgic_irq *vgic_add_lpi(struct kvm *kvm, u32 intid)
>>  out_unlock:
>>  	spin_unlock(&dist->lpi_list_lock);
>>  
>> +	/*
>> +	 * We "cache" the configuration table entries in out struct vgic_irq's.
> 
> s/out/our/ ?
> 
>> +	 * However we only have those structs for mapped IRQs, so we read in
>> +	 * the respective config data from memory here upon mapping the LPI.
>> +	 */
>> +	ret = update_lpi_config(kvm, irq, NULL);
>> +	if (ret)
>> +		return ERR_PTR(ret);
>> +
>> +	ret = vgic_v3_lpi_sync_pending_status(kvm, irq);
>> +	if (ret)
>> +		return ERR_PTR(ret);
>> +
>>  	return irq;
>>  }
>>  
>> @@ -769,6 +787,7 @@ static int vgic_its_cmd_handle_mapi(struct kvm *kvm, struct vgic_its *its,
>>  	u32 event_id = its_cmd_get_id(its_cmd);
>>  	u32 coll_id = its_cmd_get_collection(its_cmd);
>>  	struct its_ite *ite;
>> +	struct kvm_vcpu *vcpu = NULL;
>>  	struct its_device *device;
>>  	struct its_collection *collection, *new_coll = NULL;
>>  	int lpi_nr;
>> @@ -814,7 +833,10 @@ static int vgic_its_cmd_handle_mapi(struct kvm *kvm, struct vgic_its *its,
>>  	ite->collection = collection;
>>  	ite->lpi = lpi_nr;
>>  
>> -	irq = vgic_add_lpi(kvm, lpi_nr);
>> +	if (its_is_collection_mapped(collection))
>> +		vcpu = kvm_get_vcpu(kvm, collection->target_addr);
>> +
>> +	irq = vgic_add_lpi(kvm, lpi_nr, vcpu);
> 
> So, if we don't have the collection and therefore end up with irq->vcpu
> == NULL, how do we ever read the pending bit from memory as the affinity
> may later change?
> 
> Is this a problem with our idea of only looking at the pending bit on
> vgic_add_lpi, or does it just mean that we should sample the pending bit
> whenever we move LPIs around to VCPUs and if the bit is set, then also
> set the pennding_latch (if not already set), or what should happen here?

It means that we would need to sample that bit on MOVI and maybe MOVALL
as well, but this feels a bit odd. How did that bit land there the first
place?

Thanks,

	M.
-- 
Jazz is not dead. It just smells funny...



More information about the linux-arm-kernel mailing list