89 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			89 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===---------- UsingInserter.cpp - clang-tidy ----------------------------===//
 | |
| //
 | |
| // 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 "UsingInserter.h"
 | |
| 
 | |
| #include "ASTUtils.h"
 | |
| #include "clang/ASTMatchers/ASTMatchFinder.h"
 | |
| #include "clang/ASTMatchers/ASTMatchers.h"
 | |
| #include "clang/Lex/Lexer.h"
 | |
| 
 | |
| namespace clang {
 | |
| namespace tidy {
 | |
| namespace utils {
 | |
| 
 | |
| using namespace ast_matchers;
 | |
| 
 | |
| static StringRef getUnqualifiedName(StringRef QualifiedName) {
 | |
|   size_t LastSeparatorPos = QualifiedName.rfind("::");
 | |
|   if (LastSeparatorPos == StringRef::npos)
 | |
|     return QualifiedName;
 | |
|   return QualifiedName.drop_front(LastSeparatorPos + 2);
 | |
| }
 | |
| 
 | |
| UsingInserter::UsingInserter(const SourceManager &SourceMgr)
 | |
|     : SourceMgr(SourceMgr) {}
 | |
| 
 | |
| Optional<FixItHint> UsingInserter::createUsingDeclaration(
 | |
|     ASTContext &Context, const Stmt &Statement, StringRef QualifiedName) {
 | |
|   StringRef UnqualifiedName = getUnqualifiedName(QualifiedName);
 | |
|   const FunctionDecl *Function = getSurroundingFunction(Context, Statement);
 | |
|   if (!Function)
 | |
|     return None;
 | |
| 
 | |
|   if (AddedUsing.count(std::make_pair(Function, QualifiedName.str())) != 0)
 | |
|     return None;
 | |
| 
 | |
|   SourceLocation InsertLoc = Lexer::getLocForEndOfToken(
 | |
|       Function->getBody()->getBeginLoc(), 0, SourceMgr, Context.getLangOpts());
 | |
| 
 | |
|   // Only use using declarations in the main file, not in includes.
 | |
|   if (SourceMgr.getFileID(InsertLoc) != SourceMgr.getMainFileID())
 | |
|     return None;
 | |
| 
 | |
|   // FIXME: This declaration could be masked. Investigate if
 | |
|   // there is a way to avoid using Sema.
 | |
|   bool AlreadyHasUsingDecl =
 | |
|       !match(stmt(hasAncestor(decl(has(usingDecl(hasAnyUsingShadowDecl(
 | |
|                  hasTargetDecl(hasName(QualifiedName.str())))))))),
 | |
|              Statement, Context)
 | |
|            .empty();
 | |
|   if (AlreadyHasUsingDecl) {
 | |
|     AddedUsing.emplace(NameInFunction(Function, QualifiedName.str()));
 | |
|     return None;
 | |
|   }
 | |
|   // Find conflicting declarations and references.
 | |
|   auto ConflictingDecl = namedDecl(hasName(UnqualifiedName));
 | |
|   bool HasConflictingDeclaration =
 | |
|       !match(findAll(ConflictingDecl), *Function, Context).empty();
 | |
|   bool HasConflictingDeclRef =
 | |
|       !match(findAll(declRefExpr(to(ConflictingDecl))), *Function, Context)
 | |
|            .empty();
 | |
|   if (HasConflictingDeclaration || HasConflictingDeclRef)
 | |
|     return None;
 | |
| 
 | |
|   std::string Declaration =
 | |
|       (llvm::Twine("\nusing ") + QualifiedName + ";").str();
 | |
| 
 | |
|   AddedUsing.emplace(std::make_pair(Function, QualifiedName.str()));
 | |
|   return FixItHint::CreateInsertion(InsertLoc, Declaration);
 | |
| }
 | |
| 
 | |
| StringRef UsingInserter::getShortName(ASTContext &Context,
 | |
|                                       const Stmt &Statement,
 | |
|                                       StringRef QualifiedName) {
 | |
|   const FunctionDecl *Function = getSurroundingFunction(Context, Statement);
 | |
|   if (AddedUsing.count(NameInFunction(Function, QualifiedName.str())) != 0)
 | |
|     return getUnqualifiedName(QualifiedName);
 | |
|   return QualifiedName;
 | |
| }
 | |
| 
 | |
| } // namespace utils
 | |
| } // namespace tidy
 | |
| } // namespace clang
 |