precedence_test.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  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/precedence.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include "toolchain/lex/token_kind.h"
  8. namespace Carbon::Testing {
  9. namespace {
  10. using Parse::Associativity;
  11. using Parse::OperatorPriority;
  12. using Parse::PrecedenceGroup;
  13. using ::testing::Eq;
  14. TEST(PrecedenceTest, OperatorsAreRecognized) {
  15. EXPECT_TRUE(PrecedenceGroup::ForLeading(Lex::TokenKind::Minus).has_value());
  16. EXPECT_TRUE(PrecedenceGroup::ForLeading(Lex::TokenKind::Caret).has_value());
  17. EXPECT_FALSE(PrecedenceGroup::ForLeading(Lex::TokenKind::Slash).has_value());
  18. EXPECT_FALSE(
  19. PrecedenceGroup::ForLeading(Lex::TokenKind::Identifier).has_value());
  20. EXPECT_FALSE(PrecedenceGroup::ForLeading(Lex::TokenKind::Tilde).has_value());
  21. EXPECT_TRUE(
  22. PrecedenceGroup::ForTrailing(Lex::TokenKind::Minus, false).has_value());
  23. EXPECT_TRUE(
  24. PrecedenceGroup::ForTrailing(Lex::TokenKind::Caret, false).has_value());
  25. EXPECT_FALSE(
  26. PrecedenceGroup::ForTrailing(Lex::TokenKind::Tilde, false).has_value());
  27. EXPECT_TRUE(
  28. PrecedenceGroup::ForTrailing(Lex::TokenKind::Slash, true).has_value());
  29. EXPECT_FALSE(PrecedenceGroup::ForTrailing(Lex::TokenKind::Identifier, false)
  30. .has_value());
  31. EXPECT_TRUE(
  32. PrecedenceGroup::ForTrailing(Lex::TokenKind::Minus, true)->is_binary);
  33. EXPECT_FALSE(PrecedenceGroup::ForTrailing(Lex::TokenKind::MinusMinus, false)
  34. .has_value());
  35. }
  36. TEST(PrecedenceTest, InfixVsPostfix) {
  37. // A trailing `-` is always binary, even when written with whitespace that
  38. // suggests it's postfix.
  39. EXPECT_TRUE(
  40. PrecedenceGroup::ForTrailing(Lex::TokenKind::Minus, /*infix*/ false)
  41. ->is_binary);
  42. // A trailing `*` is interpreted based on context.
  43. EXPECT_TRUE(
  44. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->is_binary);
  45. EXPECT_FALSE(
  46. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, false)->is_binary);
  47. // Infix `*` can appear in `+` contexts; postfix `*` cannot.
  48. EXPECT_THAT(
  49. PrecedenceGroup::GetPriority(
  50. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->level,
  51. PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)->level),
  52. Eq(OperatorPriority::LeftFirst));
  53. EXPECT_THAT(
  54. PrecedenceGroup::GetPriority(
  55. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, false)->level,
  56. PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)->level),
  57. Eq(OperatorPriority::Ambiguous));
  58. }
  59. TEST(PrecedenceTest, Associativity) {
  60. EXPECT_THAT(
  61. PrecedenceGroup::ForLeading(Lex::TokenKind::Minus)->GetAssociativity(),
  62. Eq(Associativity::None));
  63. EXPECT_THAT(
  64. PrecedenceGroup::ForLeading(Lex::TokenKind::Star)->GetAssociativity(),
  65. Eq(Associativity::RightToLeft));
  66. EXPECT_THAT(PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)
  67. ->level.GetAssociativity(),
  68. Eq(Associativity::LeftToRight));
  69. EXPECT_THAT(PrecedenceGroup::ForTrailing(Lex::TokenKind::Equal, true)
  70. ->level.GetAssociativity(),
  71. Eq(Associativity::None));
  72. }
  73. TEST(PrecedenceTest, DirectRelations) {
  74. EXPECT_THAT(
  75. PrecedenceGroup::GetPriority(
  76. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->level,
  77. PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)->level),
  78. Eq(OperatorPriority::LeftFirst));
  79. EXPECT_THAT(
  80. PrecedenceGroup::GetPriority(
  81. PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)->level,
  82. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->level),
  83. Eq(OperatorPriority::RightFirst));
  84. EXPECT_THAT(
  85. PrecedenceGroup::GetPriority(
  86. PrecedenceGroup::ForTrailing(Lex::TokenKind::Amp, true)->level,
  87. PrecedenceGroup::ForTrailing(Lex::TokenKind::Less, true)->level),
  88. Eq(OperatorPriority::LeftFirst));
  89. EXPECT_THAT(
  90. PrecedenceGroup::GetPriority(
  91. PrecedenceGroup::ForTrailing(Lex::TokenKind::Less, true)->level,
  92. PrecedenceGroup::ForTrailing(Lex::TokenKind::Amp, true)->level),
  93. Eq(OperatorPriority::RightFirst));
  94. }
  95. TEST(PrecedenceTest, IndirectRelations) {
  96. EXPECT_THAT(
  97. PrecedenceGroup::GetPriority(
  98. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->level,
  99. PrecedenceGroup::ForTrailing(Lex::TokenKind::Or, true)->level),
  100. Eq(OperatorPriority::LeftFirst));
  101. EXPECT_THAT(
  102. PrecedenceGroup::GetPriority(
  103. PrecedenceGroup::ForTrailing(Lex::TokenKind::Or, true)->level,
  104. PrecedenceGroup::ForTrailing(Lex::TokenKind::Star, true)->level),
  105. Eq(OperatorPriority::RightFirst));
  106. EXPECT_THAT(
  107. PrecedenceGroup::GetPriority(
  108. *PrecedenceGroup::ForLeading(Lex::TokenKind::Caret),
  109. PrecedenceGroup::ForTrailing(Lex::TokenKind::Equal, true)->level),
  110. Eq(OperatorPriority::LeftFirst));
  111. EXPECT_THAT(
  112. PrecedenceGroup::GetPriority(
  113. PrecedenceGroup::ForTrailing(Lex::TokenKind::Equal, true)->level,
  114. *PrecedenceGroup::ForLeading(Lex::TokenKind::Caret)),
  115. Eq(OperatorPriority::RightFirst));
  116. }
  117. TEST(PrecedenceTest, IncomparableOperators) {
  118. EXPECT_THAT(PrecedenceGroup::GetPriority(
  119. *PrecedenceGroup::ForLeading(Lex::TokenKind::Caret),
  120. *PrecedenceGroup::ForLeading(Lex::TokenKind::Not)),
  121. Eq(OperatorPriority::Ambiguous));
  122. EXPECT_THAT(PrecedenceGroup::GetPriority(
  123. *PrecedenceGroup::ForLeading(Lex::TokenKind::Caret),
  124. *PrecedenceGroup::ForLeading(Lex::TokenKind::Minus)),
  125. Eq(OperatorPriority::Ambiguous));
  126. EXPECT_THAT(
  127. PrecedenceGroup::GetPriority(
  128. *PrecedenceGroup::ForLeading(Lex::TokenKind::Not),
  129. PrecedenceGroup::ForTrailing(Lex::TokenKind::Amp, true)->level),
  130. Eq(OperatorPriority::Ambiguous));
  131. EXPECT_THAT(
  132. PrecedenceGroup::GetPriority(
  133. PrecedenceGroup::ForTrailing(Lex::TokenKind::Equal, true)->level,
  134. PrecedenceGroup::ForTrailing(Lex::TokenKind::PipeEqual, true)->level),
  135. Eq(OperatorPriority::Ambiguous));
  136. EXPECT_THAT(
  137. PrecedenceGroup::GetPriority(
  138. PrecedenceGroup::ForTrailing(Lex::TokenKind::Plus, true)->level,
  139. PrecedenceGroup::ForTrailing(Lex::TokenKind::Amp, true)->level),
  140. Eq(OperatorPriority::Ambiguous));
  141. }
  142. } // namespace
  143. } // namespace Carbon::Testing