precedence_test.cpp 6.2 KB

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