forked from OSchip/llvm-project
Reapply [SCCP] Directly remove non-feasible edges
Reapply with DTU update moved after CFG update, which is a requirement of the API. ----- Non-feasible control-flow edges are currently removed by replacing the branch condition with a constant and then calling ConstantFoldTerminator. This happens in a rather roundabout manner, by inspecting the users (effectively: predecessors) of unreachable blocks, and further complicated by the need to explicitly materialize the condition for "forced" edges. I would like to extend SCCP to discard switch conditions that are non-feasible based on range information, but this is incompatible with the current approach (as there is no single constant we could use.) Instead, this patch explicitly removes non-feasible edges. It currently only needs to handle the case where there is a single feasible edge. The llvm_unreachable() branch will need to be implemented for the aforementioned switch improvement. Differential Revision: https://reviews.llvm.org/D84264
This commit is contained in:
parent
b5e14d78f1
commit
ad16e71c95
|
@ -276,7 +276,7 @@ public:
|
||||||
|
|
||||||
// isEdgeFeasible - Return true if the control flow edge from the 'From' basic
|
// isEdgeFeasible - Return true if the control flow edge from the 'From' basic
|
||||||
// block to the 'To' basic block is currently feasible.
|
// block to the 'To' basic block is currently feasible.
|
||||||
bool isEdgeFeasible(BasicBlock *From, BasicBlock *To);
|
bool isEdgeFeasible(BasicBlock *From, BasicBlock *To) const;
|
||||||
|
|
||||||
std::vector<ValueLatticeElement> getStructLatticeValueFor(Value *V) const {
|
std::vector<ValueLatticeElement> getStructLatticeValueFor(Value *V) const {
|
||||||
std::vector<ValueLatticeElement> StructValues;
|
std::vector<ValueLatticeElement> StructValues;
|
||||||
|
@ -705,7 +705,7 @@ void SCCPSolver::getFeasibleSuccessors(Instruction &TI,
|
||||||
|
|
||||||
// isEdgeFeasible - Return true if the control flow edge from the 'From' basic
|
// isEdgeFeasible - Return true if the control flow edge from the 'From' basic
|
||||||
// block to the 'To' basic block is currently feasible.
|
// block to the 'To' basic block is currently feasible.
|
||||||
bool SCCPSolver::isEdgeFeasible(BasicBlock *From, BasicBlock *To) {
|
bool SCCPSolver::isEdgeFeasible(BasicBlock *From, BasicBlock *To) const {
|
||||||
// Check if we've called markEdgeExecutable on the edge yet. (We could
|
// Check if we've called markEdgeExecutable on the edge yet. (We could
|
||||||
// be more aggressive and try to consider edges which haven't been marked
|
// be more aggressive and try to consider edges which haven't been marked
|
||||||
// yet, but there isn't any need.)
|
// yet, but there isn't any need.)
|
||||||
|
@ -1807,39 +1807,51 @@ static void findReturnsToZap(Function &F,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the condition for terminators that are branching on indeterminate
|
static bool removeNonFeasibleEdges(const SCCPSolver &Solver, BasicBlock *BB,
|
||||||
// values, forcing them to use a specific edge.
|
DomTreeUpdater &DTU) {
|
||||||
static void forceIndeterminateEdge(Instruction* I, SCCPSolver &Solver) {
|
SmallPtrSet<BasicBlock *, 8> FeasibleSuccessors;
|
||||||
BasicBlock *Dest = nullptr;
|
bool HasNonFeasibleEdges = false;
|
||||||
Constant *C = nullptr;
|
for (BasicBlock *Succ : successors(BB)) {
|
||||||
if (SwitchInst *SI = dyn_cast<SwitchInst>(I)) {
|
if (Solver.isEdgeFeasible(BB, Succ))
|
||||||
if (!isa<ConstantInt>(SI->getCondition())) {
|
FeasibleSuccessors.insert(Succ);
|
||||||
// Indeterminate switch; use first case value.
|
else
|
||||||
Dest = SI->case_begin()->getCaseSuccessor();
|
HasNonFeasibleEdges = true;
|
||||||
C = SI->case_begin()->getCaseValue();
|
|
||||||
}
|
}
|
||||||
} else if (BranchInst *BI = dyn_cast<BranchInst>(I)) {
|
|
||||||
if (!isa<ConstantInt>(BI->getCondition())) {
|
|
||||||
// Indeterminate branch; use false.
|
|
||||||
Dest = BI->getSuccessor(1);
|
|
||||||
C = ConstantInt::getFalse(BI->getContext());
|
|
||||||
}
|
|
||||||
} else if (IndirectBrInst *IBR = dyn_cast<IndirectBrInst>(I)) {
|
|
||||||
if (!isa<BlockAddress>(IBR->getAddress()->stripPointerCasts())) {
|
|
||||||
// Indeterminate indirectbr; use successor 0.
|
|
||||||
Dest = IBR->getSuccessor(0);
|
|
||||||
C = BlockAddress::get(IBR->getSuccessor(0));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
llvm_unreachable("Unexpected terminator instruction");
|
|
||||||
}
|
|
||||||
if (C) {
|
|
||||||
assert(Solver.isEdgeFeasible(I->getParent(), Dest) &&
|
|
||||||
"Didn't find feasible edge?");
|
|
||||||
(void)Dest;
|
|
||||||
|
|
||||||
I->setOperand(0, C);
|
// All edges feasible, nothing to do.
|
||||||
|
if (!HasNonFeasibleEdges)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// SCCP can only determine non-feasible edges for br, switch and indirectbr.
|
||||||
|
Instruction *TI = BB->getTerminator();
|
||||||
|
assert((isa<BranchInst>(TI) || isa<SwitchInst>(TI) ||
|
||||||
|
isa<IndirectBrInst>(TI)) &&
|
||||||
|
"Terminator must be a br, switch or indirectbr");
|
||||||
|
|
||||||
|
if (FeasibleSuccessors.size() == 1) {
|
||||||
|
// Replace with an unconditional branch to the only feasible successor.
|
||||||
|
BasicBlock *OnlyFeasibleSuccessor = *FeasibleSuccessors.begin();
|
||||||
|
SmallVector<DominatorTree::UpdateType, 8> Updates;
|
||||||
|
bool HaveSeenOnlyFeasibleSuccessor = false;
|
||||||
|
for (BasicBlock *Succ : successors(BB)) {
|
||||||
|
if (Succ == OnlyFeasibleSuccessor && !HaveSeenOnlyFeasibleSuccessor) {
|
||||||
|
// Don't remove the edge to the only feasible successor the first time
|
||||||
|
// we see it. We still do need to remove any multi-edges to it though.
|
||||||
|
HaveSeenOnlyFeasibleSuccessor = true;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Succ->removePredecessor(BB);
|
||||||
|
Updates.push_back({DominatorTree::Delete, BB, Succ});
|
||||||
|
}
|
||||||
|
|
||||||
|
BranchInst::Create(OnlyFeasibleSuccessor, BB);
|
||||||
|
TI->eraseFromParent();
|
||||||
|
DTU.applyUpdatesPermissive(Updates);
|
||||||
|
} else {
|
||||||
|
llvm_unreachable("Either all successors are feasible, or exactly one is");
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool llvm::runIPSCCP(
|
bool llvm::runIPSCCP(
|
||||||
|
@ -1972,45 +1984,11 @@ bool llvm::runIPSCCP(
|
||||||
/*UseLLVMTrap=*/false,
|
/*UseLLVMTrap=*/false,
|
||||||
/*PreserveLCSSA=*/false, &DTU);
|
/*PreserveLCSSA=*/false, &DTU);
|
||||||
|
|
||||||
// Now that all instructions in the function are constant folded,
|
for (BasicBlock &BB : F)
|
||||||
// use ConstantFoldTerminator to get rid of in-edges, record DT updates and
|
removeNonFeasibleEdges(Solver, &BB, DTU);
|
||||||
// delete dead BBs.
|
|
||||||
for (BasicBlock *DeadBB : BlocksToErase) {
|
|
||||||
// If there are any PHI nodes in this successor, drop entries for BB now.
|
|
||||||
for (Value::user_iterator UI = DeadBB->user_begin(),
|
|
||||||
UE = DeadBB->user_end();
|
|
||||||
UI != UE;) {
|
|
||||||
// Grab the user and then increment the iterator early, as the user
|
|
||||||
// will be deleted. Step past all adjacent uses from the same user.
|
|
||||||
auto *I = dyn_cast<Instruction>(*UI);
|
|
||||||
do { ++UI; } while (UI != UE && *UI == I);
|
|
||||||
|
|
||||||
// Ignore blockaddress users; BasicBlock's dtor will handle them.
|
for (BasicBlock *DeadBB : BlocksToErase)
|
||||||
if (!I) continue;
|
|
||||||
|
|
||||||
// If we have forced an edge for an indeterminate value, then force the
|
|
||||||
// terminator to fold to that edge.
|
|
||||||
forceIndeterminateEdge(I, Solver);
|
|
||||||
BasicBlock *InstBB = I->getParent();
|
|
||||||
bool Folded = ConstantFoldTerminator(InstBB,
|
|
||||||
/*DeleteDeadConditions=*/false,
|
|
||||||
/*TLI=*/nullptr, &DTU);
|
|
||||||
assert(Folded &&
|
|
||||||
"Expect TermInst on constantint or blockaddress to be folded");
|
|
||||||
(void) Folded;
|
|
||||||
// If we folded the terminator to an unconditional branch to another
|
|
||||||
// dead block, replace it with Unreachable, to avoid trying to fold that
|
|
||||||
// branch again.
|
|
||||||
BranchInst *BI = cast<BranchInst>(InstBB->getTerminator());
|
|
||||||
if (BI && BI->isUnconditional() &&
|
|
||||||
!Solver.isBlockExecutable(BI->getSuccessor(0))) {
|
|
||||||
InstBB->getTerminator()->eraseFromParent();
|
|
||||||
new UnreachableInst(InstBB->getContext(), InstBB);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Mark dead BB for deletion.
|
|
||||||
DTU.deleteBB(DeadBB);
|
DTU.deleteBB(DeadBB);
|
||||||
}
|
|
||||||
|
|
||||||
for (BasicBlock &BB : F) {
|
for (BasicBlock &BB : F) {
|
||||||
for (BasicBlock::iterator BI = BB.begin(), E = BB.end(); BI != E;) {
|
for (BasicBlock::iterator BI = BB.begin(), E = BB.end(); BI != E;) {
|
||||||
|
|
|
@ -231,12 +231,12 @@ define void @f7_nested_conds(i32* %a, i32 %b) {
|
||||||
; CHECK-NEXT: [[C_1:%.*]] = icmp ne i32 [[A_V]], 0
|
; CHECK-NEXT: [[C_1:%.*]] = icmp ne i32 [[A_V]], 0
|
||||||
; CHECK-NEXT: br i1 [[C_1]], label [[TRUE:%.*]], label [[FALSE:%.*]]
|
; CHECK-NEXT: br i1 [[C_1]], label [[TRUE:%.*]], label [[FALSE:%.*]]
|
||||||
; CHECK: false:
|
; CHECK: false:
|
||||||
; CHECK-NEXT: br i1 true, label [[TRUE_2:%.*]], label [[TRUE]]
|
; CHECK-NEXT: br label [[TRUE_2:%.*]]
|
||||||
; CHECK: true.2:
|
; CHECK: true.2:
|
||||||
; CHECK-NEXT: call void @use(i1 true)
|
; CHECK-NEXT: call void @use(i1 true)
|
||||||
; CHECK-NEXT: ret void
|
; CHECK-NEXT: ret void
|
||||||
; CHECK: true:
|
; CHECK: true:
|
||||||
; CHECK-NEXT: store i32 [[B:%.*]], i32* [[A]]
|
; CHECK-NEXT: store i32 [[B:%.*]], i32* [[A]], align 4
|
||||||
; CHECK-NEXT: ret void
|
; CHECK-NEXT: ret void
|
||||||
;
|
;
|
||||||
entry:
|
entry:
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
||||||
|
; RUN: opt -S -ipsccp < %s | FileCheck %s
|
||||||
|
; RUN: opt -S -passes='ipsccp,function(verify<domtree>)' < %s | FileCheck %s
|
||||||
|
|
||||||
|
; DTU should not crash.
|
||||||
|
|
||||||
|
define i32 @test() {
|
||||||
|
; CHECK-LABEL: @test(
|
||||||
|
; CHECK-NEXT: entry:
|
||||||
|
; CHECK-NEXT: br label [[FOR_BODY:%.*]]
|
||||||
|
; CHECK: for.body:
|
||||||
|
; CHECK-NEXT: br label [[IF_THEN2:%.*]]
|
||||||
|
; CHECK: if.then2:
|
||||||
|
; CHECK-NEXT: br label [[FOR_INC:%.*]]
|
||||||
|
; CHECK: for.inc:
|
||||||
|
; CHECK-NEXT: unreachable
|
||||||
|
;
|
||||||
|
entry:
|
||||||
|
br label %for.body
|
||||||
|
|
||||||
|
for.body: ; preds = %entry
|
||||||
|
br i1 true, label %if.then2, label %if.else
|
||||||
|
|
||||||
|
if.then2: ; preds = %for.body
|
||||||
|
br label %for.inc
|
||||||
|
|
||||||
|
if.else: ; preds = %for.body
|
||||||
|
br i1 undef, label %lor.rhs, label %if.then19.critedge
|
||||||
|
|
||||||
|
lor.rhs: ; preds = %if.else
|
||||||
|
br i1 undef, label %if.then19, label %for.inc
|
||||||
|
|
||||||
|
if.then19.critedge: ; preds = %if.else
|
||||||
|
br label %if.then19
|
||||||
|
|
||||||
|
if.then19: ; preds = %if.then19.critedge, %lor.rhs
|
||||||
|
unreachable
|
||||||
|
|
||||||
|
for.inc: ; preds = %lor.rhs, %if.then2
|
||||||
|
unreachable
|
||||||
|
}
|
|
@ -105,7 +105,7 @@ define void @pr46814(i32 %a) {
|
||||||
; CHECK-NEXT: [[C3:%.*]] = and i1 [[C1]], [[C2]]
|
; CHECK-NEXT: [[C3:%.*]] = and i1 [[C1]], [[C2]]
|
||||||
; CHECK-NEXT: br i1 [[C3]], label [[IF_1:%.*]], label [[EXIT:%.*]]
|
; CHECK-NEXT: br i1 [[C3]], label [[IF_1:%.*]], label [[EXIT:%.*]]
|
||||||
; CHECK: if.1:
|
; CHECK: if.1:
|
||||||
; CHECK-NEXT: br i1 true, label [[IF_2:%.*]], label [[EXIT]]
|
; CHECK-NEXT: br label [[IF_2:%.*]]
|
||||||
; CHECK: if.2:
|
; CHECK: if.2:
|
||||||
; CHECK-NEXT: br i1 true, label [[EXIT]], label [[EXIT]]
|
; CHECK-NEXT: br i1 true, label [[EXIT]], label [[EXIT]]
|
||||||
; CHECK: exit:
|
; CHECK: exit:
|
||||||
|
|
|
@ -136,13 +136,12 @@ define internal i1 @test2_g(%t1* %h, i32 %i) {
|
||||||
; CHECK-LABEL: define {{[^@]+}}@test2_g
|
; CHECK-LABEL: define {{[^@]+}}@test2_g
|
||||||
; CHECK-SAME: (%t1* [[H:%.*]], i32 [[I:%.*]])
|
; CHECK-SAME: (%t1* [[H:%.*]], i32 [[I:%.*]])
|
||||||
; CHECK-NEXT: entry:
|
; CHECK-NEXT: entry:
|
||||||
; CHECK-NEXT: br i1 true, label [[LAND_RHS:%.*]], label [[LAND_END:%.*]]
|
; CHECK-NEXT: br label [[LAND_RHS:%.*]]
|
||||||
; CHECK: land.rhs:
|
; CHECK: land.rhs:
|
||||||
; CHECK-NEXT: [[CALL:%.*]] = call i32 (...) @test2_j()
|
; CHECK-NEXT: [[CALL:%.*]] = call i32 (...) @test2_j()
|
||||||
; CHECK-NEXT: [[TOBOOL1:%.*]] = icmp ne i32 [[CALL]], 0
|
; CHECK-NEXT: [[TOBOOL1:%.*]] = icmp ne i32 [[CALL]], 0
|
||||||
; CHECK-NEXT: br label [[LAND_END]]
|
; CHECK-NEXT: br label [[LAND_END:%.*]]
|
||||||
; CHECK: land.end:
|
; CHECK: land.end:
|
||||||
; CHECK-NEXT: [[TMP0:%.*]] = phi i1 [ false, [[ENTRY:%.*]] ], [ [[TOBOOL1]], [[LAND_RHS]] ]
|
|
||||||
; CHECK-NEXT: ret i1 undef
|
; CHECK-NEXT: ret i1 undef
|
||||||
;
|
;
|
||||||
entry:
|
entry:
|
||||||
|
@ -196,10 +195,9 @@ define internal i32 @test3_k(i8 %h, i32 %i) {
|
||||||
; CHECK-NEXT: [[TMP1:%.*]] = inttoptr i64 [[CONV]] to %t1*
|
; CHECK-NEXT: [[TMP1:%.*]] = inttoptr i64 [[CONV]] to %t1*
|
||||||
; CHECK-NEXT: br label [[LOOP:%.*]]
|
; CHECK-NEXT: br label [[LOOP:%.*]]
|
||||||
; CHECK: loop:
|
; CHECK: loop:
|
||||||
; CHECK-NEXT: [[PHI:%.*]] = phi i1 [ undef, [[ENTRY:%.*]] ], [ false, [[LOOP]] ]
|
|
||||||
; CHECK-NEXT: [[CALL:%.*]] = call i1 @test3_g(%t1* [[TMP1]], i32 0)
|
; CHECK-NEXT: [[CALL:%.*]] = call i1 @test3_g(%t1* [[TMP1]], i32 0)
|
||||||
; CHECK-NEXT: call void @use.1(i1 false)
|
; CHECK-NEXT: call void @use.1(i1 false)
|
||||||
; CHECK-NEXT: br i1 false, label [[LOOP]], label [[EXIT:%.*]]
|
; CHECK-NEXT: br label [[EXIT:%.*]]
|
||||||
; CHECK: exit:
|
; CHECK: exit:
|
||||||
; CHECK-NEXT: ret i32 undef
|
; CHECK-NEXT: ret i32 undef
|
||||||
;
|
;
|
||||||
|
|
|
@ -5,11 +5,11 @@
|
||||||
define void @barney() {
|
define void @barney() {
|
||||||
; CHECK-LABEL: @barney(
|
; CHECK-LABEL: @barney(
|
||||||
; CHECK-NEXT: bb:
|
; CHECK-NEXT: bb:
|
||||||
; CHECK-NEXT: br label %bb9
|
; CHECK-NEXT: br label [[BB9:%.*]]
|
||||||
; CHECK: bb6:
|
; CHECK: bb6:
|
||||||
; CHECK-NEXT: unreachable
|
; CHECK-NEXT: unreachable
|
||||||
; CHECK: bb9:
|
; CHECK: bb9:
|
||||||
; CHECK-NEXT: unreachable
|
; CHECK-NEXT: br label [[BB6:%.*]]
|
||||||
;
|
;
|
||||||
bb:
|
bb:
|
||||||
br label %bb9
|
br label %bb9
|
||||||
|
@ -29,9 +29,9 @@ bb9: ; preds = %bb
|
||||||
define void @blam() {
|
define void @blam() {
|
||||||
; CHECK-LABEL: @blam(
|
; CHECK-LABEL: @blam(
|
||||||
; CHECK-NEXT: bb:
|
; CHECK-NEXT: bb:
|
||||||
; CHECK-NEXT: br label %bb16
|
; CHECK-NEXT: br label [[BB16:%.*]]
|
||||||
; CHECK: bb16:
|
; CHECK: bb16:
|
||||||
; CHECK-NEXT: br label %bb38
|
; CHECK-NEXT: br label [[BB38:%.*]]
|
||||||
; CHECK: bb38:
|
; CHECK: bb38:
|
||||||
; CHECK-NEXT: unreachable
|
; CHECK-NEXT: unreachable
|
||||||
;
|
;
|
||||||
|
@ -62,9 +62,9 @@ bb38: ; preds = %bb16
|
||||||
define void @hoge() {
|
define void @hoge() {
|
||||||
; CHECK-LABEL: @hoge(
|
; CHECK-LABEL: @hoge(
|
||||||
; CHECK-NEXT: bb:
|
; CHECK-NEXT: bb:
|
||||||
; CHECK-NEXT: br label %bb2
|
; CHECK-NEXT: br label [[BB2:%.*]]
|
||||||
; CHECK: bb2:
|
; CHECK: bb2:
|
||||||
; CHECK-NEXT: unreachable
|
; CHECK-NEXT: br label [[BB3:%.*]]
|
||||||
; CHECK: bb3:
|
; CHECK: bb3:
|
||||||
; CHECK-NEXT: unreachable
|
; CHECK-NEXT: unreachable
|
||||||
;
|
;
|
||||||
|
|
|
@ -23,15 +23,11 @@ define i32 @test_duplicate_successors_phi(i1 %c, i32 %x) {
|
||||||
; CHECK-NEXT: entry:
|
; CHECK-NEXT: entry:
|
||||||
; CHECK-NEXT: br i1 [[C:%.*]], label [[SWITCH:%.*]], label [[END:%.*]]
|
; CHECK-NEXT: br i1 [[C:%.*]], label [[SWITCH:%.*]], label [[END:%.*]]
|
||||||
; CHECK: switch:
|
; CHECK: switch:
|
||||||
; CHECK-NEXT: switch i32 -1, label [[SWITCH_DEFAULT:%.*]] [
|
; CHECK-NEXT: br label [[SWITCH_DEFAULT:%.*]]
|
||||||
; CHECK-NEXT: i32 0, label [[END]]
|
|
||||||
; CHECK-NEXT: i32 1, label [[END]]
|
|
||||||
; CHECK-NEXT: ]
|
|
||||||
; CHECK: switch.default:
|
; CHECK: switch.default:
|
||||||
; CHECK-NEXT: ret i32 -1
|
; CHECK-NEXT: ret i32 -1
|
||||||
; CHECK: end:
|
; CHECK: end:
|
||||||
; CHECK-NEXT: [[PHI:%.*]] = phi i32 [ [[X:%.*]], [[ENTRY:%.*]] ], [ 1, [[SWITCH]] ], [ 1, [[SWITCH]] ]
|
; CHECK-NEXT: ret i32 [[X:%.*]]
|
||||||
; CHECK-NEXT: ret i32 [[PHI]]
|
|
||||||
;
|
;
|
||||||
entry:
|
entry:
|
||||||
br i1 %c, label %switch, label %end
|
br i1 %c, label %switch, label %end
|
||||||
|
|
|
@ -216,11 +216,11 @@ define void @rotated_loop_2(i32 %x) {
|
||||||
; IPSCCP: bb3:
|
; IPSCCP: bb3:
|
||||||
; IPSCCP-NEXT: br label [[EXIT]]
|
; IPSCCP-NEXT: br label [[EXIT]]
|
||||||
; IPSCCP: exit:
|
; IPSCCP: exit:
|
||||||
; IPSCCP-NEXT: [[P:%.*]] = phi i32 [ 1, [[ENTRY:%.*]] ], [ 3, [[BB1]] ], [ 2, [[BB2]] ], [ 5, [[BB3]] ], [ [[A:%.*]], [[EXIT]] ]
|
; IPSCCP-NEXT: [[P:%.*]] = phi i32 [ 1, [[ENTRY:%.*]] ], [ 3, [[BB1]] ], [ 2, [[BB2]] ], [ 5, [[BB3]] ]
|
||||||
; IPSCCP-NEXT: [[A]] = add i32 [[P]], 1
|
; IPSCCP-NEXT: [[A:%.*]] = add i32 [[P]], 1
|
||||||
; IPSCCP-NEXT: call void @use(i1 true)
|
; IPSCCP-NEXT: call void @use(i1 true)
|
||||||
; IPSCCP-NEXT: call void @use(i1 false)
|
; IPSCCP-NEXT: call void @use(i1 false)
|
||||||
; IPSCCP-NEXT: br i1 false, label [[EXIT]], label [[EXIT_1:%.*]]
|
; IPSCCP-NEXT: br label [[EXIT_1:%.*]]
|
||||||
; IPSCCP: exit.1:
|
; IPSCCP: exit.1:
|
||||||
; IPSCCP-NEXT: ret void
|
; IPSCCP-NEXT: ret void
|
||||||
;
|
;
|
||||||
|
@ -451,10 +451,10 @@ define void @foo(i64* %arg) {
|
||||||
; SCCP-NEXT: [[TMP7:%.*]] = sub i64 3, [[TMP6]]
|
; SCCP-NEXT: [[TMP7:%.*]] = sub i64 3, [[TMP6]]
|
||||||
; SCCP-NEXT: [[TMP8:%.*]] = shl i64 [[TMP7]], 1
|
; SCCP-NEXT: [[TMP8:%.*]] = shl i64 [[TMP7]], 1
|
||||||
; SCCP-NEXT: [[TMP9:%.*]] = trunc i64 [[TMP8]] to i32
|
; SCCP-NEXT: [[TMP9:%.*]] = trunc i64 [[TMP8]] to i32
|
||||||
; SCCP-NEXT: [[TMP10:%.*]] = zext i32 [[TMP9]] to i64
|
; SCCP-NEXT: [[TMP0:%.*]] = zext i32 [[TMP9]] to i64
|
||||||
; SCCP-NEXT: br label [[BB11:%.*]]
|
; SCCP-NEXT: br label [[BB11:%.*]]
|
||||||
; SCCP: bb11:
|
; SCCP: bb11:
|
||||||
; SCCP-NEXT: [[TMP12:%.*]] = phi i64 [ [[TMP10]], [[BB4]] ], [ [[TMP17:%.*]], [[BB18:%.*]] ]
|
; SCCP-NEXT: [[TMP12:%.*]] = phi i64 [ [[TMP0]], [[BB4]] ], [ [[TMP17:%.*]], [[BB18:%.*]] ]
|
||||||
; SCCP-NEXT: br label [[BB13:%.*]]
|
; SCCP-NEXT: br label [[BB13:%.*]]
|
||||||
; SCCP: bb13:
|
; SCCP: bb13:
|
||||||
; SCCP-NEXT: [[C_1:%.*]] = icmp eq i64 [[TMP12]], 6
|
; SCCP-NEXT: [[C_1:%.*]] = icmp eq i64 [[TMP12]], 6
|
||||||
|
@ -489,10 +489,10 @@ define void @foo(i64* %arg) {
|
||||||
; IPSCCP-NEXT: [[TMP7:%.*]] = sub i64 3, [[TMP6]]
|
; IPSCCP-NEXT: [[TMP7:%.*]] = sub i64 3, [[TMP6]]
|
||||||
; IPSCCP-NEXT: [[TMP8:%.*]] = shl i64 [[TMP7]], 1
|
; IPSCCP-NEXT: [[TMP8:%.*]] = shl i64 [[TMP7]], 1
|
||||||
; IPSCCP-NEXT: [[TMP9:%.*]] = trunc i64 [[TMP8]] to i32
|
; IPSCCP-NEXT: [[TMP9:%.*]] = trunc i64 [[TMP8]] to i32
|
||||||
; IPSCCP-NEXT: [[TMP10:%.*]] = zext i32 [[TMP9]] to i64
|
; IPSCCP-NEXT: [[TMP0:%.*]] = zext i32 [[TMP9]] to i64
|
||||||
; IPSCCP-NEXT: br label [[BB11:%.*]]
|
; IPSCCP-NEXT: br label [[BB11:%.*]]
|
||||||
; IPSCCP: bb11:
|
; IPSCCP: bb11:
|
||||||
; IPSCCP-NEXT: [[TMP12:%.*]] = phi i64 [ [[TMP10]], [[BB4]] ], [ [[TMP17:%.*]], [[BB18:%.*]] ]
|
; IPSCCP-NEXT: [[TMP12:%.*]] = phi i64 [ [[TMP0]], [[BB4]] ], [ [[TMP17:%.*]], [[BB18:%.*]] ]
|
||||||
; IPSCCP-NEXT: br label [[BB13:%.*]]
|
; IPSCCP-NEXT: br label [[BB13:%.*]]
|
||||||
; IPSCCP: bb13:
|
; IPSCCP: bb13:
|
||||||
; IPSCCP-NEXT: [[C_1:%.*]] = icmp eq i64 [[TMP12]], 6
|
; IPSCCP-NEXT: [[C_1:%.*]] = icmp eq i64 [[TMP12]], 6
|
||||||
|
|
Loading…
Reference in New Issue