Skip to content

Commit 241ffaf

Browse files
authored
Merge pull request #80245 from Azoy/ia-literal-closure
[SILGen] Use the opaque element type for address in InlineArray literal
2 parents 2f970b6 + bc6c88b commit 241ffaf

File tree

3 files changed

+35
-2
lines changed

3 files changed

+35
-2
lines changed

lib/SILGen/SILGenExpr.cpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -4834,12 +4834,11 @@ static RValue emitInlineArrayLiteral(SILGenFunction &SGF, CollectionExpr *E,
48344834
}
48354835

48364836
auto elementType = iaTy->getGenericArgs()[1]->getCanonicalType();
4837-
auto loweredElementType = SGF.getLoweredType(elementType);
48384837
auto &eltTL = SGF.getTypeLowering(AbstractionPattern::getOpaque(), elementType);
48394838

48404839
SILValue alloc = SGF.emitTemporaryAllocation(E, loweredIAType);
48414840
SILValue addr = SGF.B.createUncheckedAddrCast(E, alloc,
4842-
loweredElementType.getAddressType());
4841+
eltTL.getLoweredType().getAddressType());
48434842

48444843
// Cleanups for any elements that have been initialized so far.
48454844
SmallVector<CleanupHandle, 8> cleanups;

test/SILGen/inlinearray_literal.swift

+14
Original file line numberDiff line numberDiff line change
@@ -106,3 +106,17 @@ func nontrivial() -> InlineArray<2, String> {
106106
func noncopyable() -> InlineArray<2, Atomic<Int>> {
107107
[Atomic(0), Atomic(1)]
108108
}
109+
110+
// CHECK-LABEL: sil{{.*}} @$s19inlinearray_literal7closures11InlineArrayVy$0_S2icGyF : $@convention(thin) () -> @owned InlineArray<1, (Int) -> Int> {
111+
// CHECK: [[IA_ALLOC:%.*]] = alloc_stack $InlineArray<1, (Int) -> Int>
112+
// CHECK-NEXT: [[ADDR_CAST:%.*]] = unchecked_addr_cast [[IA_ALLOC]] to $*@callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for <Int, Int>
113+
// CHECK: [[FN_REF:%.*]] = function_ref
114+
// CHECK-NEXT: [[THIN_TO_THICK_FN:%.*]] = thin_to_thick_function [[FN_REF]] to $@callee_guaranteed @substituted <τ_0_0, τ_0_1> (@in_guaranteed τ_0_0) -> @out τ_0_1 for <Int, Int>
115+
// CHECK-NEXT: store [[THIN_TO_THICK_FN]] to [init] [[ADDR_CAST]]
116+
// CHECK-NEXT: [[IA:%.*]] = load [take] [[IA_ALLOC]]
117+
// CHECK-NEXT: dealloc_stack [[IA_ALLOC]]
118+
// CHECK-NEXT: return [[IA]]
119+
// CHECK-LABEL: } // end sil function '$s19inlinearray_literal7closures11InlineArrayVy$0_S2icGyF'
120+
func closure() -> InlineArray<1, (Int) -> Int> {
121+
[{$0 * 2}]
122+
}

test/stdlib/InlineArray.swift

+20
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ enum InlineArrayTests {
3737
testSuite.test("Noncopyable", testNoncopyable)
3838
testSuite.test("Uninhabited", testUninhabited)
3939
testSuite.test("Throws", testThrows)
40+
testSuite.test("Closures", testClosures)
4041
runAllTests()
4142
}
4243

@@ -196,6 +197,25 @@ enum InlineArrayTests {
196197
}
197198
}
198199
}
200+
201+
/// Test the handling of closure values in InlineArray literals
202+
@available(SwiftStdlib 6.2, *)
203+
static func testClosures() {
204+
let ia: InlineArray<_, (Int) -> Int> = [{$0 * 2}]
205+
206+
for i in 0 ..< 10 {
207+
expectEqual(i * 2, ia[0](i))
208+
}
209+
210+
var x = 0
211+
let ia2: InlineArray<_, () -> ()> = [{ x += 1 }]
212+
213+
for _ in 0 ..< 10 {
214+
ia2[0]()
215+
}
216+
217+
expectEqual(x, 10)
218+
}
199219
}
200220
201221
//===----------------------------------------------------------------------===//

0 commit comments

Comments
 (0)