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