Commit 54d7195f authored by Chris Wilson's avatar Chris Wilson

drm/i915: Unpin vma->obj on early error

If we inherit an error along the fence chain, we skip the main work
callback and go straight to the error. In the case of the vma bind
worker, we only dropped the pinned pages from the worker.

In the process, make sure we call the release earlier rather than wait
until the final reference to the fence is dropped (as a reference is
kept while being listened upon).
Signed-off-by: default avatarChris Wilson <chris@chris-wilson.co.uk>
Reviewed-by: default avatarJoonas Lahtinen <joonas.lahtinen@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20191216161717.2688274-1-chris@chris-wilson.co.uk
parent f20c6b27
...@@ -26,26 +26,23 @@ static void __do_clflush(struct drm_i915_gem_object *obj) ...@@ -26,26 +26,23 @@ static void __do_clflush(struct drm_i915_gem_object *obj)
static int clflush_work(struct dma_fence_work *base) static int clflush_work(struct dma_fence_work *base)
{ {
struct clflush *clflush = container_of(base, typeof(*clflush), base); struct clflush *clflush = container_of(base, typeof(*clflush), base);
struct drm_i915_gem_object *obj = fetch_and_zero(&clflush->obj); struct drm_i915_gem_object *obj = clflush->obj;
int err; int err;
err = i915_gem_object_pin_pages(obj); err = i915_gem_object_pin_pages(obj);
if (err) if (err)
goto put; return err;
__do_clflush(obj); __do_clflush(obj);
i915_gem_object_unpin_pages(obj); i915_gem_object_unpin_pages(obj);
put: return 0;
i915_gem_object_put(obj);
return err;
} }
static void clflush_release(struct dma_fence_work *base) static void clflush_release(struct dma_fence_work *base)
{ {
struct clflush *clflush = container_of(base, typeof(*clflush), base); struct clflush *clflush = container_of(base, typeof(*clflush), base);
if (clflush->obj)
i915_gem_object_put(clflush->obj); i915_gem_object_put(clflush->obj);
} }
......
...@@ -6,6 +6,13 @@ ...@@ -6,6 +6,13 @@
#include "i915_sw_fence_work.h" #include "i915_sw_fence_work.h"
static void fence_complete(struct dma_fence_work *f)
{
if (f->ops->release)
f->ops->release(f);
dma_fence_signal(&f->dma);
}
static void fence_work(struct work_struct *work) static void fence_work(struct work_struct *work)
{ {
struct dma_fence_work *f = container_of(work, typeof(*f), work); struct dma_fence_work *f = container_of(work, typeof(*f), work);
...@@ -14,7 +21,8 @@ static void fence_work(struct work_struct *work) ...@@ -14,7 +21,8 @@ static void fence_work(struct work_struct *work)
err = f->ops->work(f); err = f->ops->work(f);
if (err) if (err)
dma_fence_set_error(&f->dma, err); dma_fence_set_error(&f->dma, err);
dma_fence_signal(&f->dma);
fence_complete(f);
dma_fence_put(&f->dma); dma_fence_put(&f->dma);
} }
...@@ -32,7 +40,7 @@ fence_notify(struct i915_sw_fence *fence, enum i915_sw_fence_notify state) ...@@ -32,7 +40,7 @@ fence_notify(struct i915_sw_fence *fence, enum i915_sw_fence_notify state)
dma_fence_get(&f->dma); dma_fence_get(&f->dma);
queue_work(system_unbound_wq, &f->work); queue_work(system_unbound_wq, &f->work);
} else { } else {
dma_fence_signal(&f->dma); fence_complete(f);
} }
break; break;
...@@ -60,9 +68,6 @@ static void fence_release(struct dma_fence *fence) ...@@ -60,9 +68,6 @@ static void fence_release(struct dma_fence *fence)
{ {
struct dma_fence_work *f = container_of(fence, typeof(*f), dma); struct dma_fence_work *f = container_of(fence, typeof(*f), dma);
if (f->ops->release)
f->ops->release(f);
i915_sw_fence_fini(&f->chain); i915_sw_fence_fini(&f->chain);
BUILD_BUG_ON(offsetof(typeof(*f), dma)); BUILD_BUG_ON(offsetof(typeof(*f), dma));
......
...@@ -292,6 +292,7 @@ i915_vma_instance(struct drm_i915_gem_object *obj, ...@@ -292,6 +292,7 @@ i915_vma_instance(struct drm_i915_gem_object *obj,
struct i915_vma_work { struct i915_vma_work {
struct dma_fence_work base; struct dma_fence_work base;
struct i915_vma *vma; struct i915_vma *vma;
struct drm_i915_gem_object *pinned;
enum i915_cache_level cache_level; enum i915_cache_level cache_level;
unsigned int flags; unsigned int flags;
}; };
...@@ -306,15 +307,21 @@ static int __vma_bind(struct dma_fence_work *work) ...@@ -306,15 +307,21 @@ static int __vma_bind(struct dma_fence_work *work)
if (err) if (err)
atomic_or(I915_VMA_ERROR, &vma->flags); atomic_or(I915_VMA_ERROR, &vma->flags);
if (vma->obj)
__i915_gem_object_unpin_pages(vma->obj);
return err; return err;
} }
static void __vma_release(struct dma_fence_work *work)
{
struct i915_vma_work *vw = container_of(work, typeof(*vw), base);
if (vw->pinned)
__i915_gem_object_unpin_pages(vw->pinned);
}
static const struct dma_fence_work_ops bind_ops = { static const struct dma_fence_work_ops bind_ops = {
.name = "bind", .name = "bind",
.work = __vma_bind, .work = __vma_bind,
.release = __vma_release,
}; };
struct i915_vma_work *i915_vma_work(void) struct i915_vma_work *i915_vma_work(void)
...@@ -395,8 +402,10 @@ int i915_vma_bind(struct i915_vma *vma, ...@@ -395,8 +402,10 @@ int i915_vma_bind(struct i915_vma *vma,
i915_active_set_exclusive(&vma->active, &work->base.dma); i915_active_set_exclusive(&vma->active, &work->base.dma);
work->base.dma.error = 0; /* enable the queue_work() */ work->base.dma.error = 0; /* enable the queue_work() */
if (vma->obj) if (vma->obj) {
__i915_gem_object_pin_pages(vma->obj); __i915_gem_object_pin_pages(vma->obj);
work->pinned = vma->obj;
}
} else { } else {
GEM_BUG_ON((bind_flags & ~vma_flags) & vma->vm->bind_async_flags); GEM_BUG_ON((bind_flags & ~vma_flags) & vma->vm->bind_async_flags);
ret = vma->ops->bind_vma(vma, cache_level, bind_flags); ret = vma->ops->bind_vma(vma, cache_level, bind_flags);
......
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