[llvm] r242337 - Fix mergefunc infinite loop
JF Bastien
jfb at google.com
Wed Jul 15 14:51:33 PDT 2015
Author: jfb
Date: Wed Jul 15 16:51:33 2015
New Revision: 242337
URL: http://llvm.org/viewvc/llvm-project?rev=242337&view=rev
Log:
Fix mergefunc infinite loop
Self-referential constants containing references to a merged function
no longer cause the MergeFunctions pass to infinite loop. Also adds a
reproduction IR which would otherwise fail, which was isolated from a similar
issue in Chromium.
Author: jrkoenig
Reviewers: nlewycky, jfb
Subscribers: llvm-commits, nlewycky, jfb
Differential Revision: http://reviews.llvm.org/D11208
Added:
llvm/trunk/test/Transforms/MergeFunc/self-referential-global.ll
Modified:
llvm/trunk/lib/Transforms/IPO/MergeFunctions.cpp
Modified: llvm/trunk/lib/Transforms/IPO/MergeFunctions.cpp
URL: http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Transforms/IPO/MergeFunctions.cpp?rev=242337&r1=242336&r2=242337&view=diff
==============================================================================
--- llvm/trunk/lib/Transforms/IPO/MergeFunctions.cpp (original)
+++ llvm/trunk/lib/Transforms/IPO/MergeFunctions.cpp Wed Jul 15 16:51:33 2015
@@ -1516,6 +1516,8 @@ void MergeFunctions::remove(Function *F)
void MergeFunctions::removeUsers(Value *V) {
std::vector<Value *> Worklist;
Worklist.push_back(V);
+ SmallSet<Value*, 8> Visited;
+ Visited.insert(V);
while (!Worklist.empty()) {
Value *V = Worklist.back();
Worklist.pop_back();
@@ -1526,8 +1528,10 @@ void MergeFunctions::removeUsers(Value *
} else if (isa<GlobalValue>(U)) {
// do nothing
} else if (Constant *C = dyn_cast<Constant>(U)) {
- for (User *UU : C->users())
- Worklist.push_back(UU);
+ for (User *UU : C->users()) {
+ if (!Visited.insert(UU).second)
+ Worklist.push_back(UU);
+ }
}
}
}
Added: llvm/trunk/test/Transforms/MergeFunc/self-referential-global.ll
URL: http://llvm.org/viewvc/llvm-project/llvm/trunk/test/Transforms/MergeFunc/self-referential-global.ll?rev=242337&view=auto
==============================================================================
--- llvm/trunk/test/Transforms/MergeFunc/self-referential-global.ll (added)
+++ llvm/trunk/test/Transforms/MergeFunc/self-referential-global.ll Wed Jul 15 16:51:33 2015
@@ -0,0 +1,40 @@
+; RUN: opt -mergefunc -disable-output < %s
+
+; A linked list type and simple payload
+%LL = type { %S, %LL* }
+%S = type { void (%S*, i32)* }
+
+; Table refers to itself via GEP
+ at Table = internal global [3 x %LL] [%LL { %S { void (%S*, i32)* @B }, %LL* getelementptr inbounds ([3 x %LL], [3 x %LL]* @Table, i32 0, i32 0) }, %LL { %S { void (%S*, i32)* @A }, %LL* getelementptr inbounds ([3 x %LL], [3 x %LL]* @Table, i32 0, i32 0) }, %LL { %S { void (%S*, i32)* @A }, %LL* getelementptr inbounds ([3 x %LL], [3 x %LL]* @Table, i32 0, i32 0) }], align 16
+
+; The body of this is irrelevant; it is long so that mergefunc doesn't skip it as a small function.
+define internal void @A(%S* %self, i32 %a) {
+ %1 = add i32 %a, 32
+ %2 = add i32 %1, 32
+ %3 = add i32 %2, 32
+ %4 = add i32 %3, 32
+ %5 = add i32 %4, 32
+ %6 = add i32 %5, 32
+ %7 = add i32 %6, 32
+ %8 = add i32 %7, 32
+ %9 = add i32 %8, 32
+ %10 = add i32 %9, 32
+ %11 = add i32 %10, 32
+ ret void
+}
+
+define internal void @B(%S* %self, i32 %a) {
+ %1 = add i32 %a, 32
+ %2 = add i32 %1, 32
+ %3 = add i32 %2, 32
+ %4 = add i32 %3, 32
+ %5 = add i32 %4, 32
+ %6 = add i32 %5, 32
+ %7 = add i32 %6, 32
+ %8 = add i32 %7, 32
+ %9 = add i32 %8, 32
+ %10 = add i32 %9, 32
+ %11 = add i32 %10, 32
+ ret void
+}
+
More information about the llvm-commits
mailing list