From: Ville Syrjälä <ville.syrjala@xxxxxxxxxxxxxxx> As per the SNB and HSW PM guides, we should enable FBC render/blitter tracking only during batches targetting the front buffer. On SNB we must also update the FBC render tracking address whenever it changes. And since the register in question is stored in the context, we need to make sure we reload it with correct data after context switches. On IVB/HSW we use the render nuke mechanism, so no render tracking address updates are needed. Hoever on the blitter side we need to enable the blitter tracking like on SNB, and in addition we need to issue the cache clean messages, which we already did. v2: Introduce intel_fb_obj_has_fbc() Fix crtc locking around crtc->fb access Drop a hunk that was included by accident in v1 Set fbc_address_dirty=false not true after emitting the LRI v3: Now that fbc hangs on to the fb intel_fb_obj_has_fbc() doesn't need to upset lockdep anymore v4: Use |= instead of = to update fbc_address_dirty v5: |= for fbc_dirty too, kill fbc_obj variable, pack the intel_ringbuffer dirty bits using bitfields, skip ILK_FBC_RT_BASE write on SNB+, kill sandybridge_blit_fbc_update(), reorganize code to make future ILK FBC RT LRI support easier v6: Rework based on new fbc.obj handling Reviewed-by: Chris Wilson <chris@xxxxxxxxxxxxxxxxxx> Signed-off-by: Ville Syrjälä <ville.syrjala@xxxxxxxxxxxxxxx> --- drivers/gpu/drm/i915/i915_gem_context.c | 7 ++++ drivers/gpu/drm/i915/i915_gem_execbuffer.c | 29 +++++++++++++++ drivers/gpu/drm/i915/intel_display.c | 2 - drivers/gpu/drm/i915/intel_drv.h | 1 + drivers/gpu/drm/i915/intel_pm.c | 33 ++--------------- drivers/gpu/drm/i915/intel_ringbuffer.c | 59 +++++++++++++++++++++++++++++- drivers/gpu/drm/i915/intel_ringbuffer.h | 6 ++- 7 files changed, 102 insertions(+), 35 deletions(-) diff --git a/drivers/gpu/drm/i915/i915_gem_context.c b/drivers/gpu/drm/i915/i915_gem_context.c index 3ffe308..1aab053 100644 --- a/drivers/gpu/drm/i915/i915_gem_context.c +++ b/drivers/gpu/drm/i915/i915_gem_context.c @@ -588,6 +588,13 @@ mi_set_context(struct intel_engine_cs *ring, intel_ring_advance(ring); + /* + * FBC RT address is stored in the context, so we may have just + * restored it to an old value. Make sure we emit a new LRI + * to update the address. + */ + ring->fbc_address_dirty = true; + return ret; } diff --git a/drivers/gpu/drm/i915/i915_gem_execbuffer.c b/drivers/gpu/drm/i915/i915_gem_execbuffer.c index 93d7f72..5a91dae 100644 --- a/drivers/gpu/drm/i915/i915_gem_execbuffer.c +++ b/drivers/gpu/drm/i915/i915_gem_execbuffer.c @@ -955,6 +955,33 @@ i915_gem_validate_context(struct drm_device *dev, struct drm_file *file, } static void +i915_gem_execbuffer_mark_fbc_dirty(struct intel_engine_cs *ring, + struct list_head *vmas) +{ + struct drm_i915_private *dev_priv = ring->dev->dev_private; + struct i915_vma *vma; + u32 fbc_address = -1; + + list_for_each_entry(vma, vmas, exec_list) { + struct drm_i915_gem_object *obj = vma->obj; + + if (obj->base.pending_write_domain && + obj == dev_priv->fbc.obj) { + WARN_ON(fbc_address != -1 && + fbc_address != i915_gem_obj_ggtt_offset(obj)); + fbc_address = i915_gem_obj_ggtt_offset(obj); + } + } + + /* need to nuke/cache_clean on IVB+? */ + ring->fbc_dirty |= fbc_address != -1; + + /* need to update FBC tracking? */ + ring->fbc_address_dirty |= fbc_address != ring->fbc_address; + ring->fbc_address = fbc_address; +} + +static void i915_gem_execbuffer_move_to_active(struct list_head *vmas, struct intel_engine_cs *ring) { @@ -1322,6 +1349,8 @@ i915_gem_do_execbuffer(struct drm_device *dev, void *data, else exec_start += i915_gem_obj_offset(batch_obj, vm); + i915_gem_execbuffer_mark_fbc_dirty(ring, &eb->vmas); + ret = i915_gem_execbuffer_move_to_gpu(ring, &eb->vmas); if (ret) goto err; diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index e4cacd9..3011c35 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c @@ -8923,8 +8923,6 @@ void intel_mark_fb_busy(struct drm_i915_gem_object *obj, continue; intel_increase_pllclock(crtc); - if (ring && intel_fbc_enabled(dev)) - ring->fbc_dirty = true; } } diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h index 52e95f1..e9de8b1 100644 --- a/drivers/gpu/drm/i915/intel_drv.h +++ b/drivers/gpu/drm/i915/intel_drv.h @@ -732,6 +732,7 @@ bool intel_has_pending_fb_unpin(struct drm_device *dev); int intel_pch_rawclk(struct drm_device *dev); int valleyview_cur_cdclk(struct drm_i915_private *dev_priv); void intel_mark_busy(struct drm_device *dev); +bool intel_fb_obj_has_fbc(struct drm_i915_gem_object *obj); void intel_mark_fb_busy(struct drm_i915_gem_object *obj, struct intel_engine_cs *ring); void intel_mark_idle(struct drm_device *dev); diff --git a/drivers/gpu/drm/i915/intel_pm.c b/drivers/gpu/drm/i915/intel_pm.c index 6de283b..a5d3d9f 100644 --- a/drivers/gpu/drm/i915/intel_pm.c +++ b/drivers/gpu/drm/i915/intel_pm.c @@ -185,31 +185,6 @@ static bool g4x_fbc_enabled(struct drm_device *dev) return I915_READ(DPFC_CONTROL) & DPFC_CTL_EN; } -static void sandybridge_blit_fbc_update(struct drm_device *dev) -{ - struct drm_i915_private *dev_priv = dev->dev_private; - u32 blt_ecoskpd; - - /* Make sure blitter notifies FBC of writes */ - - /* Blitter is part of Media powerwell on VLV. No impact of - * his param in other platforms for now */ - gen6_gt_force_wake_get(dev_priv, FORCEWAKE_MEDIA); - - blt_ecoskpd = I915_READ(GEN6_BLITTER_ECOSKPD); - blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY << - GEN6_BLITTER_LOCK_SHIFT; - I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); - blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY; - I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); - blt_ecoskpd &= ~(GEN6_BLITTER_FBC_NOTIFY << - GEN6_BLITTER_LOCK_SHIFT); - I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); - POSTING_READ(GEN6_BLITTER_ECOSKPD); - - gen6_gt_force_wake_put(dev_priv, FORCEWAKE_MEDIA); -} - static void ironlake_enable_fbc(struct drm_device *dev) { struct drm_i915_private *dev_priv = dev->dev_private; @@ -229,8 +204,9 @@ static void ironlake_enable_fbc(struct drm_device *dev) dpfc_ctl |= dev_priv->fbc.fence_reg; I915_WRITE(ILK_DPFC_FENCE_YOFF, dev_priv->fbc.y); - I915_WRITE(ILK_FBC_RT_BASE, i915_gem_obj_ggtt_offset(dev_priv->fbc.obj) | - ILK_FBC_RT_VALID); + if (IS_GEN5(dev)) + I915_WRITE(ILK_FBC_RT_BASE, i915_gem_obj_ggtt_offset(dev_priv->fbc.obj) | + ILK_FBC_RT_VALID); /* enable it... */ I915_WRITE(ILK_DPFC_CONTROL, dpfc_ctl | DPFC_CTL_EN); @@ -238,7 +214,6 @@ static void ironlake_enable_fbc(struct drm_device *dev) I915_WRITE(SNB_DPFC_CTL_SA, SNB_CPU_FENCE_ENABLE | dev_priv->fbc.fence_reg); I915_WRITE(DPFC_CPU_FENCE_OFFSET, dev_priv->fbc.y); - sandybridge_blit_fbc_update(dev); } } @@ -297,8 +272,6 @@ static void gen7_enable_fbc(struct drm_device *dev) I915_WRITE(SNB_DPFC_CTL_SA, SNB_CPU_FENCE_ENABLE | dev_priv->fbc.fence_reg); I915_WRITE(DPFC_CPU_FENCE_OFFSET, dev_priv->fbc.y); - - sandybridge_blit_fbc_update(dev); } bool intel_fbc_enabled(struct drm_device *dev) diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.c b/drivers/gpu/drm/i915/intel_ringbuffer.c index 31321ae..d19738a 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.c +++ b/drivers/gpu/drm/i915/intel_ringbuffer.c @@ -69,6 +69,57 @@ void __intel_ring_advance(struct intel_engine_cs *ring) ring->write_tail(ring, ringbuf->tail); } +static int gen5_render_fbc_tracking(struct intel_engine_cs *ring) +{ + int ret; + + if (!ring->fbc_address_dirty) + return 0; + + ret = intel_ring_begin(ring, 4); + if (ret) + return ret; + + intel_ring_emit(ring, MI_NOOP); + intel_ring_emit(ring, MI_LOAD_REGISTER_IMM(1)); + intel_ring_emit(ring, ILK_FBC_RT_BASE); + if (ring->fbc_address != -1) + intel_ring_emit(ring, ring->fbc_address | + SNB_FBC_FRONT_BUFFER | ILK_FBC_RT_VALID); + else + intel_ring_emit(ring, 0); + intel_ring_advance(ring); + + ring->fbc_address_dirty = false; + + return 0; +} + +static int gen6_blt_fbc_tracking(struct intel_engine_cs *ring) +{ + int ret; + + if (!ring->fbc_address_dirty) + return 0; + + ret = intel_ring_begin(ring, 4); + if (ret) + return ret; + + intel_ring_emit(ring, MI_NOOP); + intel_ring_emit(ring, MI_LOAD_REGISTER_IMM(1)); + intel_ring_emit(ring, GEN6_BLITTER_ECOSKPD); + if (ring->fbc_address != -1) + intel_ring_emit(ring, _MASKED_BIT_ENABLE(GEN6_BLITTER_FBC_NOTIFY)); + else + intel_ring_emit(ring, _MASKED_BIT_DISABLE(GEN6_BLITTER_FBC_NOTIFY)); + intel_ring_advance(ring); + + ring->fbc_address_dirty = false; + + return 0; +} + static int gen2_render_ring_flush(struct intel_engine_cs *ring, u32 invalidate_domains, @@ -274,6 +325,9 @@ gen6_render_ring_flush(struct intel_engine_cs *ring, intel_ring_emit(ring, 0); intel_ring_advance(ring); + if (invalidate_domains) + return gen5_render_fbc_tracking(ring); + return 0; } @@ -316,6 +370,7 @@ static int gen7_ring_fbc_flush(struct intel_engine_cs *ring, u32 value) intel_ring_advance(ring); ring->fbc_dirty = false; + return 0; } @@ -1938,7 +1993,9 @@ static int gen6_ring_flush(struct intel_engine_cs *ring, } intel_ring_advance(ring); - if (IS_GEN7(dev) && !invalidate && flush) + if (invalidate) + return gen6_blt_fbc_tracking(ring); + else if (flush && IS_GEN7(dev)) return gen7_ring_fbc_flush(ring, FBC_REND_CACHE_CLEAN); return 0; diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.h b/drivers/gpu/drm/i915/intel_ringbuffer.h index 273abf3..80406ab 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.h +++ b/drivers/gpu/drm/i915/intel_ringbuffer.h @@ -176,8 +176,10 @@ struct intel_engine_cs { */ struct drm_i915_gem_request *preallocated_lazy_request; u32 outstanding_lazy_seqno; - bool gpu_caches_dirty; - bool fbc_dirty; + u32 fbc_address; + bool gpu_caches_dirty:1; + bool fbc_dirty:1; + bool fbc_address_dirty:1; wait_queue_head_t irq_queue; -- 1.8.5.5 _______________________________________________ Intel-gfx mailing list Intel-gfx@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/intel-gfx