// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #ifndef CARBON_EXPLORER_AST_IMPL_BINDING_H_ #define CARBON_EXPLORER_AST_IMPL_BINDING_H_ #include #include "common/check.h" #include "common/ostream.h" #include "explorer/ast/ast_node.h" #include "explorer/ast/expression_category.h" namespace Carbon { class Value; class Expression; class GenericBinding; // `ImplBinding` plays the role of the parameter for passing witness // tables to a generic. However, unlike regular parameters // (`BindingPattern`) there is no explicit syntax that corresponds to // an `ImplBinding`, so they are not created during parsing. Instances // of `ImplBinding` are created during type checking, when processing // a type parameter (a `GenericBinding`), or an `impls` requirement in // a `where` clause. class ImplBinding : public AstNode { public: using ImplementsCarbonValueNode = void; ImplBinding(SourceLocation source_loc, Nonnull type_var, std::optional> iface) : AstNode(AstNodeKind::ImplBinding, source_loc), type_var_(type_var), iface_(iface) {} explicit ImplBinding(CloneContext& context, const ImplBinding& other); static auto classof(const AstNode* node) -> bool { return InheritsFromImplBinding(node->kind()); } void Print(llvm::raw_ostream& out) const override; void PrintID(llvm::raw_ostream& out) const override; // The binding for the type variable. auto type_var() const -> Nonnull { return type_var_; } // The constraint being implemented. // TODO: Rename this to `constraint`. auto interface() const -> Nonnull { CARBON_CHECK(iface_, "interface has not been set yet"); return *iface_; } // Set the interface being implemented, if not set by the constructor. Should // only be called by typechecking. void set_interface(Nonnull iface) { CARBON_CHECK(!iface_, "interface set twice"); iface_ = iface; } // Required for the ValueNode interface auto constant_value() const -> std::optional> { return std::nullopt; } auto symbolic_identity() const -> std::optional> { return symbolic_identity_; } void set_symbolic_identity(Nonnull value) { CARBON_CHECK(!symbolic_identity_.has_value()); symbolic_identity_ = value; } // These functions exist only so that an `ImplBinding` can be used as a // `ValueNodeView` as a key in a `StaticScope`. auto static_type() const -> const Value& { CARBON_FATAL("an ImplBinding has no type"); } auto expression_category() const -> ExpressionCategory { return ExpressionCategory::Value; } // Return the original impl binding. auto original() const -> Nonnull { if (original_.has_value()) { return *original_; } else { return this; } } // Set the original impl binding. void set_original(Nonnull orig) { original_ = orig; } private: Nonnull type_var_; std::optional> iface_; std::optional> symbolic_identity_; std::optional> original_; }; } // namespace Carbon #endif // CARBON_EXPLORER_AST_IMPL_BINDING_H_