// RUN: %target-swift-frontend -Xllvm -sil-print-types -emit-silgen %s | %FileCheck %s import _Differentiation @_silgen_name("triggerReabstraction1") func triggerReabstraction1(_ f: @escaping @differentiable(reverse) (T) -> T) {} @_silgen_name("triggerReabstraction2") func triggerReabstraction2(_ t: T) {} @_silgen_name("differentiable1") func differentiable1(_ x: Float) -> Float { x } @_silgen_name("differentiable2") func differentiable2(_ t: T) -> T { t } @_silgen_name("makeSignatureAbstract") func makeSignatureAbstract() { triggerReabstraction1(differentiable1) } // CHECK-LABEL: sil{{.*}}@makeSignatureAbstract // CHECK: [[BEFORE:%.*]] = differentiable_function [parameters 0] [results 0] // CHECK: [[ORIG_0:%.*]] = differentiable_function_extract [original] [[BEFORE]] // CHECK: [[ORIG_1:%.*]] = copy_value [[ORIG_0]] // CHECK: [[ORIG_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> Float) -> @out Float // CHECK: [[ORIG_2:%.*]] = partial_apply [callee_guaranteed] [[ORIG_THUNK]]([[ORIG_1]]) // CHECK: [[ORIG_3:%.*]] = convert_function [[ORIG_2]] // CHECK: [[JVP_0:%.*]] = differentiable_function_extract [jvp] [[BEFORE]] // CHECK: [[JVP_1:%.*]] = copy_value [[JVP_0]] // CHECK: [[JVP_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> (Float, @owned @callee_guaranteed (Float) -> Float)) -> (@out Float, @owned @callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for ) // CHECK: [[JVP_2:%.*]] = partial_apply [callee_guaranteed] [[JVP_THUNK]]([[JVP_1]]) // CHECK: [[JVP_3:%.*]] = convert_function [[JVP_2]] // CHECK: [[VJP_0:%.*]] = differentiable_function_extract [vjp] [[BEFORE]] // CHECK: [[VJP_1:%.*]] = copy_value [[VJP_0]] // CHECK: [[VJP_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> (Float, @owned @callee_guaranteed (Float) -> Float)) -> (@out Float, @owned @callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for ) // CHECK: [[VJP_2:%.*]] = partial_apply [callee_guaranteed] [[VJP_THUNK]]([[VJP_1]]) // CHECK: [[VJP_3:%.*]] = convert_function [[VJP_2]] // CHECK: [[AFTER:%.*]] = differentiable_function [parameters 0] [results 0] [[ORIG_3]] {{.*}} with_derivative {[[JVP_3]] {{.*}}, [[VJP_3]] {{.*}}} // CHECK: [[TRIGGER:%.*]] = function_ref @triggerReabstraction1 // CHECK: apply [[TRIGGER]]([[AFTER]]) @_silgen_name("makeOpaque") func makeOpaque() { let f: @differentiable(reverse) (Float) -> Float = differentiable1 triggerReabstraction2(f) } // CHECK-LABEL: sil{{.*}}@makeOpaque // CHECK: [[STACK_ADDR:%.*]] = alloc_stack $@differentiable(reverse) @callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for // CHECK: [[ORIG_0:%.*]] = differentiable_function_extract [original] [[BEFORE:%.*]] : $@differentiable(reverse) @callee_guaranteed (Float) -> Float // CHECK: [[ORIG_1:%.*]] = copy_value [[ORIG_0]] // CHECK: [[ORIG_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> Float) -> @out Float // CHECK: [[ORIG_2:%.*]] = partial_apply [callee_guaranteed] [[ORIG_THUNK]]([[ORIG_1]]) // CHECK: [[ORIG_3:%.*]] = convert_function [[ORIG_2]] // CHECK: [[JVP_0:%.*]] = differentiable_function_extract [jvp] [[BEFORE]] // CHECK: [[JVP_1:%.*]] = copy_value [[JVP_0]] // CHECK: [[JVP_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> (Float, @owned @callee_guaranteed (Float) -> Float)) -> (@out Float, @owned @callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for ) // CHECK: [[JVP_2:%.*]] = partial_apply [callee_guaranteed] [[JVP_THUNK]]([[JVP_1]]) // CHECK: [[JVP_3:%.*]] = convert_function [[JVP_2]] // CHECK: [[VJP_0:%.*]] = differentiable_function_extract [vjp] [[BEFORE]] // CHECK: [[VJP_1:%.*]] = copy_value [[VJP_0]] // CHECK: [[VJP_THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (@in_guaranteed Float, @guaranteed @callee_guaranteed (Float) -> (Float, @owned @callee_guaranteed (Float) -> Float)) -> (@out Float, @owned @callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for ) // CHECK: [[VJP_2:%.*]] = partial_apply [callee_guaranteed] [[VJP_THUNK]]([[VJP_1]]) // CHECK: [[VJP_3:%.*]] = convert_function [[VJP_2]] // CHECK: [[AFTER:%.*]] = differentiable_function [parameters 0] [results 0] [[ORIG_3]] {{.*}} with_derivative {[[JVP_3]] {{.*}}, [[VJP_3]] {{.*}}} // CHECK: store [[AFTER]] to [init] [[STACK_ADDR]] // CHECK: [[TRIGGER:%.*]] = function_ref @triggerReabstraction2 // CHECK: apply [[TRIGGER]]<@differentiable(reverse) (Float) -> Float>([[STACK_ADDR]]) @_silgen_name("makeSignatureDirect") func makeSignatureDirect() { let _: @differentiable(reverse) (Float) -> Float = differentiable2 } // CHECK-LABEL: sil{{.*}}@makeSignatureDirect // CHECK: [[ORIG_0:%.*]] = function_ref @differentiable2 // CHECK: [[ORIG_1:%.*]] = partial_apply [callee_guaranteed] [[ORIG_0]]() // CHECK: [[THUNK:%.*]] = function_ref {{.*}} : $@convention(thin) (Float, @guaranteed @callee_guaranteed (@in_guaranteed Float) -> @out Float) -> Float // CHECK: [[ORIG_2:%.*]] = partial_apply [callee_guaranteed] [[THUNK]]([[ORIG_1]]) // CHECK: differentiable_function [parameters 0] [results 0] [[ORIG_2]]