Files
swift-mirror/test/SILOptimizer/sil_combine_global_addr.sil
Arnold Schwaighofer 25a071efc8 Add experimental feature @inline(always)
The intent for `@inline(always)` is to act as an optimization control.
The user can rely on inlining to happen or the compiler will emit an error
message.

Because function values can be dynamic (closures, protocol/class lookup)
this guarantee can only be upheld for direct function references.

In cases where the optimizer can resolve dynamic function values the
attribute shall be respected.

rdar://148608854
2025-09-30 08:36:26 -07:00

141 lines
5.7 KiB
Plaintext

// RUN: %target-sil-opt -enforce-exclusivity=none -enable-sil-verify-all %s -sil-combine -devirtualizer | %FileCheck %s
// Test to see if concrete type can be propagated from
// global_addr in sil_combiner.
sil_stage canonical
import Builtin
import Swift
protocol SomeProtocol {
func foo() -> Int
}
class SomeClass : SomeProtocol {
init()
func foo() -> Int
}
sil hidden [thunk] [heuristic_always_inline] @foo_ : $@convention(witness_method:SomeProtocol) (@in_guaranteed SomeClass) -> Int {
bb0(%0 : $*SomeClass):
%1 = load %0 : $*SomeClass
%2 = class_method %1 : $SomeClass, #SomeClass.foo : (SomeClass) -> () -> Int, $@convention(method) (@guaranteed SomeClass) -> Int
%3 = apply %2(%1) : $@convention(method) (@guaranteed SomeClass) -> Int
return %3 : $Int
}
sil hidden_external @foo : $@convention(method) (@guaranteed SomeClass) -> Int
sil_global hidden [let] @$global_var : $SomeProtocol
// CHECK-LABEL: sil @witness_global_addr
// CHECK: bb0
// CHECK: alloc_global
// CHECK: global_addr
// CHECK: init_existential_addr
// CHECK: alloc_ref
// CHECK: store
// CHECK: function_ref
// CHECK: apply
// CHECK: return
// CHECK: } // end sil function 'witness_global_addr'
sil @witness_global_addr : $@convention(thin) () -> Int {
bb0:
alloc_global @$global_var
%3 = global_addr @$global_var : $*SomeProtocol
%4 = init_existential_addr %3 : $*SomeProtocol, $SomeClass
%5 = alloc_ref $SomeClass
store %5 to %4 : $*SomeClass
%8 = alloc_stack $SomeProtocol
copy_addr %3 to [init] %8 : $*SomeProtocol
%9 = open_existential_addr immutable_access %8 : $*SomeProtocol to $*@opened("1B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self
%10 = witness_method $@opened("1B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self, #SomeProtocol.foo : <Self where Self : SomeProtocol> (Self) -> () -> Int, %9 : $*@opened("1B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
%11 = apply %10<@opened("1B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self>(%9) : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
destroy_addr %8 : $*SomeProtocol
dealloc_stack %8 : $*SomeProtocol
return %11 : $Int
}
// CHECK-LABEL: sil @witness_global_addr_fail_1
// CHECK: bb0
// CHECK: alloc_global
// CHECK: global_addr
// CHECK: bb1
// CHECK: bb2
// CHECK: bb3
// CHECK: alloc_ref
// CHECK: store
// CHECK: alloc_stack
// CHECK: copy_addr
// CHECK: witness_method
// CHECK: apply
// CHECK: destroy_addr
// CHECK: dealloc_stack
// CHECK: return
// CHECK: } // end sil function 'witness_global_addr_fail_1'
sil @witness_global_addr_fail_1: $@convention(thin) (Builtin.Int1) -> Int {
bb0(%0 : $Builtin.Int1):
alloc_global @$global_var
%3 = global_addr @$global_var : $*SomeProtocol
cond_br %0, bb1, bb2
bb1:
br bb3
bb2:
br bb3
bb3:
%6 = init_existential_addr %3 : $*SomeProtocol, $SomeClass
%7 = alloc_ref $SomeClass
store %7 to %6 : $*SomeClass
%8 = alloc_stack $SomeProtocol
copy_addr %3 to [init] %8 : $*SomeProtocol
%9 = open_existential_addr immutable_access %8 : $*SomeProtocol to $*@opened("2B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self
%10 = witness_method $@opened("2B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self, #SomeProtocol.foo : <Self where Self : SomeProtocol> (Self) -> () -> Int, %9 : $*@opened("2B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
%11 = apply %10<@opened("2B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self>(%9) : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
destroy_addr %8 : $*SomeProtocol
dealloc_stack %8 : $*SomeProtocol
return %11 : $Int
}
// CHECK-LABEL: sil @witness_global_addr_fail_2
// CHECK: bb0
// CHECK: alloc_global
// CHECK: global_addr
// CHECK: init_existential_addr
// CHECK: alloc_ref
// CHECK: store
// CHECK: init_existential_addr
// CHECK: alloc_stack
// CHECK: copy_addr
// CHECK: open_existential_addr
// CHECK: witness_method
// CHECK: apply
// CHECK: destroy_addr
// CHECK: dealloc_stack
// CHECK: return
// CHECK: } // end sil function 'witness_global_addr_fail_2'
sil @witness_global_addr_fail_2 : $@convention(thin) () -> Int {
bb0:
alloc_global @$global_var
%3 = global_addr @$global_var : $*SomeProtocol
%4 = init_existential_addr %3 : $*SomeProtocol, $SomeClass
%5 = alloc_ref $SomeClass
store %5 to %4 : $*SomeClass
%6 = init_existential_addr %3 : $*SomeProtocol, $SomeClass
%8 = alloc_stack $SomeProtocol
copy_addr %3 to [init] %8 : $*SomeProtocol
%9 = open_existential_addr immutable_access %8 : $*SomeProtocol to $*@opened("3B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self
%10 = witness_method $@opened("3B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self, #SomeProtocol.foo : <Self where Self : SomeProtocol> (Self) -> () -> Int, %9 : $*@opened("3B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
%11 = apply %10<@opened("3B0A5B84-3441-11E8-AC03-DCA9048B1C6D", SomeProtocol) Self>(%9) : $@convention(witness_method: SomeProtocol) <τ_0_0 where τ_0_0 : SomeProtocol> (@in_guaranteed τ_0_0) -> Int
destroy_addr %8 : $*SomeProtocol
dealloc_stack %8 : $*SomeProtocol
return %11 : $Int
}
sil_vtable SomeClass {
#SomeClass.foo: (SomeClass) -> () -> Int : @foo
}
sil_witness_table hidden SomeClass: SomeProtocol module test {
method #SomeProtocol.foo: <Self where Self : SomeProtocol> (Self) -> () -> Int : @foo_
}