2
0
mirror of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-09-04 20:19:47 +08:00

drm/i915: Protect i915_request_await_start from early waits

We need to be extremely careful inside i915_request_await_start() as it
needs to walk the list of requests in the foreign timeline with very
little protection. As we hold our own timeline mutex, we can not nest
inside the signaler's timeline mutex, so all that remains is our RCU
protection. However, to be safe we need to tell the compiler that we may
be traversing the list only under RCU protection, and furthermore we
need to start declaring requests as elements of the timeline from their
construction.

Fixes: 9ddc8ec027 ("drm/i915: Eliminate the trylock for awaiting an earlier request")
Fixes: 6a79d84840 ("drm/i915: Lock signaler timeline while navigating")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200227085723.1961649-11-chris@chris-wilson.co.uk
This commit is contained in:
Chris Wilson 2020-02-27 08:57:14 +00:00
parent 24eba7a998
commit d22d2d073e

View File

@ -290,7 +290,7 @@ bool i915_request_retire(struct i915_request *rq)
spin_unlock_irq(&rq->lock); spin_unlock_irq(&rq->lock);
remove_from_client(rq); remove_from_client(rq);
list_del(&rq->link); list_del_rcu(&rq->link);
intel_context_exit(rq->context); intel_context_exit(rq->context);
intel_context_unpin(rq->context); intel_context_unpin(rq->context);
@ -736,6 +736,8 @@ __i915_request_create(struct intel_context *ce, gfp_t gfp)
rq->infix = rq->ring->emit; /* end of header; start of user payload */ rq->infix = rq->ring->emit; /* end of header; start of user payload */
intel_context_mark_active(ce); intel_context_mark_active(ce);
list_add_tail_rcu(&rq->link, &tl->requests);
return rq; return rq;
err_unwind: err_unwind:
@ -792,13 +794,23 @@ i915_request_await_start(struct i915_request *rq, struct i915_request *signal)
GEM_BUG_ON(i915_request_timeline(rq) == GEM_BUG_ON(i915_request_timeline(rq) ==
rcu_access_pointer(signal->timeline)); rcu_access_pointer(signal->timeline));
if (i915_request_started(signal))
return 0;
fence = NULL; fence = NULL;
rcu_read_lock(); rcu_read_lock();
spin_lock_irq(&signal->lock); spin_lock_irq(&signal->lock);
if (!i915_request_started(signal) && do {
!list_is_first(&signal->link, struct list_head *pos = READ_ONCE(signal->link.prev);
&rcu_dereference(signal->timeline)->requests)) { struct i915_request *prev;
struct i915_request *prev = list_prev_entry(signal, link);
/* Confirm signal has not been retired, the link is valid */
if (unlikely(i915_request_started(signal)))
break;
/* Is signal the earliest request on its timeline? */
if (pos == &rcu_dereference(signal->timeline)->requests)
break;
/* /*
* Peek at the request before us in the timeline. That * Peek at the request before us in the timeline. That
@ -806,13 +818,18 @@ i915_request_await_start(struct i915_request *rq, struct i915_request *signal)
* after acquiring a reference to it, confirm that it is * after acquiring a reference to it, confirm that it is
* still part of the signaler's timeline. * still part of the signaler's timeline.
*/ */
if (i915_request_get_rcu(prev)) { prev = list_entry(pos, typeof(*prev), link);
if (list_next_entry(prev, link) == signal) if (!i915_request_get_rcu(prev))
fence = &prev->fence; break;
else
/* After the strong barrier, confirm prev is still attached */
if (unlikely(READ_ONCE(prev->link.next) != &signal->link)) {
i915_request_put(prev); i915_request_put(prev);
break;
} }
}
fence = &prev->fence;
} while (0);
spin_unlock_irq(&signal->lock); spin_unlock_irq(&signal->lock);
rcu_read_unlock(); rcu_read_unlock();
if (!fence) if (!fence)
@ -1253,8 +1270,6 @@ __i915_request_add_to_timeline(struct i915_request *rq)
0); 0);
} }
list_add_tail(&rq->link, &timeline->requests);
/* /*
* Make sure that no request gazumped us - if it was allocated after * Make sure that no request gazumped us - if it was allocated after
* our i915_request_alloc() and called __i915_request_add() before * our i915_request_alloc() and called __i915_request_add() before