Commit b358c3b9 authored by Ville Syrjälä's avatar Ville Syrjälä

drm/i915: Make DSB lower level

We could have many different uses for the DSB(s) during a
single commit, so the current approach of passing the whole
crtc_state to the DSB functions is far too high level. Lower
the abstraction a little bit so each DSB user can decide where
to stick the command buffer/etc.

v2: Document the intel_dsb_prepare() return value (Ankit)
Signed-off-by: default avatarVille Syrjälä <ville.syrjala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20221123152638.20622-10-ville.syrjala@linux.intel.comReviewed-by: default avatarAnkit Nautiyal <ankit.k.nautiyal@intel.com>
Reviewed-by: default avatarUma Shankar <uma.shankar@intel.com>
parent efb2b57e
...@@ -842,7 +842,7 @@ static void ilk_lut_write(const struct intel_crtc_state *crtc_state, ...@@ -842,7 +842,7 @@ static void ilk_lut_write(const struct intel_crtc_state *crtc_state,
struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev); struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev);
if (crtc_state->dsb) if (crtc_state->dsb)
intel_dsb_reg_write(crtc_state, reg, val); intel_dsb_reg_write(crtc_state->dsb, reg, val);
else else
intel_de_write_fw(i915, reg, val); intel_de_write_fw(i915, reg, val);
} }
...@@ -853,7 +853,7 @@ static void ilk_lut_write_indexed(const struct intel_crtc_state *crtc_state, ...@@ -853,7 +853,7 @@ static void ilk_lut_write_indexed(const struct intel_crtc_state *crtc_state,
struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev); struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev);
if (crtc_state->dsb) if (crtc_state->dsb)
intel_dsb_indexed_reg_write(crtc_state, reg, val); intel_dsb_indexed_reg_write(crtc_state->dsb, reg, val);
else else
intel_de_write_fw(i915, reg, val); intel_de_write_fw(i915, reg, val);
} }
...@@ -1273,7 +1273,8 @@ static void icl_load_luts(const struct intel_crtc_state *crtc_state) ...@@ -1273,7 +1273,8 @@ static void icl_load_luts(const struct intel_crtc_state *crtc_state)
break; break;
} }
intel_dsb_commit(crtc_state); if (crtc_state->dsb)
intel_dsb_commit(crtc_state->dsb);
} }
static u32 chv_cgm_degamma_ldw(const struct drm_color_lut *color) static u32 chv_cgm_degamma_ldw(const struct drm_color_lut *color)
...@@ -1391,12 +1392,18 @@ void intel_color_commit_arm(const struct intel_crtc_state *crtc_state) ...@@ -1391,12 +1392,18 @@ void intel_color_commit_arm(const struct intel_crtc_state *crtc_state)
void intel_color_prepare_commit(struct intel_crtc_state *crtc_state) void intel_color_prepare_commit(struct intel_crtc_state *crtc_state)
{ {
intel_dsb_prepare(crtc_state); struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
crtc_state->dsb = intel_dsb_prepare(crtc);
} }
void intel_color_cleanup_commit(struct intel_crtc_state *crtc_state) void intel_color_cleanup_commit(struct intel_crtc_state *crtc_state)
{ {
intel_dsb_cleanup(crtc_state); if (!crtc_state->dsb)
return;
intel_dsb_cleanup(crtc_state->dsb);
crtc_state->dsb = NULL;
} }
static bool intel_can_preload_luts(const struct intel_crtc_state *new_crtc_state) static bool intel_can_preload_luts(const struct intel_crtc_state *new_crtc_state)
......
...@@ -24,8 +24,10 @@ enum dsb_id { ...@@ -24,8 +24,10 @@ enum dsb_id {
struct intel_dsb { struct intel_dsb {
enum dsb_id id; enum dsb_id id;
u32 *cmd_buf; u32 *cmd_buf;
struct i915_vma *vma; struct i915_vma *vma;
struct intel_crtc *crtc;
/* /*
* free_pos will point the first free entry position * free_pos will point the first free entry position
...@@ -113,7 +115,7 @@ static bool intel_dsb_disable_engine(struct drm_i915_private *i915, ...@@ -113,7 +115,7 @@ static bool intel_dsb_disable_engine(struct drm_i915_private *i915,
/** /**
* intel_dsb_indexed_reg_write() -Write to the DSB context for auto * intel_dsb_indexed_reg_write() -Write to the DSB context for auto
* increment register. * increment register.
* @crtc_state: intel_crtc_state structure * @dsb: DSB context
* @reg: register address. * @reg: register address.
* @val: value. * @val: value.
* *
...@@ -123,11 +125,10 @@ static bool intel_dsb_disable_engine(struct drm_i915_private *i915, ...@@ -123,11 +125,10 @@ static bool intel_dsb_disable_engine(struct drm_i915_private *i915,
* is done through mmio write. * is done through mmio write.
*/ */
void intel_dsb_indexed_reg_write(const struct intel_crtc_state *crtc_state, void intel_dsb_indexed_reg_write(struct intel_dsb *dsb,
i915_reg_t reg, u32 val) i915_reg_t reg, u32 val)
{ {
struct intel_dsb *dsb = crtc_state->dsb; struct intel_crtc *crtc = dsb->crtc;
struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
struct drm_i915_private *dev_priv = to_i915(crtc->base.dev); struct drm_i915_private *dev_priv = to_i915(crtc->base.dev);
u32 *buf = dsb->cmd_buf; u32 *buf = dsb->cmd_buf;
u32 reg_val; u32 reg_val;
...@@ -195,12 +196,11 @@ void intel_dsb_indexed_reg_write(const struct intel_crtc_state *crtc_state, ...@@ -195,12 +196,11 @@ void intel_dsb_indexed_reg_write(const struct intel_crtc_state *crtc_state,
* and rest all erroneous condition register programming is done * and rest all erroneous condition register programming is done
* through mmio write. * through mmio write.
*/ */
void intel_dsb_reg_write(const struct intel_crtc_state *crtc_state, void intel_dsb_reg_write(struct intel_dsb *dsb,
i915_reg_t reg, u32 val) i915_reg_t reg, u32 val)
{ {
struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc); struct intel_crtc *crtc = dsb->crtc;
struct drm_i915_private *dev_priv = to_i915(crtc->base.dev); struct drm_i915_private *dev_priv = to_i915(crtc->base.dev);
struct intel_dsb *dsb = crtc_state->dsb;
u32 *buf = dsb->cmd_buf; u32 *buf = dsb->cmd_buf;
if (drm_WARN_ON(&dev_priv->drm, dsb->free_pos >= DSB_BUF_SIZE)) { if (drm_WARN_ON(&dev_priv->drm, dsb->free_pos >= DSB_BUF_SIZE)) {
...@@ -217,17 +217,14 @@ void intel_dsb_reg_write(const struct intel_crtc_state *crtc_state, ...@@ -217,17 +217,14 @@ void intel_dsb_reg_write(const struct intel_crtc_state *crtc_state,
/** /**
* intel_dsb_commit() - Trigger workload execution of DSB. * intel_dsb_commit() - Trigger workload execution of DSB.
* @crtc_state: intel_crtc_state structure * @dsb: DSB context
* *
* This function is used to do actual write to hardware using DSB. * This function is used to do actual write to hardware using DSB.
* On errors, fall back to MMIO. Also this function help to reset the context.
*/ */
void intel_dsb_commit(const struct intel_crtc_state *crtc_state) void intel_dsb_commit(struct intel_dsb *dsb)
{ {
struct intel_dsb *dsb = crtc_state->dsb; struct intel_crtc *crtc = dsb->crtc;
struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc); struct drm_i915_private *dev_priv = to_i915(crtc->base.dev);
struct drm_device *dev = crtc->base.dev;
struct drm_i915_private *dev_priv = to_i915(dev);
enum pipe pipe = crtc->pipe; enum pipe pipe = crtc->pipe;
u32 tail; u32 tail;
...@@ -274,14 +271,16 @@ void intel_dsb_commit(const struct intel_crtc_state *crtc_state) ...@@ -274,14 +271,16 @@ void intel_dsb_commit(const struct intel_crtc_state *crtc_state)
/** /**
* intel_dsb_prepare() - Allocate, pin and map the DSB command buffer. * intel_dsb_prepare() - Allocate, pin and map the DSB command buffer.
* @crtc_state: intel_crtc_state structure to prepare associated dsb instance. * @crtc: the CRTC
* *
* This function prepare the command buffer which is used to store dsb * This function prepare the command buffer which is used to store dsb
* instructions with data. * instructions with data.
*
* Returns:
* DSB context, NULL on failure
*/ */
void intel_dsb_prepare(struct intel_crtc_state *crtc_state) struct intel_dsb *intel_dsb_prepare(struct intel_crtc *crtc)
{ {
struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
struct drm_i915_private *i915 = to_i915(crtc->base.dev); struct drm_i915_private *i915 = to_i915(crtc->base.dev);
struct intel_dsb *dsb; struct intel_dsb *dsb;
struct drm_i915_gem_object *obj; struct drm_i915_gem_object *obj;
...@@ -290,63 +289,60 @@ void intel_dsb_prepare(struct intel_crtc_state *crtc_state) ...@@ -290,63 +289,60 @@ void intel_dsb_prepare(struct intel_crtc_state *crtc_state)
intel_wakeref_t wakeref; intel_wakeref_t wakeref;
if (!HAS_DSB(i915)) if (!HAS_DSB(i915))
return; return NULL;
dsb = kmalloc(sizeof(*dsb), GFP_KERNEL); dsb = kmalloc(sizeof(*dsb), GFP_KERNEL);
if (!dsb) { if (!dsb)
drm_err(&i915->drm, "DSB object creation failed\n"); goto out;
return;
}
wakeref = intel_runtime_pm_get(&i915->runtime_pm); wakeref = intel_runtime_pm_get(&i915->runtime_pm);
obj = i915_gem_object_create_internal(i915, DSB_BUF_SIZE); obj = i915_gem_object_create_internal(i915, DSB_BUF_SIZE);
if (IS_ERR(obj)) { if (IS_ERR(obj))
kfree(dsb); goto out_put_rpm;
goto out;
}
vma = i915_gem_object_ggtt_pin(obj, NULL, 0, 0, 0); vma = i915_gem_object_ggtt_pin(obj, NULL, 0, 0, 0);
if (IS_ERR(vma)) { if (IS_ERR(vma)) {
i915_gem_object_put(obj); i915_gem_object_put(obj);
kfree(dsb); goto out_put_rpm;
goto out;
} }
buf = i915_gem_object_pin_map_unlocked(vma->obj, I915_MAP_WC); buf = i915_gem_object_pin_map_unlocked(vma->obj, I915_MAP_WC);
if (IS_ERR(buf)) { if (IS_ERR(buf)) {
i915_vma_unpin_and_release(&vma, I915_VMA_RELEASE_MAP); i915_vma_unpin_and_release(&vma, I915_VMA_RELEASE_MAP);
kfree(dsb); goto out_put_rpm;
goto out;
} }
intel_runtime_pm_put(&i915->runtime_pm, wakeref);
dsb->id = DSB1; dsb->id = DSB1;
dsb->vma = vma; dsb->vma = vma;
dsb->crtc = crtc;
dsb->cmd_buf = buf; dsb->cmd_buf = buf;
dsb->free_pos = 0; dsb->free_pos = 0;
dsb->ins_start_offset = 0; dsb->ins_start_offset = 0;
crtc_state->dsb = dsb;
out:
if (!crtc_state->dsb)
drm_info(&i915->drm,
"DSB queue setup failed, will fallback to MMIO for display HW programming\n");
return dsb;
out_put_rpm:
intel_runtime_pm_put(&i915->runtime_pm, wakeref); intel_runtime_pm_put(&i915->runtime_pm, wakeref);
kfree(dsb);
out:
drm_info_once(&i915->drm,
"DSB queue setup failed, will fallback to MMIO for display HW programming\n");
return NULL;
} }
/** /**
* intel_dsb_cleanup() - To cleanup DSB context. * intel_dsb_cleanup() - To cleanup DSB context.
* @crtc_state: intel_crtc_state structure to cleanup associated dsb instance. * @dsb: DSB context
* *
* This function cleanup the DSB context by unpinning and releasing * This function cleanup the DSB context by unpinning and releasing
* the VMA object associated with it. * the VMA object associated with it.
*/ */
void intel_dsb_cleanup(struct intel_crtc_state *crtc_state) void intel_dsb_cleanup(struct intel_dsb *dsb)
{ {
if (!crtc_state->dsb) i915_vma_unpin_and_release(&dsb->vma, I915_VMA_RELEASE_MAP);
return; kfree(dsb);
i915_vma_unpin_and_release(&crtc_state->dsb->vma, I915_VMA_RELEASE_MAP);
kfree(crtc_state->dsb);
crtc_state->dsb = NULL;
} }
...@@ -10,14 +10,15 @@ ...@@ -10,14 +10,15 @@
#include "i915_reg_defs.h" #include "i915_reg_defs.h"
struct intel_crtc_state; struct intel_crtc;
struct intel_dsb;
void intel_dsb_prepare(struct intel_crtc_state *crtc_state); struct intel_dsb *intel_dsb_prepare(struct intel_crtc *crtc);
void intel_dsb_cleanup(struct intel_crtc_state *crtc_state); void intel_dsb_cleanup(struct intel_dsb *dsb);
void intel_dsb_reg_write(const struct intel_crtc_state *crtc_state, void intel_dsb_reg_write(struct intel_dsb *dsb,
i915_reg_t reg, u32 val); i915_reg_t reg, u32 val);
void intel_dsb_indexed_reg_write(const struct intel_crtc_state *crtc_state, void intel_dsb_indexed_reg_write(struct intel_dsb *dsb,
i915_reg_t reg, u32 val); i915_reg_t reg, u32 val);
void intel_dsb_commit(const struct intel_crtc_state *crtc_state); void intel_dsb_commit(struct intel_dsb *dsb);
#endif #endif
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