//===-- UseAuto/UseAutoActions.cpp - Matcher callback impl ----------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// /// /// \file /// \brief This file contains the implementation of callbacks for the UseAuto /// transform. /// //===----------------------------------------------------------------------===// #include "UseAutoActions.h" #include "UseAutoMatchers.h" #include "clang/AST/ASTContext.h" using namespace clang::ast_matchers; using namespace clang::tooling; using namespace clang; void IteratorReplacer::run(const MatchFinder::MatchResult &Result) { const DeclStmt *D = Result.Nodes.getNodeAs(IteratorDeclStmtId); assert(D && "Bad Callback. No node provided"); SourceManager &SM = *Result.SourceManager; if (!SM.isFromMainFile(D->getLocStart())) return; for (clang::DeclStmt::const_decl_iterator I = D->decl_begin(), E = D->decl_end(); I != E; ++I) { const VarDecl *V = cast(*I); const Expr *ExprInit = V->getInit(); // Skip expressions with cleanups from the initializer expression. if (const ExprWithCleanups *E = dyn_cast(ExprInit)) ExprInit = E->getSubExpr(); const CXXConstructExpr *Construct = cast(ExprInit); assert(Construct->getNumArgs() == 1u && "Expected constructor with single argument"); // Drill down to the as-written initializer. const Expr *E = Construct->arg_begin()->IgnoreParenImpCasts(); if (E != E->IgnoreConversionOperator()) // We hit a conversion operator. Early-out now as they imply an implicit // conversion from a different type. Could also mean an explicit conversion // from the same type but that's pretty rare. return; if (const CXXConstructExpr *NestedConstruct = dyn_cast(E)) // If we ran into an implicit conversion constructor, can't convert. // // FIXME: The following only checks if the constructor can be used // implicitly, not if it actually was. Cases where the converting constructor // was used explicitly won't get converted. if (NestedConstruct->getConstructor()->isConvertingConstructor(false)) return; if (!Result.Context->hasSameType(V->getType(), E->getType())) return; } // Get the type location using the first declartion. const VarDecl *V = cast(*D->decl_begin()); TypeLoc TL = V->getTypeSourceInfo()->getTypeLoc(); // WARNING: TypeLoc::getSourceRange() will include the identifier for things // like function pointers. Not a concern since this action only works with // iterators but something to keep in mind in the future. CharSourceRange Range(TL.getSourceRange(), true); Replace.insert(tooling::Replacement(SM, Range, "auto")); ++AcceptedChanges; } void NewReplacer::run(const MatchFinder::MatchResult &Result) { const VarDecl *D = Result.Nodes.getNodeAs(DeclWithNewId); assert(D && "Bad Callback. No node provided"); SourceManager &SM = *Result.SourceManager; if (!SM.isFromMainFile(D->getLocStart())) return; const CXXNewExpr *NewExpr = Result.Nodes.getNodeAs(NewExprId); assert(NewExpr && "Bad Callback. No CXXNewExpr bound"); // If declaration and initializer have exactly the same type, just replace // with 'auto'. if (Result.Context->hasSameType(D->getType(), NewExpr->getType())) { TypeLoc TL = D->getTypeSourceInfo()->getTypeLoc(); CharSourceRange Range(TL.getSourceRange(), /*IsTokenRange=*/ true); // Space after 'auto' to handle cases where the '*' in the pointer type // is next to the identifier. This avoids changing 'int *p' into 'autop'. Replace.insert(tooling::Replacement(SM, Range, "auto ")); ++AcceptedChanges; return; } // If the CV qualifiers for the pointer differ then we still use auto, just // need to leave the qualifier behind. if (Result.Context->hasSameUnqualifiedType(D->getType(), NewExpr->getType())) { TypeLoc TL = D->getTypeSourceInfo()->getTypeLoc(); CharSourceRange Range(TL.getSourceRange(), /*IsTokenRange=*/ true); // Space after 'auto' to handle cases where the '*' in the pointer type // is next to the identifier. This avoids changing 'int *p' into 'autop'. Replace.insert(tooling::Replacement(SM, Range, "auto ")); ++AcceptedChanges; return; } // The VarDecl and Initializer have mismatching types. return; // FIXME: There is, however, one case we can address: when the VarDecl // pointee is the same as the initializer, just more CV-qualified. However, // TypeLoc information is not reliable where CV qualifiers are concerned so // we can't do anything about this case for now. }