mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
This is essentially a long-belated follow-up to Arnold's #12606. The key observation here is that the enum-tag-single-payload witnesses are strictly more powerful than the XI witnesses: you can simulate the XI witnesses by using an extra case count that's <= the XI count. Of course the result is less efficient than the XI witnesses, but that's less important than overall code size, and we can work on fast-paths for that. The extra inhabitant count is stored in a 32-bit field (always present) following the ValueWitnessFlags, which now occupy a fixed 32 bits. This inflates non-XI VWTs on 32-bit targets by a word, but the net effect on XI VWTs is to shrink them by two words, which is likely to be the more important change. Also, being able to access the XI count directly should be a nice win.
254 lines
8.2 KiB
C++
254 lines
8.2 KiB
C++
//===--- KnownMetadata.cpp - Swift Language ABI Known Metadata 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Definitions of some builtin metadata objects.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/Runtime/Metadata.h"
|
|
#include "swift/Runtime/HeapObject.h"
|
|
#include "swift/Runtime/Numeric.h"
|
|
#include "MetadataImpl.h"
|
|
#include "Private.h"
|
|
#include <cstring>
|
|
#include <climits>
|
|
|
|
using namespace swift;
|
|
using namespace metadataimpl;
|
|
|
|
/// Copy a value from one object to another based on the size in the
|
|
/// given type metadata.
|
|
OpaqueValue *swift::swift_copyPOD(OpaqueValue *dest, OpaqueValue *src,
|
|
const Metadata *type) {
|
|
return (OpaqueValue*) memcpy(dest, src, type->getValueWitnesses()->size);
|
|
}
|
|
|
|
namespace {
|
|
// A type sized and aligned the way Swift wants Int128 (and Float80/Float128)
|
|
// to be sized and aligned.
|
|
struct alignas(16) int128_like {
|
|
char data[16];
|
|
};
|
|
|
|
static_assert(MaximumAlignment == 16, "max alignment was hardcoded");
|
|
struct alignas(16) int256_like {
|
|
char data[32];
|
|
};
|
|
struct alignas(16) int512_like {
|
|
char data[64];
|
|
};
|
|
|
|
struct alignas(16) float80_like {
|
|
char data[10];
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
namespace ctypes {
|
|
namespace {
|
|
// Type definitions that map each names Swift builtin type to their
|
|
// C counterparts.
|
|
using Bi1_ = uint8_t;
|
|
using Bi7_ = uint8_t;
|
|
using Bi8_ = uint8_t;
|
|
using Bi16_ = uint16_t;
|
|
using Bi32_ = uint32_t;
|
|
using Bi63_ = uint64_t;
|
|
using Bi64_ = uint64_t;
|
|
using Bi128_ = int128_like;
|
|
using Bi256_ = int256_like;
|
|
using Bi512_ = int512_like;
|
|
|
|
using Bw = intptr_t;
|
|
using BL_ = IntegerLiteral;
|
|
|
|
using Bf16_ = uint16_t;
|
|
using Bf32_ = float;
|
|
using Bf64_ = double;
|
|
using Bf80_ = float80_like;
|
|
using Bf128_ = int128_like;
|
|
|
|
/// The value-witness table for UnsafeValueBuffer. You can do layout
|
|
/// with this, but the type isn't copyable, so most of the value
|
|
/// operations are meaningless.
|
|
using BB = ValueBuffer;
|
|
}
|
|
}
|
|
|
|
namespace pointer_types {
|
|
namespace {
|
|
/// The basic value-witness table for Swift object pointers.
|
|
using Bo = SwiftRetainableBox;
|
|
|
|
/// The value-witness table for raw pointers.
|
|
using Bp = RawPointerBox;
|
|
|
|
/// The value-witness table for BridgeObject.
|
|
using Bb = BridgeObjectBox;
|
|
|
|
#if SWIFT_OBJC_INTEROP
|
|
/*** Objective-C pointers *************************************************/
|
|
|
|
// This section can reasonably be suppressed in builds that don't
|
|
// need to support Objective-C.
|
|
|
|
/// The basic value-witness table for ObjC object pointers.
|
|
using BO = ObjCRetainableBox;
|
|
#else
|
|
using BO = UnknownObjectRetainableBox;
|
|
#endif
|
|
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
template <typename T>
|
|
struct BuiltinType {
|
|
static constexpr const size_t Alignment = alignof(T);
|
|
};
|
|
|
|
#define SET_FIXED_ALIGNMENT(Type, Align) \
|
|
template <> \
|
|
struct BuiltinType<Type> { \
|
|
static constexpr const size_t Alignment = Align; \
|
|
};
|
|
|
|
SET_FIXED_ALIGNMENT(uint8_t, 1)
|
|
SET_FIXED_ALIGNMENT(uint16_t, 2)
|
|
SET_FIXED_ALIGNMENT(uint32_t, 4)
|
|
SET_FIXED_ALIGNMENT(uint64_t, 8)
|
|
SET_FIXED_ALIGNMENT(int128_like, 16)
|
|
static_assert(MaximumAlignment == 16, "max alignment was hardcoded");
|
|
SET_FIXED_ALIGNMENT(int256_like, 16)
|
|
SET_FIXED_ALIGNMENT(int512_like, 16)
|
|
|
|
#undef SET_FIXED_ALIGNMENT
|
|
|
|
template <typename T, unsigned N>
|
|
struct SIMDVector {
|
|
using Type = T __attribute__((__ext_vector_type__(N)));
|
|
};
|
|
|
|
template <>
|
|
struct SIMDVector<float, 3> {
|
|
using Type = float __attribute__((__ext_vector_type__(4)));
|
|
};
|
|
|
|
template <>
|
|
struct SIMDVector<double, 3> {
|
|
using Type = double __attribute__((__ext_vector_type__(4)));
|
|
};
|
|
|
|
template <typename T, unsigned N>
|
|
using SIMDVectorType = typename SIMDVector<T, N>::Type;
|
|
}
|
|
|
|
#define BUILTIN_TYPE(Symbol, Name) \
|
|
const ValueWitnessTable swift::VALUE_WITNESS_SYM(Symbol) = \
|
|
ValueWitnessTableForBox<NativeBox<ctypes::Symbol, \
|
|
BuiltinType<ctypes::Symbol>::Alignment>>::table;
|
|
|
|
#define BUILTIN_POINTER_TYPE(Symbol, Name) \
|
|
const ValueWitnessTable swift::VALUE_WITNESS_SYM(Symbol) = \
|
|
ValueWitnessTableForBox<pointer_types::Symbol>::table;
|
|
|
|
#define BUILTIN_VECTOR_TYPE(ElementSymbol, _, Width) \
|
|
const ValueWitnessTable \
|
|
swift::VALUE_WITNESS_SYM(VECTOR_BUILTIN_SYMBOL_NAME(ElementSymbol,Width)) = \
|
|
ValueWitnessTableForBox<NativeBox<SIMDVectorType<ctypes::ElementSymbol, \
|
|
Width>>>::table;
|
|
|
|
#include "swift/Runtime/BuiltinTypes.def"
|
|
|
|
/// The value-witness table for pointer-aligned unmanaged pointer types.
|
|
const ValueWitnessTable swift::METATYPE_VALUE_WITNESS_SYM(Bo) =
|
|
ValueWitnessTableForBox<PointerPointerBox>::table;
|
|
|
|
/*** Functions ***************************************************************/
|
|
|
|
namespace {
|
|
// @escaping function types.
|
|
struct ThickFunctionBox
|
|
: AggregateBox<FunctionPointerBox, SwiftRetainableBox> {
|
|
|
|
static constexpr unsigned numExtraInhabitants =
|
|
FunctionPointerBox::numExtraInhabitants;
|
|
|
|
static void storeExtraInhabitantTag(char *dest, unsigned tag) {
|
|
FunctionPointerBox::storeExtraInhabitantTag((void**) dest, tag);
|
|
}
|
|
|
|
static unsigned getExtraInhabitantTag(const char *src) {
|
|
return FunctionPointerBox::getExtraInhabitantTag((void * const *) src);
|
|
}
|
|
};
|
|
/// @noescape function types.
|
|
struct TrivialThickFunctionBox
|
|
: AggregateBox<FunctionPointerBox, RawPointerBox> {
|
|
|
|
static constexpr unsigned numExtraInhabitants =
|
|
FunctionPointerBox::numExtraInhabitants;
|
|
|
|
static void storeExtraInhabitantTag(char *dest, unsigned tag) {
|
|
FunctionPointerBox::storeExtraInhabitantTag((void **)dest, tag);
|
|
}
|
|
|
|
static unsigned getExtraInhabitantTag(const char *src) {
|
|
return FunctionPointerBox::getExtraInhabitantTag((void *const *)src);
|
|
}
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
/// The basic value-witness table for escaping function types.
|
|
const ValueWitnessTable
|
|
swift::VALUE_WITNESS_SYM(FUNCTION_MANGLING) =
|
|
ValueWitnessTableForBox<ThickFunctionBox>::table;
|
|
|
|
/// The basic value-witness table for @noescape function types.
|
|
const ValueWitnessTable
|
|
swift::VALUE_WITNESS_SYM(NOESCAPE_FUNCTION_MANGLING) =
|
|
ValueWitnessTableForBox<TrivialThickFunctionBox>::table;
|
|
|
|
/// The basic value-witness table for thin function types.
|
|
const ValueWitnessTable
|
|
swift::VALUE_WITNESS_SYM(THIN_FUNCTION_MANGLING) =
|
|
ValueWitnessTableForBox<FunctionPointerBox>::table;
|
|
|
|
/*** Empty tuples ************************************************************/
|
|
|
|
/// The basic value-witness table for empty types.
|
|
const ValueWitnessTable swift::VALUE_WITNESS_SYM(EMPTY_TUPLE_MANGLING) =
|
|
ValueWitnessTableForBox<AggregateBox<>>::table;
|
|
|
|
/*** Known metadata **********************************************************/
|
|
|
|
// Define some builtin opaque metadata.
|
|
#define OPAQUE_METADATA(TYPE) \
|
|
const FullOpaqueMetadata swift::METADATA_SYM(TYPE) = { \
|
|
{ &VALUE_WITNESS_SYM(TYPE) }, \
|
|
{ { MetadataKind::Opaque } } \
|
|
};
|
|
#define BUILTIN_TYPE(Symbol, Name) \
|
|
OPAQUE_METADATA(Symbol)
|
|
#include "swift/Runtime/BuiltinTypes.def"
|
|
|
|
/// The standard metadata for the empty tuple.
|
|
const FullMetadata<TupleTypeMetadata> swift::
|
|
METADATA_SYM(EMPTY_TUPLE_MANGLING) = {
|
|
{ &VALUE_WITNESS_SYM(EMPTY_TUPLE_MANGLING) }, // ValueWitnesses
|
|
{
|
|
{ MetadataKind::Tuple }, // Kind
|
|
0, // NumElements
|
|
nullptr // Labels
|
|
}
|
|
};
|
|
|