Commit a829f033 authored by Tvrtko Ursulin's avatar Tvrtko Ursulin Committed by Dave Airlie

drm/i915: Wedge the GPU if command parser setup fails

Commit 311a50e7 ("drm/i915: Add support for mandatory cmdparsing")
introduced mandatory command parsing but setup failures were not
translated into wedging the GPU which was probably the intent.

Possible errors come in two categories. Either the sanity check on
internal tables has failed, which should be caught in CI unless an
affected platform would be missed in testing; or memory allocation failure
happened during driver load, which should be extremely unlikely but for
correctness should still be handled.

v2:
 * Tidy coding style. (Chris)

[airlied: cherry-picked to avoid rc1 base]
Signed-off-by: default avatarTvrtko Ursulin <tvrtko.ursulin@intel.com>
Fixes: 311a50e7 ("drm/i915: Add support for mandatory cmdparsing")
Cc: Jon Bloomfield <jon.bloomfield@intel.com>
Cc: Joonas Lahtinen <joonas.lahtinen@linux.intel.com>
Cc: Chris Wilson <chris.p.wilson@intel.com>
Reviewed-by: default avatarChris Wilson <chris.p.wilson@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20210302114213.1102223-1-tvrtko.ursulin@linux.intel.com
(cherry picked from commit 5a1a659762d35a6dc51047c9127c011303c77b7f)
Signed-off-by: default avatarRodrigo Vivi <rodrigo.vivi@intel.com>
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>
parent fb198483
...@@ -713,9 +713,12 @@ static int engine_setup_common(struct intel_engine_cs *engine) ...@@ -713,9 +713,12 @@ static int engine_setup_common(struct intel_engine_cs *engine)
goto err_status; goto err_status;
} }
err = intel_engine_init_cmd_parser(engine);
if (err)
goto err_cmd_parser;
intel_engine_init_active(engine, ENGINE_PHYSICAL); intel_engine_init_active(engine, ENGINE_PHYSICAL);
intel_engine_init_execlists(engine); intel_engine_init_execlists(engine);
intel_engine_init_cmd_parser(engine);
intel_engine_init__pm(engine); intel_engine_init__pm(engine);
intel_engine_init_retire(engine); intel_engine_init_retire(engine);
...@@ -732,6 +735,8 @@ static int engine_setup_common(struct intel_engine_cs *engine) ...@@ -732,6 +735,8 @@ static int engine_setup_common(struct intel_engine_cs *engine)
return 0; return 0;
err_cmd_parser:
intel_breadcrumbs_free(engine->breadcrumbs);
err_status: err_status:
cleanup_status_page(engine); cleanup_status_page(engine);
return err; return err;
......
...@@ -940,7 +940,7 @@ static void fini_hash_table(struct intel_engine_cs *engine) ...@@ -940,7 +940,7 @@ static void fini_hash_table(struct intel_engine_cs *engine)
* struct intel_engine_cs based on whether the platform requires software * struct intel_engine_cs based on whether the platform requires software
* command parsing. * command parsing.
*/ */
void intel_engine_init_cmd_parser(struct intel_engine_cs *engine) int intel_engine_init_cmd_parser(struct intel_engine_cs *engine)
{ {
const struct drm_i915_cmd_table *cmd_tables; const struct drm_i915_cmd_table *cmd_tables;
int cmd_table_count; int cmd_table_count;
...@@ -948,7 +948,7 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine) ...@@ -948,7 +948,7 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine)
if (!IS_GEN(engine->i915, 7) && !(IS_GEN(engine->i915, 9) && if (!IS_GEN(engine->i915, 7) && !(IS_GEN(engine->i915, 9) &&
engine->class == COPY_ENGINE_CLASS)) engine->class == COPY_ENGINE_CLASS))
return; return 0;
switch (engine->class) { switch (engine->class) {
case RENDER_CLASS: case RENDER_CLASS:
...@@ -1013,19 +1013,19 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine) ...@@ -1013,19 +1013,19 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine)
break; break;
default: default:
MISSING_CASE(engine->class); MISSING_CASE(engine->class);
return; goto out;
} }
if (!validate_cmds_sorted(engine, cmd_tables, cmd_table_count)) { if (!validate_cmds_sorted(engine, cmd_tables, cmd_table_count)) {
drm_err(&engine->i915->drm, drm_err(&engine->i915->drm,
"%s: command descriptions are not sorted\n", "%s: command descriptions are not sorted\n",
engine->name); engine->name);
return; goto out;
} }
if (!validate_regs_sorted(engine)) { if (!validate_regs_sorted(engine)) {
drm_err(&engine->i915->drm, drm_err(&engine->i915->drm,
"%s: registers are not sorted\n", engine->name); "%s: registers are not sorted\n", engine->name);
return; goto out;
} }
ret = init_hash_table(engine, cmd_tables, cmd_table_count); ret = init_hash_table(engine, cmd_tables, cmd_table_count);
...@@ -1033,10 +1033,17 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine) ...@@ -1033,10 +1033,17 @@ void intel_engine_init_cmd_parser(struct intel_engine_cs *engine)
drm_err(&engine->i915->drm, drm_err(&engine->i915->drm,
"%s: initialised failed!\n", engine->name); "%s: initialised failed!\n", engine->name);
fini_hash_table(engine); fini_hash_table(engine);
return; goto out;
} }
engine->flags |= I915_ENGINE_USING_CMD_PARSER; engine->flags |= I915_ENGINE_USING_CMD_PARSER;
out:
if (intel_engine_requires_cmd_parser(engine) &&
!intel_engine_using_cmd_parser(engine))
return -EINVAL;
return 0;
} }
/** /**
......
...@@ -1952,7 +1952,7 @@ const char *i915_cache_level_str(struct drm_i915_private *i915, int type); ...@@ -1952,7 +1952,7 @@ const char *i915_cache_level_str(struct drm_i915_private *i915, int type);
/* i915_cmd_parser.c */ /* i915_cmd_parser.c */
int i915_cmd_parser_get_version(struct drm_i915_private *dev_priv); int i915_cmd_parser_get_version(struct drm_i915_private *dev_priv);
void intel_engine_init_cmd_parser(struct intel_engine_cs *engine); int intel_engine_init_cmd_parser(struct intel_engine_cs *engine);
void intel_engine_cleanup_cmd_parser(struct intel_engine_cs *engine); void intel_engine_cleanup_cmd_parser(struct intel_engine_cs *engine);
int intel_engine_cmd_parser(struct intel_engine_cs *engine, int intel_engine_cmd_parser(struct intel_engine_cs *engine,
struct i915_vma *batch, struct i915_vma *batch,
......
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