// RUN: %empty-directory(%t) // RUN: %target-build-swift-dylib(%t/%target-library-name(PrintShims)) %S/../../Inputs/print-shims.swift -module-name PrintShims -emit-module -emit-module-path %t/PrintShims.swiftmodule // RUN: %target-codesign %t/%target-library-name(PrintShims) // RUN: %target-build-swift -g -parse-sil %s -emit-ir -I %t -L %t -lPrintShim | %FileCheck %s --check-prefix=CHECK-LL // RUN: %target-build-swift -g -parse-sil %s -module-name main -o %t/main -I %t -L %t -lPrintShims %target-rpath(%t) // RUN: %target-codesign %t/main // RUN: %target-run %t/main %t/%target-library-name(PrintShims) | %FileCheck %s // REQUIRES: executable_test // REQUIRES: swift_test_mode_optimize_none // REQUIRES: concurrency // REQUIRES: concurrency_runtime // UNSUPPORTED: back_deployment_runtime import Builtin import Swift import PrintShims import _Concurrency sil public_external @printGeneric : $@convention(thin) (@in_guaranteed T) -> () sil public_external @printInt64 : $@convention(thin) (Int64) -> () protocol P { func printMe(_ t: T) async -> (Int64, T) } extension P { func callPrintMe(_ t: T) async -> (Int64, T) } struct I : P { @_hasStorage let int: Int64 { get } func printMe(_ t: T) async -> (Int64, T) init(int: Int64) } sil hidden @I_printMe : $@convention(method) @async (@in_guaranteed T, I) -> (Int64, @out T) { bb0(%out_addr : $*T, %in_addr : $*T, %self : $I): %self_addr = alloc_stack $I store %self to %self_addr : $*I %printGeneric = function_ref @printGeneric : $@convention(thin) <τ_0_0> (@in_guaranteed τ_0_0) -> () %printGeneric_result = apply %printGeneric(%self_addr) : $@convention(thin) <τ_0_0> (@in_guaranteed τ_0_0) -> () dealloc_stack %self_addr : $*I %value = struct_extract %self : $I, #I.int copy_addr %in_addr to [init] %out_addr : $*T return %value : $Int64 } // CHECK-LL: @I_P_printMeTu = // CHECK-LL: define internal swift{{(tail)?}}cc void @I_P_printMe( sil private [transparent] [thunk] @I_P_printMe : $@convention(witness_method: P) @async <τ_0_0> (@in_guaranteed τ_0_0, @in_guaranteed I) -> (Int64, @out τ_0_0) { bb0(%out_addr : $*τ_0_0, %in_addr : $*τ_0_0, %self_addr : $*I): %self = load %self_addr : $*I %I_printMe = function_ref @I_printMe : $@convention(method) @async <τ_0_0> (@in_guaranteed τ_0_0, I) -> (Int64, @out τ_0_0) %result = apply %I_printMe<τ_0_0>(%out_addr, %in_addr, %self) : $@convention(method) @async <τ_0_0> (@in_guaranteed τ_0_0, I) -> (Int64, @out τ_0_0) return %result : $Int64 } sil hidden @callPrintMe : $@async @convention(thin) (@in_guaranteed T, @in_guaranteed U) -> (Int64, @out U) { bb0(%out_addr : $*U, %self_addr : $*T, %in_addr : $*U): %I_P_printMe = witness_method $T, #P.printMe : (Self) -> (T) async -> (Int64, T) : $@convention(witness_method: P) @async <τ_0_0 where τ_0_0 : P><τ_1_0> (@in_guaranteed τ_1_0, @in_guaranteed τ_0_0) -> (Int64, @out τ_1_0) %result = apply %I_P_printMe(%out_addr, %in_addr, %self_addr) : $@convention(witness_method: P) @async <τ_0_0 where τ_0_0 : P><τ_1_0> (@in_guaranteed τ_1_0, @in_guaranteed τ_0_0) -> (Int64, @out τ_1_0) return %result : $Int64 } sil @test_case : $@convention(thin) @async () -> () { %I_type = metatype $@thin I.Type %int_literal = integer_literal $Builtin.Int64, 99 %int = struct $Int64 (%int_literal : $Builtin.Int64) %i = struct $I (%int : $Int64) %out_addr = alloc_stack $I %in_addr = alloc_stack $I store %i to %in_addr : $*I %i_addr = alloc_stack $I store %i to %i_addr : $*I %callPrintMe = function_ref @callPrintMe : $@async @convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : P> (@in_guaranteed τ_0_0, @in_guaranteed τ_0_1) -> (Int64, @out τ_0_1) %result = apply %callPrintMe(%out_addr, %in_addr, %i_addr) : $@async @convention(thin) <τ_0_0, τ_0_1 where τ_0_0 : P> (@in_guaranteed τ_0_0, @in_guaranteed τ_0_1) -> (Int64, @out τ_0_1) dealloc_stack %i_addr : $*I dealloc_stack %in_addr : $*I %out = load %out_addr : $*I dealloc_stack %out_addr : $*I %printInt64 = function_ref @printInt64 : $@convention(thin) (Int64) -> () %printInt64_result = apply %printInt64(%result) : $@convention(thin) (Int64) -> () // CHECK: 99 %out_addr_2 = alloc_stack $I store %out to %out_addr_2 : $*I %printGeneric = function_ref @printGeneric : $@convention(thin) <τ_0_0> (@in_guaranteed τ_0_0) -> () %printGeneric_result = apply %printGeneric(%out_addr_2) : $@convention(thin) <τ_0_0> (@in_guaranteed τ_0_0) -> () // CHECK: I(int: 99) dealloc_stack %out_addr_2 : $*I %void = tuple() return %void : $() } // Defined in _Concurrency sil @$ss13_runAsyncMainyyyyYaKcF : $@convention(thin) (@guaranteed @async @callee_guaranteed () -> @error Error) -> () sil @no_throw_to_throw_think : $@convention(thin) @async (@guaranteed @async @callee_guaranteed () -> ()) -> @error Error { bb0(%0 :$@async @callee_guaranteed () -> ()): %void = apply %0() : $@async @callee_guaranteed () -> () return %void : $() } // main sil @main : $@convention(c) () -> Int32 { %test_case_nothrow = function_ref @test_case : $@convention(thin) @async () -> () %thick_test_case = thin_to_thick_function %test_case_nothrow : $@convention(thin) @async () -> () to $@callee_guaranteed @async () -> () %thunk = function_ref @no_throw_to_throw_think : $@convention(thin) @async (@guaranteed @async @callee_guaranteed () -> ()) -> @error Error %throwing = partial_apply [callee_guaranteed] %thunk(%thick_test_case) : $@convention(thin) @async (@guaranteed @async @callee_guaranteed () -> ()) -> @error Error %runAsyncMain = function_ref @$ss13_runAsyncMainyyyyYaKcF : $@convention(thin) (@guaranteed @async @callee_guaranteed () -> @error Error) -> () %result = apply %runAsyncMain(%throwing) : $@convention(thin) (@guaranteed @async @callee_guaranteed () -> @error Error) -> () %out_literal = integer_literal $Builtin.Int32, 0 %out = struct $Int32 (%out_literal : $Builtin.Int32) return %out : $Int32 } sil_witness_table hidden I: P module main { method #P.printMe: (Self) -> (T) async -> (Int64, T) : @I_P_printMe }