precedence_test.cpp 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  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::Tilde).has_value());
  14. EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Slash).has_value());
  15. EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Identifier).has_value());
  16. EXPECT_TRUE(
  17. PrecedenceGroup::ForTrailing(TokenKind::Minus, false).has_value());
  18. EXPECT_FALSE(
  19. PrecedenceGroup::ForTrailing(TokenKind::Tilde, false).has_value());
  20. EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Slash, true).has_value());
  21. EXPECT_FALSE(
  22. PrecedenceGroup::ForTrailing(TokenKind::Identifier, false).has_value());
  23. EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Minus, true)->is_binary);
  24. EXPECT_FALSE(
  25. PrecedenceGroup::ForTrailing(TokenKind::MinusMinus, false)->is_binary);
  26. }
  27. TEST(PrecedenceTest, InfixVsPostfix) {
  28. // A trailing `-` is always infix; a trailing `--` is always postfix.
  29. EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Minus, false)->is_binary);
  30. EXPECT_FALSE(
  31. PrecedenceGroup::ForTrailing(TokenKind::MinusMinus, true)->is_binary);
  32. // A trailing `*` is interpreted based on context.
  33. EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Star, true)->is_binary);
  34. EXPECT_FALSE(PrecedenceGroup::ForTrailing(TokenKind::Star, false)->is_binary);
  35. // Infix `*` can appear in type contexts; binary `*` cannot.
  36. EXPECT_THAT(PrecedenceGroup::GetPriority(
  37. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level,
  38. PrecedenceGroup::ForType()),
  39. Eq(OperatorPriority::Ambiguous));
  40. EXPECT_THAT(PrecedenceGroup::GetPriority(
  41. PrecedenceGroup::ForTrailing(TokenKind::Star, false)->level,
  42. PrecedenceGroup::ForType()),
  43. Eq(OperatorPriority::LeftFirst));
  44. // Binary `*` can appear in `+` contexts; binary `*` cannot.
  45. EXPECT_THAT(PrecedenceGroup::GetPriority(
  46. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level,
  47. PrecedenceGroup::ForTrailing(TokenKind::Plus, true)->level),
  48. Eq(OperatorPriority::LeftFirst));
  49. EXPECT_THAT(PrecedenceGroup::GetPriority(
  50. PrecedenceGroup::ForTrailing(TokenKind::Star, false)->level,
  51. PrecedenceGroup::ForTrailing(TokenKind::Plus, true)->level),
  52. Eq(OperatorPriority::Ambiguous));
  53. }
  54. TEST(PrecedenceTest, Associativity) {
  55. EXPECT_THAT(PrecedenceGroup::ForLeading(TokenKind::Minus)->GetAssociativity(),
  56. Eq(Associativity::RightToLeft));
  57. EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusPlus, false)
  58. ->level.GetAssociativity(),
  59. Eq(Associativity::LeftToRight));
  60. EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::Plus, true)
  61. ->level.GetAssociativity(),
  62. Eq(Associativity::LeftToRight));
  63. EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::Equal, true)
  64. ->level.GetAssociativity(),
  65. Eq(Associativity::RightToLeft));
  66. EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusEqual, true)
  67. ->level.GetAssociativity(),
  68. Eq(Associativity::None));
  69. }
  70. TEST(PrecedenceTest, DirectRelations) {
  71. EXPECT_THAT(PrecedenceGroup::GetPriority(
  72. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level,
  73. PrecedenceGroup::ForTrailing(TokenKind::Plus, true)->level),
  74. Eq(OperatorPriority::LeftFirst));
  75. EXPECT_THAT(PrecedenceGroup::GetPriority(
  76. PrecedenceGroup::ForTrailing(TokenKind::Plus, true)->level,
  77. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level),
  78. Eq(OperatorPriority::RightFirst));
  79. EXPECT_THAT(PrecedenceGroup::GetPriority(
  80. PrecedenceGroup::ForTrailing(TokenKind::Amp, true)->level,
  81. PrecedenceGroup::ForTrailing(TokenKind::Less, true)->level),
  82. Eq(OperatorPriority::LeftFirst));
  83. EXPECT_THAT(PrecedenceGroup::GetPriority(
  84. PrecedenceGroup::ForTrailing(TokenKind::Less, true)->level,
  85. PrecedenceGroup::ForTrailing(TokenKind::Amp, true)->level),
  86. Eq(OperatorPriority::RightFirst));
  87. }
  88. TEST(PrecedenceTest, IndirectRelations) {
  89. EXPECT_THAT(PrecedenceGroup::GetPriority(
  90. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level,
  91. PrecedenceGroup::ForTrailing(TokenKind::Or, true)->level),
  92. Eq(OperatorPriority::LeftFirst));
  93. EXPECT_THAT(PrecedenceGroup::GetPriority(
  94. PrecedenceGroup::ForTrailing(TokenKind::Or, true)->level,
  95. PrecedenceGroup::ForTrailing(TokenKind::Star, true)->level),
  96. Eq(OperatorPriority::RightFirst));
  97. EXPECT_THAT(PrecedenceGroup::GetPriority(
  98. *PrecedenceGroup::ForLeading(TokenKind::Tilde),
  99. PrecedenceGroup::ForTrailing(TokenKind::Equal, true)->level),
  100. Eq(OperatorPriority::LeftFirst));
  101. EXPECT_THAT(PrecedenceGroup::GetPriority(
  102. PrecedenceGroup::ForTrailing(TokenKind::Equal, true)->level,
  103. *PrecedenceGroup::ForLeading(TokenKind::Tilde)),
  104. Eq(OperatorPriority::RightFirst));
  105. }
  106. TEST(PrecedenceTest, IncomparableOperators) {
  107. EXPECT_THAT(PrecedenceGroup::GetPriority(
  108. *PrecedenceGroup::ForLeading(TokenKind::Tilde),
  109. *PrecedenceGroup::ForLeading(TokenKind::Not)),
  110. Eq(OperatorPriority::Ambiguous));
  111. EXPECT_THAT(PrecedenceGroup::GetPriority(
  112. *PrecedenceGroup::ForLeading(TokenKind::Tilde),
  113. *PrecedenceGroup::ForLeading(TokenKind::Minus)),
  114. Eq(OperatorPriority::Ambiguous));
  115. EXPECT_THAT(PrecedenceGroup::GetPriority(
  116. *PrecedenceGroup::ForLeading(TokenKind::Minus),
  117. PrecedenceGroup::ForTrailing(TokenKind::Amp, true)->level),
  118. Eq(OperatorPriority::Ambiguous));
  119. EXPECT_THAT(
  120. PrecedenceGroup::GetPriority(
  121. PrecedenceGroup::ForTrailing(TokenKind::Equal, true)->level,
  122. PrecedenceGroup::ForTrailing(TokenKind::PipeEqual, true)->level),
  123. Eq(OperatorPriority::Ambiguous));
  124. EXPECT_THAT(PrecedenceGroup::GetPriority(
  125. PrecedenceGroup::ForTrailing(TokenKind::Plus, true)->level,
  126. PrecedenceGroup::ForTrailing(TokenKind::Amp, true)->level),
  127. Eq(OperatorPriority::Ambiguous));
  128. }
  129. } // namespace
  130. } // namespace Carbon::Testing