#ifndef LLDB_SOURCE_PLUGINS_EXPRESSIONPARSER_CLANG_CLANGASTIMPORTER_H
#define LLDB_SOURCE_PLUGINS_EXPRESSIONPARSER_CLANG_CLANGASTIMPORTER_H
#include <map>
#include <memory>
#include <set>
#include <vector>
#include "clang/AST/ASTContext.h"
#include "clang/AST/ASTImporter.h"
#include "clang/AST/CharUnits.h"
#include "clang/AST/Decl.h"
#include "clang/AST/DeclCXX.h"
#include "clang/Basic/FileManager.h"
#include "clang/Basic/FileSystemOptions.h"
#include "lldb/Host/FileSystem.h"
#include "lldb/Symbol/CompilerDeclContext.h"
#include "lldb/Utility/LLDBAssert.h"
#include "lldb/lldb-types.h"
#include "Plugins/ExpressionParser/Clang/CxxModuleHandler.h"
#include "llvm/ADT/DenseMap.h"
namespace lldb_private {
class ClangASTMetadata;
class TypeSystemClang;
class ClangASTImporter {
public:
struct LayoutInfo {
LayoutInfo() = default;
typedef llvm::DenseMap<const clang::CXXRecordDecl *, clang::CharUnits>
OffsetMap;
uint64_t bit_size = 0;
uint64_t alignment = 0;
llvm::DenseMap<const clang::FieldDecl *, uint64_t> field_offsets;
OffsetMap base_offsets;
OffsetMap vbase_offsets;
};
ClangASTImporter()
: m_file_manager(clang::FileSystemOptions(),
FileSystem::Instance().GetVirtualFileSystem()) {}
CompilerType CopyType(TypeSystemClang &dst, const CompilerType &src_type);
clang::Decl *CopyDecl(clang::ASTContext *dst_ctx, clang::Decl *decl);
CompilerType DeportType(TypeSystemClang &dst, const CompilerType &src_type);
clang::Decl *DeportDecl(clang::ASTContext *dst_ctx, clang::Decl *decl);
void SetRecordLayout(clang::RecordDecl *decl, const LayoutInfo &layout);
bool LayoutRecordType(
const clang::RecordDecl *record_decl, uint64_t &bit_size,
uint64_t &alignment,
llvm::DenseMap<const clang::FieldDecl *, uint64_t> &field_offsets,
llvm::DenseMap<const clang::CXXRecordDecl *, clang::CharUnits>
&base_offsets,
llvm::DenseMap<const clang::CXXRecordDecl *, clang::CharUnits>
&vbase_offsets);
bool importRecordLayoutFromOrigin(
const clang::RecordDecl *record, uint64_t &size, uint64_t &alignment,
llvm::DenseMap<const clang::FieldDecl *, uint64_t> &field_offsets,
llvm::DenseMap<const clang::CXXRecordDecl *, clang::CharUnits>
&base_offsets,
llvm::DenseMap<const clang::CXXRecordDecl *, clang::CharUnits>
&vbase_offsets);
bool CanImport(const CompilerType &type);
bool Import(const CompilerType &type);
bool CompleteType(const CompilerType &compiler_type);
bool CompleteTagDecl(clang::TagDecl *decl);
bool CompleteTagDeclWithOrigin(clang::TagDecl *decl, clang::TagDecl *origin);
bool CompleteObjCInterfaceDecl(clang::ObjCInterfaceDecl *interface_decl);
bool CompleteAndFetchChildren(clang::QualType type);
bool RequireCompleteType(clang::QualType type);
void SetDeclOrigin(const clang::Decl *decl, clang::Decl *original_decl);
ClangASTMetadata *GetDeclMetadata(const clang::Decl *decl);
typedef std::pair<lldb::ModuleSP, CompilerDeclContext> NamespaceMapItem;
typedef std::vector<NamespaceMapItem> NamespaceMap;
typedef std::shared_ptr<NamespaceMap> NamespaceMapSP;
void RegisterNamespaceMap(const clang::NamespaceDecl *decl,
NamespaceMapSP &namespace_map);
NamespaceMapSP GetNamespaceMap(const clang::NamespaceDecl *decl);
void BuildNamespaceMap(const clang::NamespaceDecl *decl);
class MapCompleter {
public:
virtual ~MapCompleter();
virtual void CompleteNamespaceMap(NamespaceMapSP &namespace_map,
ConstString name,
NamespaceMapSP &parent_map) const = 0;
};
void InstallMapCompleter(clang::ASTContext *dst_ctx,
MapCompleter &completer) {
ASTContextMetadataSP context_md;
ContextMetadataMap::iterator context_md_iter = m_metadata_map.find(dst_ctx);
if (context_md_iter == m_metadata_map.end()) {
context_md = ASTContextMetadataSP(new ASTContextMetadata(dst_ctx));
m_metadata_map[dst_ctx] = context_md;
} else {
context_md = context_md_iter->second;
}
context_md->m_map_completer = &completer;
}
void ForgetDestination(clang::ASTContext *dst_ctx);
void ForgetSource(clang::ASTContext *dst_ctx, clang::ASTContext *src_ctx);
struct DeclOrigin {
DeclOrigin() = default;
DeclOrigin(clang::ASTContext *_ctx, clang::Decl *_decl)
: ctx(_ctx), decl(_decl) {
assert(_decl == nullptr || &_decl->getASTContext() == _ctx);
}
DeclOrigin(const DeclOrigin &rhs) {
ctx = rhs.ctx;
decl = rhs.decl;
}
void operator=(const DeclOrigin &rhs) {
ctx = rhs.ctx;
decl = rhs.decl;
}
bool Valid() const { return (ctx != nullptr || decl != nullptr); }
clang::ASTContext *ctx = nullptr;
clang::Decl *decl = nullptr;
};
struct NewDeclListener {
virtual ~NewDeclListener() = default;
virtual void NewDeclImported(clang::Decl *from, clang::Decl *to) = 0;
};
struct ASTImporterDelegate : public clang::ASTImporter {
ASTImporterDelegate(ClangASTImporter &main, clang::ASTContext *target_ctx,
clang::ASTContext *source_ctx)
: clang::ASTImporter(*target_ctx, main.m_file_manager, *source_ctx,
main.m_file_manager, true ),
m_main(main), m_source_ctx(source_ctx) {
lldbassert(target_ctx != source_ctx && "Can't import into itself");
assert(target_ctx->getExternalSource() && "Missing ExternalSource");
setODRHandling(clang::ASTImporter::ODRHandlingType::Liberal);
}
class CxxModuleScope {
CxxModuleHandler m_handler;
ASTImporterDelegate &m_delegate;
bool m_valid = false;
public:
CxxModuleScope(ASTImporterDelegate &delegate, clang::ASTContext *dst_ctx)
: m_delegate(delegate) {
if (!delegate.m_std_handler) {
m_handler = CxxModuleHandler(delegate, dst_ctx);
m_valid = true;
delegate.m_std_handler = &m_handler;
}
}
~CxxModuleScope() {
if (m_valid) {
assert(m_delegate.m_std_handler == &m_handler);
m_delegate.m_std_handler = nullptr;
}
}
};
void ImportDefinitionTo(clang::Decl *to, clang::Decl *from);
void Imported(clang::Decl *from, clang::Decl *to) override;
clang::Decl *GetOriginalDecl(clang::Decl *To) override;
void SetImportListener(NewDeclListener *listener) {
assert(m_new_decl_listener == nullptr && "Already attached a listener?");
m_new_decl_listener = listener;
}
void RemoveImportListener() { m_new_decl_listener = nullptr; }
protected:
llvm::Expected<clang::Decl *> ImportImpl(clang::Decl *From) override;
private:
llvm::SmallPtrSet<clang::Decl *, 16> m_decls_to_ignore;
ClangASTImporter &m_main;
clang::ASTContext *m_source_ctx;
CxxModuleHandler *m_std_handler = nullptr;
NewDeclListener *m_new_decl_listener = nullptr;
};
typedef std::shared_ptr<ASTImporterDelegate> ImporterDelegateSP;
typedef llvm::DenseMap<clang::ASTContext *, ImporterDelegateSP> DelegateMap;
typedef llvm::DenseMap<const clang::NamespaceDecl *, NamespaceMapSP>
NamespaceMetaMap;
class ASTContextMetadata {
typedef llvm::DenseMap<const clang::Decl *, DeclOrigin> OriginMap;
public:
ASTContextMetadata(clang::ASTContext *dst_ctx) : m_dst_ctx(dst_ctx) {}
clang::ASTContext *m_dst_ctx;
DelegateMap m_delegates;
NamespaceMetaMap m_namespace_maps;
MapCompleter *m_map_completer = nullptr;
void setOrigin(const clang::Decl *decl, DeclOrigin origin) {
assert(&decl->getASTContext() != origin.ctx &&
"Trying to set decl origin to its own ASTContext?");
assert(decl != origin.decl && "Trying to set decl origin to itself?");
m_origins[decl] = origin;
}
void removeOrigin(const clang::Decl *decl) { m_origins.erase(decl); }
void removeOriginsWithContext(clang::ASTContext *ctx) {
for (OriginMap::iterator iter = m_origins.begin();
iter != m_origins.end();) {
if (iter->second.ctx == ctx)
m_origins.erase(iter++);
else
++iter;
}
}
DeclOrigin getOrigin(const clang::Decl *decl) const {
auto iter = m_origins.find(decl);
if (iter == m_origins.end())
return DeclOrigin();
return iter->second;
}
bool hasOrigin(const clang::Decl *decl) const {
return getOrigin(decl).Valid();
}
private:
OriginMap m_origins;
};
typedef std::shared_ptr<ASTContextMetadata> ASTContextMetadataSP;
typedef llvm::DenseMap<const clang::ASTContext *, ASTContextMetadataSP>
ContextMetadataMap;
ContextMetadataMap m_metadata_map;
ASTContextMetadataSP GetContextMetadata(clang::ASTContext *dst_ctx) {
ContextMetadataMap::iterator context_md_iter = m_metadata_map.find(dst_ctx);
if (context_md_iter == m_metadata_map.end()) {
ASTContextMetadataSP context_md =
ASTContextMetadataSP(new ASTContextMetadata(dst_ctx));
m_metadata_map[dst_ctx] = context_md;
return context_md;
}
return context_md_iter->second;
}
ASTContextMetadataSP MaybeGetContextMetadata(clang::ASTContext *dst_ctx) {
ContextMetadataMap::iterator context_md_iter = m_metadata_map.find(dst_ctx);
if (context_md_iter != m_metadata_map.end())
return context_md_iter->second;
return ASTContextMetadataSP();
}
ImporterDelegateSP GetDelegate(clang::ASTContext *dst_ctx,
clang::ASTContext *src_ctx) {
ASTContextMetadataSP context_md = GetContextMetadata(dst_ctx);
DelegateMap &delegates = context_md->m_delegates;
DelegateMap::iterator delegate_iter = delegates.find(src_ctx);
if (delegate_iter == delegates.end()) {
ImporterDelegateSP delegate =
ImporterDelegateSP(new ASTImporterDelegate(*this, dst_ctx, src_ctx));
delegates[src_ctx] = delegate;
return delegate;
}
return delegate_iter->second;
}
DeclOrigin GetDeclOrigin(const clang::Decl *decl);
clang::FileManager m_file_manager;
typedef llvm::DenseMap<const clang::RecordDecl *, LayoutInfo>
RecordDeclToLayoutMap;
RecordDeclToLayoutMap m_record_decl_to_layout_map;
};
template <class D> class TaggedASTDecl {
public:
TaggedASTDecl() : decl(nullptr) {}
TaggedASTDecl(D *_decl) : decl(_decl) {}
bool IsValid() const { return (decl != nullptr); }
bool IsInvalid() const { return !IsValid(); }
D *operator->() const { return decl; }
D *decl;
};
template <class D2, template <class D> class TD, class D1>
TD<D2> DynCast(TD<D1> source) {
return TD<D2>(llvm::dyn_cast<D2>(source.decl));
}
template <class D = clang::Decl> class DeclFromParser;
template <class D = clang::Decl> class DeclFromUser;
template <class D> class DeclFromParser : public TaggedASTDecl<D> {
public:
DeclFromParser() : TaggedASTDecl<D>() {}
DeclFromParser(D *_decl) : TaggedASTDecl<D>(_decl) {}
DeclFromUser<D> GetOrigin(ClangASTImporter &importer);
};
template <class D> class DeclFromUser : public TaggedASTDecl<D> {
public:
DeclFromUser() : TaggedASTDecl<D>() {}
DeclFromUser(D *_decl) : TaggedASTDecl<D>(_decl) {}
DeclFromParser<D> Import(clang::ASTContext *dest_ctx,
ClangASTImporter &importer);
};
template <class D>
DeclFromUser<D> DeclFromParser<D>::GetOrigin(ClangASTImporter &importer) {
ClangASTImporter::DeclOrigin origin = importer.GetDeclOrigin(this->decl);
if (!origin.Valid())
return DeclFromUser<D>();
return DeclFromUser<D>(llvm::dyn_cast<D>(origin.decl));
}
template <class D>
DeclFromParser<D> DeclFromUser<D>::Import(clang::ASTContext *dest_ctx,
ClangASTImporter &importer) {
DeclFromParser<> parser_generic_decl(importer.CopyDecl(dest_ctx, this->decl));
if (parser_generic_decl.IsInvalid())
return DeclFromParser<D>();
return DeclFromParser<D>(llvm::dyn_cast<D>(parser_generic_decl.decl));
}
}
#endif