intel/brw: Do not create empty basic blocks when removing instructions
If there's only a single instruction in a basic block, then removing it would create an empty block. We seem to have trouble representing those as there are no instructions with an IP inside the block; several places mess up connections. While most blocks end in control flow instructions (which are rarely eliminated), ones preceding a DO instruction may end in an ordinary instruction. This makes such blocks tricky to merge with adjacent blocks - they may be between loops. Any optimization pass may may find such an instruction and want to eliminate it, and most of them are unprepared to perform such CFG link surgery. Nor do we want to make every pass aware of this issue. To work around this, we simply replace an instruction with a NOP when removing it from a block containing only that instruction, leaving the block in place. Reviewed-by: Ian Romanick <ian.d.romanick@intel.com> Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/28971>
This commit is contained in:
@@ -110,6 +110,13 @@ brw_fs_opt_dead_control_flow_eliminate(fs_visitor &s)
|
|||||||
if_inst->predicate_inverse = !if_inst->predicate_inverse;
|
if_inst->predicate_inverse = !if_inst->predicate_inverse;
|
||||||
else_inst->remove(else_block);
|
else_inst->remove(else_block);
|
||||||
|
|
||||||
|
progress = true;
|
||||||
|
} else if (inst->opcode == BRW_OPCODE_NOP &&
|
||||||
|
prev_block->can_combine_with(block) &&
|
||||||
|
exec_list_is_singular(&block->parents) &&
|
||||||
|
exec_list_is_singular(&prev_block->children)) {
|
||||||
|
prev_block->combine_with(block);
|
||||||
|
inst->remove(prev_block);
|
||||||
progress = true;
|
progress = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -106,7 +106,9 @@ brw_fs_opt_dead_code_eliminate(fs_visitor &s)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (inst->dst.is_null() && can_eliminate(devinfo, inst, flag_live)) {
|
if (inst->dst.is_null() && can_eliminate(devinfo, inst, flag_live) &&
|
||||||
|
!(inst->opcode == BRW_OPCODE_NOP &&
|
||||||
|
exec_list_is_singular(&block->instructions))) {
|
||||||
inst->opcode = BRW_OPCODE_NOP;
|
inst->opcode = BRW_OPCODE_NOP;
|
||||||
progress = true;
|
progress = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -901,6 +901,9 @@ fs_generator::generate_code(const cfg_t *cfg, int dispatch_width,
|
|||||||
assert(inst->mlen <= BRW_MAX_MSG_LENGTH * reg_unit(devinfo));
|
assert(inst->mlen <= BRW_MAX_MSG_LENGTH * reg_unit(devinfo));
|
||||||
|
|
||||||
switch (inst->opcode) {
|
switch (inst->opcode) {
|
||||||
|
case BRW_OPCODE_NOP:
|
||||||
|
brw_NOP(p);
|
||||||
|
break;
|
||||||
case BRW_OPCODE_SYNC:
|
case BRW_OPCODE_SYNC:
|
||||||
assert(src[0].file == BRW_IMMEDIATE_VALUE);
|
assert(src[0].file == BRW_IMMEDIATE_VALUE);
|
||||||
brw_SYNC(p, tgl_sync_function(src[0].ud));
|
brw_SYNC(p, tgl_sync_function(src[0].ud));
|
||||||
|
|||||||
@@ -577,6 +577,14 @@ fs_inst::remove(bblock_t *block, bool defer_later_block_ip_updates)
|
|||||||
{
|
{
|
||||||
assert(inst_is_in_block(block, this) || !"Instruction not in block");
|
assert(inst_is_in_block(block, this) || !"Instruction not in block");
|
||||||
|
|
||||||
|
if (exec_list_is_singular(&block->instructions)) {
|
||||||
|
this->opcode = BRW_OPCODE_NOP;
|
||||||
|
this->resize_sources(0);
|
||||||
|
this->dst = fs_reg();
|
||||||
|
this->size_written = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (defer_later_block_ip_updates) {
|
if (defer_later_block_ip_updates) {
|
||||||
block->end_ip_delta--;
|
block->end_ip_delta--;
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
Reference in New Issue
Block a user