| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- #include "toolchain/parser/precedence.h"
- #include <gmock/gmock.h>
- #include <gtest/gtest.h>
- #include "toolchain/lexer/token_kind.h"
- namespace Carbon::Testing {
- namespace {
- using ::testing::Eq;
- TEST(PrecedenceTest, OperatorsAreRecognized) {
- EXPECT_TRUE(PrecedenceGroup::ForLeading(TokenKind::Minus()).hasValue());
- EXPECT_TRUE(PrecedenceGroup::ForLeading(TokenKind::Tilde()).hasValue());
- EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Slash()).hasValue());
- EXPECT_FALSE(PrecedenceGroup::ForLeading(TokenKind::Identifier()).hasValue());
- EXPECT_TRUE(
- PrecedenceGroup::ForTrailing(TokenKind::Minus(), false).hasValue());
- EXPECT_FALSE(
- 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);
- }
- 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(), 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(), false)
- ->level.GetAssociativity(),
- Eq(Associativity::LeftToRight));
- 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(), true)
- ->level.GetAssociativity(),
- Eq(Associativity::None));
- }
- TEST(PrecedenceTest, DirectRelations) {
- 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::Plus(), true)->level,
- PrecedenceGroup::ForTrailing(TokenKind::Star(), true)->level),
- Eq(OperatorPriority::RightFirst));
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level,
- PrecedenceGroup::ForTrailing(TokenKind::Less(), true)->level),
- Eq(OperatorPriority::LeftFirst));
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- 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(), true)->level,
- PrecedenceGroup::ForTrailing(TokenKind::Or(), true)->level),
- Eq(OperatorPriority::LeftFirst));
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- PrecedenceGroup::ForTrailing(TokenKind::Or(), 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::ForTrailing(TokenKind::Equal(), true)->level,
- *PrecedenceGroup::ForLeading(TokenKind::Tilde())),
- Eq(OperatorPriority::RightFirst));
- }
- TEST(PrecedenceTest, IncomparableOperators) {
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- *PrecedenceGroup::ForLeading(TokenKind::Tilde()),
- *PrecedenceGroup::ForLeading(TokenKind::Not())),
- Eq(OperatorPriority::Ambiguous));
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- *PrecedenceGroup::ForLeading(TokenKind::Tilde()),
- *PrecedenceGroup::ForLeading(TokenKind::Minus())),
- Eq(OperatorPriority::Ambiguous));
- EXPECT_THAT(PrecedenceGroup::GetPriority(
- *PrecedenceGroup::ForLeading(TokenKind::Minus()),
- 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(), true)->level,
- PrecedenceGroup::ForTrailing(TokenKind::Amp(), true)->level),
- Eq(OperatorPriority::Ambiguous));
- }
- } // namespace
- } // namespace Carbon::Testing
|