[PM/Inliner] Add a test case that encapsulates the core issue addressed

in r297374.

I've extracted a small version of this from the C++ metaprogram Richard
came up with to exercise these kinds of issues and written comments to
describe both how to reproduce a fresh version of the test case and what
likely failure modes are.

The test case is still a bit brittle as it depends on the particular
inline cost modeling and SCC visitation order, but it definitely would
have caught the bug right away when developing things so it seems
a really valuable test case to have.

llvm-svn: 297935
This commit is contained in:
Chandler Carruth 2017-03-16 10:13:55 +00:00
parent 47bed8c9f7
commit 6ef42cc6bb
1 changed files with 460 additions and 0 deletions

View File

@ -0,0 +1,460 @@
; This test creates a monster SCC with a very pernicious call graph. It builds
; a cycle of cross-connected pairs of functions with interesting inlining
; decisions throughout, but ultimately trivial code complexity.
;
; Typically, a greedy approach to inlining works well for bottom-up inliners
; such as LLVM's. However, there is no way to be bottom-up over an SCC: it's
; a cycle! Greedily inlining as much as possible into each function of this
; *SCC* will have the disasterous effect of inlining all N-1 functions into the
; first one visited, N-2 functions into the second one visited, N-3 into the
; third, and so on. This is because until inlining occurs, each function in
; isolation appears to be an excellent inline candidate.
;
; Note that the exact number of calls in each function doesn't really matter.
; It is mostly a function of cost thresholds and visit order. Because this is an
; SCC there is no "right" or "wrong" answer here as long as no function blows up
; to be *huge*. The specific concerning pattern is if one or more functions get
; more than 16 calls in them.
;
; This test is extracted from the following C++ program compiled with Clang.
; The IR is simplified with SROA, instcombine, and simplify-cfg. Then C++
; linkage stuff, attributes, target specific things, metadata and comments were
; removed. The order of the fuctions is also made more predictable than Clang's
; output order.
;
; void g(int);
;
; template <bool K, int N> void f(bool *B, bool *E) {
; if (K)
; g(N);
; if (B == E)
; return;
; if (*B)
; f<true, N + 1>(B + 1, E);
; else
; f<false, N + 1>(B + 1, E);
; }
; template <> void f<false, MAX>(bool *B, bool *E) { return f<false, 0>(B, E); }
; template <> void f<true, MAX>(bool *B, bool *E) { return f<true, 0>(B, E); }
;
; void test(bool *B, bool *E) { f<false, 0>(B, E); }
;
; RUN: opt -S < %s -inline -inline-threshold=150 | FileCheck %s --check-prefixes=CHECK,OLD
; RUN: opt -S < %s -passes=inline -inline-threshold=150 | FileCheck %s --check-prefixes=CHECK,NEW
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
declare void @_Z1gi(i32)
; CHECK-LABEL: define void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi1EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi2EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb0ELi0EEvPbS0_(i8* %B, i8* %E) {
entry:
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi1EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi1EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi1EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi2EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb1ELi0EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1gi(i32 0)
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi1EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi1EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb0ELi1EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi2EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi3EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb0ELi1EEvPbS0_(i8* %B, i8* %E) {
entry:
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi2EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi2EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb1ELi1EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi2EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi3EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi3EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb1ELi1EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1gi(i32 1)
%cmp = icmp eq i8* %B, %E
; CHECK-NOT: call
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi2EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi2EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb0ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi4EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb0ELi2EEvPbS0_(i8* %B, i8* %E) {
entry:
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi3EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi3EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb1ELi2EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1gi(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb1ELi2EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1gi(i32 2)
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi3EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi3EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb0ELi3EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb0ELi0EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb0ELi3EEvPbS0_(i8* %B, i8* %E) {
entry:
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi4EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi4EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb1ELi3EEvPbS0_(
; CHECK-NOT: call
; CHECK: call void @_Z1gi(
; CHECK-NOT: call
; CHECK: call void @_Z1fILb1ELi0EEvPbS0_(
; CHECK-NOT: call
; CHECK: call void @_Z1fILb0ELi0EEvPbS0_(
; CHECK-NOT: call
define void @_Z1fILb1ELi3EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1gi(i32 3)
%cmp = icmp eq i8* %B, %E
br i1 %cmp, label %if.end3, label %if.end
if.end:
%0 = load i8, i8* %B, align 1
%tobool = icmp eq i8 %0, 0
%add.ptr2 = getelementptr inbounds i8, i8* %B, i64 1
br i1 %tobool, label %if.else, label %if.then1
if.then1:
call void @_Z1fILb1ELi4EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.else:
call void @_Z1fILb0ELi4EEvPbS0_(i8* %add.ptr2, i8* %E)
br label %if.end3
if.end3:
ret void
}
; CHECK-LABEL: define void @_Z1fILb0ELi4EEvPbS0_(
; CHECK-NOT: call
; CHECK: call void @_Z1fILb0ELi0EEvPbS0_(
; CHECK-NOT: call
define void @_Z1fILb0ELi4EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1fILb0ELi0EEvPbS0_(i8* %B, i8* %E)
ret void
}
; CHECK-LABEL: define void @_Z1fILb1ELi4EEvPbS0_(
; OLD-NOT: call
; OLD: call void @_Z1fILb1ELi0EEvPbS0_(
; OLD-NOT: call
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi1EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi2EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1gi(
; NEW-NOT: call
; NEW: call void @_Z1fILb1ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi0EEvPbS0_(
; NEW-NOT: call
; NEW: call void @_Z1fILb0ELi3EEvPbS0_(
; NEW-NOT: call
define void @_Z1fILb1ELi4EEvPbS0_(i8* %B, i8* %E) {
entry:
call void @_Z1fILb1ELi0EEvPbS0_(i8* %B, i8* %E)
ret void
}
; CHECK-LABEL: define void @_Z4testPbS_(
; CHECK: call
; CHECK-NOT: call
define void @_Z4testPbS_(i8* %B, i8* %E) {
entry:
call void @_Z1fILb0ELi0EEvPbS0_(i8* %B, i8* %E)
ret void
}