// RUN: %empty-directory(%t) // RUN: %build-silgen-test-overlays // RUN: %target-swift-emit-silgen(mock-sdk: -sdk %S/Inputs -I %t) -Xllvm -sil-print-types -module-name objc_currying %s | %FileCheck %s // REQUIRES: objc_interop import Foundation import gizmo func curry_pod(_ x: CurryTest) -> (Int) -> Int { return x.pod } // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying9curry_podyS2icSo9CurryTestCF : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed (Int) -> Int // CHECK: bb0([[ARG1:%.*]] : @guaranteed $CurryTest): // CHECK: [[THUNK:%.*]] = function_ref @$s13objc_currying9curry_podyS2icSo9CurryTestCFS2icADcfu_ : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed (Int) -> Int // CHECK: [[FN:%.*]] = apply [[THUNK]]([[ARG1]]) // CHECK-NOT: destroy_value // CHECK: return [[FN]] // CHECK: } // end sil function '$s13objc_currying9curry_podyS2icSo9CurryTestCF' // CHECK: sil private [ossa] @$s13objc_currying9curry_podyS2icSo9CurryTestCFS2icADcfu_S2icfu0_ : $@convention(thin) (Int, @guaranteed CurryTest) -> Int // CHECK: bb0([[ARG1:%.*]] : $Int, [[ARG2:%.*]] : @closureCapture @guaranteed $CurryTest): // CHECK: [[METHOD:%.*]] = objc_method [[ARG2]] : $CurryTest, #CurryTest.pod!foreign // CHECK: [[RESULT:%.*]] = apply [[METHOD]]([[ARG1]], [[ARG2]]) // CHECK: return [[RESULT]] // CHECK: } func curry_bridged(_ x: CurryTest) -> (String?) -> String? { return x.bridged } // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying13curry_bridgedySSSgACcSo9CurryTestCF : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed (@guaranteed Optional) -> @owned Optional // CHECK: bb0([[ARG1:%.*]] : @guaranteed $CurryTest): // CHECK: [[THUNK:%.*]] = function_ref @$s13objc_currying13curry_bridgedySSSgACcSo9CurryTestCFA2CcAEcfu_ : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed (@guaranteed Optional) -> @owned Optional // CHECK: [[FN:%.*]] = apply [[THUNK]]([[ARG1]]) // CHECK-NOT: destroy_value [[ARG1]] // CHECK: return [[FN]] // CHECK: } // end sil function '$s13objc_currying13curry_bridgedySSSgACcSo9CurryTestCF' // CHECK: sil private [ossa] @$s13objc_currying13curry_bridgedySSSgACcSo9CurryTestCFA2CcAEcfu_A2Ccfu0_ : $@convention(thin) (@guaranteed Optional, @guaranteed CurryTest) -> @owned Optional // CHECK: bb0([[OPT_STRING:%.*]] : @guaranteed $Optional, [[SELF:%.*]] : @closureCapture @guaranteed $CurryTest): // CHECK: [[COPY_OPT_STRING:%.*]] = copy_value [[OPT_STRING]] // CHECK: switch_enum [[COPY_OPT_STRING]] : $Optional, case #Optional.some!enumelt: [[SOME_BB:bb[0-9]+]], // // CHECK: [[SOME_BB]]([[STRING:%.*]] : @owned $String): // CHECK: [[BRIDGING_FUNC:%.*]] = function_ref @$sSS10FoundationE19_bridgeToObjectiveCSo8NSStringCyF // CHECK: [[BORROWED_STRING:%.*]] = begin_borrow [[STRING]] // CHECK: [[NSSTRING:%.*]] = apply [[BRIDGING_FUNC]]([[BORROWED_STRING]]) // CHECK: end_borrow [[BORROWED_STRING]] // CHECK: [[OPT_NSSTRING:%.*]] = enum $Optional, #Optional.some!enumelt, [[NSSTRING]] : $NSString // CHECK: destroy_value [[STRING]] // CHECK: br bb3([[OPT_NSSTRING]] : $Optional) // CHECK: bb2: // CHECK: [[OPT_NONE:%.*]] = enum $Optional, #Optional.none!enumelt // CHECK: br bb3([[OPT_NONE]] : $Optional) // CHECK: bb3([[OPT_NSSTRING:%.*]] : @owned $Optional): // CHECK: [[METHOD:%.*]] = objc_method [[SELF]] : $CurryTest, #CurryTest.bridged!foreign // CHECK: [[RESULT_OPT_NSSTRING:%.*]] = apply [[METHOD]]([[OPT_NSSTRING]], [[SELF]]) : $@convention(objc_method) (Optional, CurryTest) -> @autoreleased Optional // CHECK: destroy_value [[OPT_NSSTRING]] // CHECK: switch_enum [[RESULT_OPT_NSSTRING]] : $Optional, case #Optional.some!enumelt: [[SOME_BB:bb[0-9]+]], // CHECK: [[SOME_BB]]([[RESULT_NSSTRING:%.*]] : @owned $NSString): // CHECK: [[BRIDGE_FUNC:%.*]] = function_ref @$sSS10FoundationE36_unconditionallyBridgeFromObjectiveCySSSo8NSStringCSgFZ // CHECK: [[REWRAP_RESULT_NSSTRING:%.*]] = enum $Optional, #Optional.some!enumelt, [[RESULT_NSSTRING]] // CHECK: [[RESULT_STRING:%.*]] = apply [[BRIDGE_FUNC]]([[REWRAP_RESULT_NSSTRING]] // CHECK: [[WRAPPED_RESULT_STRING:%.*]] = enum $Optional, #Optional.some!enumelt, [[RESULT_STRING]] // CHECK: destroy_value [[REWRAP_RESULT_NSSTRING]] // CHECK: br bb6([[WRAPPED_RESULT_STRING]] : $Optional) // CHECK: bb5: // CHECK: [[OPT_NONE:%.*]] = enum $Optional, #Optional.none!enumelt // CHECK: br bb6([[OPT_NONE]] : $Optional) // CHECK: bb6([[FINAL_RESULT:%.*]] : @owned $Optional): // CHECK: return [[FINAL_RESULT]] : $Optional // CHECK: } func curry_returnsInnerPointer(_ x: CurryTest) -> () -> UnsafeMutableRawPointer? { return x.returnsInnerPointer } // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying25curry_returnsInnerPointerySvSgycSo9CurryTestCF : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed () -> Optional { // CHECK: bb0([[SELF:%.*]] : @guaranteed $CurryTest): // CHECK: [[THUNK:%.*]] = function_ref @$s13objc_currying25curry_returnsInnerPointerySvSgycSo9CurryTestCFACycAEcfu_ : $@convention(thin) (@guaranteed CurryTest) -> @owned @callee_guaranteed () -> Optional // CHECK: [[FN:%.*]] = apply [[THUNK]]([[SELF]]) // CHECK-NOT: destroy_value [[SELF]] // CHECK: return [[FN]] // CHECK: } // end sil function '$s13objc_currying25curry_returnsInnerPointerySvSgycSo9CurryTestCF' // CHECK: sil private [ossa] @$s13objc_currying25curry_returnsInnerPointerySvSgycSo9CurryTestCFACycAEcfu_ACycfu0_ : $@convention(thin) (@guaranteed CurryTest) -> Optional // CHECK: bb0([[ARG1:%.*]] : @closureCapture @guaranteed $CurryTest): // CHECK: [[METHOD:%.*]] = objc_method [[ARG1]] : $CurryTest, #CurryTest.returnsInnerPointer!foreign // CHECK: [[ARG1_COPY:%.*]] = copy_value [[ARG1]] // CHECK: [[RES:%.*]] = apply [[METHOD]]([[ARG1]]) : $@convention(objc_method) (CurryTest) -> @unowned_inner_pointer Optional // CHECK: autorelease_value [[ARG1_COPY]] // CHECK: return [[RES]] // CHECK: } // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying19curry_pod_AnyObjectyS2icyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed (Int) -> Int // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[OPENED_ANY_COPY]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.pod!foreign, [[HAS_METHOD:bb[0-9]+]] // CHECK: [[HAS_METHOD]]([[METHOD:%.*]] : $@convention(objc_method) (Int, @opened({{.*}}, AnyObject) Self) -> Int): // CHECK: [[OPENED_ANY_COPY_2:%.*]] = copy_value [[OPENED_ANY_COPY]] // CHECK: partial_apply [callee_guaranteed] [[METHOD]]([[OPENED_ANY_COPY_2]]) // CHECK: } // end sil function '$s13objc_currying19curry_pod_AnyObjectyS2icyXlF' func curry_pod_AnyObject(_ x: AnyObject) -> (Int) -> Int { return x.pod! } // normalOwnership requires a thunk to bring the method to Swift conventions // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying31curry_normalOwnership_AnyObjectySo9CurryTestCSgAEcyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed (@guaranteed Optional) -> @owned Optional { // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[OPENED_ANY_COPY]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.normalOwnership!foreign, [[HAS_METHOD:bb[0-9]+]] // CHECK: [[HAS_METHOD]]([[METHOD:%.*]] : $@convention(objc_method) (Optional, @opened({{.*}}, AnyObject) Self) -> @autoreleased Optional): // CHECK: [[OPENED_ANY_COPY_2:%.*]] = copy_value [[OPENED_ANY_COPY]] // CHECK: [[PA:%.*]] = partial_apply [callee_guaranteed] [[METHOD]]([[OPENED_ANY_COPY_2]]) // CHECK: [[THUNK:%.*]] = function_ref @$sSo9CurryTestCSgACIegyo_A2CIeggo_TR // CHECK: partial_apply [callee_guaranteed] [[THUNK]]([[PA]]) // CHECK: } // end sil function '$s13objc_currying31curry_normalOwnership_AnyObjectySo9CurryTestCSgAEcyXlF' func curry_normalOwnership_AnyObject(_ x: AnyObject) -> (CurryTest?) -> CurryTest? { return x.normalOwnership! } // weirdOwnership is NS_RETURNS_RETAINED and NS_CONSUMES_SELF so already // follows Swift conventions // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying30curry_weirdOwnership_AnyObjectySo9CurryTestCSgAEcyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed (@guaranteed Optional) -> @owned Optional // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[SELF:%.*]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.weirdOwnership!foreign, [[HAS_METHOD:bb[0-9]+]] // // CHECK: bb1([[METHOD:%.*]] : $@convention(objc_method) (@owned Optional, @owned @opened({{.*}}, AnyObject) Self) -> @owned Optional): // CHECK: [[OPENED_ANY_COPY_2:%.*]] = copy_value [[OPENED_ANY_COPY]] // CHECK: partial_apply [callee_guaranteed] [[METHOD]]([[OPENED_ANY_COPY_2]]) // CHECK: } // end sil function '$s13objc_currying30curry_weirdOwnership_AnyObjectySo9CurryTestCSgAEcyXlF' func curry_weirdOwnership_AnyObject(_ x: AnyObject) -> (CurryTest?) -> CurryTest? { return x.weirdOwnership! } // bridged requires a thunk to handle bridging conversions // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying23curry_bridged_AnyObjectySSSgACcyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed (@guaranteed Optional) -> @owned Optional // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[OPENED_ANY_COPY]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.bridged!foreign, [[HAS_METHOD:bb[0-9]+]] // CHECK: } // end sil function '$s13objc_currying23curry_bridged_AnyObjectySSSgACcyXlF' func curry_bridged_AnyObject(_ x: AnyObject) -> (String?) -> String? { return x.bridged! } // check that we substitute Self = AnyObject correctly for Self-returning // methods // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying27curry_returnsSelf_AnyObjectyyXlSgycyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed () -> @owned Optional { // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[OPENED_ANY_COPY]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.returnsSelf!foreign, [[HAS_METHOD:bb[0-9]+]] // CHECK: [[HAS_METHOD]]([[METHOD:%.*]] : $@convention(objc_method) (@opened({{.*}}, AnyObject) Self) -> @autoreleased Optional): // CHECK: [[OPENED_ANY_COPY_2:%.*]] = copy_value [[OPENED_ANY_COPY]] // CHECK: [[PA:%.*]] = partial_apply [callee_guaranteed] [[METHOD]]([[OPENED_ANY_COPY_2]]) // CHECK: } // end sil function '$s13objc_currying27curry_returnsSelf_AnyObjectyyXlSgycyXlF' func curry_returnsSelf_AnyObject(_ x: AnyObject) -> () -> AnyObject? { return x.returnsSelf! } // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying35curry_returnsInnerPointer_AnyObjectySvSgycyXlF : $@convention(thin) (@guaranteed AnyObject) -> @owned @callee_guaranteed () -> Optional { // CHECK: bb0([[ANY:%.*]] : @guaranteed $AnyObject): // CHECK: [[OPENED_ANY:%.*]] = open_existential_ref [[ANY]] // CHECK: [[OPENED_ANY_COPY:%.*]] = copy_value [[OPENED_ANY]] // CHECK: dynamic_method_br [[OPENED_ANY_COPY]] : $@opened({{.*}}, AnyObject) Self, #CurryTest.returnsInnerPointer!foreign, [[HAS_METHOD:bb[0-9]+]] // CHECK: [[HAS_METHOD]]([[METHOD:%.*]] : $@convention(objc_method) (@opened({{.*}}, AnyObject) Self) -> @unowned_inner_pointer Optional): // CHECK: [[OPENED_ANY_COPY_2:%.*]] = copy_value [[OPENED_ANY_COPY]] // CHECK: [[PA:%.*]] = partial_apply [callee_guaranteed] [[METHOD]]([[OPENED_ANY_COPY_2]]) // CHECK: } // end sil function '$s13objc_currying35curry_returnsInnerPointer_AnyObjectySvSgycyXlF' func curry_returnsInnerPointer_AnyObject(_ x: AnyObject) -> () -> UnsafeMutableRawPointer? { return x.returnsInnerPointer! } // Make sure we don't crash if the same imported declaration is referenced // normally and as a curry thunk // CHECK-LABEL: sil hidden [ossa] @$s13objc_currying17curry_initializerSo5GizmoCSgSicyF : $@convention(thin) () -> @owned @callee_guaranteed (Int) -> @owned Optional { func curry_initializer() -> (Int) -> Gizmo? { _ = Gizmo.init(bellsOn: 10) return Gizmo.init(bellsOn:) } // CHECK-LABEL: sil shared [serialized] [ossa] @$sSo5GizmoC7bellsOnABSgSi_tcfC : $@convention(method) (Int, @thick Gizmo.Type) -> @owned Optional { // CHECK-LABEL: sil private [ossa] @$s13objc_currying17curry_initializerSo5GizmoCSgSicyFAESicfu_ : $@convention(thin) (Int) -> @owned Optional { // CHECK-LABEL: sil shared [serialized] [thunk] [ossa] @$sSo5GizmoC7bellsOnABSgSi_tcfcTO : $@convention(method) (Int, @owned Gizmo) -> @owned Optional {