mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-17 22:06:38 +00:00

Add the Data Inspection Language (DIL) implementation pieces for handling plain local and global variable names. See https://discourse.llvm.org/t/rfc-data-inspection-language/69893 for information about DIL. This change includes the basic AST, Lexer, Parser and Evaluator pieces, as well as some tests.
157 lines
5.8 KiB
C++
157 lines
5.8 KiB
C++
//===-- DILLexerTests.cpp --------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "lldb/ValueObject/DILLexer.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/Testing/Support/Error.h"
|
|
#include "gtest/gtest.h"
|
|
#include <string>
|
|
|
|
using llvm::StringRef;
|
|
|
|
using namespace lldb_private::dil;
|
|
|
|
llvm::Expected<std::vector<std::pair<Token::Kind, std::string>>>
|
|
ExtractTokenData(llvm::StringRef input_expr) {
|
|
|
|
llvm::Expected<DILLexer> maybe_lexer = DILLexer::Create(input_expr);
|
|
if (!maybe_lexer)
|
|
return maybe_lexer.takeError();
|
|
DILLexer lexer(*maybe_lexer);
|
|
|
|
std::vector<std::pair<Token::Kind, std::string>> data;
|
|
do {
|
|
Token tok = lexer.GetCurrentToken();
|
|
data.push_back(std::make_pair(tok.GetKind(), tok.GetSpelling()));
|
|
lexer.Advance();
|
|
} while (data.back().first != Token::eof);
|
|
// Don't return the eof token.
|
|
data.pop_back();
|
|
return data;
|
|
}
|
|
|
|
TEST(DILLexerTests, SimpleTest) {
|
|
StringRef input_expr("simple_var");
|
|
llvm::Expected<DILLexer> maybe_lexer = DILLexer::Create(input_expr);
|
|
ASSERT_THAT_EXPECTED(maybe_lexer, llvm::Succeeded());
|
|
DILLexer lexer(*maybe_lexer);
|
|
Token token = lexer.GetCurrentToken();
|
|
|
|
EXPECT_EQ(token.GetKind(), Token::identifier);
|
|
EXPECT_EQ(token.GetSpelling(), "simple_var");
|
|
lexer.Advance();
|
|
token = lexer.GetCurrentToken();
|
|
EXPECT_EQ(token.GetKind(), Token::eof);
|
|
}
|
|
|
|
TEST(DILLexerTests, TokenKindTest) {
|
|
Token token = Token(Token::identifier, "ident", 0);
|
|
|
|
EXPECT_TRUE(token.Is(Token::identifier));
|
|
EXPECT_FALSE(token.Is(Token::l_paren));
|
|
EXPECT_TRUE(token.IsOneOf({Token::eof, Token::identifier}));
|
|
EXPECT_FALSE(token.IsOneOf(
|
|
{Token::l_paren, Token::r_paren, Token::coloncolon, Token::eof}));
|
|
}
|
|
|
|
TEST(DILLexerTests, LookAheadTest) {
|
|
StringRef input_expr("(anonymous namespace)::some_var");
|
|
llvm::Expected<DILLexer> maybe_lexer = DILLexer::Create(input_expr);
|
|
ASSERT_THAT_EXPECTED(maybe_lexer, llvm::Succeeded());
|
|
DILLexer lexer(*maybe_lexer);
|
|
Token token = lexer.GetCurrentToken();
|
|
|
|
// Current token is '('; check the next 4 tokens, to make
|
|
// sure they are the identifier 'anonymous', the identifier 'namespace'
|
|
// ')' and '::', in that order.
|
|
EXPECT_EQ(token.GetKind(), Token::l_paren);
|
|
EXPECT_EQ(lexer.LookAhead(1).GetKind(), Token::identifier);
|
|
EXPECT_EQ(lexer.LookAhead(1).GetSpelling(), "anonymous");
|
|
EXPECT_EQ(lexer.LookAhead(2).GetKind(), Token::identifier);
|
|
EXPECT_EQ(lexer.LookAhead(2).GetSpelling(), "namespace");
|
|
EXPECT_EQ(lexer.LookAhead(3).GetKind(), Token::r_paren);
|
|
EXPECT_EQ(lexer.LookAhead(4).GetKind(), Token::coloncolon);
|
|
|
|
// Our current index should still be 0, as we only looked ahead; we are still
|
|
// officially on the '('.
|
|
EXPECT_EQ(lexer.GetCurrentTokenIdx(), 0u);
|
|
|
|
// Accept the 'lookahead', so our current token is '::', which has the index
|
|
// 4 in our vector of tokens (which starts at zero).
|
|
lexer.Advance(4);
|
|
token = lexer.GetCurrentToken();
|
|
EXPECT_EQ(token.GetKind(), Token::coloncolon);
|
|
EXPECT_EQ(lexer.GetCurrentTokenIdx(), 4u);
|
|
|
|
lexer.Advance();
|
|
token = lexer.GetCurrentToken();
|
|
EXPECT_EQ(token.GetKind(), Token::identifier);
|
|
EXPECT_EQ(token.GetSpelling(), "some_var");
|
|
EXPECT_EQ(lexer.GetCurrentTokenIdx(), 5u);
|
|
EXPECT_EQ(token.GetLocation(), strlen("(anonymous namespace)::"));
|
|
|
|
lexer.Advance();
|
|
token = lexer.GetCurrentToken();
|
|
EXPECT_EQ(token.GetKind(), Token::eof);
|
|
}
|
|
|
|
TEST(DILLexerTests, MultiTokenLexTest) {
|
|
EXPECT_THAT_EXPECTED(
|
|
ExtractTokenData("This string has (several ) ::identifiers"),
|
|
llvm::HasValue(testing::ElementsAre(
|
|
testing::Pair(Token::identifier, "This"),
|
|
testing::Pair(Token::identifier, "string"),
|
|
testing::Pair(Token::identifier, "has"),
|
|
testing::Pair(Token::l_paren, "("),
|
|
testing::Pair(Token::identifier, "several"),
|
|
testing::Pair(Token::r_paren, ")"),
|
|
testing::Pair(Token::coloncolon, "::"),
|
|
testing::Pair(Token::identifier, "identifiers"))));
|
|
}
|
|
|
|
TEST(DILLexerTests, IdentifiersTest) {
|
|
// These strings should lex into identifier tokens.
|
|
std::vector<std::string> valid_identifiers = {
|
|
"$My_name1", "$pc", "abcd", "_", "_a", "_a_", "$",
|
|
"a_b", "this", "self", "a", "MyName", "namespace"};
|
|
|
|
// The lexer can lex these strings, but they should not be identifiers.
|
|
std::vector<std::string> invalid_identifiers = {"", "::", "(", ")"};
|
|
|
|
// The lexer is expected to fail attempting to lex these strings (it cannot
|
|
// create valid tokens out of them).
|
|
std::vector<std::string> invalid_tok_strings = {"234", "2a", "2", "1MyName"};
|
|
|
|
// Verify that all of the valid identifiers come out as identifier tokens.
|
|
for (auto &str : valid_identifiers) {
|
|
SCOPED_TRACE(str);
|
|
EXPECT_THAT_EXPECTED(ExtractTokenData(str),
|
|
llvm::HasValue(testing::ElementsAre(
|
|
testing::Pair(Token::identifier, str))));
|
|
}
|
|
|
|
// Verify that the lexer fails on invalid token strings.
|
|
for (auto &str : invalid_tok_strings) {
|
|
SCOPED_TRACE(str);
|
|
auto maybe_lexer = DILLexer::Create(str);
|
|
EXPECT_THAT_EXPECTED(maybe_lexer, llvm::Failed());
|
|
}
|
|
|
|
// Verify that none of the invalid identifiers come out as identifier tokens.
|
|
for (auto &str : invalid_identifiers) {
|
|
SCOPED_TRACE(str);
|
|
llvm::Expected<DILLexer> maybe_lexer = DILLexer::Create(str);
|
|
EXPECT_THAT_EXPECTED(maybe_lexer, llvm::Succeeded());
|
|
DILLexer lexer(*maybe_lexer);
|
|
Token token = lexer.GetCurrentToken();
|
|
EXPECT_TRUE(token.IsNot(Token::identifier));
|
|
EXPECT_TRUE(token.IsOneOf(
|
|
{Token::eof, Token::coloncolon, Token::l_paren, Token::r_paren}));
|
|
}
|
|
}
|