[PATCH] drm/amdgpu: fix old fence check in amdgpu_fence_emit

Christian König ckoenig.leichtzumerken at gmail.com
Mon Apr 1 13:05:41 UTC 2019


Am 01.04.19 um 04:54 schrieb Zhou, David(ChunMing):
>
>> -----Original Message-----
>> From: amd-gfx <amd-gfx-bounces at lists.freedesktop.org> On Behalf Of
>> Christian K?nig
>> Sent: Saturday, March 30, 2019 2:33 AM
>> To: amd-gfx at lists.freedesktop.org
>> Subject: [PATCH] drm/amdgpu: fix old fence check in amdgpu_fence_emit
>>
>> We don't hold a reference to the old fence, so it can go away any time we are
>> waiting for it to signal.
>>
>> Signed-off-by: Christian König <christian.koenig at amd.com>
>> ---
>>   drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c | 24 ++++++++++++++++-
>> ------
>>   1 file changed, 17 insertions(+), 7 deletions(-)
>>
>> diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c
>> b/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c
>> index ee47c11e92ce..4dee2326b29c 100644
>> --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c
>> +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_fence.c
>> @@ -136,8 +136,9 @@ int amdgpu_fence_emit(struct amdgpu_ring *ring,
>> struct dma_fence **f,  {
>>   	struct amdgpu_device *adev = ring->adev;
>>   	struct amdgpu_fence *fence;
>> -	struct dma_fence *old, **ptr;
>> +	struct dma_fence __rcu **ptr;
>>   	uint32_t seq;
>> +	int r;
>>
>>   	fence = kmem_cache_alloc(amdgpu_fence_slab, GFP_KERNEL);
>>   	if (fence == NULL)
>> @@ -153,15 +154,24 @@ int amdgpu_fence_emit(struct amdgpu_ring *ring,
>> struct dma_fence **f,
>>   			       seq, flags | AMDGPU_FENCE_FLAG_INT);
>>
>>   	ptr = &ring->fence_drv.fences[seq & ring-
>>> fence_drv.num_fences_mask];
>> +	if (unlikely(rcu_dereference_protected(*ptr, 1))) {
> Isn't this line redundant with dma_fence_get_rcu_safe? I think it's unnecessary.
> Otherwise looks ok to me.

The key point is lock()+dma_fence_get_rcu_safe(ptr)+unlock() is rather 
expensive for something which is really unlikely.

So we check here if we already see the variable as NULL and if that is 
true, then we can just skip the whole expensive dance.

Christian.

>
> -David
>> +		struct dma_fence *old;
>> +
>> +		rcu_read_lock();
>> +		old = dma_fence_get_rcu_safe(ptr);
>> +		rcu_read_unlock();
>> +
>> +		if (old) {
>> +			r = dma_fence_wait(old, false);
>> +			dma_fence_put(old);
>> +			if (r)
>> +				return r;
>> +		}
>> +	}
>> +
>>   	/* This function can't be called concurrently anyway, otherwise
>>   	 * emitting the fence would mess up the hardware ring buffer.
>>   	 */
>> -	old = rcu_dereference_protected(*ptr, 1);
>> -	if (old && !dma_fence_is_signaled(old)) {
>> -		DRM_INFO("rcu slot is busy\n");
>> -		dma_fence_wait(old, false);
>> -	}
>> -
>>   	rcu_assign_pointer(*ptr, dma_fence_get(&fence->base));
>>
>>   	*f = &fence->base;
>> --
>> 2.17.1
>>
>> _______________________________________________
>> amd-gfx mailing list
>> amd-gfx at lists.freedesktop.org
>> https://lists.freedesktop.org/mailman/listinfo/amd-gfx



More information about the amd-gfx mailing list