dma-buf: remove shared fence staging in reservation object
No need for that any more. Just replace the list when there isn't enough room any more for the additional fence. Signed-off-by: Christian König <christian.koenig@amd.com> Reviewed-by: Junwei Zhang <Jerry.Zhang@amd.com> Reviewed-by: Huang Rui <ray.huang@amd.com> Link: https://patchwork.kernel.org/patch/10626143/
This commit is contained in:
parent
9d32bdbb5b
commit
27836b641c
|
@ -68,105 +68,23 @@ EXPORT_SYMBOL(reservation_seqcount_string);
|
||||||
*/
|
*/
|
||||||
int reservation_object_reserve_shared(struct reservation_object *obj)
|
int reservation_object_reserve_shared(struct reservation_object *obj)
|
||||||
{
|
{
|
||||||
struct reservation_object_list *fobj, *old;
|
struct reservation_object_list *old, *new;
|
||||||
u32 max;
|
unsigned int i, j, k, max;
|
||||||
|
|
||||||
old = reservation_object_get_list(obj);
|
old = reservation_object_get_list(obj);
|
||||||
|
|
||||||
if (old && old->shared_max) {
|
if (old && old->shared_max) {
|
||||||
if (old->shared_count < old->shared_max) {
|
if (old->shared_count < old->shared_max)
|
||||||
/* perform an in-place update */
|
|
||||||
kfree(obj->staged);
|
|
||||||
obj->staged = NULL;
|
|
||||||
return 0;
|
return 0;
|
||||||
} else
|
else
|
||||||
max = old->shared_max * 2;
|
max = old->shared_max * 2;
|
||||||
} else
|
|
||||||
max = 4;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* resize obj->staged or allocate if it doesn't exist,
|
|
||||||
* noop if already correct size
|
|
||||||
*/
|
|
||||||
fobj = krealloc(obj->staged, offsetof(typeof(*fobj), shared[max]),
|
|
||||||
GFP_KERNEL);
|
|
||||||
if (!fobj)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
obj->staged = fobj;
|
|
||||||
fobj->shared_max = max;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(reservation_object_reserve_shared);
|
|
||||||
|
|
||||||
static void
|
|
||||||
reservation_object_add_shared_inplace(struct reservation_object *obj,
|
|
||||||
struct reservation_object_list *fobj,
|
|
||||||
struct dma_fence *fence)
|
|
||||||
{
|
|
||||||
struct dma_fence *signaled = NULL;
|
|
||||||
u32 i, signaled_idx;
|
|
||||||
|
|
||||||
dma_fence_get(fence);
|
|
||||||
|
|
||||||
preempt_disable();
|
|
||||||
write_seqcount_begin(&obj->seq);
|
|
||||||
|
|
||||||
for (i = 0; i < fobj->shared_count; ++i) {
|
|
||||||
struct dma_fence *old_fence;
|
|
||||||
|
|
||||||
old_fence = rcu_dereference_protected(fobj->shared[i],
|
|
||||||
reservation_object_held(obj));
|
|
||||||
|
|
||||||
if (old_fence->context == fence->context) {
|
|
||||||
/* memory barrier is added by write_seqcount_begin */
|
|
||||||
RCU_INIT_POINTER(fobj->shared[i], fence);
|
|
||||||
write_seqcount_end(&obj->seq);
|
|
||||||
preempt_enable();
|
|
||||||
|
|
||||||
dma_fence_put(old_fence);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!signaled && dma_fence_is_signaled(old_fence)) {
|
|
||||||
signaled = old_fence;
|
|
||||||
signaled_idx = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* memory barrier is added by write_seqcount_begin,
|
|
||||||
* fobj->shared_count is protected by this lock too
|
|
||||||
*/
|
|
||||||
if (signaled) {
|
|
||||||
RCU_INIT_POINTER(fobj->shared[signaled_idx], fence);
|
|
||||||
} else {
|
} else {
|
||||||
BUG_ON(fobj->shared_count >= fobj->shared_max);
|
max = 4;
|
||||||
RCU_INIT_POINTER(fobj->shared[fobj->shared_count], fence);
|
|
||||||
fobj->shared_count++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
write_seqcount_end(&obj->seq);
|
new = kmalloc(offsetof(typeof(*new), shared[max]), GFP_KERNEL);
|
||||||
preempt_enable();
|
if (!new)
|
||||||
|
return -ENOMEM;
|
||||||
dma_fence_put(signaled);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
reservation_object_add_shared_replace(struct reservation_object *obj,
|
|
||||||
struct reservation_object_list *old,
|
|
||||||
struct reservation_object_list *fobj,
|
|
||||||
struct dma_fence *fence)
|
|
||||||
{
|
|
||||||
unsigned i, j, k;
|
|
||||||
|
|
||||||
dma_fence_get(fence);
|
|
||||||
|
|
||||||
if (!old) {
|
|
||||||
RCU_INIT_POINTER(fobj->shared[0], fence);
|
|
||||||
fobj->shared_count = 1;
|
|
||||||
goto done;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* no need to bump fence refcounts, rcu_read access
|
* no need to bump fence refcounts, rcu_read access
|
||||||
|
@ -174,46 +92,45 @@ reservation_object_add_shared_replace(struct reservation_object *obj,
|
||||||
* references from the old struct are carried over to
|
* references from the old struct are carried over to
|
||||||
* the new.
|
* the new.
|
||||||
*/
|
*/
|
||||||
for (i = 0, j = 0, k = fobj->shared_max; i < old->shared_count; ++i) {
|
for (i = 0, j = 0, k = max; i < (old ? old->shared_count : 0); ++i) {
|
||||||
struct dma_fence *check;
|
struct dma_fence *fence;
|
||||||
|
|
||||||
check = rcu_dereference_protected(old->shared[i],
|
fence = rcu_dereference_protected(old->shared[i],
|
||||||
reservation_object_held(obj));
|
reservation_object_held(obj));
|
||||||
|
if (dma_fence_is_signaled(fence))
|
||||||
if (check->context == fence->context ||
|
RCU_INIT_POINTER(new->shared[--k], fence);
|
||||||
dma_fence_is_signaled(check))
|
|
||||||
RCU_INIT_POINTER(fobj->shared[--k], check);
|
|
||||||
else
|
else
|
||||||
RCU_INIT_POINTER(fobj->shared[j++], check);
|
RCU_INIT_POINTER(new->shared[j++], fence);
|
||||||
}
|
}
|
||||||
fobj->shared_count = j;
|
new->shared_count = j;
|
||||||
RCU_INIT_POINTER(fobj->shared[fobj->shared_count], fence);
|
new->shared_max = max;
|
||||||
fobj->shared_count++;
|
|
||||||
|
|
||||||
done:
|
|
||||||
preempt_disable();
|
preempt_disable();
|
||||||
write_seqcount_begin(&obj->seq);
|
write_seqcount_begin(&obj->seq);
|
||||||
/*
|
/*
|
||||||
* RCU_INIT_POINTER can be used here,
|
* RCU_INIT_POINTER can be used here,
|
||||||
* seqcount provides the necessary barriers
|
* seqcount provides the necessary barriers
|
||||||
*/
|
*/
|
||||||
RCU_INIT_POINTER(obj->fence, fobj);
|
RCU_INIT_POINTER(obj->fence, new);
|
||||||
write_seqcount_end(&obj->seq);
|
write_seqcount_end(&obj->seq);
|
||||||
preempt_enable();
|
preempt_enable();
|
||||||
|
|
||||||
if (!old)
|
if (!old)
|
||||||
return;
|
return 0;
|
||||||
|
|
||||||
/* Drop the references to the signaled fences */
|
/* Drop the references to the signaled fences */
|
||||||
for (i = k; i < fobj->shared_max; ++i) {
|
for (i = k; i < new->shared_max; ++i) {
|
||||||
struct dma_fence *f;
|
struct dma_fence *fence;
|
||||||
|
|
||||||
f = rcu_dereference_protected(fobj->shared[i],
|
fence = rcu_dereference_protected(new->shared[i],
|
||||||
reservation_object_held(obj));
|
reservation_object_held(obj));
|
||||||
dma_fence_put(f);
|
dma_fence_put(fence);
|
||||||
}
|
}
|
||||||
kfree_rcu(old, rcu);
|
kfree_rcu(old, rcu);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
EXPORT_SYMBOL(reservation_object_reserve_shared);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* reservation_object_add_shared_fence - Add a fence to a shared slot
|
* reservation_object_add_shared_fence - Add a fence to a shared slot
|
||||||
|
@ -226,15 +143,39 @@ done:
|
||||||
void reservation_object_add_shared_fence(struct reservation_object *obj,
|
void reservation_object_add_shared_fence(struct reservation_object *obj,
|
||||||
struct dma_fence *fence)
|
struct dma_fence *fence)
|
||||||
{
|
{
|
||||||
struct reservation_object_list *old, *fobj = obj->staged;
|
struct reservation_object_list *fobj;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
old = reservation_object_get_list(obj);
|
dma_fence_get(fence);
|
||||||
obj->staged = NULL;
|
|
||||||
|
|
||||||
if (!fobj)
|
fobj = reservation_object_get_list(obj);
|
||||||
reservation_object_add_shared_inplace(obj, old, fence);
|
|
||||||
else
|
preempt_disable();
|
||||||
reservation_object_add_shared_replace(obj, old, fobj, fence);
|
write_seqcount_begin(&obj->seq);
|
||||||
|
|
||||||
|
for (i = 0; i < fobj->shared_count; ++i) {
|
||||||
|
struct dma_fence *old_fence;
|
||||||
|
|
||||||
|
old_fence = rcu_dereference_protected(fobj->shared[i],
|
||||||
|
reservation_object_held(obj));
|
||||||
|
if (old_fence->context == fence->context ||
|
||||||
|
dma_fence_is_signaled(old_fence)) {
|
||||||
|
dma_fence_put(old_fence);
|
||||||
|
goto replace;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BUG_ON(fobj->shared_count >= fobj->shared_max);
|
||||||
|
fobj->shared_count++;
|
||||||
|
|
||||||
|
replace:
|
||||||
|
/*
|
||||||
|
* memory barrier is added by write_seqcount_begin,
|
||||||
|
* fobj->shared_count is protected by this lock too
|
||||||
|
*/
|
||||||
|
RCU_INIT_POINTER(fobj->shared[i], fence);
|
||||||
|
write_seqcount_end(&obj->seq);
|
||||||
|
preempt_enable();
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(reservation_object_add_shared_fence);
|
EXPORT_SYMBOL(reservation_object_add_shared_fence);
|
||||||
|
|
||||||
|
@ -343,9 +284,6 @@ retry:
|
||||||
new = dma_fence_get_rcu_safe(&src->fence_excl);
|
new = dma_fence_get_rcu_safe(&src->fence_excl);
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
|
|
||||||
kfree(dst->staged);
|
|
||||||
dst->staged = NULL;
|
|
||||||
|
|
||||||
src_list = reservation_object_get_list(dst);
|
src_list = reservation_object_get_list(dst);
|
||||||
old = reservation_object_get_excl(dst);
|
old = reservation_object_get_excl(dst);
|
||||||
|
|
||||||
|
|
|
@ -68,7 +68,6 @@ struct reservation_object_list {
|
||||||
* @seq: sequence count for managing RCU read-side synchronization
|
* @seq: sequence count for managing RCU read-side synchronization
|
||||||
* @fence_excl: the exclusive fence, if there is one currently
|
* @fence_excl: the exclusive fence, if there is one currently
|
||||||
* @fence: list of current shared fences
|
* @fence: list of current shared fences
|
||||||
* @staged: staged copy of shared fences for RCU updates
|
|
||||||
*/
|
*/
|
||||||
struct reservation_object {
|
struct reservation_object {
|
||||||
struct ww_mutex lock;
|
struct ww_mutex lock;
|
||||||
|
@ -76,7 +75,6 @@ struct reservation_object {
|
||||||
|
|
||||||
struct dma_fence __rcu *fence_excl;
|
struct dma_fence __rcu *fence_excl;
|
||||||
struct reservation_object_list __rcu *fence;
|
struct reservation_object_list __rcu *fence;
|
||||||
struct reservation_object_list *staged;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#define reservation_object_held(obj) lockdep_is_held(&(obj)->lock.base)
|
#define reservation_object_held(obj) lockdep_is_held(&(obj)->lock.base)
|
||||||
|
@ -95,7 +93,6 @@ reservation_object_init(struct reservation_object *obj)
|
||||||
__seqcount_init(&obj->seq, reservation_seqcount_string, &reservation_seqcount_class);
|
__seqcount_init(&obj->seq, reservation_seqcount_string, &reservation_seqcount_class);
|
||||||
RCU_INIT_POINTER(obj->fence, NULL);
|
RCU_INIT_POINTER(obj->fence, NULL);
|
||||||
RCU_INIT_POINTER(obj->fence_excl, NULL);
|
RCU_INIT_POINTER(obj->fence_excl, NULL);
|
||||||
obj->staged = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -124,7 +121,6 @@ reservation_object_fini(struct reservation_object *obj)
|
||||||
|
|
||||||
kfree(fobj);
|
kfree(fobj);
|
||||||
}
|
}
|
||||||
kfree(obj->staged);
|
|
||||||
|
|
||||||
ww_mutex_destroy(&obj->lock);
|
ww_mutex_destroy(&obj->lock);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue