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/gt: Close race between engine_park and intel_gt_retire_requests
The general concept was that intel_timeline.active_count was locked by
the intel_timeline.mutex. The exception was for power management, where
the engine->kernel_context->timeline could be manipulated under the
global wakeref.mutex.
This was quite solid, as we always manipulated the timeline only while
we held an engine wakeref.
And then we started retiring requests outside of struct_mutex, only
using the timelines.active_list and the timeline->mutex. There we
started manipulating intel_timeline.active_count outside of an engine
wakeref, and so introduced a race between __engine_park() and
intel_gt_retire_requests(), a race that could result in the
engine->kernel_context not being added to the active timelines and so
losing requests, which caused us to keep the system permanently powered
up [and unloadable].
The race would be easy to close if we could take the engine wakeref for
the timeline before we retire -- except timelines are not bound to any
engine and so we would need to keep all active engines awake. The
alternative is to guard intel_timeline_enter/intel_timeline_exit for use
outside of the timeline->mutex.
Fixes: e5dadff4b0 ("drm/i915: Protect request retirement with timeline->mutex")
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Matthew Auld <matthew.auld@intel.com>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20191120165514.3955081-1-chris@chris-wilson.co.uk
			
			
This commit is contained in:
		
							parent
							
								
									07779a76ee
								
							
						
					
					
						commit
						a6edbca74b
					
				| @ -49,8 +49,8 @@ long intel_gt_retire_requests_timeout(struct intel_gt *gt, long timeout) | |||||||
| 			continue; | 			continue; | ||||||
| 
 | 
 | ||||||
| 		intel_timeline_get(tl); | 		intel_timeline_get(tl); | ||||||
| 		GEM_BUG_ON(!tl->active_count); | 		GEM_BUG_ON(!atomic_read(&tl->active_count)); | ||||||
| 		tl->active_count++; /* pin the list element */ | 		atomic_inc(&tl->active_count); /* pin the list element */ | ||||||
| 		spin_unlock_irqrestore(&timelines->lock, flags); | 		spin_unlock_irqrestore(&timelines->lock, flags); | ||||||
| 
 | 
 | ||||||
| 		if (timeout > 0) { | 		if (timeout > 0) { | ||||||
| @ -71,14 +71,14 @@ long intel_gt_retire_requests_timeout(struct intel_gt *gt, long timeout) | |||||||
| 
 | 
 | ||||||
| 		/* Resume iteration after dropping lock */ | 		/* Resume iteration after dropping lock */ | ||||||
| 		list_safe_reset_next(tl, tn, link); | 		list_safe_reset_next(tl, tn, link); | ||||||
| 		if (!--tl->active_count) | 		if (atomic_dec_and_test(&tl->active_count)) | ||||||
| 			list_del(&tl->link); | 			list_del(&tl->link); | ||||||
| 
 | 
 | ||||||
| 		mutex_unlock(&tl->mutex); | 		mutex_unlock(&tl->mutex); | ||||||
| 
 | 
 | ||||||
| 		/* Defer the final release to after the spinlock */ | 		/* Defer the final release to after the spinlock */ | ||||||
| 		if (refcount_dec_and_test(&tl->kref.refcount)) { | 		if (refcount_dec_and_test(&tl->kref.refcount)) { | ||||||
| 			GEM_BUG_ON(tl->active_count); | 			GEM_BUG_ON(atomic_read(&tl->active_count)); | ||||||
| 			list_add(&tl->link, &free); | 			list_add(&tl->link, &free); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -334,14 +334,32 @@ void intel_timeline_enter(struct intel_timeline *tl) | |||||||
| 	struct intel_gt_timelines *timelines = &tl->gt->timelines; | 	struct intel_gt_timelines *timelines = &tl->gt->timelines; | ||||||
| 	unsigned long flags; | 	unsigned long flags; | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Pretend we are serialised by the timeline->mutex. | ||||||
|  | 	 * | ||||||
|  | 	 * While generally true, there are a few exceptions to the rule | ||||||
|  | 	 * for the engine->kernel_context being used to manage power | ||||||
|  | 	 * transitions. As the engine_park may be called from under any | ||||||
|  | 	 * timeline, it uses the power mutex as a global serialisation | ||||||
|  | 	 * lock to prevent any other request entering its timeline. | ||||||
|  | 	 * | ||||||
|  | 	 * The rule is generally tl->mutex, otherwise engine->wakeref.mutex. | ||||||
|  | 	 * | ||||||
|  | 	 * However, intel_gt_retire_request() does not know which engine | ||||||
|  | 	 * it is retiring along and so cannot partake in the engine-pm | ||||||
|  | 	 * barrier, and there we use the tl->active_count as a means to | ||||||
|  | 	 * pin the timeline in the active_list while the locks are dropped. | ||||||
|  | 	 * Ergo, as that is outside of the engine-pm barrier, we need to | ||||||
|  | 	 * use atomic to manipulate tl->active_count. | ||||||
|  | 	 */ | ||||||
| 	lockdep_assert_held(&tl->mutex); | 	lockdep_assert_held(&tl->mutex); | ||||||
| 
 |  | ||||||
| 	GEM_BUG_ON(!atomic_read(&tl->pin_count)); | 	GEM_BUG_ON(!atomic_read(&tl->pin_count)); | ||||||
| 	if (tl->active_count++) | 
 | ||||||
|  | 	if (atomic_add_unless(&tl->active_count, 1, 0)) | ||||||
| 		return; | 		return; | ||||||
| 	GEM_BUG_ON(!tl->active_count); /* overflow? */ |  | ||||||
| 
 | 
 | ||||||
| 	spin_lock_irqsave(&timelines->lock, flags); | 	spin_lock_irqsave(&timelines->lock, flags); | ||||||
|  | 	if (!atomic_fetch_inc(&tl->active_count)) | ||||||
| 		list_add_tail(&tl->link, &timelines->active_list); | 		list_add_tail(&tl->link, &timelines->active_list); | ||||||
| 	spin_unlock_irqrestore(&timelines->lock, flags); | 	spin_unlock_irqrestore(&timelines->lock, flags); | ||||||
| } | } | ||||||
| @ -351,13 +369,15 @@ void intel_timeline_exit(struct intel_timeline *tl) | |||||||
| 	struct intel_gt_timelines *timelines = &tl->gt->timelines; | 	struct intel_gt_timelines *timelines = &tl->gt->timelines; | ||||||
| 	unsigned long flags; | 	unsigned long flags; | ||||||
| 
 | 
 | ||||||
|  | 	/* See intel_timeline_enter() */ | ||||||
| 	lockdep_assert_held(&tl->mutex); | 	lockdep_assert_held(&tl->mutex); | ||||||
| 
 | 
 | ||||||
| 	GEM_BUG_ON(!tl->active_count); | 	GEM_BUG_ON(!atomic_read(&tl->active_count)); | ||||||
| 	if (--tl->active_count) | 	if (atomic_add_unless(&tl->active_count, -1, 1)) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	spin_lock_irqsave(&timelines->lock, flags); | 	spin_lock_irqsave(&timelines->lock, flags); | ||||||
|  | 	if (atomic_dec_and_test(&tl->active_count)) | ||||||
| 		list_del(&tl->link); | 		list_del(&tl->link); | ||||||
| 	spin_unlock_irqrestore(&timelines->lock, flags); | 	spin_unlock_irqrestore(&timelines->lock, flags); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -42,7 +42,7 @@ struct intel_timeline { | |||||||
| 	 * from the intel_context caller plus internal atomicity. | 	 * from the intel_context caller plus internal atomicity. | ||||||
| 	 */ | 	 */ | ||||||
| 	atomic_t pin_count; | 	atomic_t pin_count; | ||||||
| 	unsigned int active_count; | 	atomic_t active_count; | ||||||
| 
 | 
 | ||||||
| 	const u32 *hwsp_seqno; | 	const u32 *hwsp_seqno; | ||||||
| 	struct i915_vma *hwsp_ggtt; | 	struct i915_vma *hwsp_ggtt; | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Chris Wilson
						Chris Wilson