+ nir_block *first_then = nir_if_first_then_block(nif);
+ if (use_src->parent_instr->block->index > first_then->index) {
+ nir_block *first_else = nir_if_first_else_block(nif);
+ if (use_src->parent_instr->block->index < first_else->index) {
+ nir_instr *instr = nir_block_first_instr(first_then);
+ replace_if_condition_use_with_const(use_src, instr,
+ NIR_TRUE, mem_ctx);
+
+ progress = true;
+ } else if (use_src->parent_instr->block->index <
+ nir_if_last_else_block(nif)->index) {
+ nir_instr *instr = nir_block_first_instr(first_else);
+ replace_if_condition_use_with_const(use_src, instr,
+ NIR_FALSE, mem_ctx);
+
+ progress = true;
+ }
+ }
+ }
+
+ return progress;
+}
+
+static bool
+opt_if_cf_list(nir_builder *b, struct exec_list *cf_list, void *mem_ctx,
+ bool *md_altered)
{
bool progress = false;
foreach_list_typed(nir_cf_node, cf_node, node, cf_list) {
@@ -361,17 +406,30 @@ opt_if_cf_list(nir_builder *b, struct exec_list *cf_list)
case nir_cf_node_if: {
nir_if *nif = nir_cf_node_as_if(cf_node);
- progress |= opt_if_cf_list(b, &nif->then_list);
- progress |= opt_if_cf_list(b, &nif->else_list);
- progress |= opt_if_loop_terminator(nif);
- progress |= opt_if_simplification(b, nif);
+ progress |= opt_if_cf_list(b, &nif->then_list, mem_ctx, md_altered);
+ progress |= opt_if_cf_list(b, &nif->else_list, mem_ctx, md_altered);
+
+ if (*md_altered == false)
+ progress |= opt_if_evaluate_condition_use(nif, mem_ctx);
+
+ bool lp_term_progress = opt_if_loop_terminator(nif);
+ bool if_simp_progress = opt_if_simplification(b, nif);
+ if (lp_term_progress || if_simp_progress) {
+ *md_altered = true;
+ progress = true;
+ }
+
break;
}
case nir_cf_node_loop: {
nir_loop *loop = nir_cf_node_as_loop(cf_node);
- progress |= opt_if_cf_list(b, &loop->body);
- progress |= opt_peel_loop_initial_if(loop);
+ progress |= opt_if_cf_list(b, &loop->body, mem_ctx, md_altered);
+ bool lp_peel_progress = opt_peel_loop_initial_if(loop);
+ if (lp_peel_progress) {
+ *md_altered = true;
+ progress = true;
+ }
break;
}
@@ -395,7 +453,17 @@ nir_opt_if(nir_shader *shader)
nir_builder b;
nir_builder_init(&b, function->impl);
- if (opt_if_cf_list(&b, &function->impl->body)) {
+ nir_metadata_require(function->impl, nir_metadata_block_index);
+
+ /* opt_if_evaluate_condition_use() depends on nir_metadata_block_index
+ * if one of the other opts has altered the block in some way we need to
+ * make sure we skip this opt.
+ */
+ bool index_metadata_altered = false;
+
+ void *mem_ctx = ralloc_parent(function->impl);
+ if (opt_if_cf_list(&b, &function->impl->body, mem_ctx,
+ &index_metadata_altered)) {
nir_metadata_preserve(function->impl, nir_metadata_none);
/* If that made progress, we're no longer really in SSA form. We