//===--- UnusedUsingDeclsCheck.cpp - clang-tidy----------------------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "UnusedUsingDeclsCheck.h" #include "clang/AST/ASTContext.h" #include "clang/ASTMatchers/ASTMatchFinder.h" #include "clang/Lex/Lexer.h" using namespace clang::ast_matchers; namespace clang { namespace tidy { namespace misc { void UnusedUsingDeclsCheck::registerMatchers(MatchFinder *Finder) { Finder->addMatcher(usingDecl(isExpansionInMainFile()).bind("using"), this); auto DeclMatcher = hasDeclaration(namedDecl().bind("used")); Finder->addMatcher(loc(recordType(DeclMatcher)), this); Finder->addMatcher(loc(templateSpecializationType(DeclMatcher)), this); } void UnusedUsingDeclsCheck::check(const MatchFinder::MatchResult &Result) { if (const auto *Using = Result.Nodes.getNodeAs("using")) { // FIXME: Implement the correct behavior for using declarations with more // than one shadow. if (Using->shadow_size() != 1) return; const auto *TargetDecl = Using->shadow_begin()->getTargetDecl()->getCanonicalDecl(); // FIXME: Handle other target types. if (!isa(TargetDecl) && !isa(TargetDecl)) return; FoundDecls[TargetDecl] = Using; FoundRanges[TargetDecl] = CharSourceRange::getCharRange( Using->getLocStart(), Lexer::findLocationAfterToken( Using->getLocEnd(), tok::semi, *Result.SourceManager, Result.Context->getLangOpts(), /*SkipTrailingWhitespaceAndNewLine=*/true)); return; } // Mark using declarations as used by setting FoundDecls' value to zero. As // the AST is walked in order, usages are only marked after a the // corresponding using declaration has been found. // FIXME: This currently doesn't look at whether the type reference is // actually found with the help of the using declaration. if (const auto *Used = Result.Nodes.getNodeAs("used")) { if (const auto *Specialization = dyn_cast(Used)) Used = Specialization->getSpecializedTemplate(); auto I = FoundDecls.find(Used->getCanonicalDecl()); if (I != FoundDecls.end()) I->second = nullptr; } } void UnusedUsingDeclsCheck::onEndOfTranslationUnit() { for (const auto &FoundDecl : FoundDecls) { if (FoundDecl.second == nullptr) continue; diag(FoundDecl.second->getLocation(), "using decl %0 is unused") << FoundDecl.second << FixItHint::CreateRemoval(FoundRanges[FoundDecl.first]); } FoundDecls.clear(); } } // namespace misc } // namespace tidy } // namespace clang