Split `Importer` into individual classes for each decl type.
This is one step towards a more query-based importer architecture, by
encapsulating the individual import methods and extracting an interface for
their interaction with the importer.
The extracted interfaces of the `DeclImporter` and `ImportContext` are not meant
to be permanent. They merely extract the status quo and provide a basis for
future improvements.
I also plan on moving individual `DeclImporter`s into their own files. Maybe
also the `Invocation` and the `ImportContext`. I haven't done that in this CL to
make review easier.
PiperOrigin-RevId: 443642590
diff --git a/rs_bindings_from_cc/importer.h b/rs_bindings_from_cc/importer.h
index 50aa64e..c2824c8 100644
--- a/rs_bindings_from_cc/importer.h
+++ b/rs_bindings_from_cc/importer.h
@@ -33,154 +33,133 @@
namespace crubit {
-// Iterates over the AST created from the invocation's entry headers and
-// creates an intermediate representation of the import (`IR`) into the
-// invocation object.
-class Importer {
+// Top-level parameters as well as return value of an importer invocation.
+class Invocation {
public:
- // Top-level parameters as well as return value of an importer invocation.
- class Invocation {
- public:
- Invocation(BazelLabel target, absl::Span<const HeaderName> entry_headers,
- const absl::flat_hash_map<const HeaderName, const BazelLabel>&
- header_targets)
- : target_(target),
- entry_headers_(entry_headers),
- lifetime_context_(
- std::make_shared<
- clang::tidy::lifetimes::LifetimeAnnotationContext>()),
- header_targets_(header_targets) {
- // Caller should verify that the inputs are non-empty.
- CRUBIT_CHECK(!entry_headers_.empty());
- CRUBIT_CHECK(!header_targets_.empty());
+ Invocation(BazelLabel target, absl::Span<const HeaderName> entry_headers,
+ const absl::flat_hash_map<const HeaderName, const BazelLabel>&
+ header_targets)
+ : target_(target),
+ entry_headers_(entry_headers),
+ lifetime_context_(std::make_shared<
+ clang::tidy::lifetimes::LifetimeAnnotationContext>()),
+ header_targets_(header_targets) {
+ // Caller should verify that the inputs are non-empty.
+ CRUBIT_CHECK(!entry_headers_.empty());
+ CRUBIT_CHECK(!header_targets_.empty());
- ir_.used_headers.insert(ir_.used_headers.end(), entry_headers_.begin(),
- entry_headers.end());
- ir_.current_target = target_;
- }
+ ir_.used_headers.insert(ir_.used_headers.end(), entry_headers_.begin(),
+ entry_headers.end());
+ ir_.current_target = target_;
+ }
- // Returns the target of a header, if any.
- std::optional<BazelLabel> header_target(const HeaderName header) const {
- auto it = header_targets_.find(header);
- return (it != header_targets_.end()) ? std::optional(it->second)
- : std::nullopt;
- }
+ // Returns the target of a header, if any.
+ std::optional<BazelLabel> header_target(const HeaderName header) const {
+ auto it = header_targets_.find(header);
+ return (it != header_targets_.end()) ? std::optional(it->second)
+ : std::nullopt;
+ }
- // The main target from which we are importing.
- const BazelLabel target_;
+ // The main target from which we are importing.
+ const BazelLabel target_;
- // The headers from which the import starts (a collection of
- // paths in the format suitable for a google3-relative quote include).
- const absl::Span<const HeaderName> entry_headers_;
+ // The headers from which the import starts (a collection of
+ // paths in the format suitable for a google3-relative quote include).
+ const absl::Span<const HeaderName> entry_headers_;
- const std::shared_ptr<clang::tidy::lifetimes::LifetimeAnnotationContext>
- lifetime_context_;
+ const std::shared_ptr<clang::tidy::lifetimes::LifetimeAnnotationContext>
+ lifetime_context_;
- // The main output of the import process
- IR ir_;
-
- private:
- const absl::flat_hash_map<const HeaderName, const BazelLabel>&
- header_targets_;
- };
-
- // The currently known canonical type decls that we know how to map into
- // Rust.
- class TypeMapper {
- public:
- TypeMapper(const clang::ASTContext* ctx) : ctx_(ctx) {}
-
- TypeMapper(const TypeMapper& other) = default;
- TypeMapper& operator=(const TypeMapper& other) = default;
-
- // Converts the Clang type `qual_type` into an equivalent `MappedType`.
- // Lifetimes for the type can optionally be specified using `lifetimes`.
- // If `qual_type` is a pointer type, `nullable` specifies whether the
- // pointer can be null.
- // TODO(b/209390498): Currently, we're able to specify nullability only for
- // top-level pointers. Extend this so that we can specify nullability for
- // all pointers contained in `qual_type`, in the same way that `lifetimes`
- // specifies lifetimes for all these pointers. Once this is done, make sure
- // that all callers pass in the appropriate information, derived from
- // nullability annotations.
- absl::StatusOr<MappedType> ConvertQualType(
- clang::QualType qual_type,
- std::optional<clang::tidy::lifetimes::ValueLifetimes>& lifetimes,
- bool nullable = true) const;
- absl::StatusOr<MappedType> ConvertType(
- const clang::Type* type,
- std::optional<clang::tidy::lifetimes::ValueLifetimes>& lifetimes,
- bool nullable) const;
- absl::StatusOr<MappedType> ConvertTypeDecl(
- const clang::TypeDecl* decl) const;
-
- bool Contains(const clang::TypeDecl* decl) const {
- return known_type_decls_.contains(
- clang::cast<clang::TypeDecl>(decl->getCanonicalDecl()));
- }
-
- void Insert(const clang::TypeDecl* decl) {
- known_type_decls_.insert(
- clang::cast<clang::TypeDecl>(decl->getCanonicalDecl()));
- }
-
- private:
- const clang::ASTContext* ctx_;
- absl::flat_hash_set<const clang::TypeDecl*> known_type_decls_;
- };
-
- explicit Importer(Invocation& invocation, clang::ASTContext& ctx,
- clang::Sema& sema)
- : invocation_(invocation),
- ctx_(ctx),
- sema_(sema),
- mangler_(CRUBIT_DIE_IF_NULL(ctx_.createMangleContext())),
- type_mapper_(&ctx) {}
-
- // Import all visible declarations from a translation unit.
- void Import(clang::TranslationUnitDecl* decl);
+ // The main output of the import process
+ IR ir_;
private:
+ const absl::flat_hash_map<const HeaderName, const BazelLabel>&
+ header_targets_;
+};
+
+// The currently known canonical type decls that we know how to map into
+// Rust.
+class TypeMapper {
+ public:
+ TypeMapper(const clang::ASTContext* ctx) : ctx_(ctx) {}
+
+ TypeMapper(const TypeMapper& other) = default;
+ TypeMapper& operator=(const TypeMapper& other) = default;
+
+ // Converts the Clang type `qual_type` into an equivalent `MappedType`.
+ // Lifetimes for the type can optionally be specified using `lifetimes`.
+ // If `qual_type` is a pointer type, `nullable` specifies whether the
+ // pointer can be null.
+ // TODO(b/209390498): Currently, we're able to specify nullability only for
+ // top-level pointers. Extend this so that we can specify nullability for
+ // all pointers contained in `qual_type`, in the same way that `lifetimes`
+ // specifies lifetimes for all these pointers. Once this is done, make sure
+ // that all callers pass in the appropriate information, derived from
+ // nullability annotations.
+ absl::StatusOr<MappedType> ConvertQualType(
+ clang::QualType qual_type,
+ std::optional<clang::tidy::lifetimes::ValueLifetimes>& lifetimes,
+ bool nullable = true) const;
+ absl::StatusOr<MappedType> ConvertType(
+ const clang::Type* type,
+ std::optional<clang::tidy::lifetimes::ValueLifetimes>& lifetimes,
+ bool nullable) const;
+ absl::StatusOr<MappedType> ConvertTypeDecl(const clang::TypeDecl* decl) const;
+ std::optional<absl::string_view> MapKnownCcTypeToRsType(
+ absl::string_view cc_type) const;
+
+ bool Contains(const clang::TypeDecl* decl) const {
+ return known_type_decls_.contains(
+ clang::cast<clang::TypeDecl>(decl->getCanonicalDecl()));
+ }
+
+ void Insert(const clang::TypeDecl* decl) {
+ known_type_decls_.insert(
+ clang::cast<clang::TypeDecl>(decl->getCanonicalDecl()));
+ }
+
+ private:
+ const clang::ASTContext* ctx_;
+ absl::flat_hash_set<const clang::TypeDecl*> known_type_decls_;
+};
+
+// Explicitly defined interface that defines how `DeclImporter`s are allowed to
+// interface with the global state of the importer.
+class ImportContext {
+ public:
+ ImportContext(Invocation& invocation, clang::ASTContext& ctx,
+ clang::Sema& sema)
+ : invocation_(invocation), ctx_(ctx), sema_(sema), type_mapper_(&ctx){};
+ virtual ~ImportContext(){};
+
// Imports all decls contained in a `DeclContext`.
- void ImportDeclsFromDeclContext(const clang::DeclContext* decl_context);
+ virtual void ImportDeclsFromDeclContext(
+ const clang::DeclContext* decl_context) = 0;
- // Returns the Item of a Decl, importing it first if necessary.
- std::optional<IR::Item> GetDeclItem(clang::Decl* decl);
+ // Imports an unsupported item with a single error message.
+ virtual IR::Item ImportUnsupportedItem(const clang::Decl* decl,
+ std::string error) = 0;
- // Imports a decl and creates an IR item (or error messages).
- // Does not use or update the cache.
- std::optional<IR::Item> ImportDecl(clang::Decl* decl);
+ // Imports an unsupported item with multiple error messages.
+ virtual IR::Item ImportUnsupportedItem(const clang::Decl* decl,
+ std::set<std::string> errors) = 0;
- // These functions import specific `Decl` subtypes. They use `LookupDecl` to
- // lookup dependencies. They don't use or update the cache themselves.
- std::optional<IR::Item> ImportNamespace(clang::NamespaceDecl* namespace_decl);
- std::optional<IR::Item> ImportFunction(
- clang::FunctionDecl* function_decl,
- clang::FunctionTemplateDecl* function_template_decl = nullptr);
- std::optional<IR::Item> ImportRecord(clang::CXXRecordDecl* record_decl);
- std::optional<IR::Item> ImportTypedefName(
- clang::TypedefNameDecl* typedef_name_decl);
- std::optional<IR::Item> ImportEnum(clang::EnumDecl* enum_decl);
-
- IR::Item ImportUnsupportedItem(const clang::Decl* decl, std::string error);
- IR::Item ImportUnsupportedItem(const clang::Decl* decl,
- std::set<std::string> errors);
-
- absl::StatusOr<std::vector<Field>> ImportFields(
- clang::CXXRecordDecl* record_decl);
- // Stores the comments of this target in source order.
- void ImportFreeComments();
// Returns the item ids of the children and comments of the given decl in
// source order. This method assumes that the children decls have already been
// imported.
- std::vector<ItemId> GetItemIdsInSourceOrder(clang::Decl* decl);
+ virtual std::vector<ItemId> GetItemIdsInSourceOrder(clang::Decl* decl) = 0;
- std::string GetMangledName(const clang::NamedDecl* named_decl) const;
- BazelLabel GetOwningTarget(const clang::Decl* decl) const;
+ // Mangles the name of a named decl.
+ virtual std::string GetMangledName(
+ const clang::NamedDecl* named_decl) const = 0;
+
+ // Returs the label of the target that contains a decl.
+ virtual BazelLabel GetOwningTarget(const clang::Decl* decl) const = 0;
// Checks if the given decl belongs to the current target. Does not look into
// other redeclarations of the decl.
- bool IsFromCurrentTarget(const clang::Decl* decl) const;
+ virtual bool IsFromCurrentTarget(const clang::Decl* decl) const = 0;
// Gets an IR UnqualifiedIdentifier for the named decl.
//
@@ -190,36 +169,184 @@
// a SpecialName.
//
// If the translated name is not yet implemented, this returns null.
- std::optional<UnqualifiedIdentifier> GetTranslatedName(
- const clang::NamedDecl* named_decl) const;
+ virtual std::optional<UnqualifiedIdentifier> GetTranslatedName(
+ const clang::NamedDecl* named_decl) const = 0;
// GetTranslatedName, but only for identifier names. This is the common case.
+ virtual std::optional<Identifier> GetTranslatedIdentifier(
+ const clang::NamedDecl* named_decl) const = 0;
+
+ // Gets the doc comment of the declaration.
+ virtual llvm::Optional<std::string> GetComment(
+ const clang::Decl* decl) const = 0;
+
+ // Converts a Clang source location to IR.
+ virtual SourceLoc ConvertSourceLocation(clang::SourceLocation loc) const = 0;
+
+ Invocation& invocation_;
+ clang::ASTContext& ctx_;
+ clang::Sema& sema_;
+ TypeMapper type_mapper_;
+};
+
+// Interface for components that can import decls of a certain category.
+class DeclImporter {
+ public:
+ DeclImporter(ImportContext& ictx) : ictx_(ictx){};
+ virtual ~DeclImporter(){};
+
+ // Determines whether this importer is autoritative for a decl. This does not
+ // imply that the import will be succesful.
+ virtual bool CanImport(clang::Decl*) = 0;
+
+ // Returns an IR item for a decl, or `std::nullopt` if importing failed.
+ // This member function may only be called after `CanImport` returned `true`.
+ virtual std::optional<IR::Item> ImportDecl(clang::Decl*) = 0;
+
+ protected:
+ ImportContext& ictx_;
+};
+
+// Common implementation for defining `DeclImporter`s that determine their
+// applicability by the dynamic type of the decl.
+template <typename D>
+class DeclImporterBase : public DeclImporter {
+ public:
+ DeclImporterBase(ImportContext& context) : DeclImporter(context) {}
+
+ protected:
+ bool CanImport(clang::Decl* decl) { return clang::isa<D>(decl); }
+ std::optional<IR::Item> ImportDecl(clang::Decl* decl) {
+ return Import(clang::cast<D>(decl));
+ }
+ virtual std::optional<IR::Item> Import(D*);
+};
+
+// TODO(forster): Move those implementations into separate files.
+
+// A `DeclImporter` for `ClassTemplateDecl`s.
+class ClassTemplateDeclImporter
+ : public DeclImporterBase<clang::ClassTemplateDecl> {
+ public:
+ ClassTemplateDeclImporter(ImportContext& context)
+ : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::ClassTemplateDecl*);
+};
+
+// A `DeclImporter` for `CXXRecordDecl`s.
+class CXXRecordDeclImporter : public DeclImporterBase<clang::CXXRecordDecl> {
+ public:
+ CXXRecordDeclImporter(ImportContext& context) : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::CXXRecordDecl*);
+
+ private:
+ absl::StatusOr<std::vector<Field>> ImportFields(clang::CXXRecordDecl*);
+ std::vector<BaseClass> GetUnambiguousPublicBases(
+ const clang::CXXRecordDecl& record_decl) const;
+};
+
+// A `DeclImporter` for `EnumDecl`s.
+class EnumDeclImporter : public DeclImporterBase<clang::EnumDecl> {
+ public:
+ EnumDeclImporter(ImportContext& context) : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::EnumDecl*);
+};
+
+// A `DeclImporter` for `FunctionTemplateDecl`s.
+class FunctionTemplateDeclImporter
+ : public DeclImporterBase<clang::FunctionTemplateDecl> {
+ public:
+ FunctionTemplateDeclImporter(ImportContext& context)
+ : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::FunctionTemplateDecl*);
+};
+
+// A `DeclImporter` for `FunctionDecl`s.
+class FunctionDeclImporter : public DeclImporterBase<clang::FunctionDecl> {
+ public:
+ FunctionDeclImporter(ImportContext& context) : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::FunctionDecl*);
+};
+
+// A `DeclImporter` for `NamespaceDecl`s.
+class NamespaceDeclImporter : public DeclImporterBase<clang::NamespaceDecl> {
+ public:
+ NamespaceDeclImporter(ImportContext& context) : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::NamespaceDecl*);
+};
+
+// A `DeclImporter` for `TypedefNameDecl`s.
+class TypedefNameDeclImporter
+ : public DeclImporterBase<clang::TypedefNameDecl> {
+ public:
+ TypedefNameDeclImporter(ImportContext& context) : DeclImporterBase(context){};
+ std::optional<IR::Item> Import(clang::TypedefNameDecl*);
+};
+
+// Iterates over the AST created from the invocation's entry headers and
+// creates an intermediate representation of the import (`IR`) into the
+// invocation object.
+class Importer : public ImportContext {
+ public:
+ explicit Importer(Invocation& invocation, clang::ASTContext& ctx,
+ clang::Sema& sema)
+ : ImportContext(invocation, ctx, sema),
+ mangler_(CRUBIT_DIE_IF_NULL(ctx_.createMangleContext())) {
+ decl_importers_.push_back(
+ std::make_unique<ClassTemplateDeclImporter>(*this));
+ decl_importers_.push_back(std::make_unique<CXXRecordDeclImporter>(*this));
+ decl_importers_.push_back(std::make_unique<EnumDeclImporter>(*this));
+ decl_importers_.push_back(std::make_unique<FunctionDeclImporter>(*this));
+ decl_importers_.push_back(
+ std::make_unique<FunctionTemplateDeclImporter>(*this));
+ decl_importers_.push_back(std::make_unique<NamespaceDeclImporter>(*this));
+ decl_importers_.push_back(std::make_unique<TypedefNameDeclImporter>(*this));
+ }
+
+ // Import all visible declarations from a translation unit.
+ void Import(clang::TranslationUnitDecl* decl);
+
+ protected:
+ // Implementation of `ImportContext`
+ void ImportDeclsFromDeclContext(
+ const clang::DeclContext* decl_context) override;
+ IR::Item ImportUnsupportedItem(const clang::Decl* decl,
+ std::string error) override;
+ IR::Item ImportUnsupportedItem(const clang::Decl* decl,
+ std::set<std::string> errors) override;
+ std::vector<ItemId> GetItemIdsInSourceOrder(clang::Decl* decl) override;
+ std::string GetMangledName(const clang::NamedDecl* named_decl) const override;
+ BazelLabel GetOwningTarget(const clang::Decl* decl) const override;
+ bool IsFromCurrentTarget(const clang::Decl* decl) const override;
+ std::optional<UnqualifiedIdentifier> GetTranslatedName(
+ const clang::NamedDecl* named_decl) const override;
std::optional<Identifier> GetTranslatedIdentifier(
- const clang::NamedDecl* named_decl) const {
+ const clang::NamedDecl* named_decl) const override {
if (std::optional<UnqualifiedIdentifier> name =
GetTranslatedName(named_decl)) {
return std::move(*std::get_if<Identifier>(&*name));
}
return std::nullopt;
}
+ llvm::Optional<std::string> GetComment(
+ const clang::Decl* decl) const override;
+ SourceLoc ConvertSourceLocation(clang::SourceLocation loc) const override;
- // Gets the doc comment of the declaration.
- llvm::Optional<std::string> GetComment(const clang::Decl* decl) const;
+ private:
+ // Returns the Item of a Decl, importing it first if necessary.
+ std::optional<IR::Item> GetDeclItem(clang::Decl* decl);
- SourceLoc ConvertSourceLocation(clang::SourceLocation loc) const;
+ // Imports a decl and creates an IR item (or error messages).
+ // Does not use or update the cache.
+ std::optional<IR::Item> ImportDecl(clang::Decl* decl);
- std::vector<BaseClass> GetUnambiguousPublicBases(
- const clang::CXXRecordDecl& record_decl) const;
+ // Stores the comments of this target in source order.
+ void ImportFreeComments();
- Invocation& invocation_;
-
- clang::ASTContext& ctx_;
- clang::Sema& sema_;
-
+ std::vector<std::unique_ptr<DeclImporter>> decl_importers_;
std::unique_ptr<clang::MangleContext> mangler_;
absl::flat_hash_map<const clang::Decl*, std::optional<IR::Item>>
import_cache_;
- TypeMapper type_mapper_;
std::vector<const clang::RawComment*> comments_;
}; // class Importer