// RUN: %target-swift-frontend -emit-silgen -sdk %S/Inputs -I %S/Inputs -enable-source-import %s | %FileCheck %s // REQUIRES: objc_interop import Foundation // FIXME: Should go into the standard library. public extension _ObjectiveCBridgeable { static func _unconditionallyBridgeFromObjectiveC(_ source: _ObjectiveCType?) -> Self { var result: Self? _forceBridgeFromObjectiveC(source!, result: &result) return result! } } class BridgedObjC : NSObject { } func == (x: BridgedObjC, y: BridgedObjC) -> Bool { return true } struct BridgedSwift : Hashable, _ObjectiveCBridgeable { var hashValue: Int { return 0 } func _bridgeToObjectiveC() -> BridgedObjC { return BridgedObjC() } static func _forceBridgeFromObjectiveC( _ x: BridgedObjC, result: inout BridgedSwift? ) { } static func _conditionallyBridgeFromObjectiveC( _ x: BridgedObjC, result: inout BridgedSwift? ) -> Bool { return true } } func == (x: BridgedSwift, y: BridgedSwift) -> Bool { return true } // CHECK-LABEL: sil hidden @_T019collection_downcast17testArrayDowncast{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array): func testArrayDowncast(_ array: [AnyObject]) -> [BridgedObjC] { // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s15_arrayForceCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Array<τ_0_1> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array as! [BridgedObjC] } // CHECK-LABEL: sil hidden @_T019collection_downcast27testArrayDowncastFromObject{{.*}}F // CHECK: bb0([[OBJ:%[0-9]+]] : $AnyObject): func testArrayDowncastFromObject(_ obj: AnyObject) -> [BridgedObjC] { // CHECK: unconditional_checked_cast_addr take_always AnyObject in [[OBJECT_ALLOC:%[0-9]+]] : $*AnyObject to Array in [[VALUE_ALLOC:%[0-9]+]] : $*Array return obj as! [BridgedObjC] } // CHECK-LABEL: sil hidden @_T019collection_downcast28testArrayDowncastFromNSArray{{.*}}F // CHECK: bb0([[NSARRAY_OBJ:%[0-9]+]] : $NSArray): func testArrayDowncastFromNSArray(_ obj: NSArray) -> [BridgedObjC] { // CHECK: unconditional_checked_cast_addr take_always NSArray in [[OBJECT_ALLOC:%[0-9]+]] : $*NSArray to Array in [[VALUE_ALLOC:%[0-9]+]] : $*Array return obj as! [BridgedObjC] } // CHECK-LABEL: sil hidden @_T019collection_downcast28testArrayDowncastConditional{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array): func testArrayDowncastConditional(_ array: [AnyObject]) -> [BridgedObjC]? { // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s21_arrayConditionalCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array as? [BridgedObjC] } // CHECK: } // end sil function '_T019collection_downcast28testArrayDowncastConditional{{.*}}F' // CHECK-LABEL: sil hidden @_T019collection_downcast12testArrayIsa{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array) func testArrayIsa(_ array: [AnyObject]) -> Bool { // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s21_arrayConditionalCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array is [BridgedObjC] ? true : false } // CHECK-LABEL: sil hidden @_T019collection_downcast24testArrayDowncastBridged{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array): func testArrayDowncastBridged(_ array: [AnyObject]) -> [BridgedSwift] { // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s15_arrayForceCast{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Array<τ_0_1> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array as! [BridgedSwift] } // CHECK-LABEL: sil hidden @_T019collection_downcast35testArrayDowncastBridgedConditional{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array): func testArrayDowncastBridgedConditional(_ array: [AnyObject]) -> [BridgedSwift]?{ // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s21_arrayConditionalCast{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array as? [BridgedSwift] } // CHECK-LABEL: sil hidden @_T019collection_downcast19testArrayIsaBridged{{.*}}F // CHECK: bb0([[ARRAY:%[0-9]+]] : $Array) func testArrayIsaBridged(_ array: [AnyObject]) -> Bool { // CHECK: [[BORROWED_ARRAY:%.*]] = begin_borrow [[ARRAY]] // CHECK: [[ARRAY_COPY:%.*]] = copy_value [[BORROWED_ARRAY]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s21_arrayConditionalCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[ARRAY_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1> (@owned Array<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_ARRAY]] from [[ARRAY]] // CHECK: destroy_value [[ARRAY]] return array is [BridgedSwift] ? true : false } // CHECK-LABEL: sil hidden @_T019collection_downcast32testDictionaryDowncastFromObject{{.*}}F // CHECK: bb0([[OBJ:%[0-9]+]] : $AnyObject): func testDictionaryDowncastFromObject(_ obj: AnyObject) -> Dictionary { // CHECK: unconditional_checked_cast_addr take_always AnyObject in [[OBJECT_ALLOC:%[0-9]+]] : $*AnyObject to Dictionary in [[VALUE_ALLOC:%[0-9]+]] : $*Dictionary return obj as! Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast22testDictionaryDowncast{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncast(_ dict: Dictionary) -> Dictionary { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s19_dictionaryDownCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Dictionary<τ_0_2, τ_0_3> // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as! Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast33testDictionaryDowncastConditional{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncastConditional(_ dict: Dictionary) -> Dictionary? { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s30_dictionaryDownCastConditional{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as? Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast41testDictionaryDowncastBridgedVConditional{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncastBridgedVConditional(_ dict: Dictionary) -> Dictionary? { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s30_dictionaryDownCastConditional{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Optional>{{.*}} // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as? Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast41testDictionaryDowncastBridgedKConditional{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncastBridgedKConditional(_ dict: Dictionary) -> Dictionary? { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s30_dictionaryDownCastConditional{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as? Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast31testDictionaryDowncastBridgedKV{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncastBridgedKV(_ dict: Dictionary) -> Dictionary { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s19_dictionaryDownCast{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Dictionary<τ_0_2, τ_0_3> // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as! Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast42testDictionaryDowncastBridgedKVConditional{{.*}}F // CHECK: bb0([[DICT:%[0-9]+]] : $Dictionary) func testDictionaryDowncastBridgedKVConditional(_ dict: Dictionary) -> Dictionary? { // CHECK: [[BORROWED_DICT:%.*]] = begin_borrow [[DICT]] // CHECK: [[DICT_COPY:%.*]] = copy_value [[BORROWED_DICT]] // CHECK: [[BRIDGE_FN:%[0-9]+]] = function_ref @_T0s30_dictionaryDownCastConditional{{.*}}F // CHECK: apply [[BRIDGE_FN]]([[DICT_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1, τ_0_2, τ_0_3 where τ_0_0 : Hashable, τ_0_2 : Hashable> (@owned Dictionary<τ_0_0, τ_0_1>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_DICT]] from [[DICT]] // CHECK: destroy_value [[DICT]] return dict as? Dictionary } // CHECK-LABEL: sil hidden @_T019collection_downcast25testSetDowncastFromObject{{.*}}F // CHECK: bb0([[OBJ:%[0-9]+]] : $AnyObject): func testSetDowncastFromObject(_ obj: AnyObject) -> Set { // CHECK: unconditional_checked_cast_addr take_always AnyObject in [[OBJECT_ALLOC:%[0-9]+]] : $*AnyObject to Set in [[VALUE_ALLOC:%[0-9]+]] : $*Set return obj as! Set } // CHECK-LABEL: sil hidden @_T019collection_downcast15testSetDowncast{{.*}}F // CHECK: bb0([[SET:%[0-9]+]] : $Set) func testSetDowncast(_ dict: Set) -> Set { // CHECK: [[BORROWED_SET:%.*]] = begin_borrow [[SET]] // CHECK: [[SET_COPY:%.*]] = copy_value [[BORROWED_SET]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s12_setDownCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[SET_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : Hashable, τ_0_1 : Hashable> (@owned Set<τ_0_0>) -> @owned Set<τ_0_1> // CHECK: end_borrow [[BORROWED_SET]] from [[SET]] // CHECK: destroy_value [[SET]] return dict as! Set } // CHECK-LABEL: sil hidden @_T019collection_downcast26testSetDowncastConditional{{.*}}F // CHECK: bb0([[SET:%[0-9]+]] : $Set) func testSetDowncastConditional(_ dict: Set) -> Set? { // CHECK: [[BORROWED_SET:%.*]] = begin_borrow [[SET]] // CHECK: [[SET_COPY:%.*]] = copy_value [[BORROWED_SET]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s23_setDownCastConditional{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[SET_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : Hashable, τ_0_1 : Hashable> (@owned Set<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_SET]] from [[SET]] // CHECK: destroy_value [[SET]] return dict as? Set } // CHECK-LABEL: sil hidden @_T019collection_downcast22testSetDowncastBridged{{.*}}F // CHECK: bb0([[SET:%[0-9]+]] : $Set) func testSetDowncastBridged(_ dict: Set) -> Set { // CHECK: [[BORROWED_SET:%.*]] = begin_borrow [[SET]] // CHECK: [[SET_COPY:%.*]] = copy_value [[BORROWED_SET]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s12_setDownCast{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[SET_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : Hashable, τ_0_1 : Hashable> (@owned Set<τ_0_0>) -> @owned Set<τ_0_1> // CHECK: end_borrow [[BORROWED_SET]] from [[SET]] // CHECK: destroy_value [[SET]] return dict as! Set } // CHECK-LABEL: sil hidden @_T019collection_downcast33testSetDowncastBridgedConditional{{.*}}F // CHECK: bb0([[SET:%[0-9]+]] : $Set) func testSetDowncastBridgedConditional(_ dict: Set) -> Set? { // CHECK: [[BORROWED_SET:%.*]] = begin_borrow [[SET]] // CHECK: [[SET_COPY:%.*]] = copy_value [[BORROWED_SET]] // CHECK: [[DOWNCAST_FN:%[0-9]+]] = function_ref @_T0s23_setDownCastConditional{{.*}}F // CHECK: apply [[DOWNCAST_FN]]([[SET_COPY]]) : $@convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : Hashable, τ_0_1 : Hashable> (@owned Set<τ_0_0>) -> @owned Optional> // CHECK: end_borrow [[BORROWED_SET]] from [[SET]] // CHECK: destroy_value [[SET]] return dict as? Set }