diff options
author | Christian König <christian.koenig@amd.com> | 2014-07-18 07:48:10 -0400 |
---|---|---|
committer | Alex Deucher <alexander.deucher@amd.com> | 2014-07-21 13:17:38 -0400 |
commit | cc9e67e3d7000c1efbaf929c6bdaf78707407b3b (patch) | |
tree | 821067ec1bd97e60b1ff140b0f5227c436fa3808 /drivers/gpu | |
parent | 036bf46a3962c87fc6ab5e6dbc65f469730b4cf0 (diff) |
drm/radeon: fix VM IB handling
Calling radeon_vm_bo_find on the IB BO during CS
is illegal and can lead to an crash.
Signed-off-by: Christian König <christian.koenig@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
Diffstat (limited to 'drivers/gpu')
-rw-r--r-- | drivers/gpu/drm/radeon/radeon.h | 2 | ||||
-rw-r--r-- | drivers/gpu/drm/radeon/radeon_cs.c | 6 | ||||
-rw-r--r-- | drivers/gpu/drm/radeon/radeon_kms.c | 26 | ||||
-rw-r--r-- | drivers/gpu/drm/radeon/radeon_vm.c | 1 |
4 files changed, 19 insertions, 16 deletions
diff --git a/drivers/gpu/drm/radeon/radeon.h b/drivers/gpu/drm/radeon/radeon.h index 3d5e1a93b947..60c47f829122 100644 --- a/drivers/gpu/drm/radeon/radeon.h +++ b/drivers/gpu/drm/radeon/radeon.h | |||
@@ -879,6 +879,8 @@ struct radeon_vm { | |||
879 | /* array of page tables, one for each page directory entry */ | 879 | /* array of page tables, one for each page directory entry */ |
880 | struct radeon_vm_pt *page_tables; | 880 | struct radeon_vm_pt *page_tables; |
881 | 881 | ||
882 | struct radeon_bo_va *ib_bo_va; | ||
883 | |||
882 | struct mutex mutex; | 884 | struct mutex mutex; |
883 | /* last fence for cs using this vm */ | 885 | /* last fence for cs using this vm */ |
884 | struct radeon_fence *fence; | 886 | struct radeon_fence *fence; |
diff --git a/drivers/gpu/drm/radeon/radeon_cs.c b/drivers/gpu/drm/radeon/radeon_cs.c index 09fcf4dcdfdb..ae763f60c8a0 100644 --- a/drivers/gpu/drm/radeon/radeon_cs.c +++ b/drivers/gpu/drm/radeon/radeon_cs.c | |||
@@ -472,13 +472,13 @@ static int radeon_bo_vm_update_pte(struct radeon_cs_parser *p, | |||
472 | if (r) | 472 | if (r) |
473 | return r; | 473 | return r; |
474 | 474 | ||
475 | bo_va = radeon_vm_bo_find(vm, rdev->ring_tmp_bo.bo); | 475 | if (vm->ib_bo_va == NULL) { |
476 | if (bo_va == NULL) { | ||
477 | DRM_ERROR("Tmp BO not in VM!\n"); | 476 | DRM_ERROR("Tmp BO not in VM!\n"); |
478 | return -EINVAL; | 477 | return -EINVAL; |
479 | } | 478 | } |
480 | 479 | ||
481 | r = radeon_vm_bo_update(rdev, bo_va, &rdev->ring_tmp_bo.bo->tbo.mem); | 480 | r = radeon_vm_bo_update(rdev, vm->ib_bo_va, |
481 | &rdev->ring_tmp_bo.bo->tbo.mem); | ||
482 | if (r) | 482 | if (r) |
483 | return r; | 483 | return r; |
484 | 484 | ||
diff --git a/drivers/gpu/drm/radeon/radeon_kms.c b/drivers/gpu/drm/radeon/radeon_kms.c index 35d931881b4b..d25ae6acfd5a 100644 --- a/drivers/gpu/drm/radeon/radeon_kms.c +++ b/drivers/gpu/drm/radeon/radeon_kms.c | |||
@@ -579,7 +579,7 @@ int radeon_driver_open_kms(struct drm_device *dev, struct drm_file *file_priv) | |||
579 | /* new gpu have virtual address space support */ | 579 | /* new gpu have virtual address space support */ |
580 | if (rdev->family >= CHIP_CAYMAN) { | 580 | if (rdev->family >= CHIP_CAYMAN) { |
581 | struct radeon_fpriv *fpriv; | 581 | struct radeon_fpriv *fpriv; |
582 | struct radeon_bo_va *bo_va; | 582 | struct radeon_vm *vm; |
583 | int r; | 583 | int r; |
584 | 584 | ||
585 | fpriv = kzalloc(sizeof(*fpriv), GFP_KERNEL); | 585 | fpriv = kzalloc(sizeof(*fpriv), GFP_KERNEL); |
@@ -587,7 +587,8 @@ int radeon_driver_open_kms(struct drm_device *dev, struct drm_file *file_priv) | |||
587 | return -ENOMEM; | 587 | return -ENOMEM; |
588 | } | 588 | } |
589 | 589 | ||
590 | r = radeon_vm_init(rdev, &fpriv->vm); | 590 | vm = &fpriv->vm; |
591 | r = radeon_vm_init(rdev, vm); | ||
591 | if (r) { | 592 | if (r) { |
592 | kfree(fpriv); | 593 | kfree(fpriv); |
593 | return r; | 594 | return r; |
@@ -596,22 +597,23 @@ int radeon_driver_open_kms(struct drm_device *dev, struct drm_file *file_priv) | |||
596 | if (rdev->accel_working) { | 597 | if (rdev->accel_working) { |
597 | r = radeon_bo_reserve(rdev->ring_tmp_bo.bo, false); | 598 | r = radeon_bo_reserve(rdev->ring_tmp_bo.bo, false); |
598 | if (r) { | 599 | if (r) { |
599 | radeon_vm_fini(rdev, &fpriv->vm); | 600 | radeon_vm_fini(rdev, vm); |
600 | kfree(fpriv); | 601 | kfree(fpriv); |
601 | return r; | 602 | return r; |
602 | } | 603 | } |
603 | 604 | ||
604 | /* map the ib pool buffer read only into | 605 | /* map the ib pool buffer read only into |
605 | * virtual address space */ | 606 | * virtual address space */ |
606 | bo_va = radeon_vm_bo_add(rdev, &fpriv->vm, | 607 | vm->ib_bo_va = radeon_vm_bo_add(rdev, vm, |
607 | rdev->ring_tmp_bo.bo); | 608 | rdev->ring_tmp_bo.bo); |
608 | r = radeon_vm_bo_set_addr(rdev, bo_va, RADEON_VA_IB_OFFSET, | 609 | r = radeon_vm_bo_set_addr(rdev, vm->ib_bo_va, |
610 | RADEON_VA_IB_OFFSET, | ||
609 | RADEON_VM_PAGE_READABLE | | 611 | RADEON_VM_PAGE_READABLE | |
610 | RADEON_VM_PAGE_SNOOPED); | 612 | RADEON_VM_PAGE_SNOOPED); |
611 | 613 | ||
612 | radeon_bo_unreserve(rdev->ring_tmp_bo.bo); | 614 | radeon_bo_unreserve(rdev->ring_tmp_bo.bo); |
613 | if (r) { | 615 | if (r) { |
614 | radeon_vm_fini(rdev, &fpriv->vm); | 616 | radeon_vm_fini(rdev, vm); |
615 | kfree(fpriv); | 617 | kfree(fpriv); |
616 | return r; | 618 | return r; |
617 | } | 619 | } |
@@ -640,21 +642,19 @@ void radeon_driver_postclose_kms(struct drm_device *dev, | |||
640 | /* new gpu have virtual address space support */ | 642 | /* new gpu have virtual address space support */ |
641 | if (rdev->family >= CHIP_CAYMAN && file_priv->driver_priv) { | 643 | if (rdev->family >= CHIP_CAYMAN && file_priv->driver_priv) { |
642 | struct radeon_fpriv *fpriv = file_priv->driver_priv; | 644 | struct radeon_fpriv *fpriv = file_priv->driver_priv; |
643 | struct radeon_bo_va *bo_va; | 645 | struct radeon_vm *vm = &fpriv->vm; |
644 | int r; | 646 | int r; |
645 | 647 | ||
646 | if (rdev->accel_working) { | 648 | if (rdev->accel_working) { |
647 | r = radeon_bo_reserve(rdev->ring_tmp_bo.bo, false); | 649 | r = radeon_bo_reserve(rdev->ring_tmp_bo.bo, false); |
648 | if (!r) { | 650 | if (!r) { |
649 | bo_va = radeon_vm_bo_find(&fpriv->vm, | 651 | if (vm->ib_bo_va) |
650 | rdev->ring_tmp_bo.bo); | 652 | radeon_vm_bo_rmv(rdev, vm->ib_bo_va); |
651 | if (bo_va) | ||
652 | radeon_vm_bo_rmv(rdev, bo_va); | ||
653 | radeon_bo_unreserve(rdev->ring_tmp_bo.bo); | 653 | radeon_bo_unreserve(rdev->ring_tmp_bo.bo); |
654 | } | 654 | } |
655 | } | 655 | } |
656 | 656 | ||
657 | radeon_vm_fini(rdev, &fpriv->vm); | 657 | radeon_vm_fini(rdev, vm); |
658 | kfree(fpriv); | 658 | kfree(fpriv); |
659 | file_priv->driver_priv = NULL; | 659 | file_priv->driver_priv = NULL; |
660 | } | 660 | } |
diff --git a/drivers/gpu/drm/radeon/radeon_vm.c b/drivers/gpu/drm/radeon/radeon_vm.c index 2726b46ac819..fa41e0d7d17d 100644 --- a/drivers/gpu/drm/radeon/radeon_vm.c +++ b/drivers/gpu/drm/radeon/radeon_vm.c | |||
@@ -1007,6 +1007,7 @@ int radeon_vm_init(struct radeon_device *rdev, struct radeon_vm *vm) | |||
1007 | int r; | 1007 | int r; |
1008 | 1008 | ||
1009 | vm->id = 0; | 1009 | vm->id = 0; |
1010 | vm->ib_bo_va = NULL; | ||
1010 | vm->fence = NULL; | 1011 | vm->fence = NULL; |
1011 | vm->last_flush = NULL; | 1012 | vm->last_flush = NULL; |
1012 | vm->last_id_use = NULL; | 1013 | vm->last_id_use = NULL; |