// RUN: %target-swift-frontend -emit-silgen %s | %FileCheck %s class B { } class D : B { } // CHECK-LABEL: sil hidden @_T05casts6upcast{{[_0-9a-zA-Z]*}}F func upcast(d: D) -> B { // CHECK: {{%.*}} = upcast return d } // CHECK-LABEL: sil hidden @_T05casts8downcast{{[_0-9a-zA-Z]*}}F func downcast(b: B) -> D { // CHECK: {{%.*}} = unconditional_checked_cast return b as! D } // CHECK-LABEL: sil hidden @_T05casts3isa{{[_0-9a-zA-Z]*}}F func isa(b: B) -> Bool { // CHECK: bb0([[ARG:%.*]] : $B): // CHECK: [[BORROWED_ARG:%.*]] = begin_borrow [[ARG]] // CHECK: [[COPIED_BORROWED_ARG:%.*]] = copy_value [[BORROWED_ARG]] // CHECK: checked_cast_br [[COPIED_BORROWED_ARG]] : $B to $D, [[YES:bb[0-9]+]], [[NO:bb[0-9]+]] // // CHECK: [[YES]]([[CASTED_VALUE:%.*]] : $D): // CHECK: integer_literal {{.*}} -1 // CHECK: destroy_value [[CASTED_VALUE]] // CHECK: end_borrow [[BORROWED_ARG]] from [[ARG]] // // CHECK: [[NO]]([[ORIGINAL_VALUE:%.*]] : $B): // CHECK: destroy_value [[ORIGINAL_VALUE]] // CHECK: integer_literal {{.*}} 0 // CHECK: end_borrow [[BORROWED_ARG]] from [[ARG]] return b is D } // CHECK-LABEL: sil hidden @_T05casts16upcast_archetype{{[_0-9a-zA-Z]*}}F func upcast_archetype(t: T) -> B { // CHECK: {{%.*}} = upcast return t } // CHECK-LABEL: sil hidden @_T05casts25upcast_archetype_metatype{{[_0-9a-zA-Z]*}}F func upcast_archetype_metatype(t: T.Type) -> B.Type { // CHECK: {{%.*}} = upcast return t } // CHECK-LABEL: sil hidden @_T05casts18downcast_archetype{{[_0-9a-zA-Z]*}}F func downcast_archetype(b: B) -> T { // CHECK: {{%.*}} = unconditional_checked_cast return b as! T } // This is making sure that we do not have the default propagating behavior in // the address case. // // CHECK-LABEL: sil hidden @_T05casts12is_archetype{{[_0-9a-zA-Z]*}}F func is_archetype(b: B, _: T) -> Bool { // CHECK: bb0([[ARG1:%.*]] : $B, [[ARG2:%.*]] : $T): // CHECK: checked_cast_br {{%.*}}, [[YES:bb[0-9]+]], [[NO:bb[0-9]+]] // CHECK: [[YES]]([[CASTED_ARG:%.*]] : $T): // CHECK: integer_literal {{.*}} -1 // CHECK: destroy_value [[CASTED_ARG]] // CHECK: [[NO]]([[ORIGINAL_VALUE:%.*]] : $B): // CHCEK: destroy_value [[CASTED_ARG]] // CHECK: integer_literal {{.*}} 0 return b is T } // CHECK: } // end sil function '_T05casts12is_archetype{{[_0-9a-zA-Z]*}}F' // CHECK: sil hidden @_T05casts20downcast_conditional{{[_0-9a-zA-Z]*}}F // CHECK: checked_cast_br {{%.*}} : $B to $D // CHECK: bb{{[0-9]+}}({{.*}} : $Optional) func downcast_conditional(b: B) -> D? { return b as? D } protocol P {} struct S : P {} // CHECK: sil hidden @_T05casts32downcast_existential_conditional{{[_0-9a-zA-Z]*}}F // CHECK: bb0([[IN:%.*]] : $*P): // CHECK: [[COPY:%.*]] = alloc_stack $P // CHECK: copy_addr [[IN]] to [initialization] [[COPY]] // CHECK: [[TMP:%.*]] = alloc_stack $S // CHECK: checked_cast_addr_br take_always P in [[COPY]] : $*P to S in [[TMP]] : $*S, bb1, bb2 // Success block. // CHECK: bb1: // CHECK: [[T0:%.*]] = load [trivial] [[TMP]] : $*S // CHECK: [[T1:%.*]] = enum $Optional, #Optional.some!enumelt.1, [[T0]] : $S // CHECK: dealloc_stack [[TMP]] // CHECK: br bb3([[T1]] : $Optional) // Failure block. // CHECK: bb2: // CHECK: [[T0:%.*]] = enum $Optional, #Optional.none!enumelt // CHECK: dealloc_stack [[TMP]] // CHECK: br bb3([[T0]] : $Optional) // Continuation block. // CHECK: bb3([[RESULT:%.*]] : $Optional): // CHECK: dealloc_stack [[COPY]] // CHECK: destroy_addr [[IN]] : $*P // CHECK: return [[RESULT]] func downcast_existential_conditional(p: P) -> S? { return p as? S }