// 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 EXECUTABLE_SEMANTICS_AST_GENERIC_BINDING_H_ #define EXECUTABLE_SEMANTICS_AST_GENERIC_BINDING_H_ #include #include "common/check.h" #include "common/ostream.h" #include "executable_semantics/ast/ast_node.h" #include "executable_semantics/ast/pattern.h" #include "executable_semantics/ast/value_category.h" namespace Carbon { class Value; class Expression; class ImplBinding; // The run-time counterpart of a `GenericBinding`. // // Once a generic binding has been declared, it can be used // in two different ways: as a compile-time constant with a // symbolic value (such as a `VariableType`), or as a run-time // variable with a concrete value that is stored on the stack. // An `ImplBinding` is used in contexts where the second // interpretation is intended. class ImplBinding : public AstNode { public: using ImplementsCarbonValueNode = void; ImplBinding(SourceLocation source_loc, Nonnull type_var, Nonnull iface) : AstNode(AstNodeKind::ImplBinding, source_loc), type_var_(type_var), iface_(iface) {} 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 interface being implemented. auto interface() const -> Nonnull { return iface_; } // Required for the the ValueNode interface auto constant_value() const -> std::optional> { return std::nullopt; } auto symbolic_identity() const -> std::optional> { return std::nullopt; } // The static type of the impl. Cannot be called before typechecking. auto static_type() const -> const Value& { return **static_type_; } // Sets the static type of the impl. Can only be called once, during // typechecking. void set_static_type(Nonnull type) { CHECK(!static_type_.has_value()); static_type_ = type; } auto value_category() const -> ValueCategory { return ValueCategory::Let; } private: Nonnull type_var_; Nonnull iface_; std::optional> static_type_; }; } // namespace Carbon #endif // EXECUTABLE_SEMANTICS_AST_GENERIC_BINDING_H_