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 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185
|
// RUN: %target-sil-opt -enable-sil-verify-all %s -compute-side-effects -cse | %FileCheck %s
// REQUIRES: swift_in_compiler
sil_stage canonical
import Builtin
import Swift
// "return (x + y) + (x + y)"
// make sure we are only calculating x+y once.
//CHECK-LABEL: simply_arith
//CHECK: builtin
//CHECK: cond_fail
//CHECK: builtin
//CHECK: cond_fail
//CHECK-NOT: builtin
//CHECK-NOT: cond_fail
//CHECK: return
sil @simply_arith : $@convention(thin) (Int32, Int32) -> Int32 {
bb0(%0 : $Int32, %1 : $Int32):
debug_value %0 : $Int32, let, name "x" // id: %2
debug_value %1 : $Int32, let, name "y" // id: %3
%5 = struct_extract %0 : $Int32, #Int32._value // user: %8
%6 = struct_extract %1 : $Int32, #Int32._value // user: %8
%7 = integer_literal $Builtin.Int1, -1 // user: %8
%8 = builtin "sadd_with_overflow_Int32"(%5 : $Builtin.Int32, %6 : $Builtin.Int32, %7 : $Builtin.Int1) : $(Builtin.Int32, Builtin.Int1) // users: %9, %10
%9 = tuple_extract %8 : $(Builtin.Int32, Builtin.Int1), 0 // user: %12
%10 = tuple_extract %8 : $(Builtin.Int32, Builtin.Int1), 1 // user: %11
cond_fail %10 : $Builtin.Int1 // id: %11
%12 = struct $Int32 (%9 : $Builtin.Int32) // user: %23
%14 = struct_extract %0 : $Int32, #Int32._value // user: %17
%15 = struct_extract %1 : $Int32, #Int32._value // user: %17
%16 = integer_literal $Builtin.Int1, -1 // user: %17
%17 = builtin "sadd_with_overflow_Int32"(%14 : $Builtin.Int32, %15 : $Builtin.Int32, %16 : $Builtin.Int1) : $(Builtin.Int32, Builtin.Int1) // users: %18, %19
%18 = tuple_extract %17 : $(Builtin.Int32, Builtin.Int1), 0 // user: %21
%19 = tuple_extract %17 : $(Builtin.Int32, Builtin.Int1), 1 // user: %20
cond_fail %19 : $Builtin.Int1 // id: %20
%21 = struct $Int32 (%18 : $Builtin.Int32) // user: %24
%23 = struct_extract %12 : $Int32, #Int32._value // user: %26
%24 = struct_extract %21 : $Int32, #Int32._value // user: %26
%25 = integer_literal $Builtin.Int1, -1 // user: %26
%26 = builtin "sadd_with_overflow_Int32"(%23 : $Builtin.Int32, %24 : $Builtin.Int32, %25 : $Builtin.Int1) : $(Builtin.Int32, Builtin.Int1) // users: %27, %28
%27 = tuple_extract %26 : $(Builtin.Int32, Builtin.Int1), 0 // user: %30
%28 = tuple_extract %26 : $(Builtin.Int32, Builtin.Int1), 1 // user: %29
cond_fail %28 : $Builtin.Int1 // id: %29
%30 = struct $Int32 (%27 : $Builtin.Int32) // user: %31
return %30 : $Int32 // id: %31
}
sil @readnone : $@convention(thin) (Int64, Int64) -> Int64 {
bb0(%0 : $Int64, %1 : $Int64):
return %0 : $Int64
}
sil_global @gg : $Int64
sil @readsome : $@convention(thin) (Int64, Int64) -> Int64 {
bb0(%0 : $Int64, %1 : $Int64):
%g = global_addr @gg : $*Int64
%l = load %g : $*Int64
return %l : $Int64
}
struct Storage {
var a: Int64
var b: Int64
var c: Int64
}
class XX {
}
sil_global @ggs : $Storage
sil @retain_only : $@convention(thin) () -> @owned XX {
bb0:
%g = global_addr @ggs : $*Storage
%a = address_to_pointer %g : $*Storage to $Builtin.RawPointer
%r = raw_pointer_to_ref %a : $Builtin.RawPointer to $XX
strong_retain %r : $XX
return %r : $XX
}
//CHECK-LABEL: sil @cse_readnone_apply
//CHECK: [[A:%[0-9]+]] = apply
//CHECK: [[S:%[0-9]+]] = struct_extract [[A]]
//CHECK: builtin "sadd_with_overflow_Int64"([[S]] : $Builtin.Int64, [[S]] : $Builtin.Int64
//CHECK: return
sil @cse_readnone_apply : $@convention(thin) (Int64) -> Int64 {
bb0(%0 : $Int64):
%2 = function_ref @readnone : $@convention(thin) (Int64, Int64) -> Int64
%3 = integer_literal $Builtin.Int64, 3
%4 = struct $Int64 (%3 : $Builtin.Int64)
%5 = apply %2(%0, %4) : $@convention(thin) (Int64, Int64) -> Int64
%6 = apply %2(%0, %4) : $@convention(thin) (Int64, Int64) -> Int64
%7 = struct_extract %5 : $Int64, #Int64._value
%8 = struct_extract %6 : $Int64, #Int64._value
%9 = integer_literal $Builtin.Int1, 0
%10 = builtin "sadd_with_overflow_Int64"(%7 : $Builtin.Int64, %8 : $Builtin.Int64, %9 : $Builtin.Int1) : $(Builtin.Int64, Builtin.Int1)
%11 = tuple_extract %10 : $(Builtin.Int64, Builtin.Int1), 0
%14 = struct $Int64 (%11 : $Builtin.Int64)
return %14 : $Int64
}
//CHECK-LABEL: sil @dont_cse_readsome_apply
//CHECK: %{{[0-9]+}} = apply
//CHECK: %{{[0-9]+}} = apply
//CHECK: return
sil @dont_cse_readsome_apply : $@convention(thin) (Int64) -> Int64 {
bb0(%0 : $Int64):
%2 = function_ref @readsome : $@convention(thin) (Int64, Int64) -> Int64
%3 = integer_literal $Builtin.Int64, 3
%4 = struct $Int64 (%3 : $Builtin.Int64)
%5 = apply %2(%0, %4) : $@convention(thin) (Int64, Int64) -> Int64
%6 = apply %2(%0, %4) : $@convention(thin) (Int64, Int64) -> Int64
%7 = struct_extract %5 : $Int64, #Int64._value
%8 = struct_extract %6 : $Int64, #Int64._value
%9 = integer_literal $Builtin.Int1, 0
%10 = builtin "sadd_with_overflow_Int64"(%7 : $Builtin.Int64, %8 : $Builtin.Int64, %9 : $Builtin.Int1) : $(Builtin.Int64, Builtin.Int1)
%11 = tuple_extract %10 : $(Builtin.Int64, Builtin.Int1), 0
%14 = struct $Int64 (%11 : $Builtin.Int64)
return %14 : $Int64
}
//CHECK-LABEL: sil @dont_cse_retain_only_apply :
//CHECK: %{{[0-9]+}} = apply
//CHECK: %{{[0-9]+}} = apply
//CHECK: return
sil @dont_cse_retain_only_apply : $@convention(thin) () -> () {
bb0:
%f = function_ref @retain_only : $@convention(thin) () -> @owned XX
%a1 = apply %f() : $@convention(thin) () -> @owned XX
%a2 = apply %f() : $@convention(thin) () -> @owned XX
strong_release %a1 : $XX
strong_release %a2 : $XX
%r = tuple()
return %r : $()
}
sil @get_tuple : $@convention(thin) () -> (@owned XX, Int) {
[global: ]
}
//CHECK-LABEL: sil @dont_cse_retain_only_apply2 :
//CHECK: %{{[0-9]+}} = apply %0
//CHECK: %{{[0-9]+}} = apply %0
//CHECK: } // end sil function 'dont_cse_retain_only_apply2'
sil @dont_cse_retain_only_apply2 : $@convention(thin) () -> () {
bb0:
%0 = function_ref @get_tuple : $@convention(thin) () -> (@owned XX, Int)
%1 = apply %0() : $@convention(thin) () -> (@owned XX, Int)
%2 = tuple_extract %1 : $(XX, Int), 0
release_value %2 : $XX
%4 = apply %0() : $@convention(thin) () -> (@owned XX, Int)
%5 = tuple_extract %4 : $(XX, Int), 0
release_value %5 : $XX
%6 = tuple ()
return %6 : $()
}
sil @get_trivial_tuple : $@convention(thin) () -> (Int, Int) {
[global: ]
}
//CHECK-LABEL: sil @cse_apply_with_trivial_result :
//CHECK: [[R:%.*]] = apply %0
//CHECK-NEXT: [[I:%.*]] = tuple_extract [[R]] : $(Int, Int), 0
//CHECK-NEXT: [[T:%.*]] = tuple ([[I]] : $Int, [[I]] : $Int)
//CHECK-NEXT: return [[T]]
//CHECK: } // end sil function 'cse_apply_with_trivial_result'
sil @cse_apply_with_trivial_result : $@convention(thin) () -> (Int, Int) {
bb0:
%0 = function_ref @get_trivial_tuple : $@convention(thin) () -> (Int, Int)
%1 = apply %0() : $@convention(thin) () -> (Int, Int)
%2 = tuple_extract %1 : $(Int, Int), 0
%4 = apply %0() : $@convention(thin) () -> (Int, Int)
%5 = tuple_extract %4 : $(Int, Int), 0
%6 = tuple (%2 : $Int, %5 : $Int)
return %6 : $(Int, Int)
}
|