diff options
author | Oded Gabbay <oded.gabbay@gmail.com> | 2016-05-26 01:41:08 -0400 |
---|---|---|
committer | Oded Gabbay <oded.gabbay@gmail.com> | 2016-06-03 01:50:40 -0400 |
commit | 121b78e679ee3ffab780115e260b2775d0cc1f73 (patch) | |
tree | 89fe5381f5d37eb84603d05a4a303b7742e6a38b | |
parent | ab3ab68493aaac6fea4ad1bb597def9f48f41c71 (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.c | 60 |
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 | ||
445 | struct kfd_process_device *kfd_get_first_process_device_data(struct kfd_process *p) | 455 | struct kfd_process_device *kfd_get_first_process_device_data(struct kfd_process *p) |