semantics_ir_factory.h 3.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #ifndef CARBON_TOOLCHAIN_SEMANTICS_SEMANTICS_IR_FACTORY_H_
  5. #define CARBON_TOOLCHAIN_SEMANTICS_SEMANTICS_IR_FACTORY_H_
  6. #include "common/check.h"
  7. #include "toolchain/parser/parse_tree.h"
  8. #include "toolchain/semantics/semantics_ir.h"
  9. namespace Carbon {
  10. // The main semantic analysis entry.
  11. class SemanticsIRFactory {
  12. public:
  13. // Builds the SemanticsIR without doing any substantial semantic analysis.
  14. static auto Build(const TokenizedBuffer& tokens, const ParseTree& parse_tree)
  15. -> SemanticsIR;
  16. private:
  17. struct TraversalStackEntry {
  18. ParseTree::Node parse_node;
  19. llvm::Optional<SemanticsNodeId> result_id;
  20. };
  21. explicit SemanticsIRFactory(const TokenizedBuffer& tokens,
  22. const ParseTree& parse_tree)
  23. : tokens_(&tokens), semantics_(parse_tree) {}
  24. auto Build() -> void;
  25. auto Push(ParseTree::Node parse_node) -> void {
  26. node_stack_.push_back({parse_node, llvm::None});
  27. }
  28. auto Push(ParseTree::Node parse_node, SemanticsNode node) -> void {
  29. auto node_id = semantics_.AddNode(node);
  30. node_stack_.push_back({parse_node, node_id});
  31. }
  32. auto Pop(ParseNodeKind pop_parse_kind) -> void {
  33. auto back = node_stack_.back();
  34. auto parse_kind = parse_tree().node_kind(back.parse_node);
  35. CARBON_CHECK(parse_kind == pop_parse_kind)
  36. << "Expected " << pop_parse_kind << ", found " << parse_kind;
  37. CARBON_CHECK(!back.result_id) << "Expected no result ID on " << parse_kind;
  38. node_stack_.pop_back();
  39. }
  40. auto PopWithResult() -> SemanticsNodeId {
  41. auto back = node_stack_.back();
  42. auto node_id = *back.result_id;
  43. node_stack_.pop_back();
  44. return node_id;
  45. }
  46. auto PopWithResult(ParseNodeKind pop_parse_kind) -> SemanticsNodeId {
  47. auto back = node_stack_.back();
  48. auto parse_kind = parse_tree().node_kind(back.parse_node);
  49. auto node_id = *back.result_id;
  50. CARBON_CHECK(parse_kind == pop_parse_kind)
  51. << "Expected " << pop_parse_kind << ", found " << parse_kind;
  52. node_stack_.pop_back();
  53. return node_id;
  54. }
  55. // Parse node handlers.
  56. auto HandleDeclaredName(ParseTree::Node parse_node) -> void;
  57. auto HandleFunctionDefinition(ParseTree::Node parse_node) -> void;
  58. auto HandleFunctionDefinitionStart(ParseTree::Node parse_node) -> void;
  59. auto HandleInfixOperator(ParseTree::Node parse_node) -> void;
  60. auto HandleLiteral(ParseTree::Node parse_node) -> void;
  61. auto HandleParameterList(ParseTree::Node parse_node) -> void;
  62. auto HandleReturnStatement(ParseTree::Node parse_node) -> void;
  63. // Convenience accessor.
  64. auto parse_tree() -> const ParseTree& { return *semantics_.parse_tree_; }
  65. // Tokens for getting data on literals.
  66. const TokenizedBuffer* tokens_;
  67. // The SemanticsIR being constructed.
  68. SemanticsIR semantics_;
  69. // The stack during Build. Will contain file-level parse nodes on return.
  70. llvm::SmallVector<TraversalStackEntry> node_stack_;
  71. };
  72. } // namespace Carbon
  73. #endif // CARBON_TOOLCHAIN_SEMANTICS_SEMANTICS_IR_FACTORY_H_