mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
and use it to update LeaksChecker in a robust way to handle new metadata kinds. I also fixed a small typo where the native ErrorType was not included in the range of non-ObjC isa metadata kinds. Swift SVN r27718
409 lines
13 KiB
C++
409 lines
13 KiB
C++
//===--- MetadataValues.h - Compiler/runtime ABI Metadata -------*- C++ -*-===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This header is shared between the runtime and the compiler and
|
|
// includes target-independent information which can be usefully shared
|
|
// between them.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SWIFT_ABI_METADATAVALUES_H
|
|
#define SWIFT_ABI_METADATAVALUES_H
|
|
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
|
|
namespace swift {
|
|
|
|
/// Kinds of Swift metadata records. Some of these are types, some
|
|
/// aren't.
|
|
enum class MetadataKind : uintptr_t {
|
|
#define METADATAKIND(name, value) name = value,
|
|
#define ABSTRACTMETADATAKIND(name, start, end) \
|
|
name##_Start = start, name##_End = end,
|
|
#include "MetadataKind.def"
|
|
};
|
|
|
|
/// Kinds of Swift nominal type descriptor records.
|
|
enum class NominalTypeKind : uintptr_t {
|
|
#define NOMINALTYPEMETADATAKIND(name, value) name = value,
|
|
#include "MetadataKind.def"
|
|
};
|
|
|
|
/// Flags for dynamic-cast operations.
|
|
enum class DynamicCastFlags : size_t {
|
|
/// All flags clear.
|
|
Default = 0x0,
|
|
|
|
/// True if the cast is not permitted to fail.
|
|
Unconditional = 0x1,
|
|
|
|
/// True if the cast should 'take' the source value on success;
|
|
/// false if the value should be copied.
|
|
TakeOnSuccess = 0x2,
|
|
|
|
/// True if the cast should destroy the source value on failure;
|
|
/// false if the value should be left in place.
|
|
DestroyOnFailure = 0x4,
|
|
};
|
|
inline bool operator&(DynamicCastFlags a, DynamicCastFlags b) {
|
|
return (size_t(a) & size_t(b)) != 0;
|
|
}
|
|
inline DynamicCastFlags operator|(DynamicCastFlags a, DynamicCastFlags b) {
|
|
return DynamicCastFlags(size_t(a) | size_t(b));
|
|
}
|
|
inline DynamicCastFlags operator-(DynamicCastFlags a, DynamicCastFlags b) {
|
|
return DynamicCastFlags(size_t(a) & ~size_t(b));
|
|
}
|
|
inline DynamicCastFlags &operator|=(DynamicCastFlags &a, DynamicCastFlags b) {
|
|
return a = (a | b);
|
|
}
|
|
|
|
/// Swift class flags.
|
|
enum class ClassFlags : uint32_t {
|
|
/// Is this a Swift 1 class?
|
|
IsSwift1 = 0x1,
|
|
|
|
/// Does this class use Swift 1.0 refcounting?
|
|
UsesSwift1Refcounting = 0x2,
|
|
};
|
|
inline bool operator&(ClassFlags a, ClassFlags b) {
|
|
return (uint32_t(a) & uint32_t(b)) != 0;
|
|
}
|
|
inline ClassFlags operator|(ClassFlags a, ClassFlags b) {
|
|
return ClassFlags(uint32_t(a) | uint32_t(b));
|
|
}
|
|
inline ClassFlags &operator|=(ClassFlags &a, ClassFlags b) {
|
|
return a = (a | b);
|
|
}
|
|
|
|
enum : unsigned {
|
|
/// Number of words reserved in generic metadata patterns.
|
|
NumGenericMetadataPrivateDataWords = 16,
|
|
};
|
|
|
|
/// Kinds of protocol conformance record.
|
|
enum class ProtocolConformanceTypeKind : unsigned {
|
|
/// The conformance is universal and might apply to any type.
|
|
/// getDirectType() is nil.
|
|
Universal,
|
|
|
|
/// The conformance is for a nongeneric native struct or enum type.
|
|
/// getDirectType() points to the canonical metadata for the type.
|
|
UniqueDirectType,
|
|
|
|
/// The conformance is for a nongeneric foreign struct or enum type.
|
|
/// getDirectType() points to a nonunique metadata record for the type, which
|
|
/// needs to be uniqued by the runtime.
|
|
NonuniqueDirectType,
|
|
|
|
/// The conformance is for a nongeneric class type.
|
|
/// getIndirectClass() points to a variable that contains the pointer to the
|
|
/// class object, which may be ObjC and thus require a runtime call to get
|
|
/// metadata.
|
|
///
|
|
/// On platforms without ObjC interop, this indirection isn't necessary,
|
|
/// and classes could be emitted as UniqueDirectType.
|
|
UniqueIndirectClass,
|
|
|
|
/// The conformance is for a generic type.
|
|
/// getGenericPattern() points to the generic metadata pattern used to
|
|
/// form instances of the type.
|
|
UniqueGenericPattern,
|
|
|
|
/// The conformance is for a nongeneric class type.
|
|
/// getDirectType() points to the unique class object.
|
|
///
|
|
/// FIXME: This shouldn't exist. On ObjC interop platforms, class references
|
|
/// must be indirected (using UniqueIndirectClass). On non-ObjC interop
|
|
/// platforms, the class object always is the type metadata.
|
|
UniqueDirectClass = 0xF,
|
|
};
|
|
|
|
/// Kinds of reference to protocol conformance.
|
|
enum class ProtocolConformanceReferenceKind : unsigned {
|
|
/// A direct reference to a protocol witness table.
|
|
WitnessTable,
|
|
/// A function pointer that can be called to access the protocol witness
|
|
/// table.
|
|
WitnessTableAccessor,
|
|
};
|
|
|
|
struct ProtocolConformanceFlags {
|
|
private:
|
|
using int_type = unsigned;
|
|
int_type Data;
|
|
|
|
enum : int_type {
|
|
TypeKindMask = 0x0000000FU,
|
|
TypeKindShift = 0,
|
|
ConformanceKindMask = 0x00000010U,
|
|
ConformanceKindShift = 4,
|
|
};
|
|
|
|
public:
|
|
constexpr ProtocolConformanceFlags() : Data(0) {}
|
|
constexpr ProtocolConformanceFlags(int_type Data) : Data(Data) {}
|
|
|
|
constexpr ProtocolConformanceTypeKind getTypeKind() const {
|
|
return ProtocolConformanceTypeKind((Data >> TypeKindShift) & TypeKindMask);
|
|
}
|
|
constexpr ProtocolConformanceFlags withTypeKind(
|
|
ProtocolConformanceTypeKind ptk) const {
|
|
return ProtocolConformanceFlags(
|
|
(Data & ~TypeKindMask) | (int_type(ptk) << TypeKindShift));
|
|
}
|
|
|
|
constexpr ProtocolConformanceReferenceKind getConformanceKind() const {
|
|
return ProtocolConformanceReferenceKind((Data >> ConformanceKindShift)
|
|
& ConformanceKindMask);
|
|
}
|
|
constexpr ProtocolConformanceFlags withConformanceKind(
|
|
ProtocolConformanceReferenceKind pck) const {
|
|
return ProtocolConformanceFlags(
|
|
(Data & ~ConformanceKindMask) | (int_type(pck) << ConformanceKindShift));
|
|
}
|
|
|
|
int_type getValue() const { return Data; }
|
|
};
|
|
|
|
/// Flag that indicates whether an existential type is class-constrained or not.
|
|
enum class ProtocolClassConstraint : bool {
|
|
/// The protocol is class-constrained, so only class types can conform to it.
|
|
///
|
|
/// This must be 0 for ABI compatibility with Objective-C protocol_t records.
|
|
Class = false,
|
|
/// Any type can conform to the protocol.
|
|
Any = true,
|
|
};
|
|
|
|
/// Identifiers for protocols with special meaning to the Swift runtime.
|
|
enum class SpecialProtocol: uint8_t {
|
|
/// Not a special protocol.
|
|
///
|
|
/// This must be 0 for ABI compatibility with Objective-C protocol_t records.
|
|
None = 0,
|
|
/// The AnyObject protocol.
|
|
AnyObject = 1,
|
|
/// The ErrorType protocol.
|
|
ErrorType = 2,
|
|
};
|
|
|
|
/// Identifiers for protocol method dispatch strategies.
|
|
enum class ProtocolDispatchStrategy: uint8_t {
|
|
/// Uses ObjC method dispatch.
|
|
///
|
|
/// This must be 0 for ABI compatibility with Objective-C protocol_t records.
|
|
ObjC = 0,
|
|
|
|
/// Uses Swift protocol witness table dispatch.
|
|
///
|
|
/// To invoke methods of this protocol, a pointer to a protocol witness table
|
|
/// corresponding to the protocol conformance must be available.
|
|
Swift = 1,
|
|
|
|
/// The protocol guarantees that it has no methods to dispatch. It requires
|
|
/// neither Objective-C metadata nor a witness table.
|
|
Empty = 2,
|
|
};
|
|
|
|
/// Flags for protocol descriptors.
|
|
class ProtocolDescriptorFlags {
|
|
typedef uint32_t int_type;
|
|
enum : int_type {
|
|
IsSwift = 1U << 0U,
|
|
ClassConstraint = 1U << 1U,
|
|
|
|
DispatchStrategyMask = 0xFU << 2U,
|
|
DispatchStrategyShift = 2,
|
|
|
|
SpecialProtocolMask = 0x7F000000U,
|
|
SpecialProtocolShift = 24,
|
|
|
|
/// Reserved by the ObjC runtime.
|
|
_ObjC_FixedUp = 1U << 31U,
|
|
};
|
|
|
|
int_type Data;
|
|
|
|
constexpr ProtocolDescriptorFlags(int_type Data) : Data(Data) {}
|
|
public:
|
|
constexpr ProtocolDescriptorFlags() : Data(0) {}
|
|
constexpr ProtocolDescriptorFlags withSwift(bool s) const {
|
|
return ProtocolDescriptorFlags((Data & ~IsSwift) | (s ? IsSwift : 0));
|
|
}
|
|
constexpr ProtocolDescriptorFlags withClassConstraint(
|
|
ProtocolClassConstraint c) const {
|
|
return ProtocolDescriptorFlags((Data & ~ClassConstraint)
|
|
| (bool(c) ? ClassConstraint : 0));
|
|
}
|
|
constexpr ProtocolDescriptorFlags withDispatchStrategy(
|
|
ProtocolDispatchStrategy s) const {
|
|
return ProtocolDescriptorFlags((Data & ~DispatchStrategyMask)
|
|
| (int_type(s) << DispatchStrategyShift));
|
|
}
|
|
constexpr ProtocolDescriptorFlags
|
|
withSpecialProtocol(SpecialProtocol sp) const {
|
|
return ProtocolDescriptorFlags((Data & ~SpecialProtocolMask)
|
|
| (int_type(sp) << SpecialProtocolShift));
|
|
}
|
|
|
|
/// Was the protocol defined in Swift?
|
|
bool isSwift() const { return Data & IsSwift; }
|
|
|
|
/// Is the protocol class-constrained?
|
|
ProtocolClassConstraint getClassConstraint() const {
|
|
return ProtocolClassConstraint(bool(Data & ClassConstraint));
|
|
}
|
|
|
|
/// What dispatch strategy does this protocol use?
|
|
ProtocolDispatchStrategy getDispatchStrategy() const {
|
|
return ProtocolDispatchStrategy((Data & DispatchStrategyMask)
|
|
>> DispatchStrategyShift);
|
|
}
|
|
|
|
/// Does the protocol require a witness table for method dispatch?
|
|
bool needsWitnessTable() const {
|
|
return needsWitnessTable(getDispatchStrategy());
|
|
}
|
|
|
|
static bool needsWitnessTable(ProtocolDispatchStrategy strategy) {
|
|
switch (strategy) {
|
|
case ProtocolDispatchStrategy::ObjC:
|
|
case ProtocolDispatchStrategy::Empty:
|
|
return false;
|
|
case ProtocolDispatchStrategy::Swift:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/// Return the identifier if this is a special runtime-known protocol.
|
|
SpecialProtocol getSpecialProtocol() const {
|
|
return SpecialProtocol(uint8_t((Data & SpecialProtocolMask)
|
|
>> SpecialProtocolShift));
|
|
}
|
|
|
|
int_type getIntValue() const {
|
|
return Data;
|
|
}
|
|
};
|
|
|
|
/// Flags in an existential type metadata record.
|
|
class ExistentialTypeFlags {
|
|
typedef size_t int_type;
|
|
enum : int_type {
|
|
NumWitnessTablesMask = 0x00FFFFFFU,
|
|
ClassConstraintMask = 0x80000000U,
|
|
SpecialProtocolMask = 0x7F000000U,
|
|
SpecialProtocolShift = 24U,
|
|
};
|
|
int_type Data;
|
|
|
|
constexpr ExistentialTypeFlags(int_type Data) : Data(Data) {}
|
|
public:
|
|
constexpr ExistentialTypeFlags() : Data(0) {}
|
|
constexpr ExistentialTypeFlags withNumWitnessTables(unsigned numTables) const {
|
|
return ExistentialTypeFlags((Data & ~NumWitnessTablesMask) | numTables);
|
|
}
|
|
constexpr ExistentialTypeFlags
|
|
withClassConstraint(ProtocolClassConstraint c) const {
|
|
return ExistentialTypeFlags((Data & ~ClassConstraintMask)
|
|
| (bool(c) ? ClassConstraintMask : 0));
|
|
}
|
|
constexpr ExistentialTypeFlags
|
|
withSpecialProtocol(SpecialProtocol sp) const {
|
|
return ExistentialTypeFlags((Data & ~SpecialProtocolMask)
|
|
| (int_type(sp) << SpecialProtocolShift));
|
|
}
|
|
|
|
unsigned getNumWitnessTables() const {
|
|
return Data & NumWitnessTablesMask;
|
|
}
|
|
|
|
ProtocolClassConstraint getClassConstraint() const {
|
|
return ProtocolClassConstraint(bool(Data & ClassConstraintMask));
|
|
}
|
|
|
|
/// Return whether this existential type represents an uncomposed special
|
|
/// protocol.
|
|
SpecialProtocol getSpecialProtocol() const {
|
|
return SpecialProtocol(uint8_t((Data & SpecialProtocolMask)
|
|
>> SpecialProtocolShift));
|
|
}
|
|
|
|
int_type getIntValue() const {
|
|
return Data;
|
|
}
|
|
};
|
|
|
|
/// Convention values for function type metadata.
|
|
enum class FunctionMetadataConvention: uint8_t {
|
|
Swift = 0,
|
|
Block = 1,
|
|
Thin = 2,
|
|
CFunctionPointer = 3,
|
|
};
|
|
|
|
/// Flags in a function type metadata record.
|
|
class FunctionTypeFlags {
|
|
typedef size_t int_type;
|
|
enum : int_type {
|
|
NumArgumentsMask = 0x00FFFFFFU,
|
|
ConventionMask = 0x0F000000U,
|
|
ConventionShift = 24U,
|
|
ThrowsMask = 0x10000000U,
|
|
};
|
|
int_type Data;
|
|
|
|
constexpr FunctionTypeFlags(int_type Data) : Data(Data) {}
|
|
public:
|
|
constexpr FunctionTypeFlags() : Data(0) {}
|
|
|
|
constexpr FunctionTypeFlags withNumArguments(unsigned numArguments) const {
|
|
return FunctionTypeFlags((Data & ~NumArgumentsMask) | numArguments);
|
|
}
|
|
|
|
constexpr FunctionTypeFlags withConvention(FunctionMetadataConvention c) const {
|
|
return FunctionTypeFlags((Data & ~ConventionMask)
|
|
| (int_type(c) << ConventionShift));
|
|
}
|
|
|
|
constexpr FunctionTypeFlags withThrows(bool throws) const {
|
|
return FunctionTypeFlags((Data & ~ThrowsMask)
|
|
| (throws ? ThrowsMask : 0));
|
|
}
|
|
|
|
unsigned getNumArguments() const {
|
|
return Data & NumArgumentsMask;
|
|
}
|
|
|
|
FunctionMetadataConvention getConvention() const {
|
|
return FunctionMetadataConvention((Data&ConventionMask) >> ConventionShift);
|
|
}
|
|
|
|
bool throws() const {
|
|
return bool(Data & ThrowsMask);
|
|
}
|
|
|
|
int_type getIntValue() const {
|
|
return Data;
|
|
}
|
|
|
|
static FunctionTypeFlags fromIntValue(int_type Data) {
|
|
return FunctionTypeFlags(Data);
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|