precedence_test.cpp 6.9 KB

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