2019-03-08 20:25:19 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: MIT
|
|
|
|
*
|
|
|
|
* Copyright © 2019 Intel Corporation
|
|
|
|
*/
|
|
|
|
|
2019-05-28 16:29:49 +07:00
|
|
|
#include "gem/i915_gem_context.h"
|
|
|
|
#include "gem/i915_gem_pm.h"
|
|
|
|
|
2019-03-08 20:25:19 +07:00
|
|
|
#include "i915_drv.h"
|
|
|
|
#include "i915_globals.h"
|
2019-04-25 00:48:39 +07:00
|
|
|
|
2019-03-08 20:25:19 +07:00
|
|
|
#include "intel_context.h"
|
2019-04-25 00:48:39 +07:00
|
|
|
#include "intel_engine.h"
|
drm/i915: Invert the GEM wakeref hierarchy
In the current scheme, on submitting a request we take a single global
GEM wakeref, which trickles down to wake up all GT power domains. This
is undesirable as we would like to be able to localise our power
management to the available power domains and to remove the global GEM
operations from the heart of the driver. (The intent there is to push
global GEM decisions to the boundary as used by the GEM user interface.)
Now during request construction, each request is responsible via its
logical context to acquire a wakeref on each power domain it intends to
utilize. Currently, each request takes a wakeref on the engine(s) and
the engines themselves take a chipset wakeref. This gives us a
transition on each engine which we can extend if we want to insert more
powermangement control (such as soft rc6). The global GEM operations
that currently require a struct_mutex are reduced to listening to pm
events from the chipset GT wakeref. As we reduce the struct_mutex
requirement, these listeners should evaporate.
Perhaps the biggest immediate change is that this removes the
struct_mutex requirement around GT power management, allowing us greater
flexibility in request construction. Another important knock-on effect,
is that by tracking engine usage, we can insert a switch back to the
kernel context on that engine immediately, avoiding any extra delay or
inserting global synchronisation barriers. This makes tracking when an
engine and its associated contexts are idle much easier -- important for
when we forgo our assumed execution ordering and need idle barriers to
unpin used contexts. In the process, it means we remove a large chunk of
code whose only purpose was to switch back to the kernel context.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Imre Deak <imre.deak@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190424200717.1686-5-chris@chris-wilson.co.uk
2019-04-25 03:07:17 +07:00
|
|
|
#include "intel_engine_pm.h"
|
2019-10-24 17:03:44 +07:00
|
|
|
#include "intel_ring.h"
|
2019-03-08 20:25:19 +07:00
|
|
|
|
|
|
|
static struct i915_global_context {
|
|
|
|
struct i915_global base;
|
|
|
|
struct kmem_cache *slab_ce;
|
|
|
|
} global;
|
|
|
|
|
2019-04-26 23:33:34 +07:00
|
|
|
static struct intel_context *intel_context_alloc(void)
|
2019-03-08 20:25:19 +07:00
|
|
|
{
|
|
|
|
return kmem_cache_zalloc(global.slab_ce, GFP_KERNEL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void intel_context_free(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
kmem_cache_free(global.slab_ce, ce);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct intel_context *
|
2019-12-21 23:03:24 +07:00
|
|
|
intel_context_create(struct intel_engine_cs *engine)
|
2019-03-08 20:25:19 +07:00
|
|
|
{
|
2019-04-26 23:33:34 +07:00
|
|
|
struct intel_context *ce;
|
2019-03-08 20:25:19 +07:00
|
|
|
|
|
|
|
ce = intel_context_alloc();
|
|
|
|
if (!ce)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
2019-12-21 23:03:24 +07:00
|
|
|
intel_context_init(ce, engine);
|
2019-04-26 23:33:34 +07:00
|
|
|
return ce;
|
2019-03-08 20:25:19 +07:00
|
|
|
}
|
|
|
|
|
2020-01-09 15:57:15 +07:00
|
|
|
int intel_context_alloc_state(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (mutex_lock_interruptible(&ce->pin_mutex))
|
|
|
|
return -EINTR;
|
|
|
|
|
|
|
|
if (!test_bit(CONTEXT_ALLOC_BIT, &ce->flags)) {
|
|
|
|
err = ce->ops->alloc(ce);
|
|
|
|
if (unlikely(err))
|
|
|
|
goto unlock;
|
|
|
|
|
|
|
|
set_bit(CONTEXT_ALLOC_BIT, &ce->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
unlock:
|
|
|
|
mutex_unlock(&ce->pin_mutex);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-09 15:57:16 +07:00
|
|
|
static int intel_context_active_acquire(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2020-01-27 22:28:29 +07:00
|
|
|
__i915_active_acquire(&ce->active);
|
|
|
|
|
|
|
|
if (intel_context_is_barrier(ce))
|
|
|
|
return 0;
|
2020-01-09 15:57:16 +07:00
|
|
|
|
|
|
|
/* Preallocate tracking nodes */
|
2020-01-27 22:28:29 +07:00
|
|
|
err = i915_active_acquire_preallocate_barrier(&ce->active,
|
|
|
|
ce->engine);
|
|
|
|
if (err)
|
|
|
|
i915_active_release(&ce->active);
|
2020-01-09 15:57:16 +07:00
|
|
|
|
2020-01-27 22:28:29 +07:00
|
|
|
return err;
|
2020-01-09 15:57:16 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void intel_context_active_release(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
/* Nodes preallocated in intel_context_active() */
|
|
|
|
i915_active_acquire_barrier(&ce->active);
|
|
|
|
i915_active_release(&ce->active);
|
|
|
|
}
|
|
|
|
|
2019-04-26 23:33:29 +07:00
|
|
|
int __intel_context_do_pin(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
int err;
|
2019-03-08 20:25:22 +07:00
|
|
|
|
2020-01-09 15:57:15 +07:00
|
|
|
if (unlikely(!test_bit(CONTEXT_ALLOC_BIT, &ce->flags))) {
|
|
|
|
err = intel_context_alloc_state(ce);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-01-27 22:28:29 +07:00
|
|
|
err = i915_active_acquire(&ce->active);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2019-03-08 20:25:22 +07:00
|
|
|
|
2020-01-27 22:28:29 +07:00
|
|
|
if (mutex_lock_interruptible(&ce->pin_mutex)) {
|
|
|
|
err = -EINTR;
|
|
|
|
goto out_release;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (likely(!atomic_add_unless(&ce->pin_count, 1, 0))) {
|
2020-01-09 15:57:16 +07:00
|
|
|
err = intel_context_active_acquire(ce);
|
|
|
|
if (unlikely(err))
|
2020-01-27 22:28:29 +07:00
|
|
|
goto out_unlock;
|
2020-01-09 15:57:16 +07:00
|
|
|
|
2020-01-09 15:57:17 +07:00
|
|
|
err = ce->ops->pin(ce);
|
2020-01-09 15:57:16 +07:00
|
|
|
if (unlikely(err))
|
|
|
|
goto err_active;
|
2019-03-08 20:25:20 +07:00
|
|
|
|
2019-12-13 22:51:52 +07:00
|
|
|
CE_TRACE(ce, "pin ring:{head:%04x, tail:%04x}\n",
|
|
|
|
ce->ring->head, ce->ring->tail);
|
2019-06-26 02:48:59 +07:00
|
|
|
|
2019-03-08 20:25:22 +07:00
|
|
|
smp_mb__before_atomic(); /* flush pin before it is visible */
|
2020-01-27 22:28:29 +07:00
|
|
|
atomic_inc(&ce->pin_count);
|
2019-03-08 20:25:20 +07:00
|
|
|
}
|
|
|
|
|
2019-03-08 20:25:22 +07:00
|
|
|
GEM_BUG_ON(!intel_context_is_pinned(ce)); /* no overflow! */
|
2020-01-27 22:28:29 +07:00
|
|
|
GEM_BUG_ON(i915_active_is_idle(&ce->active));
|
|
|
|
goto out_unlock;
|
2019-03-08 20:25:20 +07:00
|
|
|
|
2020-01-09 15:57:16 +07:00
|
|
|
err_active:
|
|
|
|
intel_context_active_release(ce);
|
2020-01-27 22:28:29 +07:00
|
|
|
out_unlock:
|
2019-03-08 20:25:22 +07:00
|
|
|
mutex_unlock(&ce->pin_mutex);
|
2020-01-27 22:28:29 +07:00
|
|
|
out_release:
|
|
|
|
i915_active_release(&ce->active);
|
2019-04-26 23:33:29 +07:00
|
|
|
return err;
|
2019-03-08 20:25:20 +07:00
|
|
|
}
|
|
|
|
|
2019-03-08 20:25:22 +07:00
|
|
|
void intel_context_unpin(struct intel_context *ce)
|
|
|
|
{
|
2020-01-06 18:42:33 +07:00
|
|
|
if (!atomic_dec_and_test(&ce->pin_count))
|
2019-03-08 20:25:22 +07:00
|
|
|
return;
|
|
|
|
|
2020-01-06 18:42:33 +07:00
|
|
|
CE_TRACE(ce, "unpin\n");
|
|
|
|
ce->ops->unpin(ce);
|
2019-03-19 04:23:46 +07:00
|
|
|
|
2020-01-06 18:42:33 +07:00
|
|
|
/*
|
|
|
|
* Once released, we may asynchronously drop the active reference.
|
|
|
|
* As that may be the only reference keeping the context alive,
|
|
|
|
* take an extra now so that it is not freed before we finish
|
|
|
|
* dereferencing it.
|
|
|
|
*/
|
|
|
|
intel_context_get(ce);
|
|
|
|
intel_context_active_release(ce);
|
2019-03-19 04:23:47 +07:00
|
|
|
intel_context_put(ce);
|
2019-03-08 20:25:22 +07:00
|
|
|
}
|
|
|
|
|
2019-06-22 01:38:00 +07:00
|
|
|
static int __context_pin_state(struct i915_vma *vma)
|
2019-03-08 20:25:19 +07:00
|
|
|
{
|
2019-12-05 18:37:25 +07:00
|
|
|
unsigned int bias = i915_ggtt_pin_bias(vma) | PIN_OFFSET_BIAS;
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
int err;
|
2019-03-08 20:25:19 +07:00
|
|
|
|
2019-12-05 18:37:25 +07:00
|
|
|
err = i915_ggtt_pin(vma, 0, bias | PIN_HIGH);
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
2020-01-10 18:04:01 +07:00
|
|
|
err = i915_active_acquire(&vma->active);
|
|
|
|
if (err)
|
|
|
|
goto err_unpin;
|
|
|
|
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
/*
|
|
|
|
* And mark it as a globally pinned object to let the shrinker know
|
|
|
|
* it cannot reclaim the object until we release it.
|
|
|
|
*/
|
2019-08-03 04:21:36 +07:00
|
|
|
i915_vma_make_unshrinkable(vma);
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
vma->obj->mm.dirty = true;
|
|
|
|
|
|
|
|
return 0;
|
2020-01-10 18:04:01 +07:00
|
|
|
|
|
|
|
err_unpin:
|
|
|
|
i915_vma_unpin(vma);
|
|
|
|
return err;
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void __context_unpin_state(struct i915_vma *vma)
|
|
|
|
{
|
2019-08-03 04:21:36 +07:00
|
|
|
i915_vma_make_shrinkable(vma);
|
2020-01-10 18:04:01 +07:00
|
|
|
i915_active_release(&vma->active);
|
2019-09-11 04:22:04 +07:00
|
|
|
__i915_vma_unpin(vma);
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
}
|
|
|
|
|
2020-01-10 18:04:02 +07:00
|
|
|
static int __ring_active(struct intel_ring *ring)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = i915_active_acquire(&ring->vma->active);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = intel_ring_pin(ring);
|
|
|
|
if (err)
|
|
|
|
goto err_active;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_active:
|
|
|
|
i915_active_release(&ring->vma->active);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __ring_retire(struct intel_ring *ring)
|
|
|
|
{
|
|
|
|
intel_ring_unpin(ring);
|
|
|
|
i915_active_release(&ring->vma->active);
|
|
|
|
}
|
|
|
|
|
2019-10-04 20:39:59 +07:00
|
|
|
__i915_active_call
|
2019-06-22 01:38:00 +07:00
|
|
|
static void __intel_context_retire(struct i915_active *active)
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
{
|
|
|
|
struct intel_context *ce = container_of(active, typeof(*ce), active);
|
|
|
|
|
2019-12-13 22:51:52 +07:00
|
|
|
CE_TRACE(ce, "retire\n");
|
2019-06-26 02:48:59 +07:00
|
|
|
|
2019-12-03 19:41:55 +07:00
|
|
|
set_bit(CONTEXT_VALID_BIT, &ce->flags);
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
if (ce->state)
|
|
|
|
__context_unpin_state(ce->state);
|
|
|
|
|
2019-08-10 01:25:18 +07:00
|
|
|
intel_timeline_unpin(ce->timeline);
|
2020-01-10 18:04:02 +07:00
|
|
|
__ring_retire(ce->ring);
|
2019-10-04 20:39:59 +07:00
|
|
|
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
intel_context_put(ce);
|
2019-03-08 20:25:19 +07:00
|
|
|
}
|
|
|
|
|
2019-06-22 01:38:00 +07:00
|
|
|
static int __intel_context_active(struct i915_active *active)
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
{
|
2019-06-22 01:38:00 +07:00
|
|
|
struct intel_context *ce = container_of(active, typeof(*ce), active);
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
int err;
|
|
|
|
|
2020-01-06 18:42:32 +07:00
|
|
|
CE_TRACE(ce, "active\n");
|
|
|
|
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
intel_context_get(ce);
|
|
|
|
|
2020-01-10 18:04:02 +07:00
|
|
|
err = __ring_active(ce->ring);
|
2019-06-20 00:01:35 +07:00
|
|
|
if (err)
|
|
|
|
goto err_put;
|
|
|
|
|
2019-08-10 01:25:18 +07:00
|
|
|
err = intel_timeline_pin(ce->timeline);
|
|
|
|
if (err)
|
|
|
|
goto err_ring;
|
|
|
|
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
if (!ce->state)
|
|
|
|
return 0;
|
|
|
|
|
2019-06-22 01:38:00 +07:00
|
|
|
err = __context_pin_state(ce->state);
|
2019-06-20 00:01:35 +07:00
|
|
|
if (err)
|
2019-08-10 01:25:18 +07:00
|
|
|
goto err_timeline;
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
|
2019-08-02 17:00:15 +07:00
|
|
|
return 0;
|
|
|
|
|
2019-08-10 01:25:18 +07:00
|
|
|
err_timeline:
|
|
|
|
intel_timeline_unpin(ce->timeline);
|
2019-08-02 17:00:15 +07:00
|
|
|
err_ring:
|
2020-01-10 18:04:02 +07:00
|
|
|
__ring_retire(ce->ring);
|
2019-08-02 17:00:15 +07:00
|
|
|
err_put:
|
|
|
|
intel_context_put(ce);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2019-06-22 01:38:00 +07:00
|
|
|
void
|
|
|
|
intel_context_init(struct intel_context *ce,
|
|
|
|
struct intel_engine_cs *engine)
|
drm/i915: Keep contexts pinned until after the next kernel context switch
We need to keep the context image pinned in memory until after the GPU
has finished writing into it. Since it continues to write as we signal
the final breadcrumb, we need to keep it pinned until the request after
it is complete. Currently we know the order in which requests execute on
each engine, and so to remove that presumption we need to identify a
request/context-switch we know must occur after our completion. Any
request queued after the signal must imply a context switch, for
simplicity we use a fresh request from the kernel context.
The sequence of operations for keeping the context pinned until saved is:
- On context activation, we preallocate a node for each physical engine
the context may operate on. This is to avoid allocations during
unpinning, which may be from inside FS_RECLAIM context (aka the
shrinker)
- On context deactivation on retirement of the last active request (which
is before we know the context has been saved), we add the
preallocated node onto a barrier list on each engine
- On engine idling, we emit a switch to kernel context. When this
switch completes, we know that all previous contexts must have been
saved, and so on retiring this request we can finally unpin all the
contexts that were marked as deactivated prior to the switch.
We can enhance this in future by flushing all the idle contexts on a
regular heartbeat pulse of a switch to kernel context, which will also
be used to check for hung engines.
v2: intel_context_active_acquire/_release
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Reviewed-by: Mika Kuoppala <mika.kuoppala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190614164606.15633-1-chris@chris-wilson.co.uk
2019-06-14 23:46:04 +07:00
|
|
|
{
|
2019-06-22 01:38:00 +07:00
|
|
|
GEM_BUG_ON(!engine->cops);
|
2019-12-21 23:03:24 +07:00
|
|
|
GEM_BUG_ON(!engine->gt->vm);
|
2019-06-22 01:38:00 +07:00
|
|
|
|
|
|
|
kref_init(&ce->ref);
|
|
|
|
|
|
|
|
ce->engine = engine;
|
|
|
|
ce->ops = engine->cops;
|
|
|
|
ce->sseu = engine->sseu;
|
2019-12-21 23:03:24 +07:00
|
|
|
ce->ring = __intel_context_ring_size(SZ_4K);
|
|
|
|
|
|
|
|
ce->vm = i915_vm_get(engine->gt->vm);
|
2019-06-22 01:38:00 +07:00
|
|
|
|
|
|
|
INIT_LIST_HEAD(&ce->signal_link);
|
|
|
|
INIT_LIST_HEAD(&ce->signals);
|
|
|
|
|
|
|
|
mutex_init(&ce->pin_mutex);
|
|
|
|
|
2019-10-04 20:40:00 +07:00
|
|
|
i915_active_init(&ce->active,
|
2019-06-22 01:38:00 +07:00
|
|
|
__intel_context_active, __intel_context_retire);
|
2019-03-08 20:25:19 +07:00
|
|
|
}
|
|
|
|
|
2019-07-18 14:00:06 +07:00
|
|
|
void intel_context_fini(struct intel_context *ce)
|
|
|
|
{
|
2019-08-10 01:25:18 +07:00
|
|
|
if (ce->timeline)
|
|
|
|
intel_timeline_put(ce->timeline);
|
2019-07-30 21:32:09 +07:00
|
|
|
i915_vm_put(ce->vm);
|
|
|
|
|
2019-07-18 14:00:06 +07:00
|
|
|
mutex_destroy(&ce->pin_mutex);
|
|
|
|
i915_active_fini(&ce->active);
|
|
|
|
}
|
|
|
|
|
2019-03-08 20:25:19 +07:00
|
|
|
static void i915_global_context_shrink(void)
|
|
|
|
{
|
|
|
|
kmem_cache_shrink(global.slab_ce);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void i915_global_context_exit(void)
|
|
|
|
{
|
|
|
|
kmem_cache_destroy(global.slab_ce);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct i915_global_context global = { {
|
|
|
|
.shrink = i915_global_context_shrink,
|
|
|
|
.exit = i915_global_context_exit,
|
|
|
|
} };
|
|
|
|
|
|
|
|
int __init i915_global_context_init(void)
|
|
|
|
{
|
|
|
|
global.slab_ce = KMEM_CACHE(intel_context, SLAB_HWCACHE_ALIGN);
|
|
|
|
if (!global.slab_ce)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
i915_global_register(&global.base);
|
|
|
|
return 0;
|
|
|
|
}
|
2019-04-25 03:07:15 +07:00
|
|
|
|
|
|
|
void intel_context_enter_engine(struct intel_context *ce)
|
|
|
|
{
|
drm/i915: Invert the GEM wakeref hierarchy
In the current scheme, on submitting a request we take a single global
GEM wakeref, which trickles down to wake up all GT power domains. This
is undesirable as we would like to be able to localise our power
management to the available power domains and to remove the global GEM
operations from the heart of the driver. (The intent there is to push
global GEM decisions to the boundary as used by the GEM user interface.)
Now during request construction, each request is responsible via its
logical context to acquire a wakeref on each power domain it intends to
utilize. Currently, each request takes a wakeref on the engine(s) and
the engines themselves take a chipset wakeref. This gives us a
transition on each engine which we can extend if we want to insert more
powermangement control (such as soft rc6). The global GEM operations
that currently require a struct_mutex are reduced to listening to pm
events from the chipset GT wakeref. As we reduce the struct_mutex
requirement, these listeners should evaporate.
Perhaps the biggest immediate change is that this removes the
struct_mutex requirement around GT power management, allowing us greater
flexibility in request construction. Another important knock-on effect,
is that by tracking engine usage, we can insert a switch back to the
kernel context on that engine immediately, avoiding any extra delay or
inserting global synchronisation barriers. This makes tracking when an
engine and its associated contexts are idle much easier -- important for
when we forgo our assumed execution ordering and need idle barriers to
unpin used contexts. In the process, it means we remove a large chunk of
code whose only purpose was to switch back to the kernel context.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Imre Deak <imre.deak@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190424200717.1686-5-chris@chris-wilson.co.uk
2019-04-25 03:07:17 +07:00
|
|
|
intel_engine_pm_get(ce->engine);
|
2019-08-16 03:57:06 +07:00
|
|
|
intel_timeline_enter(ce->timeline);
|
2019-04-25 03:07:15 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
void intel_context_exit_engine(struct intel_context *ce)
|
|
|
|
{
|
2019-08-16 03:57:06 +07:00
|
|
|
intel_timeline_exit(ce->timeline);
|
drm/i915: Invert the GEM wakeref hierarchy
In the current scheme, on submitting a request we take a single global
GEM wakeref, which trickles down to wake up all GT power domains. This
is undesirable as we would like to be able to localise our power
management to the available power domains and to remove the global GEM
operations from the heart of the driver. (The intent there is to push
global GEM decisions to the boundary as used by the GEM user interface.)
Now during request construction, each request is responsible via its
logical context to acquire a wakeref on each power domain it intends to
utilize. Currently, each request takes a wakeref on the engine(s) and
the engines themselves take a chipset wakeref. This gives us a
transition on each engine which we can extend if we want to insert more
powermangement control (such as soft rc6). The global GEM operations
that currently require a struct_mutex are reduced to listening to pm
events from the chipset GT wakeref. As we reduce the struct_mutex
requirement, these listeners should evaporate.
Perhaps the biggest immediate change is that this removes the
struct_mutex requirement around GT power management, allowing us greater
flexibility in request construction. Another important knock-on effect,
is that by tracking engine usage, we can insert a switch back to the
kernel context on that engine immediately, avoiding any extra delay or
inserting global synchronisation barriers. This makes tracking when an
engine and its associated contexts are idle much easier -- important for
when we forgo our assumed execution ordering and need idle barriers to
unpin used contexts. In the process, it means we remove a large chunk of
code whose only purpose was to switch back to the kernel context.
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Cc: Imre Deak <imre.deak@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190424200717.1686-5-chris@chris-wilson.co.uk
2019-04-25 03:07:17 +07:00
|
|
|
intel_engine_pm_put(ce->engine);
|
2019-04-25 03:07:15 +07:00
|
|
|
}
|
2019-04-26 23:33:34 +07:00
|
|
|
|
2019-07-17 04:34:43 +07:00
|
|
|
int intel_context_prepare_remote_request(struct intel_context *ce,
|
|
|
|
struct i915_request *rq)
|
|
|
|
{
|
2019-08-10 01:25:18 +07:00
|
|
|
struct intel_timeline *tl = ce->timeline;
|
2019-07-17 04:34:43 +07:00
|
|
|
int err;
|
|
|
|
|
|
|
|
/* Only suitable for use in remotely modifying this context */
|
2019-12-20 17:12:29 +07:00
|
|
|
GEM_BUG_ON(rq->context == ce);
|
2019-07-17 04:34:43 +07:00
|
|
|
|
drm/i915: Mark i915_request.timeline as a volatile, rcu pointer
The request->timeline is only valid until the request is retired (i.e.
before it is completed). Upon retiring the request, the context may be
unpinned and freed, and along with it the timeline may be freed. We
therefore need to be very careful when chasing rq->timeline that the
pointer does not disappear beneath us. The vast majority of users are in
a protected context, either during request construction or retirement,
where the timeline->mutex is held and the timeline cannot disappear. It
is those few off the beaten path (where we access a second timeline) that
need extra scrutiny -- to be added in the next patch after first adding
the warnings about dangerous access.
One complication, where we cannot use the timeline->mutex itself, is
during request submission onto hardware (under spinlocks). Here, we want
to check on the timeline to finalize the breadcrumb, and so we need to
impose a second rule to ensure that the request->timeline is indeed
valid. As we are submitting the request, it's context and timeline must
be pinned, as it will be used by the hardware. Since it is pinned, we
know the request->timeline must still be valid, and we cannot submit the
idle barrier until after we release the engine->active.lock, ergo while
submitting and holding that spinlock, a second thread cannot release the
timeline.
v2: Don't be lazy inside selftests; hold the timeline->mutex for as long
as we need it, and tidy up acquiring the timeline with a bit of
refactoring (i915_active_add_request)
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190919111912.21631-1-chris@chris-wilson.co.uk
2019-09-19 18:19:10 +07:00
|
|
|
if (rcu_access_pointer(rq->timeline) != tl) { /* timeline sharing! */
|
2019-07-25 20:14:46 +07:00
|
|
|
/* Queue this switch after current activity by this context. */
|
2019-10-04 20:40:00 +07:00
|
|
|
err = i915_active_fence_set(&tl->last_request, rq);
|
2019-07-25 20:14:46 +07:00
|
|
|
if (err)
|
2019-08-16 19:10:00 +07:00
|
|
|
return err;
|
2019-07-25 20:14:46 +07:00
|
|
|
}
|
2019-07-17 04:34:43 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Guarantee context image and the timeline remains pinned until the
|
|
|
|
* modifying request is retired by setting the ce activity tracker.
|
|
|
|
*
|
|
|
|
* But we only need to take one pin on the account of it. Or in other
|
|
|
|
* words transfer the pinned ce object to tracked active request.
|
|
|
|
*/
|
|
|
|
GEM_BUG_ON(i915_active_is_idle(&ce->active));
|
drm/i915: Mark i915_request.timeline as a volatile, rcu pointer
The request->timeline is only valid until the request is retired (i.e.
before it is completed). Upon retiring the request, the context may be
unpinned and freed, and along with it the timeline may be freed. We
therefore need to be very careful when chasing rq->timeline that the
pointer does not disappear beneath us. The vast majority of users are in
a protected context, either during request construction or retirement,
where the timeline->mutex is held and the timeline cannot disappear. It
is those few off the beaten path (where we access a second timeline) that
need extra scrutiny -- to be added in the next patch after first adding
the warnings about dangerous access.
One complication, where we cannot use the timeline->mutex itself, is
during request submission onto hardware (under spinlocks). Here, we want
to check on the timeline to finalize the breadcrumb, and so we need to
impose a second rule to ensure that the request->timeline is indeed
valid. As we are submitting the request, it's context and timeline must
be pinned, as it will be used by the hardware. Since it is pinned, we
know the request->timeline must still be valid, and we cannot submit the
idle barrier until after we release the engine->active.lock, ergo while
submitting and holding that spinlock, a second thread cannot release the
timeline.
v2: Don't be lazy inside selftests; hold the timeline->mutex for as long
as we need it, and tidy up acquiring the timeline with a bit of
refactoring (i915_active_add_request)
Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk>
Cc: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Reviewed-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20190919111912.21631-1-chris@chris-wilson.co.uk
2019-09-19 18:19:10 +07:00
|
|
|
return i915_active_add_request(&ce->active, rq);
|
2019-07-17 04:34:43 +07:00
|
|
|
}
|
|
|
|
|
2019-04-26 23:33:34 +07:00
|
|
|
struct i915_request *intel_context_create_request(struct intel_context *ce)
|
|
|
|
{
|
|
|
|
struct i915_request *rq;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = intel_context_pin(ce);
|
|
|
|
if (unlikely(err))
|
|
|
|
return ERR_PTR(err);
|
|
|
|
|
|
|
|
rq = i915_request_create(ce);
|
|
|
|
intel_context_unpin(ce);
|
|
|
|
|
|
|
|
return rq;
|
|
|
|
}
|
2019-08-02 17:00:15 +07:00
|
|
|
|
|
|
|
#if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
|
|
|
|
#include "selftest_context.c"
|
|
|
|
#endif
|