aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorOded Gabbay <oded.gabbay@gmail.com>2016-05-26 01:41:08 -0400
committerOded Gabbay <oded.gabbay@gmail.com>2016-06-03 01:50:40 -0400
commit121b78e679ee3ffab780115e260b2775d0cc1f73 (patch)
tree89fe5381f5d37eb84603d05a4a303b7742e6a38b
parentab3ab68493aaac6fea4ad1bb597def9f48f41c71 (diff)
drm/amdkfd: unbind only existing processes
When unbinding a process from a device (initiated by amd_iommu_v2), the driver needs to make sure that process still exists in the process table. There is a possibility that amdkfd's own notifier handler - kfd_process_notifier_release() - was called before the unbind function and it already removed the process from the process table. v2: Because there can be only one process with the specified pasid, and because *p can't be NULL inside the hash_for_each_rcu macro, it is more reasonable to just put the whole code inside the if statement that compares the pasid value. That way, when we exit hash_for_each_rcu, we simply exit the function as well. Signed-off-by: Oded Gabbay <oded.gabbay@gmail.com> CC: Stable <stable@vger.kernel.org>
-rw-r--r--drivers/gpu/drm/amd/amdkfd/kfd_process.c60
1 files changed, 35 insertions, 25 deletions
diff --git a/drivers/gpu/drm/amd/amdkfd/kfd_process.c b/drivers/gpu/drm/amd/amdkfd/kfd_process.c
index ac005796b71c..a64bc619ed82 100644
--- a/drivers/gpu/drm/amd/amdkfd/kfd_process.c
+++ b/drivers/gpu/drm/amd/amdkfd/kfd_process.c
@@ -404,42 +404,52 @@ void kfd_unbind_process_from_device(struct kfd_dev *dev, unsigned int pasid)
404 404
405 idx = srcu_read_lock(&kfd_processes_srcu); 405 idx = srcu_read_lock(&kfd_processes_srcu);
406 406
407 /*
408 * Look for the process that matches the pasid. If there is no such
409 * process, we either released it in amdkfd's own notifier, or there
410 * is a bug. Unfortunately, there is no way to tell...
411 */
407 hash_for_each_rcu(kfd_processes_table, i, p, kfd_processes) 412 hash_for_each_rcu(kfd_processes_table, i, p, kfd_processes)
408 if (p->pasid == pasid) 413 if (p->pasid == pasid) {
409 break;
410 414
411 srcu_read_unlock(&kfd_processes_srcu, idx); 415 srcu_read_unlock(&kfd_processes_srcu, idx);
412 416
413 BUG_ON(p->pasid != pasid); 417 pr_debug("Unbinding process %d from IOMMU\n", pasid);
414 418
415 mutex_lock(&p->mutex); 419 mutex_lock(&p->mutex);
416 420
417 if ((dev->dbgmgr) && (dev->dbgmgr->pasid == p->pasid)) 421 if ((dev->dbgmgr) && (dev->dbgmgr->pasid == p->pasid))
418 kfd_dbgmgr_destroy(dev->dbgmgr); 422 kfd_dbgmgr_destroy(dev->dbgmgr);
419 423
420 pqm_uninit(&p->pqm); 424 pqm_uninit(&p->pqm);
421 425
422 pdd = kfd_get_process_device_data(dev, p); 426 pdd = kfd_get_process_device_data(dev, p);
423 427
424 if (!pdd) { 428 if (!pdd) {
425 mutex_unlock(&p->mutex); 429 mutex_unlock(&p->mutex);
426 return; 430 return;
427 } 431 }
428 432
429 if (pdd->reset_wavefronts) { 433 if (pdd->reset_wavefronts) {
430 dbgdev_wave_reset_wavefronts(pdd->dev, p); 434 dbgdev_wave_reset_wavefronts(pdd->dev, p);
431 pdd->reset_wavefronts = false; 435 pdd->reset_wavefronts = false;
432 } 436 }
433 437
434 /* 438 /*
435 * Just mark pdd as unbound, because we still need it to call 439 * Just mark pdd as unbound, because we still need it
436 * amd_iommu_unbind_pasid() in when the process exits. 440 * to call amd_iommu_unbind_pasid() in when the
437 * We don't call amd_iommu_unbind_pasid() here 441 * process exits.
438 * because the IOMMU called us. 442 * We don't call amd_iommu_unbind_pasid() here
439 */ 443 * because the IOMMU called us.
440 pdd->bound = false; 444 */
445 pdd->bound = false;
441 446
442 mutex_unlock(&p->mutex); 447 mutex_unlock(&p->mutex);
448
449 return;
450 }
451
452 srcu_read_unlock(&kfd_processes_srcu, idx);
443} 453}
444 454
445struct kfd_process_device *kfd_get_first_process_device_data(struct kfd_process *p) 455struct kfd_process_device *kfd_get_first_process_device_data(struct kfd_process *p)