Commit 0d3abd45 authored by Danilo Krummrich's avatar Danilo Krummrich Committed by Maxime Ripard

drm/imagination: vm: fix drm_gpuvm reference count

The driver specific reference count indicates whether the VM should be
teared down, whereas GPUVM's reference count indicates whether the VM
structure can finally be freed.

Hence, free the VM structure in pvr_gpuvm_free() and drop the last
GPUVM reference after tearing down the VM. Generally, this prevents
lifetime issues such as the VM being freed as long as drm_gpuvm_bo
structures still hold references to the VM.

Fixes: ff5f643d ("drm/imagination: Add GEM and VM related code")
Signed-off-by: default avatarDanilo Krummrich <dakr@redhat.com>
Reviewed-by: default avatarDonald Robson <donald.robson@imgtec.com>
Tested-by: default avatarFrank Binns <frank.binns@imgtec.com>
Signed-off-by: default avatarMaxime Ripard <mripard@kernel.org>
Link: https://patchwork.freedesktop.org/patch/msgid/20231124233650.152653-4-dakr@redhat.com
parent 4550d66d
...@@ -64,6 +64,12 @@ struct pvr_vm_context { ...@@ -64,6 +64,12 @@ struct pvr_vm_context {
struct drm_gem_object dummy_gem; struct drm_gem_object dummy_gem;
}; };
static inline
struct pvr_vm_context *to_pvr_vm_context(struct drm_gpuvm *gpuvm)
{
return container_of(gpuvm, struct pvr_vm_context, gpuvm_mgr);
}
struct pvr_vm_context *pvr_vm_context_get(struct pvr_vm_context *vm_ctx) struct pvr_vm_context *pvr_vm_context_get(struct pvr_vm_context *vm_ctx)
{ {
if (vm_ctx) if (vm_ctx)
...@@ -535,7 +541,7 @@ pvr_device_addr_and_size_are_valid(struct pvr_vm_context *vm_ctx, ...@@ -535,7 +541,7 @@ pvr_device_addr_and_size_are_valid(struct pvr_vm_context *vm_ctx,
void pvr_gpuvm_free(struct drm_gpuvm *gpuvm) void pvr_gpuvm_free(struct drm_gpuvm *gpuvm)
{ {
kfree(to_pvr_vm_context(gpuvm));
} }
static const struct drm_gpuvm_ops pvr_vm_gpuva_ops = { static const struct drm_gpuvm_ops pvr_vm_gpuva_ops = {
...@@ -655,12 +661,11 @@ pvr_vm_context_release(struct kref *ref_count) ...@@ -655,12 +661,11 @@ pvr_vm_context_release(struct kref *ref_count)
WARN_ON(pvr_vm_unmap(vm_ctx, vm_ctx->gpuvm_mgr.mm_start, WARN_ON(pvr_vm_unmap(vm_ctx, vm_ctx->gpuvm_mgr.mm_start,
vm_ctx->gpuvm_mgr.mm_range)); vm_ctx->gpuvm_mgr.mm_range));
drm_gpuvm_put(&vm_ctx->gpuvm_mgr);
pvr_mmu_context_destroy(vm_ctx->mmu_ctx); pvr_mmu_context_destroy(vm_ctx->mmu_ctx);
drm_gem_private_object_fini(&vm_ctx->dummy_gem); drm_gem_private_object_fini(&vm_ctx->dummy_gem);
mutex_destroy(&vm_ctx->lock); mutex_destroy(&vm_ctx->lock);
kfree(vm_ctx); drm_gpuvm_put(&vm_ctx->gpuvm_mgr);
} }
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment