3

我正在使用 clang 匹配器来获取结果节点。从结果节点中,我可以得到行号,让我们说 17。现在,我想获得该行中的整个源代码。请帮忙。

让我详细解释一下。我有一个 clang 匹配器,可以在源代码中找到浮动文字。例如第 17 行,sr = 2.0 * rt_urand_Upu32_Yd_f_pw_snf(u);是源代码,那么它匹配 2.0。这是我的匹配器:

const auto PA = floatLiteral(
                                isExpansionInMainFile(), 
                                unless(hasAncestor(arraySubscriptExpr()))
                            ).bind("pa");
            MatchFinder MatchFinder;
            MatchFinder.addMatcher(PA, &Handler);
            MatchFinder.matchAST(Context);

从匹配器中,我可以获得匹配的节点。我能够检索行号(第 17 行)和列号(6)。请在下面找到我的代码:

const clang::FloatingLiteral* Variable = Result.Nodes.getNodeAs<clang::FloatingLiteral>("pa");
clang::SourceRange loc = Variable16->getSourceRange();
locStart = srcMgr.getPresumedLoc(loc.getBegin());
locEnd = srcMgr.getPresumedLoc(loc.getEnd());
std::cout << locStart.getLine()<< ":" << locEnd.getLine() << std::endl;
std::cout << locStart.getColumn() <<":" << locEnd.getColumn() << std::endl;

现在,如果我尝试检索源代码,我只会得到部分数据。在网上做了一些研究后,我尝试通过两种方式检索源代码。第一种方法是使用词法分析器,请找到以下代码:

llvm::StringRef ref = Lexer::getSourceText(CharSourceRange::getCharRange(statement->getSourceRange()), srcMgr, LangOptions());
cout << ref.str() << endl;

第二种方法是使用重写器,请找到以下代码:

clang::Rewriter rewriter;
rewriter.setSourceMgr(Result.Context->getSourceManager(),Result.Context->getLangOpts());
cout<<rewriter.getRewrittenText (loc)<<endl;

据我了解,似乎我需要从第 17 行的第 0 列到第 17 行的列末尾的源范围。AST 匹配器仅匹配特定节点,所以我的问题是:

1) 是否可以获得第 17 行的最终列号?

2)还有其他方法可以从行号获取源代码吗?

3)还有其他方法可以从匹配器获取源代码吗?

谢谢您的帮助。

4

1 回答 1

0
  1. 没有直接的方法来获取一行的最后一列,但是有一个解决方案来获取行的开头。我假设你有SourceManager ( SM) 和SourceLocation ( SL)
SM.translateLineCol(SM.getMainFileID(),SM.getSpellingLineNumber(SL),1);

如果你真的想得到特定行的结尾,你可以将列号设置为一个不可思议的数字,比如 1000。如果该行少于1000个字符,会卡在最后一个字符,但是如果该行多于1000个字符,则不会出现在行尾,所以这个解决方案是不稳定的。该示例正在获取第 17 行的结束位置。

SM.translateLineCol(SM.getMainFileID(),17,1000);
  1. 通过使用上面的解决方案,您可以使用Rewriter ( Rw) 获取源代码。例如,如果您想获取第 17 行的代码,并将其存储在字符串结果中。
SourceLocation thisline=SM.translateLineCol(SM.getMainFileID(),17,1); // get the beginning of line 17
SourceLocation nextline=SM.translateLineCol(SM.getMainFileID(),18,1); // get the beginning of line 18
string result = Rw.getRewrittenText(SourceRange(thisline,nextline));
  1. 是的,如果你有合适的 SourceRange 和 Rewriter,你可以使用getRewrittenText()from match result。
于 2021-10-26T12:34:02.943 回答