mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
IRGen: support read-only statically initialized arrays in embedded swift
Arrays buffers need to be initialized with a (minimal) metatype and with an immortal reference count.
This commit is contained in:
@@ -456,22 +456,37 @@ llvm::Constant *irgen::emitConstantObject(IRGenModule &IGM, ObjectInst *OI,
|
|||||||
|
|
||||||
if (IGM.canMakeStaticObjectReadOnly(OI->getType())) {
|
if (IGM.canMakeStaticObjectReadOnly(OI->getType())) {
|
||||||
if (!IGM.swiftImmortalRefCount) {
|
if (!IGM.swiftImmortalRefCount) {
|
||||||
auto *var = new llvm::GlobalVariable(IGM.Module, IGM.Int8Ty,
|
if (IGM.Context.LangOpts.hasFeature(Feature::Embedded)) {
|
||||||
/*constant*/ true, llvm::GlobalValue::ExternalLinkage,
|
// = HeapObject.immortalRefCount
|
||||||
/*initializer*/ nullptr, "_swiftImmortalRefCount");
|
IGM.swiftImmortalRefCount = llvm::ConstantInt::get(IGM.IntPtrTy, -1);
|
||||||
IGM.swiftImmortalRefCount = var;
|
} else {
|
||||||
|
IGM.swiftImmortalRefCount = llvm::ConstantExpr::getPtrToInt(
|
||||||
|
new llvm::GlobalVariable(IGM.Module, IGM.Int8Ty,
|
||||||
|
/*constant*/ true, llvm::GlobalValue::ExternalLinkage,
|
||||||
|
/*initializer*/ nullptr, "_swiftImmortalRefCount"),
|
||||||
|
IGM.IntPtrTy);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (!IGM.swiftStaticArrayMetadata) {
|
if (!IGM.swiftStaticArrayMetadata) {
|
||||||
auto *classDecl = IGM.getStaticArrayStorageDecl();
|
auto *classDecl = IGM.getStaticArrayStorageDecl();
|
||||||
assert(classDecl && "no __StaticArrayStorage in stdlib");
|
assert(classDecl && "no __StaticArrayStorage in stdlib");
|
||||||
CanType classTy = CanType(ClassType::get(classDecl, Type(), IGM.Context));
|
CanType classTy = CanType(ClassType::get(classDecl, Type(), IGM.Context));
|
||||||
LinkEntity entity = LinkEntity::forTypeMetadata(classTy, TypeMetadataAddress::AddressPoint);
|
if (IGM.Context.LangOpts.hasFeature(Feature::Embedded)) {
|
||||||
auto *metatype = IGM.getAddrOfLLVMVariable(entity, NotForDefinition, DebugTypeInfo());
|
LinkEntity entity = LinkEntity::forTypeMetadata(classTy, TypeMetadataAddress::AddressPoint,
|
||||||
IGM.swiftStaticArrayMetadata = cast<llvm::GlobalVariable>(metatype);
|
/*forceShared=*/ true);
|
||||||
|
// In embedded swift, the metadata for the array buffer class only needs to be very minimal:
|
||||||
|
// No vtable needed, because the object is never destructed. It only contains the null super-
|
||||||
|
// class pointer.
|
||||||
|
llvm::Constant *superClass = llvm::ConstantPointerNull::get(IGM.Int8PtrTy);
|
||||||
|
IGM.swiftStaticArrayMetadata = IGM.getAddrOfLLVMVariable(entity, superClass, DebugTypeInfo());
|
||||||
|
} else {
|
||||||
|
LinkEntity entity = LinkEntity::forTypeMetadata(classTy, TypeMetadataAddress::AddressPoint);
|
||||||
|
IGM.swiftStaticArrayMetadata = IGM.getAddrOfLLVMVariable(entity, NotForDefinition, DebugTypeInfo());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
elements[0].add(llvm::ConstantStruct::get(ObjectHeaderTy, {
|
elements[0].add(llvm::ConstantStruct::get(ObjectHeaderTy, {
|
||||||
IGM.swiftStaticArrayMetadata,
|
IGM.swiftStaticArrayMetadata,
|
||||||
llvm::ConstantExpr::getPtrToInt(IGM.swiftImmortalRefCount, IGM.IntPtrTy)}));
|
IGM.swiftImmortalRefCount }));
|
||||||
} else {
|
} else {
|
||||||
elements[0].add(llvm::Constant::getNullValue(ObjectHeaderTy));
|
elements[0].add(llvm::Constant::getNullValue(ObjectHeaderTy));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -837,8 +837,8 @@ public:
|
|||||||
|
|
||||||
llvm::GlobalVariable *TheTrivialPropertyDescriptor = nullptr;
|
llvm::GlobalVariable *TheTrivialPropertyDescriptor = nullptr;
|
||||||
|
|
||||||
llvm::GlobalVariable *swiftImmortalRefCount = nullptr;
|
llvm::Constant *swiftImmortalRefCount = nullptr;
|
||||||
llvm::GlobalVariable *swiftStaticArrayMetadata = nullptr;
|
llvm::Constant *swiftStaticArrayMetadata = nullptr;
|
||||||
|
|
||||||
/// Used to create unique names for class layout types with tail allocated
|
/// Used to create unique names for class layout types with tail allocated
|
||||||
/// elements.
|
/// elements.
|
||||||
|
|||||||
@@ -46,6 +46,7 @@ public struct HeapObject {
|
|||||||
static let refcountMask = Int(bitPattern: 0x7fff_ffff)
|
static let refcountMask = Int(bitPattern: 0x7fff_ffff)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Note: The immortalRefCount value of -1 is also hard-coded in IRGen in `irgen::emitConstantObject`.
|
||||||
static let immortalRefCount = -1
|
static let immortalRefCount = -1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,27 +1,128 @@
|
|||||||
// RUN: %target-swift-frontend -O -emit-irgen %s -module-name main -parse-as-library -enable-experimental-feature Embedded | %FileCheck %s --check-prefix CHECK-IR
|
// RUN: %target-swift-frontend -parse-as-library -enable-experimental-feature Embedded %s -O -wmo -sil-verify-all -module-name=test -emit-ir | %FileCheck %s
|
||||||
// RUN: %target-run-simple-swift(-O -enable-experimental-feature Embedded -parse-as-library -runtime-compatibility-version none -wmo -Xfrontend -disable-objc-interop) | %FileCheck %s
|
|
||||||
|
|
||||||
// REQUIRES: swift_in_compiler
|
// Also do an end-to-end test to check all components, including IRGen.
|
||||||
// REQUIRES: executable_test
|
// RUN: %empty-directory(%t)
|
||||||
// REQUIRES: optimized_stdlib
|
// RUN: %target-build-swift -parse-as-library -enable-experimental-feature Embedded -O -wmo -module-name=test %s -o %t/a.out
|
||||||
// REQUIRES: OS=macosx || OS=linux-gnu
|
// RUN: %target-run %t/a.out | %FileCheck %s -check-prefix=CHECK-OUTPUT
|
||||||
|
|
||||||
|
// REQUIRES: executable_test,swift_stdlib_no_asserts,optimized_stdlib
|
||||||
|
|
||||||
|
// Check if the optimizer is able to convert array literals to constant statically initialized arrays.
|
||||||
|
|
||||||
|
// CHECK-DAG: @"$s4test11arrayLookupyS2iFTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test11returnArraySaySiGyFTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test9passArrayyyFTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test9passArrayyyFTv0_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test10storeArrayyyFTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test3StrV9staticLet_WZTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test3StrV9staticVar_WZTv_r" = {{.*}} constant {{.*}} @"$ss20__StaticArrayStorageCN", {{.*}} -1
|
||||||
|
// CHECK-DAG: @"$s4test3StrV9staticVarSaySiGvpZ" = global {{.*}} ptr @"$s4test3StrV9staticVar_WZTv_r"
|
||||||
|
// CHECK-DAG: @"$s4test3StrV14twoDimensionalSaySaySiGGvpZ" = global {{.*}} ptr @"$s4test3StrV14twoDimensional_WZTv{{[0-9]*}}_r"
|
||||||
|
|
||||||
|
// Currently, constant static arrays only work on Darwin platforms.
|
||||||
|
// REQUIRES: VENDOR=apple
|
||||||
|
|
||||||
|
|
||||||
|
public struct Str {
|
||||||
|
public static let staticLet = [ 200, 201, 202 ]
|
||||||
|
public static var staticVar = [ 300, 301, 302 ]
|
||||||
|
public static var twoDimensional = [[1, 2], [3, 4], [5, 6]]
|
||||||
|
}
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func arrayLookup(_ i: Int) -> Int {
|
||||||
|
let lookupTable = [10, 11, 12]
|
||||||
|
return lookupTable[i]
|
||||||
|
}
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func returnArray() -> [Int] {
|
||||||
|
return [20, 21]
|
||||||
|
}
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func modifyArray() -> [Int] {
|
||||||
|
var a = returnArray()
|
||||||
|
a[1] = 27
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
|
||||||
|
public var gg: [Int]?
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func receiveArray(_ a: [Int]) {
|
||||||
|
gg = a
|
||||||
|
}
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func passArray() {
|
||||||
|
receiveArray([27, 28])
|
||||||
|
receiveArray([29])
|
||||||
|
}
|
||||||
|
|
||||||
|
@inline(never)
|
||||||
|
public func storeArray() {
|
||||||
|
gg = [227, 228]
|
||||||
|
}
|
||||||
|
|
||||||
public func stringArray() -> [StaticString] {
|
public func stringArray() -> [StaticString] {
|
||||||
return ["a", "b", "c", "d"]
|
return ["a", "b", "c", "d"]
|
||||||
}
|
}
|
||||||
// CHECK-IR: define {{.*}}@"$s4main11stringArraySays12StaticStringVGyF"
|
|
||||||
// CHECK-IR-NEXT: entry:
|
|
||||||
// CHECK-IR-NEXT: call {{.*}}@swift_initStaticObject
|
|
||||||
|
|
||||||
@main
|
@main struct Main {
|
||||||
struct Main {
|
|
||||||
static func main() {
|
static func main() {
|
||||||
|
|
||||||
|
// CHECK-OUTPUT: [200, 201, 202]
|
||||||
|
printArray(Str.staticLet)
|
||||||
|
|
||||||
|
// CHECK-OUTPUT: [300, 301, 302]
|
||||||
|
printArray(Str.staticVar)
|
||||||
|
|
||||||
|
// CHECK-OUTPUT: [1, 2]
|
||||||
|
// CHECK-OUTPUT-NEXT: [3, 4]
|
||||||
|
// CHECK-OUTPUT-NEXT: [5, 6]
|
||||||
|
for x in Str.twoDimensional {
|
||||||
|
printArray(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK-OUTPUT-NEXT: 11
|
||||||
|
print(arrayLookup(1))
|
||||||
|
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 21]
|
||||||
|
printArray(returnArray())
|
||||||
|
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 27]
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 27]
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 27]
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 27]
|
||||||
|
// CHECK-OUTPUT-NEXT: [20, 27]
|
||||||
|
for _ in 0..<5 {
|
||||||
|
printArray(modifyArray())
|
||||||
|
}
|
||||||
|
|
||||||
|
passArray()
|
||||||
|
// CHECK-OUTPUT-NEXT: [29]
|
||||||
|
printArray(gg!)
|
||||||
|
|
||||||
|
storeArray()
|
||||||
|
// CHECK-OUTPUT-NEXT: [227, 228]
|
||||||
|
printArray(gg!)
|
||||||
|
|
||||||
for c in stringArray() {
|
for c in stringArray() {
|
||||||
|
// CHECK-OUTPUT-NEXT: a
|
||||||
|
// CHECK-OUTPUT-NEXT: b
|
||||||
|
// CHECK-OUTPUT-NEXT: c
|
||||||
|
// CHECK-OUTPUT-NEXT: d
|
||||||
print(c)
|
print(c)
|
||||||
// CHECK: a
|
|
||||||
// CHECK: b
|
|
||||||
// CHECK: c
|
|
||||||
// CHECK: d
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func printArray(_ a: [Int]) {
|
||||||
|
print("[", terminator: "")
|
||||||
|
for (i, x) in a.enumerated() {
|
||||||
|
print(x, terminator: i == a.count - 1 ? "" : ", ")
|
||||||
|
}
|
||||||
|
print("]")
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user