Fix infinite simulation cycles in SSA propagator.

This fixes https://github.com/KhronosGroup/SPIRV-Tools/issues/1159.  I
had missed a nuance in the original algorithm.  When simulating Phi
instructions, the SSA edges out of a Phi instruction should never be
added to the list of edges to simulate.

Phi instructions can be in SSA def-use cycles with other Phi
instructions.  This was causing the propagator to fall into an infinite
loop when the same def-use edge kept being added to the queue.

The original algorithm in the paper specifically separates the visit of
a Phi instruction vs the visit of a regular instruction.  This fix makes
the implementation match the original algorithm.
This commit is contained in:
Diego Novillo 2018-01-05 09:34:18 -05:00
parent 120ddffb41
commit 716718a5e9
3 changed files with 82 additions and 22 deletions

View File

@ -36,15 +36,25 @@ void SSAPropagator::AddControlEdge(const Edge& edge) {
blocks_.push(dest_bb); blocks_.push(dest_bb);
} }
void SSAPropagator::AddSSAEdges(uint32_t id) { void SSAPropagator::AddSSAEdges(ir::Instruction* instr) {
get_def_use_mgr()->ForEachUser(id, [this](ir::Instruction* instr) { if (instr->result_id() == 0 || instr->opcode() == SpvOpPhi) {
// If the basic block for |instr| has not been simulated yet, do nothing. // Ignore instructions that produce no result and Phi instructions. The SSA
if (!BlockHasBeenSimulated(ctx_->get_instr_block(instr))) { // edges out of Phi instructions are never added. Phi instructions can
// produce cycles in the def-use web and they are always simulated when a
// block is visited.
return; return;
} }
if (ShouldSimulateAgain(instr)) { get_def_use_mgr()->ForEachUser(
ssa_edge_uses_.push(instr); instr->result_id(), [this](ir::Instruction* use_instr) {
// If the basic block for |use_instr| has not been simulated yet, do
// nothing.
if (!BlockHasBeenSimulated(ctx_->get_instr_block(use_instr))) {
return;
}
if (ShouldSimulateAgain(use_instr)) {
ssa_edge_uses_.push(use_instr);
} }
}); });
} }
@ -74,9 +84,7 @@ bool SSAPropagator::Simulate(ir::Instruction* instr) {
// The statement produces a varying result, add it to the list of statements // The statement produces a varying result, add it to the list of statements
// not to simulate anymore and add its SSA def-use edges for simulation. // not to simulate anymore and add its SSA def-use edges for simulation.
DontSimulateAgain(instr); DontSimulateAgain(instr);
if (instr->result_id() > 0) { AddSSAEdges(instr);
AddSSAEdges(instr->result_id());
}
// If |instr| is a block terminator, add all the control edges out of its // If |instr| is a block terminator, add all the control edges out of its
// block. // block.
@ -88,11 +96,8 @@ bool SSAPropagator::Simulate(ir::Instruction* instr) {
} }
return false; return false;
} else if (status == kInteresting) { } else if (status == kInteresting) {
// If the instruction produced a new interesting value, add the SSA edge // Add the SSA edges coming out of this instruction.
// for its result ID. AddSSAEdges(instr);
if (instr->result_id() > 0) {
AddSSAEdges(instr->result_id());
}
// If there are multiple outgoing control flow edges and we know which one // If there are multiple outgoing control flow edges and we know which one
// will be taken, add the destination block to the CFG work list. // will be taken, add the destination block to the CFG work list.

View File

@ -248,12 +248,18 @@ class SSAPropagator {
return ctx_->get_def_use_mgr(); return ctx_->get_def_use_mgr();
} }
// If the CFG edge |e| has not been executed, add its destination block to the // If the CFG edge |e| has not been executed, this function adds |e|'s
// work list. // destination block to the work list.
void AddControlEdge(const Edge& e); void AddControlEdge(const Edge& e);
// Add all the instructions that use |id| to the SSA edges work list. // Adds all the instructions that use the result of |instr| to the SSA edges
void AddSSAEdges(uint32_t id); // work list. If |instr| produces no result id, this does nothing. This also
// does nothing if |instr| is a Phi instruction. Phi instructions are treated
// specially because (a) they can be in def-use cycles with other
// instructions, and (b) they are always executed when a basic block is
// simulated (see the description of the Sparse Conditional Constant algorithm
// in the original paper).
void AddSSAEdges(ir::Instruction* instr);
// IR context to use. // IR context to use.
ir::IRContext* ctx_; ir::IRContext* ctx_;

View File

@ -415,6 +415,55 @@ TEST_F(CCPTest, HandleAbortInstructions) {
SinglePassRunAndMatch<opt::CCPPass>(spv_asm, true); SinglePassRunAndMatch<opt::CCPPass>(spv_asm, true);
} }
TEST_F(CCPTest, SSAWebCycles) {
// Test reduced from https://github.com/KhronosGroup/SPIRV-Tools/issues/1159
// When there is a cycle in the SSA def-use web, the propagator was getting
// into an infinite loop. SSA edges for Phi instructions should not be
// added to the edges to simulate.
const std::string spv_asm = R"(
OpCapability Shader
%1 = OpExtInstImport "GLSL.std.450"
OpMemoryModel Logical GLSL450
OpEntryPoint Fragment %main "main"
OpExecutionMode %main OriginUpperLeft
OpSource GLSL 450
OpName %main "main"
%void = OpTypeVoid
%3 = OpTypeFunction %void
%int = OpTypeInt 32 1
%_ptr_Function_int = OpTypePointer Function %int
%int_0 = OpConstant %int 0
%int_4 = OpConstant %int 4
%bool = OpTypeBool
%int_1 = OpConstant %int 1
%_ptr_Output_int = OpTypePointer Output %int
%main = OpFunction %void None %3
%5 = OpLabel
OpBranch %11
%11 = OpLabel
%29 = OpPhi %int %int_0 %5 %22 %14
%30 = OpPhi %int %int_0 %5 %25 %14
OpLoopMerge %13 %14 None
OpBranch %15
%15 = OpLabel
%19 = OpSLessThan %bool %30 %int_4
; CHECK: OpBranchConditional %true {{%\d+}} {{%\d+}}
OpBranchConditional %19 %12 %13
%12 = OpLabel
; CHECK: OpIAdd %int %int_0 %int_0
%22 = OpIAdd %int %29 %30
OpBranch %14
%14 = OpLabel
; CHECK: OpPhi %int %int_0 {{%\d+}} %int_0 {{%\d+}}
%25 = OpPhi %int %int_0 %5 %30 %14
OpBranch %11
%13 = OpLabel
OpReturn
OpFunctionEnd
)";
SinglePassRunAndMatch<opt::CCPPass>(spv_asm, true);
}
#endif #endif
} // namespace } // namespace