// 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_TOOLCHAIN_SEMANTICS_SEMANTICS_NODE_H_ #define CARBON_TOOLCHAIN_SEMANTICS_SEMANTICS_NODE_H_ #include #include "common/check.h" #include "common/ostream.h" #include "toolchain/semantics/semantics_builtin_kind.h" #include "toolchain/semantics/semantics_node_kind.h" namespace Carbon { // Type-safe storage of Node IDs. struct SemanticsNodeId { static constexpr int32_t CrossReferenceBit = 0x8000'0000; // Constructs a cross-reference node ID. static auto MakeCrossReference(int32_t id) -> SemanticsNodeId { return SemanticsNodeId(id | CrossReferenceBit); } // Constructs a cross-reference node ID for a builtin. This relies on // SemanticsIR guarantees for builtin cross-reference placement. static auto MakeBuiltinReference(SemanticsBuiltinKind kind) -> SemanticsNodeId { return MakeCrossReference(kind.AsInt()); } SemanticsNodeId() : id(-1) {} explicit SemanticsNodeId(int32_t id) : id(id) {} SemanticsNodeId(SemanticsNodeId const&) = default; auto operator=(const SemanticsNodeId& other) -> SemanticsNodeId& = default; auto is_cross_reference() const -> bool { return id & CrossReferenceBit; } // Returns the ID for a cross-reference, just handling removal of the marker // bit. auto GetAsCrossReference() const -> int32_t { return id & ~CrossReferenceBit; } auto Print(llvm::raw_ostream& out) const -> void { if (is_cross_reference()) { out << "node_xref" << GetAsCrossReference(); } else { out << "node" << id; } } int32_t id; }; // Type-safe storage of identifiers. struct SemanticsIdentifierId { SemanticsIdentifierId() : id(-1) {} explicit SemanticsIdentifierId(int32_t id) : id(id) {} auto Print(llvm::raw_ostream& out) const -> void { out << "ident" << id; } int32_t id; }; // Type-safe storage of integer literals. struct SemanticsIntegerLiteralId { SemanticsIntegerLiteralId() : id(-1) {} explicit SemanticsIntegerLiteralId(int32_t id) : id(id) {} auto Print(llvm::raw_ostream& out) const -> void { out << "int" << id; } int32_t id; }; // Type-safe storage of node blocks. struct SemanticsNodeBlockId { SemanticsNodeBlockId() : id(-1) {} explicit SemanticsNodeBlockId(int32_t id) : id(id) {} auto Print(llvm::raw_ostream& out) const -> void { out << "block" << id; } int32_t id; }; // The standard structure for nodes. class SemanticsNode { public: struct NoArgs {}; auto GetAsInvalid() const -> NoArgs { CARBON_FATAL() << "Invalid access"; } static auto MakeBinaryOperatorAdd(SemanticsNodeId lhs, SemanticsNodeId rhs) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::BinaryOperatorAdd(), SemanticsNodeId(), lhs.id, rhs.id); } auto GetAsBinaryOperatorAdd() const -> std::pair { CARBON_CHECK(kind_ == SemanticsNodeKind::BinaryOperatorAdd()); return {SemanticsNodeId(arg0_), SemanticsNodeId(arg1_)}; } static auto MakeBindName(SemanticsIdentifierId name, SemanticsNodeId node) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::BindName(), SemanticsNodeId(), name.id, node.id); } auto GetAsBindName() const -> std::pair { CARBON_CHECK(kind_ == SemanticsNodeKind::BindName()); return {SemanticsIdentifierId(arg0_), SemanticsNodeId(arg1_)}; } static auto MakeBuiltin(SemanticsBuiltinKind builtin_kind, SemanticsNodeId type) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::Builtin(), type, builtin_kind.AsInt()); } auto GetAsBuiltin() const -> SemanticsBuiltinKind { CARBON_CHECK(kind_ == SemanticsNodeKind::Builtin()); return SemanticsBuiltinKind::FromInt(arg0_); } static auto MakeCodeBlock(SemanticsNodeBlockId node_block) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::CodeBlock(), SemanticsNodeId(), node_block.id); } auto GetAsCodeBlock() const -> SemanticsNodeBlockId { CARBON_CHECK(kind_ == SemanticsNodeKind::CodeBlock()); return SemanticsNodeBlockId(arg0_); } // TODO: The signature should be added as a parameter. static auto MakeFunctionDeclaration() -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::FunctionDeclaration(), SemanticsNodeId()); } auto GetAsFunctionDeclaration() const -> NoArgs { CARBON_CHECK(kind_ == SemanticsNodeKind::FunctionDeclaration()); return {}; } static auto MakeFunctionDefinition(SemanticsNodeId decl, SemanticsNodeBlockId node_block) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::FunctionDefinition(), SemanticsNodeId(), decl.id, node_block.id); } auto GetAsFunctionDefinition() const -> std::pair { CARBON_CHECK(kind_ == SemanticsNodeKind::FunctionDefinition()); return {SemanticsNodeId(arg0_), SemanticsNodeBlockId(arg1_)}; } static auto MakeIntegerLiteral(SemanticsIntegerLiteralId integer) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::IntegerLiteral(), SemanticsNodeId::MakeBuiltinReference( SemanticsBuiltinKind::IntegerLiteralType()), integer.id); } auto GetAsIntegerLiteral() const -> SemanticsIntegerLiteralId { CARBON_CHECK(kind_ == SemanticsNodeKind::IntegerLiteral()); return SemanticsIntegerLiteralId(arg0_); } static auto MakeReturn() -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::Return(), SemanticsNodeId()); } auto GetAsReturn() const -> NoArgs { CARBON_CHECK(kind_ == SemanticsNodeKind::Return()); return {}; } static auto MakeReturnExpression(SemanticsNodeId expr) -> SemanticsNode { return SemanticsNode(SemanticsNodeKind::ReturnExpression(), SemanticsNodeId(), expr.id); } auto GetAsReturnExpression() const -> SemanticsNodeId { CARBON_CHECK(kind_ == SemanticsNodeKind::ReturnExpression()); return SemanticsNodeId(arg0_); } SemanticsNode() : SemanticsNode(SemanticsNodeKind::Invalid(), SemanticsNodeId()) {} auto kind() -> SemanticsNodeKind { return kind_; } auto type() -> SemanticsNodeId { return type_; } auto Print(llvm::raw_ostream& out) const -> void; private: explicit SemanticsNode(SemanticsNodeKind kind, SemanticsNodeId type, int32_t arg0 = -1, int32_t arg1 = -1) : kind_(kind), type_(type), arg0_(arg0), arg1_(arg1) {} SemanticsNodeKind kind_; SemanticsNodeId type_; int32_t arg0_; int32_t arg1_; }; // TODO: This is currently 16 bytes because we sometimes have 2 arguments for a // pair of SemanticsNodes. However, SemanticsNodeKind is 1 byte; if args // were 3.5 bytes, we could potentially shrink SemanticsNode by 4 bytes. This // may be worth investigating further. static_assert(sizeof(SemanticsNode) == 16, "Unexpected SemanticsNode size"); } // namespace Carbon #endif // CARBON_TOOLCHAIN_SEMANTICS_SEMANTICS_NODE_H_