1 //===--- LRTableTest.cpp - ---------------------------------------*- C++-*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "clang-pseudo/grammar/LRTable.h"
10 #include "clang-pseudo/grammar/Grammar.h"
11 #include "clang/Basic/TokenKinds.h"
12 #include "llvm/Testing/Support/SupportHelpers.h"
13 #include "gmock/gmock.h"
14 #include "gtest/gtest.h"
22 using testing::ElementsAre
;
23 using StateID
= LRTable::StateID
;
25 TEST(LRTable
, Builder
) {
26 std::vector
<std::string
> GrammarDiags
;
27 Grammar G
= Grammar::parseBNF(R
"bnf(
30 expr := expr + term # rule 2
31 term := IDENTIFIER # rule 3
34 EXPECT_THAT(GrammarDiags
, testing::IsEmpty());
36 SymbolID Term
= *G
.findNonterminal("term");
37 SymbolID Eof
= tokenSymbol(tok::eof
);
38 SymbolID Identifier
= tokenSymbol(tok::identifier
);
39 SymbolID Plus
= tokenSymbol(tok::plus
);
41 LRTable::Builder
B(G
);
43 // +-------+----+-------+------+
47 // +-------+----+-------+------+-------
48 B
.Transition
[{StateID
{0}, Identifier
}] = StateID
{0};
49 B
.Transition
[{StateID
{1}, Term
}] = StateID
{3};
50 B
.Reduce
[StateID
{0}].insert(RuleID
{0});
51 B
.Reduce
[StateID
{1}].insert(RuleID
{2});
52 B
.Reduce
[StateID
{2}].insert(RuleID
{1});
53 LRTable T
= std::move(B
).build();
55 EXPECT_EQ(T
.getShiftState(0, Eof
), std::nullopt
);
56 EXPECT_THAT(T
.getShiftState(0, Identifier
), ValueIs(0));
57 EXPECT_THAT(T
.getReduceRules(0), ElementsAre(0));
59 EXPECT_EQ(T
.getShiftState(1, Eof
), std::nullopt
);
60 EXPECT_EQ(T
.getShiftState(1, Identifier
), std::nullopt
);
61 EXPECT_THAT(T
.getGoToState(1, Term
), ValueIs(3));
62 EXPECT_THAT(T
.getReduceRules(1), ElementsAre(2));
64 // Verify the behaivor for other non-available-actions terminals.
65 SymbolID Int
= tokenSymbol(tok::kw_int
);
66 EXPECT_EQ(T
.getShiftState(2, Int
), std::nullopt
);
69 EXPECT_TRUE(T
.canFollow(Term
, Plus
));
70 EXPECT_TRUE(T
.canFollow(Term
, Eof
));
71 EXPECT_FALSE(T
.canFollow(Term
, Int
));