1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
|
; RUN: opt -basic-aa -objc-arc -S < %s | FileCheck %s
target datalayout = "e-p:64:64:64"
declare i8* @llvm.objc.retain(i8*)
declare i8* @llvm.objc.retainAutorelease(i8*)
declare void @llvm.objc.release(i8*)
declare i8* @llvm.objc.autorelease(i8*)
declare void @llvm.objc.clang.arc.use(...)
declare void @llvm.objc.clang.arc.noop.use(...)
declare void @test0_helper(i8*, i8**)
declare void @can_release(i8*)
; Ensure that we honor clang.arc.use as a use and don't miscompile
; the reduced test case from <rdar://13195034>.
;
; CHECK-LABEL: define void @test0(
; CHECK: @llvm.objc.retain(i8* %x)
; CHECK-NEXT: store i8* %y, i8** %temp0
; CHECK-NEXT: @llvm.objc.retain(i8* %y)
; CHECK-NEXT: call void @test0_helper
; CHECK-NEXT: [[VAL1:%.*]] = load i8*, i8** %temp0
; CHECK-NEXT: @llvm.objc.retain(i8* [[VAL1]])
; CHECK-NEXT: call void (...) @llvm.objc.clang.arc.use(i8* %y)
; CHECK-NEXT: @llvm.objc.release(i8* %y)
; CHECK-NEXT: store i8* [[VAL1]], i8** %temp1
; CHECK-NEXT: call void @test0_helper
; CHECK-NEXT: [[VAL2:%.*]] = load i8*, i8** %temp1
; CHECK-NEXT: @llvm.objc.retain(i8* [[VAL2]])
; CHECK-NEXT: call void (...) @llvm.objc.clang.arc.use(i8* [[VAL1]])
; CHECK-NEXT: @llvm.objc.release(i8* [[VAL1]])
; CHECK-NEXT: @llvm.objc.autorelease(i8* %x)
; CHECK-NEXT: store i8* %x, i8** %out
; CHECK-NEXT: @llvm.objc.retain(i8* %x)
; CHECK-NEXT: @llvm.objc.release(i8* [[VAL2]])
; CHECK-NEXT: @llvm.objc.release(i8* %x)
; CHECK-NEXT: ret void
; CHECK-NEXT: }
define void @test0(i8** %out, i8* %x, i8* %y) {
entry:
%temp0 = alloca i8*, align 8
%temp1 = alloca i8*, align 8
%0 = call i8* @llvm.objc.retain(i8* %x) nounwind
%1 = call i8* @llvm.objc.retain(i8* %y) nounwind
store i8* %y, i8** %temp0
call void @test0_helper(i8* %x, i8** %temp0)
%val1 = load i8*, i8** %temp0
%2 = call i8* @llvm.objc.retain(i8* %val1) nounwind
call void (...) @llvm.objc.clang.arc.use(i8* %y) nounwind
call void @llvm.objc.release(i8* %y) nounwind
store i8* %val1, i8** %temp1
call void @test0_helper(i8* %x, i8** %temp1)
%val2 = load i8*, i8** %temp1
%3 = call i8* @llvm.objc.retain(i8* %val2) nounwind
call void (...) @llvm.objc.clang.arc.use(i8* %val1) nounwind
call void @llvm.objc.release(i8* %val1) nounwind
%4 = call i8* @llvm.objc.retain(i8* %x) nounwind
%5 = call i8* @llvm.objc.autorelease(i8* %x) nounwind
store i8* %x, i8** %out
call void @llvm.objc.release(i8* %val2) nounwind
call void @llvm.objc.release(i8* %x) nounwind
ret void
}
; CHECK-LABEL: define void @test0a(
; CHECK: @llvm.objc.retain(i8* %x)
; CHECK-NEXT: store i8* %y, i8** %temp0
; CHECK-NEXT: @llvm.objc.retain(i8* %y)
; CHECK-NEXT: call void @test0_helper
; CHECK-NEXT: [[VAL1:%.*]] = load i8*, i8** %temp0
; CHECK-NEXT: @llvm.objc.retain(i8* [[VAL1]])
; CHECK-NEXT: call void (...) @llvm.objc.clang.arc.use(i8* %y)
; CHECK-NEXT: @llvm.objc.release(i8* %y)
; CHECK-NEXT: store i8* [[VAL1]], i8** %temp1
; CHECK-NEXT: call void @test0_helper
; CHECK-NEXT: [[VAL2:%.*]] = load i8*, i8** %temp1
; CHECK-NEXT: @llvm.objc.retain(i8* [[VAL2]])
; CHECK-NEXT: call void (...) @llvm.objc.clang.arc.use(i8* [[VAL1]])
; CHECK-NEXT: @llvm.objc.release(i8* [[VAL1]])
; CHECK-NEXT: @llvm.objc.autorelease(i8* %x)
; CHECK-NEXT: @llvm.objc.release(i8* [[VAL2]])
; CHECK-NEXT: store i8* %x, i8** %out
; CHECK-NEXT: ret void
; CHECK-NEXT: }
define void @test0a(i8** %out, i8* %x, i8* %y) {
entry:
%temp0 = alloca i8*, align 8
%temp1 = alloca i8*, align 8
%0 = call i8* @llvm.objc.retain(i8* %x) nounwind
%1 = call i8* @llvm.objc.retain(i8* %y) nounwind
store i8* %y, i8** %temp0
call void @test0_helper(i8* %x, i8** %temp0)
%val1 = load i8*, i8** %temp0
%2 = call i8* @llvm.objc.retain(i8* %val1) nounwind
call void (...) @llvm.objc.clang.arc.use(i8* %y) nounwind
call void @llvm.objc.release(i8* %y) nounwind, !clang.imprecise_release !0
store i8* %val1, i8** %temp1
call void @test0_helper(i8* %x, i8** %temp1)
%val2 = load i8*, i8** %temp1
%3 = call i8* @llvm.objc.retain(i8* %val2) nounwind
call void (...) @llvm.objc.clang.arc.use(i8* %val1) nounwind
call void @llvm.objc.release(i8* %val1) nounwind, !clang.imprecise_release !0
%4 = call i8* @llvm.objc.retain(i8* %x) nounwind
%5 = call i8* @llvm.objc.autorelease(i8* %x) nounwind
store i8* %x, i8** %out
call void @llvm.objc.release(i8* %val2) nounwind, !clang.imprecise_release !0
call void @llvm.objc.release(i8* %x) nounwind, !clang.imprecise_release !0
ret void
}
; ARC optimizer should be able to safely remove the retain/release pair as the
; call to @llvm.objc.clang.arc.noop.use is a no-op.
; CHECK-LABEL: define void @test_arc_noop_use(
; CHECK-NEXT: call void @can_release(i8* %x)
; CHECK-NEXT: call void (...) @llvm.objc.clang.arc.noop.use(
; CHECK-NEXT: ret void
define void @test_arc_noop_use(i8** %out, i8* %x) {
call i8* @llvm.objc.retain(i8* %x)
call void @can_release(i8* %x)
call void (...) @llvm.objc.clang.arc.noop.use(i8* %x)
call void @llvm.objc.release(i8* %x), !clang.imprecise_release !0
ret void
}
!0 = !{}
|