| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 2 |
| ; RUN: opt < %s -passes=ipsccp -S | FileCheck %s |
| ; RUN: opt < %s -enable-debugify -passes=ipsccp -debugify-quiet -disable-output |
| ; RUN: opt < %s -enable-debugify -passes=ipsccp -debugify-quiet -disable-output --try-experimental-debuginfo-iterators |
| |
| ;;======================== test1 |
| |
| define internal i32 @test1a(i32 %A) { |
| ; CHECK-LABEL: define internal i32 @test1a |
| ; CHECK-SAME: (i32 [[A:%.*]]) { |
| ; CHECK-NEXT: ret i32 poison |
| ; |
| %X = add i32 1, 2 |
| ret i32 %A |
| } |
| |
| define i32 @test1b() { |
| ; CHECK-LABEL: define i32 @test1b() { |
| ; CHECK-NEXT: [[X:%.*]] = call i32 @test1a(i32 17) |
| ; CHECK-NEXT: ret i32 17 |
| ; |
| %X = call i32 @test1a( i32 17 ) |
| ret i32 %X |
| |
| } |
| |
| |
| |
| ;;======================== test2 |
| |
| define internal i32 @test2a(i32 %A) { |
| ; CHECK-LABEL: define internal i32 @test2a |
| ; CHECK-SAME: (i32 [[A:%.*]]) { |
| ; CHECK-NEXT: br label [[T:%.*]] |
| ; CHECK: T: |
| ; CHECK-NEXT: [[B:%.*]] = call i32 @test2a(i32 0) |
| ; CHECK-NEXT: ret i32 poison |
| ; |
| %C = icmp eq i32 %A, 0 |
| br i1 %C, label %T, label %F |
| T: |
| %B = call i32 @test2a( i32 0 ) |
| ret i32 0 |
| F: |
| %C.upgrd.1 = call i32 @test2a(i32 1) |
| ret i32 %C.upgrd.1 |
| } |
| |
| define i32 @test2b() { |
| ; CHECK-LABEL: define i32 @test2b() { |
| ; CHECK-NEXT: [[X:%.*]] = call i32 @test2a(i32 0) |
| ; CHECK-NEXT: ret i32 0 |
| ; |
| %X = call i32 @test2a(i32 0) |
| ret i32 %X |
| } |
| |
| ;;======================== test3 |
| |
| @G = internal global i32 undef |
| |
| define void @test3a() { |
| ; CHECK-LABEL: define void @test3a() { |
| ; CHECK-NEXT: [[X:%.*]] = load i32, ptr @G, align 4 |
| ; CHECK-NEXT: store i32 [[X]], ptr @G, align 4 |
| ; CHECK-NEXT: ret void |
| ; |
| %X = load i32, ptr @G |
| store i32 %X, ptr @G |
| ret void |
| } |
| |
| define i32 @test3b() { |
| ; CHECK-LABEL: define range(i32 0, 18) i32 @test3b() { |
| ; CHECK-NEXT: [[V:%.*]] = load i32, ptr @G, align 4 |
| ; CHECK-NEXT: [[C:%.*]] = icmp eq i32 [[V]], 17 |
| ; CHECK-NEXT: br i1 [[C]], label [[T:%.*]], label [[F:%.*]] |
| ; CHECK: T: |
| ; CHECK-NEXT: store i32 17, ptr @G, align 4 |
| ; CHECK-NEXT: ret i32 17 |
| ; CHECK: F: |
| ; CHECK-NEXT: store i32 123, ptr @G, align 4 |
| ; CHECK-NEXT: ret i32 0 |
| ; |
| %V = load i32, ptr @G |
| %C = icmp eq i32 %V, 17 |
| br i1 %C, label %T, label %F |
| T: |
| store i32 17, ptr @G |
| ret i32 %V |
| F: |
| store i32 123, ptr @G |
| ret i32 0 |
| } |
| |
| ;;======================== test4 |
| |
| define internal {i64,i64} @test4a() { |
| ; CHECK-LABEL: define internal { i64, i64 } @test4a() { |
| ; CHECK-NEXT: ret { i64, i64 } poison |
| ; |
| %a = insertvalue {i64,i64} undef, i64 4, 1 |
| %b = insertvalue {i64,i64} %a, i64 5, 0 |
| ret {i64,i64} %b |
| } |
| |
| define i64 @test4b() personality ptr @__gxx_personality_v0 { |
| ; CHECK-LABEL: define range(i64 0, 6) i64 @test4b() personality ptr @__gxx_personality_v0 { |
| ; CHECK-NEXT: [[A:%.*]] = invoke { i64, i64 } @test4a() |
| ; CHECK-NEXT: to label [[A:%.*]] unwind label [[B:%.*]] |
| ; CHECK: A: |
| ; CHECK-NEXT: [[C:%.*]] = call i64 @test4c(i64 5) |
| ; CHECK-NEXT: ret i64 5 |
| ; CHECK: B: |
| ; CHECK-NEXT: [[VAL:%.*]] = landingpad { ptr, i32 } |
| ; CHECK-NEXT: catch ptr null |
| ; CHECK-NEXT: ret i64 0 |
| ; |
| %a = invoke {i64,i64} @test4a() |
| to label %A unwind label %B |
| A: |
| %b = extractvalue {i64,i64} %a, 0 |
| %c = call i64 @test4c(i64 %b) |
| ret i64 %c |
| B: |
| %val = landingpad { ptr, i32 } |
| catch ptr null |
| ret i64 0 |
| } |
| |
| define internal i64 @test4c(i64 %a) { |
| ; CHECK-LABEL: define internal i64 @test4c |
| ; CHECK-SAME: (i64 [[A:%.*]]) { |
| ; CHECK-NEXT: ret i64 poison |
| ; |
| ret i64 %a |
| } |
| |
| ;;======================== test5 |
| |
| ; PR4313 |
| define internal {i64,i64} @test5a() { |
| ; CHECK-LABEL: define internal { i64, i64 } @test5a() { |
| ; CHECK-NEXT: ret { i64, i64 } poison |
| ; |
| %a = insertvalue {i64,i64} undef, i64 4, 1 |
| %b = insertvalue {i64,i64} %a, i64 5, 0 |
| ret {i64,i64} %b |
| } |
| |
| define i64 @test5b() personality ptr @__gxx_personality_v0 { |
| ; CHECK-LABEL: define range(i64 0, 6) i64 @test5b() personality ptr @__gxx_personality_v0 { |
| ; CHECK-NEXT: [[A:%.*]] = invoke { i64, i64 } @test5a() |
| ; CHECK-NEXT: to label [[A:%.*]] unwind label [[B:%.*]] |
| ; CHECK: A: |
| ; CHECK-NEXT: [[C:%.*]] = call i64 @test5c({ i64, i64 } { i64 5, i64 4 }) |
| ; CHECK-NEXT: ret i64 5 |
| ; CHECK: B: |
| ; CHECK-NEXT: [[VAL:%.*]] = landingpad { ptr, i32 } |
| ; CHECK-NEXT: catch ptr null |
| ; CHECK-NEXT: ret i64 0 |
| ; |
| %a = invoke {i64,i64} @test5a() |
| to label %A unwind label %B |
| A: |
| %c = call i64 @test5c({i64,i64} %a) |
| ret i64 %c |
| B: |
| %val = landingpad { ptr, i32 } |
| catch ptr null |
| ret i64 0 |
| } |
| |
| define internal i64 @test5c({i64,i64} %a) { |
| ; CHECK-LABEL: define internal i64 @test5c |
| ; CHECK-SAME: ({ i64, i64 } [[A:%.*]]) { |
| ; CHECK-NEXT: ret i64 poison |
| ; |
| %b = extractvalue {i64,i64} %a, 0 |
| ret i64 %b |
| } |
| |
| |
| ;;======================== test6 |
| |
| define i64 @test6a() { |
| ; CHECK-LABEL: define i64 @test6a() { |
| ; CHECK-NEXT: ret i64 0 |
| ; |
| ret i64 0 |
| } |
| |
| define i64 @test6b() { |
| ; CHECK-LABEL: define i64 @test6b() { |
| ; CHECK-NEXT: [[A:%.*]] = call i64 @test6a() |
| ; CHECK-NEXT: ret i64 0 |
| ; |
| %a = call i64 @test6a() |
| ret i64 %a |
| } |
| |
| ;;======================== test7 |
| |
| %T = type {i32,i32} |
| |
| define internal %T @test7a(i32 %A) { |
| ; CHECK-LABEL: define internal %T @test7a |
| ; CHECK-SAME: (i32 [[A:%.*]]) { |
| ; CHECK-NEXT: ret [[T:%.*]] poison |
| ; |
| %X = add i32 1, %A |
| %mrv0 = insertvalue %T undef, i32 %X, 0 |
| %mrv1 = insertvalue %T %mrv0, i32 %A, 1 |
| ret %T %mrv1 |
| } |
| |
| define i32 @test7b() { |
| ; CHECK-LABEL: define i32 @test7b() { |
| ; CHECK-NEXT: [[X:%.*]] = call [[T:%.*]] @[[TEST7A:[a-zA-Z0-9_$\"\\.-]*[a-zA-Z_$\"\\.-][a-zA-Z0-9_$\"\\.-]*]](i32 17) |
| ; CHECK-NEXT: ret i32 36 |
| ; |
| %X = call %T @test7a(i32 17) |
| %Y = extractvalue %T %X, 0 |
| %Z = add i32 %Y, %Y |
| ret i32 %Z |
| } |
| |
| ;;======================== test8 |
| |
| |
| define internal {} @test8a(i32 %A, ptr %P) { |
| ; CHECK-LABEL: define internal {} @test8a |
| ; CHECK-SAME: (i32 [[A:%.*]], ptr [[P:%.*]]) { |
| ; CHECK-NEXT: store i32 5, ptr [[P]], align 4 |
| ; CHECK-NEXT: ret {} poison |
| ; |
| store i32 %A, ptr %P |
| ret {} {} |
| } |
| |
| define void @test8b(ptr %P) { |
| ; CHECK-LABEL: define void @test8b |
| ; CHECK-SAME: (ptr [[P:%.*]]) { |
| ; CHECK-NEXT: [[X:%.*]] = call {} @test8a(i32 5, ptr [[P]]) |
| ; CHECK-NEXT: ret void |
| ; |
| %X = call {} @test8a(i32 5, ptr %P) |
| ret void |
| } |
| |
| ;;======================== test9 |
| |
| @test9g = internal global { } zeroinitializer |
| |
| define void @test9() { |
| ; CHECK-LABEL: define void @test9() { |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[LOCAL_FOO:%.*]] = alloca {}, align 8 |
| ; CHECK-NEXT: store {} zeroinitializer, ptr [[LOCAL_FOO]], align 1 |
| ; CHECK-NEXT: ret void |
| ; |
| entry: |
| %local_foo = alloca { } |
| load { }, ptr @test9g |
| store { } %0, ptr %local_foo |
| ret void |
| } |
| |
| declare i32 @__gxx_personality_v0(...) |
| |
| ;;======================== test10 |
| |
| define i32 @test10a() nounwind { |
| ; CHECK-LABEL: define i32 @test10a |
| ; CHECK-SAME: () #[[ATTR0:[0-9]+]] { |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[CALL:%.*]] = call i32 @test10b(i32 undef) |
| ; CHECK-NEXT: ret i32 [[CALL]] |
| ; |
| entry: |
| %call = call i32 @test10b(i32 undef) |
| ret i32 %call |
| |
| } |
| |
| define internal i32 @test10b(i32 %x) nounwind { |
| ; CHECK-LABEL: define internal i32 @test10b |
| ; CHECK-SAME: (i32 [[X:%.*]]) #[[ATTR0]] { |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[R:%.*]] = and i32 undef, 1 |
| ; CHECK-NEXT: ret i32 [[R]] |
| ; |
| entry: |
| %r = and i32 %x, 1 |
| ret i32 %r |
| } |
| |
| ;;======================== test11 |
| |
| define i64 @test11a() { |
| ; CHECK-LABEL: define i64 @test11a() { |
| ; CHECK-NEXT: [[XOR:%.*]] = xor i64 undef, undef |
| ; CHECK-NEXT: ret i64 [[XOR]] |
| ; |
| %xor = xor i64 undef, undef |
| ret i64 %xor |
| } |
| |
| define i64 @test11b() { |
| ; CHECK-LABEL: define i64 @test11b() { |
| ; CHECK-NEXT: [[CALL1:%.*]] = call i64 @test11a() |
| ; CHECK-NEXT: [[CALL2:%.*]] = call i64 @llvm.ctpop.i64(i64 [[CALL1]]) |
| ; CHECK-NEXT: ret i64 [[CALL2]] |
| ; |
| %call1 = call i64 @test11a() |
| %call2 = call i64 @llvm.ctpop.i64(i64 %call1) |
| ret i64 %call2 |
| } |
| |
| declare i64 @llvm.ctpop.i64(i64) |
| |
| ;;======================== test12 |
| ;; Ensure that a struct as an arg to a potentially constant-foldable |
| ;; function does not crash SCCP (for now it'll just ignores it) |
| |
| define i1 @test12() { |
| ; CHECK-LABEL: define i1 @test12() { |
| ; CHECK-NEXT: [[C:%.*]] = call i1 @llvm.is.constant.sl_i32i32s({ i32, i32 } { i32 -1, i32 32 }) |
| ; CHECK-NEXT: ret i1 [[C]] |
| ; |
| %c = call i1 @llvm.is.constant.sl_i32i32s({i32, i32} {i32 -1, i32 32}) |
| ret i1 %c |
| } |
| |
| declare i1 @llvm.is.constant.sl_i32i32s({i32, i32} %a) |