2

我现在正在使用LLVM + Clang。我使用ASTUnit构建树,并使用RecursiveASTVisitor遍历树。

    clang::ASTUnit* AST;
    clang::DiagnosticOptions diagOpts;
    llvm::IntrusiveRefCntPtr<clang::Diagnostic> diags = clang::CompilerInstance::createDiagnostics(diagOpts, 0, 0);

    const char** ptr = new const char*[1];
    ptr[0] = argv[1];

    clang::CompilerInvocation *ci = new clang::CompilerInvocation();
    clang::CompilerInvocation::CreateFromArgs(*ci, ptr, ptr+1, *diags);
    ci->setLangDefaults(clang::IK_CXX, clang::LangStandard::lang_cxx98);
    ci->getPreprocessorOutputOpts().ShowComments = 1;
    ci->getPreprocessorOutputOpts().ShowLineMarkers = 1;

    AST = clang::ASTUnit::LoadFromCompilerInvocation(ci, diags);

    ...
    ...

    MyRecursiveASTVisitor myvis(AST->getASTContext());
    myvis.TraverseDecl(AST->getASTContext().getTranslationUnitDecl());

我想避免访问包含的系统库。可能吗?

4

1 回答 1

3

我想我找到了某种解决方案,但不是最后一个!

在继承的 ASTVisitor 类中:

class MyRecursiveASTVisitor : public clang::RecursiveASTVisitor<MyRecursiveASTVisitor>

您可以覆盖函数TraverseDecl。在此函数中,您可以遍历声明。如果不递归调用这个函数,当声明不是来自主文件时,可以避免遍历其他文件的声明。

bool TraverseDecl ( clang::Decl *D )
{
    bool rval;

    if (!D) return true;

    if (sm.isFromMainFile(D->getLocation()) || std::string(D->getDeclKindName()) == "TranslationUnit")
    {
        bool rval = clang::RecursiveASTVisitor<MyRecursiveASTVisitor>::TraverseDecl(D);
    }
    else
        rval = true;

    return rval;
}

在这种情况下,sm是类MyRecursiveASTVisitor的clang::SourceManager。您可以避免在覆盖的TraverseStmt函数中以同样的方式遍历来自其他文件的语句。

现在我只知道如何知道语句或声明是来自系统库还是用户定义的库。

于 2012-05-07T16:41:18.217 回答