mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 10:42:39 +01:00
8be96b95c4
There was an alias between 'simplifycfg' and 'simplify-cfg' in the PassRegistry. That was the original reason for this patch, which effectively removes the alias. This patch also replaces all occurrances of 'simplify-cfg' by 'simplifycfg'. Reason for choosing that form for the name is that it matches the DEBUG_TYPE for the pass, and the legacy PM name and also how it is spelled out in other passes such as 'loop-simplifycfg', and in other options such as 'simplifycfg-merge-cond-stores'. I for some reason the name should be changed to 'simplify-cfg' in the future, then I think such a renaming should be more widely done and not only impacting the PassRegistry. Reviewed By: aeubanks Differential Revision: https://reviews.llvm.org/D105627
342 lines
6.4 KiB
LLVM
342 lines
6.4 KiB
LLVM
; RUN: opt -aa-pipeline=basic-aa -passes='cgscc(function-attrs,function(simplifycfg))' -S < %s | FileCheck %s
|
|
|
|
declare void @readnone() nofree nosync readnone
|
|
declare void @unknown()
|
|
declare void @reference_function_pointer(void()*) nofree nosync readnone
|
|
|
|
; The @test1_* set of functions checks that when we mutate functions with
|
|
; simplifycfg to delete call edges and this ends up splitting both the SCCs
|
|
; and the RefSCCs that those functions are in, we re-run the CGSCC passes to
|
|
; observe the refined call graph structure.
|
|
|
|
; CHECK: define void @test1_a() {
|
|
define void @test1_a() {
|
|
call void @test1_b1()
|
|
call void @test1_b2()
|
|
call void @test1_b3()
|
|
call void @test1_b4()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test1_b1() #0 {
|
|
define void @test1_b1() {
|
|
call void @readnone()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test1_b2() #0 {
|
|
define void @test1_b2() {
|
|
call void @readnone()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test1_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test1_b3() {
|
|
define void @test1_b3() {
|
|
call void @unknown()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test1_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test1_b4() #0 {
|
|
define void @test1_b4() {
|
|
call void @readnone()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test1_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
|
|
; The @test2_* set of functions provide similar checks to @test1_* but only
|
|
; splitting the SCCs while leaving the RefSCC intact. This is accomplished by
|
|
; having dummy ref edges to the root function.
|
|
|
|
; CHECK: define void @test2_a() {
|
|
define void @test2_a() {
|
|
call void @test2_b1()
|
|
call void @test2_b2()
|
|
call void @test2_b3()
|
|
call void @test2_b4()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test2_b1() #0 {
|
|
define void @test2_b1() {
|
|
call void @readnone()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test2_b2() #0 {
|
|
define void @test2_b2() {
|
|
call void @reference_function_pointer(void()* @test2_a)
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test2_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test2_b3() {
|
|
define void @test2_b3() {
|
|
call void @reference_function_pointer(void()* @test2_a)
|
|
call void @unknown()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test2_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test2_b4() #0 {
|
|
define void @test2_b4() {
|
|
call void @reference_function_pointer(void()* @test2_a)
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test2_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
|
|
; The @test3_* set of functions are the same challenge as @test1_* but with
|
|
; multiple layers that have to be traversed in the correct order instead of
|
|
; a single node.
|
|
|
|
; CHECK: define void @test3_a() {
|
|
define void @test3_a() {
|
|
call void @test3_b11()
|
|
call void @test3_b21()
|
|
call void @test3_b31()
|
|
call void @test3_b41()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b11() #0 {
|
|
define void @test3_b11() {
|
|
call void @test3_b12()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b12() #0 {
|
|
define void @test3_b12() {
|
|
call void @test3_b13()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b13() #0 {
|
|
define void @test3_b13() {
|
|
call void @readnone()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b21() #0 {
|
|
define void @test3_b21() {
|
|
call void @test3_b22()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b22() #0 {
|
|
define void @test3_b22() {
|
|
call void @test3_b23()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b23() #0 {
|
|
define void @test3_b23() {
|
|
call void @readnone()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test3_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b31() {
|
|
define void @test3_b31() {
|
|
call void @test3_b32()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b32() {
|
|
define void @test3_b32() {
|
|
call void @test3_b33()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b33() {
|
|
define void @test3_b33() {
|
|
call void @unknown()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test3_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b41() #0 {
|
|
define void @test3_b41() {
|
|
call void @test3_b42()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b42() #0 {
|
|
define void @test3_b42() {
|
|
call void @test3_b43()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test3_b43() #0 {
|
|
define void @test3_b43() {
|
|
call void @readnone()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test3_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
|
|
; The @test4_* functions exercise the same core challenge as the @test2_*
|
|
; functions, but again include long chains instead of single nodes and ensure
|
|
; we traverse the chains in the correct order.
|
|
|
|
; CHECK: define void @test4_a() {
|
|
define void @test4_a() {
|
|
call void @test4_b11()
|
|
call void @test4_b21()
|
|
call void @test4_b31()
|
|
call void @test4_b41()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b11() #0 {
|
|
define void @test4_b11() {
|
|
call void @test4_b12()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b12() #0 {
|
|
define void @test4_b12() {
|
|
call void @test4_b13()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b13() #0 {
|
|
define void @test4_b13() {
|
|
call void @readnone()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b21() #0 {
|
|
define void @test4_b21() {
|
|
call void @test4_b22()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b22() #0 {
|
|
define void @test4_b22() {
|
|
call void @test4_b23()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b23() #0 {
|
|
define void @test4_b23() {
|
|
call void @reference_function_pointer(void()* @test4_a)
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test4_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b31() {
|
|
define void @test4_b31() {
|
|
call void @test4_b32()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b32() {
|
|
define void @test4_b32() {
|
|
call void @test4_b33()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b33() {
|
|
define void @test4_b33() {
|
|
call void @reference_function_pointer(void()* @test4_a)
|
|
call void @unknown()
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test4_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b41() #0 {
|
|
define void @test4_b41() {
|
|
call void @test4_b42()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b42() #0 {
|
|
define void @test4_b42() {
|
|
call void @test4_b43()
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @test4_b43() #0 {
|
|
define void @test4_b43() {
|
|
call void @reference_function_pointer(void()* @test4_a)
|
|
br i1 false, label %dead, label %exit
|
|
|
|
dead:
|
|
call void @test4_a()
|
|
br label %exit
|
|
|
|
exit:
|
|
ret void
|
|
}
|
|
|
|
; CHECK: attributes #0 = { nofree nosync readnone }
|