|
|
@@ -20,71 +20,111 @@ TEST(PrecedenceTest, OperatorsAreRecognized) {
|
|
|
EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Slash()).hasValue());
|
|
|
EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Identifier()).hasValue());
|
|
|
|
|
|
- EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Minus()).hasValue());
|
|
|
- EXPECT_FALSE(PrecedenceGroup::ForTrailing(TokenKind::Tilde()).hasValue());
|
|
|
- EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Slash()).hasValue());
|
|
|
+ EXPECT_TRUE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Minus(), false).hasValue());
|
|
|
EXPECT_FALSE(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Identifier()).hasValue());
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Tilde(), false).hasValue());
|
|
|
+ EXPECT_TRUE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Slash(), true).hasValue());
|
|
|
+ EXPECT_FALSE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Identifier(), false).hasValue());
|
|
|
+
|
|
|
+ EXPECT_TRUE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Minus(), true)->is_binary);
|
|
|
+ EXPECT_FALSE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::MinusMinus(), false)->is_binary);
|
|
|
+}
|
|
|
|
|
|
- EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Minus())->is_binary);
|
|
|
+TEST(PrecedenceTest, InfixVsPostfix) {
|
|
|
+ // A trailing `-` is always infix; a trailing `--` is always postfix.
|
|
|
+ EXPECT_TRUE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Minus(), false)->is_binary);
|
|
|
EXPECT_FALSE(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::MinusMinus())->is_binary);
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::MinusMinus(), true)->is_binary);
|
|
|
+
|
|
|
+ // A trailing `*` is interpreted based on context.
|
|
|
+ EXPECT_TRUE(PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->is_binary);
|
|
|
+ EXPECT_FALSE(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), false)->is_binary);
|
|
|
+
|
|
|
+ // Infix `*` can appear in type contexts; binary `*` cannot.
|
|
|
+ EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level,
|
|
|
+ PrecedenceGroup::ForType()),
|
|
|
+ Eq(OperatorPriority::Ambiguous));
|
|
|
+ EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), false)->level,
|
|
|
+ PrecedenceGroup::ForType()),
|
|
|
+ Eq(OperatorPriority::LeftFirst));
|
|
|
+
|
|
|
+ // Binary `*` can appear in `+` contexts; binary `*` cannot.
|
|
|
+ EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)->level),
|
|
|
+ Eq(OperatorPriority::LeftFirst));
|
|
|
+ EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), false)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)->level),
|
|
|
+ Eq(OperatorPriority::Ambiguous));
|
|
|
}
|
|
|
|
|
|
TEST(PrecedenceTest, Associativity) {
|
|
|
EXPECT_THAT(
|
|
|
PrecedenceGroup::ForLeading(TokenKind::Minus())->GetAssociativity(),
|
|
|
Eq(Associativity::RightToLeft));
|
|
|
- EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusPlus())
|
|
|
+ EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusPlus(), false)
|
|
|
->level.GetAssociativity(),
|
|
|
Eq(Associativity::LeftToRight));
|
|
|
- EXPECT_THAT(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Plus())->level.GetAssociativity(),
|
|
|
- Eq(Associativity::LeftToRight));
|
|
|
- EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::Equal())
|
|
|
+ EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)
|
|
|
+ ->level.GetAssociativity(),
|
|
|
+ Eq(Associativity::LeftToRight));
|
|
|
+ EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::Equal(), true)
|
|
|
->level.GetAssociativity(),
|
|
|
Eq(Associativity::RightToLeft));
|
|
|
- EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusEqual())
|
|
|
+ EXPECT_THAT(PrecedenceGroup::ForTrailing(TokenKind::PlusEqual(), true)
|
|
|
->level.GetAssociativity(),
|
|
|
Eq(Associativity::None));
|
|
|
}
|
|
|
|
|
|
TEST(PrecedenceTest, DirectRelations) {
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Star())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Plus())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)->level),
|
|
|
Eq(OperatorPriority::LeftFirst));
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Plus())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Star())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level),
|
|
|
Eq(OperatorPriority::RightFirst));
|
|
|
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Amp())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Less())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Less(), true)->level),
|
|
|
Eq(OperatorPriority::LeftFirst));
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Less())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Amp())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Less(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level),
|
|
|
Eq(OperatorPriority::RightFirst));
|
|
|
}
|
|
|
|
|
|
TEST(PrecedenceTest, IndirectRelations) {
|
|
|
- EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Star())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::OrKeyword())->level),
|
|
|
- Eq(OperatorPriority::LeftFirst));
|
|
|
- EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::OrKeyword())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Star())->level),
|
|
|
- Eq(OperatorPriority::RightFirst));
|
|
|
+ EXPECT_THAT(
|
|
|
+ PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::OrKeyword(), true)->level),
|
|
|
+ Eq(OperatorPriority::LeftFirst));
|
|
|
+ EXPECT_THAT(
|
|
|
+ PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::OrKeyword(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level),
|
|
|
+ Eq(OperatorPriority::RightFirst));
|
|
|
|
|
|
+ EXPECT_THAT(
|
|
|
+ PrecedenceGroup::GetPriority(
|
|
|
+ *PrecedenceGroup::ForLeading(TokenKind::Tilde()),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Equal(), true)->level),
|
|
|
+ Eq(OperatorPriority::LeftFirst));
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- *PrecedenceGroup::ForLeading(TokenKind::Tilde()),
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Equal())->level),
|
|
|
- Eq(OperatorPriority::LeftFirst));
|
|
|
- EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Equal())->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Equal(), true)->level,
|
|
|
*PrecedenceGroup::ForLeading(TokenKind::Tilde())),
|
|
|
Eq(OperatorPriority::RightFirst));
|
|
|
}
|
|
|
@@ -100,15 +140,16 @@ TEST(PrecedenceTest, IncomparableOperators) {
|
|
|
Eq(OperatorPriority::Ambiguous));
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
*PrecedenceGroup::ForLeading(TokenKind::Minus()),
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Amp())->level),
|
|
|
- Eq(OperatorPriority::Ambiguous));
|
|
|
- EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Equal())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::PipeEqual())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level),
|
|
|
Eq(OperatorPriority::Ambiguous));
|
|
|
+ EXPECT_THAT(
|
|
|
+ PrecedenceGroup::GetPriority(
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Equal(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::PipeEqual(), true)->level),
|
|
|
+ Eq(OperatorPriority::Ambiguous));
|
|
|
EXPECT_THAT(PrecedenceGroup::GetPriority(
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Plus())->level,
|
|
|
- PrecedenceGroup::ForTrailing(TokenKind::Amp())->level),
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Plus(), true)->level,
|
|
|
+ PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level),
|
|
|
Eq(OperatorPriority::Ambiguous));
|
|
|
}
|
|
|
|