mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
165 lines
5.7 KiB
C++
165 lines
5.7 KiB
C++
//===--- SILDefaultWitnessTable.h -------------------------------*- 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines the SILDefaultWitnessTable class, which is used to provide
|
|
// default implementations of protocol requirements for resilient protocols,
|
|
// allowing IRGen to generate the appropriate metadata so that the runtime can
|
|
// insert those requirements to witness tables that were emitted prior to the
|
|
// requirement being added.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SWIFT_SIL_SILDEFAULTWITNESSTABLE_H
|
|
#define SWIFT_SIL_SILDEFAULTWITNESSTABLE_H
|
|
|
|
#include "swift/SIL/SILAllocated.h"
|
|
#include "swift/SIL/SILDeclRef.h"
|
|
#include "swift/SIL/SILFunction.h"
|
|
#include "swift/SIL/SILWitnessTable.h"
|
|
#include "llvm/ADT/ilist_node.h"
|
|
#include "llvm/ADT/ilist.h"
|
|
#include <string>
|
|
|
|
namespace swift {
|
|
|
|
class ProtocolDecl;
|
|
class SILFunction;
|
|
class SILModule;
|
|
|
|
/// A mapping from each requirement of a protocol to the SIL-level entity
|
|
/// satisfying the requirement for conformances which do not explicitly
|
|
/// provide a witness.
|
|
class SILDefaultWitnessTable : public llvm::ilist_node<SILDefaultWitnessTable>,
|
|
public SILAllocated<SILDefaultWitnessTable>
|
|
{
|
|
public:
|
|
/// A default witness table entry describing the default witness for a
|
|
/// requirement.
|
|
using Entry = SILWitnessTable::Entry;
|
|
|
|
private:
|
|
/// The module which contains the SILDefaultWitnessTable.
|
|
SILModule &Mod;
|
|
|
|
/// The linkage of the witness table.
|
|
SILLinkage Linkage;
|
|
|
|
/// The protocol declaration to which this default witness table applies.
|
|
const ProtocolDecl *Protocol;
|
|
|
|
/// The minimum size of a valid witness table conforming to this protocol,
|
|
/// with all resilient default requirements omitted.
|
|
unsigned MinimumWitnessTableSizeInWords;
|
|
|
|
/// The various witnesses containing in this default witness table.
|
|
MutableArrayRef<Entry> Entries;
|
|
|
|
/// Temporary state while SILGen is emitting a default witness table.
|
|
/// We can never have a true declaration since there's no way to reference
|
|
/// the default witness table from outside its defining translation unit.
|
|
bool IsDeclaration;
|
|
|
|
/// Private constructor for making SILDefaultWitnessTable declarations.
|
|
SILDefaultWitnessTable(SILModule &M, SILLinkage Linkage,
|
|
const ProtocolDecl *Protocol);
|
|
|
|
/// Private constructor for making SILDefaultWitnessTable definitions.
|
|
SILDefaultWitnessTable(SILModule &M, SILLinkage Linkage,
|
|
const ProtocolDecl *Protocol,
|
|
ArrayRef<Entry> entries);
|
|
|
|
void addDefaultWitnessTable();
|
|
|
|
public:
|
|
/// Create a new SILDefaultWitnessTable declaration.
|
|
static SILDefaultWitnessTable *create(SILModule &M, SILLinkage Linkage,
|
|
const ProtocolDecl *Protocol);
|
|
|
|
/// Create a new SILDefaultWitnessTable definition with the given entries.
|
|
static SILDefaultWitnessTable *create(SILModule &M, SILLinkage Linkage,
|
|
const ProtocolDecl *Protocol,
|
|
ArrayRef<Entry> entries);
|
|
|
|
/// Get a name that uniquely identifies this default witness table.
|
|
///
|
|
/// Note that this is /not/ valid as a symbol name; it is only guaranteed to
|
|
/// be unique among default witness tables, not all symbols.
|
|
std::string getUniqueName() const;
|
|
|
|
/// Get the linkage of the default witness table.
|
|
SILLinkage getLinkage() const { return Linkage; }
|
|
|
|
void convertToDefinition(ArrayRef<Entry> entries);
|
|
|
|
~SILDefaultWitnessTable();
|
|
|
|
SILModule &getModule() const { return Mod; }
|
|
|
|
/// Return true if this is a declaration with no body.
|
|
bool isDeclaration() const { return IsDeclaration; }
|
|
|
|
/// Return the AST ProtocolDecl this default witness table is associated with.
|
|
const ProtocolDecl *getProtocol() const { return Protocol; }
|
|
|
|
/// Clears methods in witness entries.
|
|
/// \p predicate Returns true if the passed entry should be set to null.
|
|
template <typename Predicate> void clearMethods_if(Predicate predicate) {
|
|
for (Entry &entry : Entries) {
|
|
if (!entry.isValid())
|
|
continue;
|
|
if (entry.getKind() != SILWitnessTable::Method)
|
|
continue;
|
|
|
|
auto *MW = entry.getMethodWitness().Witness;
|
|
if (MW && predicate(MW)) {
|
|
entry.removeWitnessMethod();
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Return all of the default witness table entries.
|
|
ArrayRef<Entry> getEntries() const { return Entries; }
|
|
|
|
/// Verify that the default witness table is well-formed.
|
|
void verify(const SILModule &M) const;
|
|
|
|
/// Print the default witness table.
|
|
void print(llvm::raw_ostream &OS, bool Verbose = false) const;
|
|
|
|
/// Dump the default witness table to stderr.
|
|
void dump() const;
|
|
};
|
|
|
|
} // end swift namespace
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// ilist_traits for SILDefaultWitnessTable
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
namespace llvm {
|
|
|
|
template <>
|
|
struct ilist_traits<::swift::SILDefaultWitnessTable> :
|
|
public ilist_node_traits<::swift::SILDefaultWitnessTable> {
|
|
using SILDefaultWitnessTable = ::swift::SILDefaultWitnessTable;
|
|
|
|
public:
|
|
static void deleteNode(SILDefaultWitnessTable *WT) { WT->~SILDefaultWitnessTable(); }
|
|
|
|
private:
|
|
void createNode(const SILDefaultWitnessTable &);
|
|
};
|
|
|
|
} // end llvm namespace
|
|
|
|
#endif
|