tree_test.cpp 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  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. #include "toolchain/parse/tree.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include <forward_list>
  8. #include "testing/base/test_raw_ostream.h"
  9. #include "toolchain/base/value_store.h"
  10. #include "toolchain/diagnostics/diagnostic_emitter.h"
  11. #include "toolchain/diagnostics/mocks.h"
  12. #include "toolchain/lex/lex.h"
  13. #include "toolchain/lex/tokenized_buffer.h"
  14. #include "toolchain/testing/yaml_test_helpers.h"
  15. namespace Carbon::Parse {
  16. namespace {
  17. using ::Carbon::Testing::TestRawOstream;
  18. using ::testing::ElementsAre;
  19. using ::testing::Pair;
  20. namespace Yaml = ::Carbon::Testing::Yaml;
  21. class TreeTest : public ::testing::Test {
  22. protected:
  23. auto GetSourceBuffer(llvm::StringRef t) -> SourceBuffer& {
  24. CARBON_CHECK(fs_.addFile("test.carbon", /*ModificationTime=*/0,
  25. llvm::MemoryBuffer::getMemBuffer(t)));
  26. source_storage_.push_front(std::move(
  27. *SourceBuffer::CreateFromFile(fs_, "test.carbon", consumer_)));
  28. return source_storage_.front();
  29. }
  30. auto GetTokenizedBuffer(llvm::StringRef t) -> Lex::TokenizedBuffer& {
  31. token_storage_.push_front(
  32. Lex::Lex(value_stores_, GetSourceBuffer(t), consumer_));
  33. return token_storage_.front();
  34. }
  35. SharedValueStores value_stores_;
  36. llvm::vfs::InMemoryFileSystem fs_;
  37. std::forward_list<SourceBuffer> source_storage_;
  38. std::forward_list<Lex::TokenizedBuffer> token_storage_;
  39. DiagnosticConsumer& consumer_ = ConsoleDiagnosticConsumer();
  40. };
  41. TEST_F(TreeTest, IsValid) {
  42. Lex::TokenizedBuffer& tokens = GetTokenizedBuffer("");
  43. Tree tree = Tree::Parse(tokens, consumer_, /*vlog_stream=*/nullptr);
  44. EXPECT_TRUE((*tree.postorder().begin()).is_valid());
  45. }
  46. TEST_F(TreeTest, PrintPostorderAsYAML) {
  47. Lex::TokenizedBuffer& tokens = GetTokenizedBuffer("fn F();");
  48. Tree tree = Tree::Parse(tokens, consumer_, /*vlog_stream=*/nullptr);
  49. EXPECT_FALSE(tree.has_errors());
  50. TestRawOstream print_stream;
  51. tree.Print(print_stream);
  52. auto file = Yaml::Sequence(ElementsAre(
  53. Yaml::Mapping(ElementsAre(Pair("kind", "FileStart"), Pair("text", ""))),
  54. Yaml::Mapping(
  55. ElementsAre(Pair("kind", "FunctionIntroducer"), Pair("text", "fn"))),
  56. Yaml::Mapping(
  57. ElementsAre(Pair("kind", "IdentifierName"), Pair("text", "F"))),
  58. Yaml::Mapping(
  59. ElementsAre(Pair("kind", "TuplePatternStart"), Pair("text", "("))),
  60. Yaml::Mapping(ElementsAre(Pair("kind", "TuplePattern"), Pair("text", ")"),
  61. Pair("subtree_size", "2"))),
  62. Yaml::Mapping(ElementsAre(Pair("kind", "FunctionDecl"), Pair("text", ";"),
  63. Pair("subtree_size", "5"))),
  64. Yaml::Mapping(ElementsAre(Pair("kind", "FileEnd"), Pair("text", "")))));
  65. auto root = Yaml::Sequence(ElementsAre(Yaml::Mapping(
  66. ElementsAre(Pair("filename", "test.carbon"), Pair("parse_tree", file)))));
  67. EXPECT_THAT(Yaml::Value::FromText(print_stream.TakeStr()),
  68. IsYaml(ElementsAre(root)));
  69. }
  70. TEST_F(TreeTest, PrintPreorderAsYAML) {
  71. Lex::TokenizedBuffer& tokens = GetTokenizedBuffer("fn F();");
  72. Tree tree = Tree::Parse(tokens, consumer_, /*vlog_stream=*/nullptr);
  73. EXPECT_FALSE(tree.has_errors());
  74. TestRawOstream print_stream;
  75. tree.Print(print_stream, /*preorder=*/true);
  76. auto param_list = Yaml::Sequence(ElementsAre(Yaml::Mapping(
  77. ElementsAre(Pair("node_index", "3"), Pair("kind", "TuplePatternStart"),
  78. Pair("text", "(")))));
  79. auto function_decl = Yaml::Sequence(ElementsAre(
  80. Yaml::Mapping(ElementsAre(Pair("node_index", "1"),
  81. Pair("kind", "FunctionIntroducer"),
  82. Pair("text", "fn"))),
  83. Yaml::Mapping(ElementsAre(Pair("node_index", "2"),
  84. Pair("kind", "IdentifierName"),
  85. Pair("text", "F"))),
  86. Yaml::Mapping(ElementsAre(Pair("node_index", "4"),
  87. Pair("kind", "TuplePattern"), Pair("text", ")"),
  88. Pair("subtree_size", "2"),
  89. Pair("children", param_list)))));
  90. auto file = Yaml::Sequence(ElementsAre(
  91. Yaml::Mapping(ElementsAre(Pair("node_index", "0"),
  92. Pair("kind", "FileStart"), Pair("text", ""))),
  93. Yaml::Mapping(ElementsAre(Pair("node_index", "5"),
  94. Pair("kind", "FunctionDecl"), Pair("text", ";"),
  95. Pair("subtree_size", "5"),
  96. Pair("children", function_decl))),
  97. Yaml::Mapping(ElementsAre(Pair("node_index", "6"),
  98. Pair("kind", "FileEnd"), Pair("text", "")))));
  99. auto root = Yaml::Sequence(ElementsAre(Yaml::Mapping(
  100. ElementsAre(Pair("filename", "test.carbon"), Pair("parse_tree", file)))));
  101. EXPECT_THAT(Yaml::Value::FromText(print_stream.TakeStr()),
  102. IsYaml(ElementsAre(root)));
  103. }
  104. TEST_F(TreeTest, HighRecursion) {
  105. std::string code = "fn Foo() { return ";
  106. code.append(10000, '(');
  107. code.append(10000, ')');
  108. code += "; }";
  109. Lex::TokenizedBuffer& tokens = GetTokenizedBuffer(code);
  110. ASSERT_FALSE(tokens.has_errors());
  111. Testing::MockDiagnosticConsumer consumer;
  112. Tree tree = Tree::Parse(tokens, consumer, /*vlog_stream=*/nullptr);
  113. EXPECT_FALSE(tree.has_errors());
  114. }
  115. } // namespace
  116. } // namespace Carbon::Parse