Skip to content

Commit 72c8d6a

Browse files
authored
Merge pull request #10831 from gottesmm/fix_createunsafecopyunowned_value
[silgen] Fix createUnsafeCopyUnownedValue to be correct from an ownership perspective.
2 parents 3265808 + ae2bb3e commit 72c8d6a

File tree

2 files changed

+74
-1
lines changed

2 files changed

+74
-1
lines changed

lib/SILGen/SILGenBuilder.cpp

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -247,11 +247,14 @@ ManagedValue SILGenBuilder::createCopyUnownedValue(SILLocation loc,
247247
ManagedValue
248248
SILGenBuilder::createUnsafeCopyUnownedValue(SILLocation loc,
249249
ManagedValue originalValue) {
250+
// *NOTE* The reason why this is unsafe is that we are converting and
251+
// unconditionally retaining, rather than before converting from
252+
// unmanaged->ref checking that our value is not yet uninitialized.
250253
auto unmanagedType = originalValue.getType().getAs<UnmanagedStorageType>();
251254
SILValue result = SILBuilder::createUnmanagedToRef(
252255
loc, originalValue.getValue(),
253256
SILType::getPrimitiveObjectType(unmanagedType.getReferentType()));
254-
SILBuilder::createUnmanagedRetainValue(loc, result, getDefaultAtomicity());
257+
result = SILBuilder::createCopyValue(loc, result);
255258
return SGF.emitManagedRValueWithCleanup(result);
256259
}
257260

test/SILGen/unmanaged_ownership.swift

Lines changed: 70 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
// RUN: %target-swift-frontend -enable-sil-ownership -parse-stdlib -module-name Swift -emit-silgen %s | %FileCheck %s
2+
3+
class C {}
4+
5+
enum Optional<T> {
6+
case none
7+
case some(T)
8+
}
9+
10+
precedencegroup AssignmentPrecedence {
11+
assignment: true
12+
associativity: right
13+
}
14+
15+
struct Holder {
16+
unowned(unsafe) var value: C
17+
}
18+
19+
_ = Holder(value: C())
20+
// CHECK-LABEL:sil hidden @_T0s6HolderV{{[_0-9a-zA-Z]*}}fC : $@convention(method) (@owned C, @thin Holder.Type) -> Holder
21+
// CHECK: bb0([[T0:%.*]] : @owned $C,
22+
// CHECK-NEXT: [[T1:%.*]] = ref_to_unmanaged [[T0]] : $C to $@sil_unmanaged C
23+
// CHECK-NEXT: destroy_value [[T0]] : $C
24+
// CHECK-NEXT: [[T2:%.*]] = struct $Holder ([[T1]] : $@sil_unmanaged C)
25+
// CHECK-NEXT: return [[T2]] : $Holder
26+
// CHECK-NEXT: } // end sil function '_T0s6HolderVABs1CC5value_tcfC'
27+
func set(holder holder: inout Holder) {
28+
holder.value = C()
29+
}
30+
31+
// CHECK-LABEL: sil hidden @_T0s3setys6HolderVz6holder_tF : $@convention(thin) (@inout Holder) -> () {
32+
// CHECK: bb0([[ADDR:%.*]] : @trivial $*Holder):
33+
// CHECK: [[T0:%.*]] = function_ref @_T0s1CC{{[_0-9a-zA-Z]*}}fC
34+
// CHECK: [[C:%.*]] = apply [[T0]](
35+
// CHECK-NEXT: [[WRITE:%.*]] = begin_access [modify] [unknown] [[ADDR]] : $*Holder
36+
// CHECK-NEXT: [[T0:%.*]] = struct_element_addr [[WRITE]] : $*Holder, #Holder.value
37+
// CHECK-NEXT: [[T1:%.*]] = ref_to_unmanaged [[C]]
38+
// CHECK-NEXT: assign [[T1]] to [[T0]]
39+
// CHECK-NEXT: destroy_value [[C]]
40+
// CHECK-NEXT: end_access [[WRITE]] : $*Holder
41+
// CHECK: } // end sil function '_T0s3setys6HolderVz6holder_tF'
42+
43+
func get(holder holder: inout Holder) -> C {
44+
return holder.value
45+
}
46+
// CHECK-LABEL: sil hidden @_T0s3gets1CCs6HolderVz6holder_tF : $@convention(thin) (@inout Holder) -> @owned C {
47+
// CHECK: bb0([[ADDR:%.*]] : @trivial $*Holder):
48+
// CHECK-NEXT: debug_value_addr %0 : $*Holder, var, name "holder", argno 1
49+
// CHECK-NEXT: [[READ:%.*]] = begin_access [read] [unknown] [[ADDR]] : $*Holder
50+
// CHECK-NEXT: [[T0:%.*]] = struct_element_addr [[READ]] : $*Holder, #Holder.value
51+
// CHECK-NEXT: [[T1:%.*]] = load [trivial] [[T0]] : $*@sil_unmanaged C
52+
// CHECK-NEXT: [[T2:%.*]] = unmanaged_to_ref [[T1]]
53+
// CHECK-NEXT: [[T2_COPY:%.*]] = copy_value [[T2]]
54+
// CHECK-NEXT: end_access [[READ]] : $*Holder
55+
// CHECK-NEXT: return [[T2_COPY]]
56+
57+
func project(fn fn: () -> Holder) -> C {
58+
return fn().value
59+
}
60+
// CHECK-LABEL: sil hidden @_T0s7projects1CCs6HolderVyc2fn_tF : $@convention(thin) (@owned @callee_owned () -> Holder) -> @owned C {
61+
// CHECK: bb0([[FN:%.*]] : @owned $@callee_owned () -> Holder):
62+
// CHECK: [[BORROWED_FN:%.*]] = begin_borrow [[FN]]
63+
// CHECK-NEXT: [[BORROWED_FN_COPY:%.*]] = copy_value [[BORROWED_FN]]
64+
// CHECK-NEXT: [[T0:%.*]] = apply [[BORROWED_FN_COPY]]()
65+
// CHECK-NEXT: [[T1:%.*]] = struct_extract [[T0]] : $Holder, #Holder.value
66+
// CHECK-NEXT: [[T2:%.*]] = unmanaged_to_ref [[T1]]
67+
// CHECK-NEXT: [[COPIED_T2:%.*]] = copy_value [[T2]]
68+
// CHECK-NEXT: end_borrow [[BORROWED_FN]] from [[FN]]
69+
// CHECK-NEXT: destroy_value [[FN]]
70+
// CHECK-NEXT: return [[COPIED_T2]]

0 commit comments

Comments
 (0)