Files
swift-mirror/stdlib/public/stubs/GlobalObjects.cpp
Erik Eckstein e4ea7491a2 IRGen: fix and re-enable static read-only arrays
Static read-only arrays didn't work when passed to ObjectiveC as NSArray.
The storage class of static read-only arrays doesn't carry information about the Element type.
The new `__SwiftDeferredStaticNSArray` is generic over the element type and doesn't have to rely on the element type information of the array storage.

rdar://94185998
2023-12-13 11:45:01 +01:00

169 lines
5.9 KiB
C++

//===--- GlobalObjects.cpp - Statically-initialized objects ---------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// Objects that are allocated at global scope instead of on the heap,
// and statically initialized to avoid synchronization costs, are
// defined here.
//
//===----------------------------------------------------------------------===//
#include "swift/shims/GlobalObjects.h"
#include "swift/shims/Random.h"
#include "swift/Runtime/Metadata.h"
#include "swift/Runtime/Debug.h"
#include "swift/Runtime/EnvironmentVariables.h"
#include <stdlib.h>
namespace swift {
// FIXME(ABI)#76 : does this declaration need SWIFT_RUNTIME_STDLIB_API?
// _direct type metadata for Swift.__EmptyArrayStorage
SWIFT_RUNTIME_STDLIB_API
ClassMetadata CLASS_METADATA_SYM(s19__EmptyArrayStorage);
// _direct type metadata for Swift.__EmptyDictionarySingleton
SWIFT_RUNTIME_STDLIB_API
ClassMetadata CLASS_METADATA_SYM(s26__EmptyDictionarySingleton);
// _direct type metadata for Swift.__EmptySetSingleton
SWIFT_RUNTIME_STDLIB_API
ClassMetadata CLASS_METADATA_SYM(s19__EmptySetSingleton);
} // namespace swift
SWIFT_RUNTIME_STDLIB_API
swift::_SwiftEmptyArrayStorage swift::_swiftEmptyArrayStorage = {
// HeapObject header;
{
&swift::CLASS_METADATA_SYM(s19__EmptyArrayStorage), // isa pointer
InlineRefCounts::Immortal
},
// _SwiftArrayBodyStorage body;
{
0, // int count;
1 // unsigned int _capacityAndFlags; 1 means elementTypeIsBridgedVerbatim
}
};
// Define `__swiftImmortalRefCount` which is used by constant static arrays.
// It is the bit pattern for the ref-count field of the array buffer.
//
// TODO: Support constant static arrays on other platforms, too.
// This needs a bit more work because the tricks with absolute symbols and
// symbol aliases don't work this way with other object file formats than Mach-O.
#if defined(__APPLE__)
__asm__(" .globl __swiftImmortalRefCount\n");
#if __POINTER_WIDTH__ == 64
// TODO: is there a way to avoid hard coding this constant in the inline
// assembly string?
static_assert(swift::InlineRefCountBits::immortalBits() == 0x80000004ffffffffull,
"immortal refcount bits changed: correct the inline asm below");
__asm__(".set __swiftImmortalRefCount, 0x80000004ffffffff\n");
#elif __POINTER_WIDTH__ == 32
// TODO: is there a way to avoid hard coding this constant in the inline
// assembly string?
static_assert(swift::InlineRefCountBits::immortalBits() == 0x800004fful,
"immortal refcount bits changed: correct the inline asm below");
__asm__(".set __swiftImmortalRefCount, 0x800004ff\n");
#else
#error("unsupported pointer width")
#endif
#endif
SWIFT_RUNTIME_STDLIB_API
swift::_SwiftEmptyDictionarySingleton swift::_swiftEmptyDictionarySingleton = {
// HeapObject header;
{
&swift::CLASS_METADATA_SYM(s26__EmptyDictionarySingleton), // isa pointer
InlineRefCounts::Immortal
},
// _SwiftDictionaryBodyStorage body;
{
// Setting the scale to 0 makes for a bucketCount of 1 -- so that the
// storage consists of a single unoccupied bucket. The capacity is set to
// 0 so that any insertion will lead to real storage being allocated.
0, // int count;
0, // int capacity;
0, // int8 scale;
0, // int8 reservedScale;
0, // int16 extra;
0, // int32 age;
0, // int seed;
(void *)1, // void* keys; (non-null garbage)
(void *)1 // void* values; (non-null garbage)
},
// bucket 0 is unoccupied; other buckets are out-of-bounds
static_cast<__swift_uintptr_t>(~1) // int metadata;
};
SWIFT_RUNTIME_STDLIB_API
swift::_SwiftEmptySetSingleton swift::_swiftEmptySetSingleton = {
// HeapObject header;
{
&swift::CLASS_METADATA_SYM(s19__EmptySetSingleton), // isa pointer
InlineRefCounts::Immortal
},
// _SwiftSetBodyStorage body;
{
// Setting the scale to 0 makes for a bucketCount of 1 -- so that the
// storage consists of a single unoccupied bucket. The capacity is set to
// 0 so that any insertion will lead to real storage being allocated.
0, // int count;
0, // int capacity;
0, // int8 scale;
0, // int8 reservedScale;
0, // int16 extra;
0, // int32 age;
0, // int seed;
(void *)1, // void *rawElements; (non-null garbage)
},
// bucket 0 is unoccupied; other buckets are out-of-bounds
static_cast<__swift_uintptr_t>(~1) // int metadata;
};
static swift::_SwiftHashingParameters initializeHashingParameters() {
// Setting the environment variable SWIFT_DETERMINISTIC_HASHING to "1"
// disables randomized hash seeding. This is useful in cases we need to ensure
// results are repeatable, e.g., in certain test environments. (Note that
// even if the seed override is enabled, hash values aren't guaranteed to
// remain stable across even minor stdlib releases.)
if (swift::runtime::environment::SWIFT_DETERMINISTIC_HASHING()) {
return { 0, 0, true };
}
__swift_uint64_t seed0 = 0, seed1 = 0;
swift_stdlib_random(&seed0, sizeof(seed0));
swift_stdlib_random(&seed1, sizeof(seed1));
return { seed0, seed1, false };
}
SWIFT_ALLOWED_RUNTIME_GLOBAL_CTOR_BEGIN
swift::_SwiftHashingParameters swift::_swift_stdlib_Hashing_parameters =
initializeHashingParameters();
SWIFT_ALLOWED_RUNTIME_GLOBAL_CTOR_END
SWIFT_RUNTIME_STDLIB_API
void swift::_swift_instantiateInertHeapObject(void *address,
const HeapMetadata *metadata) {
::new (address) HeapObject{metadata};
}