//===--- EnumMetadataLayout.h - CRTP for enum metadata ----------*- C++ -*-===// // // 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 // //===----------------------------------------------------------------------===// // // A CRTP class useful for laying out enum metadata. // //===----------------------------------------------------------------------===// #ifndef SWIFT_IRGEN_ENUMMETADATALAYOUT_H #define SWIFT_IRGEN_ENUMMETADATALAYOUT_H #include "MetadataLayout.h" namespace swift { namespace irgen { /// A CRTP class for laying out enum metadata. /// /// This produces an object corresponding to the EnumMetadata type. /// It does not itself doing anything special for metadata templates. template class EnumMetadataLayout : public MetadataLayout { typedef MetadataLayout super; protected: using super::IGM; using super::asImpl; /// The Enum. EnumDecl *const Target; EnumMetadataLayout(IRGenModule &IGM, EnumDecl *target) : super(IGM), Target(target) {} public: void layout() { // Metadata header. super::layout(); // EnumMetadata header. asImpl().addNominalTypeDescriptor(); asImpl().addParentMetadataRef(); // If changing this layout, you must update the magic number in // emitParentMetadataRef. // Instantiation-specific. // Add fields for generic cases. asImpl().addGenericFields(Target, Target->getDeclaredTypeInContext()); // Reserve a word to cache the payload size if the type has dynamic layout. auto &strategy = getEnumImplStrategy(IGM, Target->DeclContext::getDeclaredTypeInContext()->getCanonicalType()); if (strategy.needsPayloadSizeInMetadata()) asImpl().addPayloadSize(); } }; /// An "implementation" of EnumMetadataLayout that just scans through /// the metadata layout, maintaining the next index: the offset (in /// pointer-sized chunks) into the metadata for the next field. template class EnumMetadataScanner : public EnumMetadataLayout { typedef EnumMetadataLayout super; protected: Size NextOffset = Size(0); EnumMetadataScanner(IRGenModule &IGM, EnumDecl *target) : super(IGM, target) {} public: void addMetadataFlags() { addPointer(); } void addValueWitnessTable() { addPointer(); } void addNominalTypeDescriptor() { addPointer(); } void addParentMetadataRef() { addPointer(); } void addGenericArgument(CanType argument) { addPointer(); } void addGenericWitnessTable(CanType argument, ProtocolConformanceRef conf) { addPointer(); } void addPayloadSize() { addPointer(); } private: void addPointer() { NextOffset += super::IGM.getPointerSize(); } }; } // end namespace irgen } // end namespace swift #endif