Files
swift-mirror/include/swift/ABI/MetadataValues.h
Michael Gottesman e43570f1f0 Add MetadataKind.def, a header for metaprogramming with MetadataKinds
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
2015-04-25 19:04:55 +00:00

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