Skip to content

Commit 44908ac

Browse files
committed
[CSApply] Avoid shortcutting argument conversion when parameter has an external property wrapper
The check to see whether argument matches the parameter exactly causes two problems: prevents projected value initialized injection; and, if there are multiple parameters with property wrappers, would apply incorrect wrapper to other locations because the wrapper application index wasn't incremented. Resolves: rdar://140282980
1 parent d11db16 commit 44908ac

File tree

3 files changed

+75
-2
lines changed

3 files changed

+75
-2
lines changed

CHANGELOG.md

+23
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,29 @@
55

66
## Swift 6.1
77

8+
* Projected value initializers are now correctly injected into calls when
9+
an argument exactly matches a parameter with an external property wrapper.
10+
11+
For example:
12+
13+
```swift
14+
struct Binding {
15+
...
16+
init(projectedValue: Self) { ... }
17+
}
18+
19+
func checkValue(@Binding value: Int) {}
20+
21+
func use(v: Binding<Int>) {
22+
checkValue($value: v)
23+
// Transformed into: `checkValue(value: Binding(projectedValue: v))`
24+
}
25+
```
26+
27+
Previous versions of the Swift compiler incorrectly omitted projected value
28+
initializer injection in the call to `checkValue` because the argument type
29+
matched the parameter type exactly.
30+
831
* [SE-0444][]:
932
When the upcoming feature `MemberImportVisibility` is enabled, Swift will
1033
require that a module be directly imported in a source file when resolving

lib/Sema/CSApply.cpp

+10-2
Original file line numberDiff line numberDiff line change
@@ -6288,9 +6288,17 @@ ArgumentList *ExprRewriter::coerceCallArguments(
62886288
// `sending` parameter etc.
62896289
applyFlagsToArgument(paramIdx, argExpr);
62906290

6291-
// If the types exactly match, this is easy.
6291+
auto canShortcutConversion = [&](Type argType, Type paramType) {
6292+
if (shouldInjectWrappedValuePlaceholder ||
6293+
paramInfo.hasExternalPropertyWrapper(paramIdx))
6294+
return false;
6295+
6296+
return argType->isEqual(paramType);
6297+
};
6298+
62926299
auto paramType = param.getOldType();
6293-
if (argType->isEqual(paramType) && !shouldInjectWrappedValuePlaceholder) {
6300+
6301+
if (canShortcutConversion(argType, paramType)) {
62946302
newArgs.push_back(arg);
62956303
continue;
62966304
}

test/SILGen/property_wrapper_parameter.swift

+42
Original file line numberDiff line numberDiff line change
@@ -537,3 +537,45 @@ func testCaptures(@ClassWrapper ref: Int, @Wrapper value: Int) {
537537
// closure #2 in closure #2 in implicit closure #2 in testCaptures(ref:value:)
538538
// CHECK-LABEL: sil private [ossa] @$s26property_wrapper_parameter12testCaptures3ref5valueySi_AA7WrapperVySiGtFyAA010ProjectionH0VySiGcfu0_yAJcfU1_AJycfU0_ : $@convention(thin) (ProjectionWrapper<Int>) -> ProjectionWrapper<Int>
539539
}
540+
541+
do {
542+
@propertyWrapper
543+
struct Binding<Value> {
544+
var wrappedValue: Value {
545+
get { fatalError() }
546+
nonmutating set { }
547+
}
548+
549+
var projectedValue: Self { self }
550+
551+
init(projectedValue: Self) { self = projectedValue }
552+
}
553+
554+
final class Value {
555+
enum Kind {
556+
}
557+
558+
var kind: Binding<Kind> {
559+
fatalError()
560+
}
561+
}
562+
563+
struct Test {
564+
var value: Value
565+
566+
// CHECK-LABEL: sil private [ossa] @$s26property_wrapper_parameter4TestL_V4test5otheryAA7BindingL_VyAA5ValueL_C4KindOG_tF : $@convention(method) (Binding<Value.Kind>, @guaranteed Test) -> ()
567+
// CHECK: [[CHECK_PROJECTED_VALUE_INIT_1:%.*]] = function_ref @$s26property_wrapper_parameter4TestL_V9checkKind4kindyAA7BindingL_VyAA5ValueL_C0F0OG_tFAEL_AKvpfW
568+
// CHECK-NEXT: {{.*}} = apply [[CHECK_PROJECTED_VALUE_INIT_1]]({{.*}}) : $@convention(thin) (Binding<Value.Kind>) -> Binding<Value.Kind>
569+
// CHECK: [[CHECK_PROJECTED_VALUE_INIT_A:%.*]] = function_ref @$s26property_wrapper_parameter4TestL_V15doubleCheckKind1a1byAA7BindingL_VyAA5ValueL_C0G0OG_AMtFAEL_ALvpfW
570+
// CHECK-NEXT: {{.*}} = apply [[CHECK_PROJECTED_VALUE_INIT_A]]({{.*}}) : $@convention(thin) (Binding<Value.Kind>) -> Binding<Value.Kind>
571+
// CHECK: [[CHECK_PROJECTED_VALUE_INIT_B:%.*]] = function_ref @$s26property_wrapper_parameter4TestL_V15doubleCheckKind1a1byAA7BindingL_VyAA5ValueL_C0G0OG_AMtFAFL_ALvpfW
572+
// CHECK-NEXT: {{.*}} = apply [[CHECK_PROJECTED_VALUE_INIT_B]]({{.*}}) : $@convention(thin) (Binding<Value.Kind>) -> Binding<Value.Kind>
573+
func test(other: Binding<Value.Kind>) {
574+
checkKind($kind: value.kind) // Ok
575+
doubleCheckKind($a: value.kind, $b: other) // Ok
576+
}
577+
578+
func checkKind(@Binding kind: Value.Kind) {}
579+
func doubleCheckKind(@Binding a: Value.Kind, @Binding b: Value.Kind) {}
580+
}
581+
}

0 commit comments

Comments
 (0)