/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * Based on LLVM/Clang. * * This file is distributed under the University of Illinois Open Source * License. See LICENSE.TXT for details. * */ #include "plugin.hxx" #include #include #include #include #include #include "pluginhandler.hxx" /* Base classes for plugin actions. */ namespace loplugin { Plugin::Plugin( const InstantiationData& data ) : compiler( data.compiler ), handler( data.handler ), name( data.name ) { } DiagnosticBuilder Plugin::report( DiagnosticsEngine::Level level, StringRef message, SourceLocation loc ) const { return handler.report( level, name, message, compiler, loc ); } bool Plugin::ignoreLocation( SourceLocation loc ) { SourceLocation expansionLoc = compiler.getSourceManager().getExpansionLoc( loc ); if( compiler.getSourceManager().isInSystemHeader( expansionLoc )) return true; const char* bufferName = compiler.getSourceManager().getPresumedLoc( expansionLoc ).getFilename(); if (bufferName == NULL || hasPathnamePrefix(bufferName, SRCDIR "/external/") || isSamePathname(bufferName, SRCDIR "/sdext/source/pdfimport/wrapper/keyword_list") ) // workdir/CustomTarget/sdext/pdfimport/hash.cxx is generated from // sdext/source/pdfimport/wrapper/keyword_list by gperf, which // inserts various #line directives denoting the latter into the // former, but fails to add a #line directive returning back to // hash.cxx itself before the gperf generated boilerplate, so // compilers erroneously consider errors in the boilerplate to come // from keyword_list instead of hash.cxx (for Clang on Linux/macOS // this is not an issue due to the '#pragma GCC system_header' // generated into the start of hash.cxx, #if'ed for __GNUC__, but // for clang-cl it is an issue) return true; if( hasPathnamePrefix(bufferName, WORKDIR) ) { // workdir/CustomTarget/vcl/unx/kde4/tst_exclude_socket_notifiers.moc // includes // "../../../../../vcl/unx/kde4/tst_exclude_socket_notifiers.hxx", // making the latter file erroneously match here; so strip any ".." // segments: if (strstr(bufferName, "/..") == nullptr) { return true; } std::string s(bufferName); normalizeDotDotInFilePath(s); if (hasPathnamePrefix(s, WORKDIR)) return true; } if( hasPathnamePrefix(bufferName, BUILDDIR) || hasPathnamePrefix(bufferName, SRCDIR) ) return false; // ok return true; } void Plugin::normalizeDotDotInFilePath( std::string & s ) { for (std::string::size_type i = 0;;) { i = s.find("/.", i); if (i == std::string::npos) { break; } if (i + 2 == s.length() || s[i + 2] == '/') { s.erase(i, 2); // [AAA]/.[/CCC] -> [AAA][/CCC] } else if (s[i + 2] == '.' && (i + 3 == s.length() || s[i + 3] == '/')) { if (i == 0) { // /..[/CCC] -> /..[/CCC] break; } auto j = s.rfind('/', i - 1); if (j == std::string::npos) { // BBB/..[/CCC] -> BBB/..[/CCC] (instead of BBB/../CCC -> // CCC, to avoid wrong ../../CCC -> CCC; relative paths // shouldn't happen anyway, and even if they did, wouldn't // match against WORKDIR anyway, as WORKDIR should be // absolute): break; } s.erase(j, i + 3 - j); // AAA/BBB/..[/CCC] -> AAA[/CCC] i = j; } else { i += 2; } } } void Plugin::registerPlugin( Plugin* (*create)( const InstantiationData& ), const char* optionName, bool isPPCallback, bool byDefault ) { PluginHandler::registerPlugin( create, optionName, isPPCallback, byDefault ); } std::unordered_map< const Stmt*, const Stmt* > Plugin::parents; const Stmt* Plugin::parentStmt( const Stmt* stmt ) { if( parents.empty()) buildParents( compiler ); //if(parents.count(stmt)!=1)stmt->dump(); //assert( parents.count( stmt ) == 1 ); return parents[ stmt ]; } Stmt* Plugin::parentStmt( Stmt* stmt ) { if( parents.empty()) buildParents( compiler ); //assert( parents.count( stmt ) == 1 ); return const_cast< Stmt* >( parents[ stmt ] ); } static const Decl* getDeclContext(ASTContext& context, const Stmt* stmt) { auto it = context.getParents(*stmt).begin(); if (it == context.getParents(*stmt).end()) return nullptr; const Decl *aDecl = it->get(); if (aDecl) return aDecl; const Stmt *aStmt = it->get(); if (aStmt) return getDeclContext(context, aStmt); return nullptr; } const FunctionDecl* Plugin::parentFunctionDecl( const Stmt* stmt ) { const Decl *decl = getDeclContext(compiler.getASTContext(), stmt); if (decl) return static_cast(decl->getNonClosureContext()); return nullptr; } bool Plugin::isInUnoIncludeFile(SourceLocation spellingLocation) const { StringRef name { compiler.getSourceManager().getFilename(spellingLocation) }; return compiler.getSourceManager().isInMainFile(spellingLocation) ? (isSamePathname(name, SRCDIR "/cppu/source/cppu/compat.cxx") || isSamePathname(name, SRCDIR "/cppuhelper/source/compat.cxx") || isSamePathname(name, SRCDIR "/sal/osl/all/compat.cxx")) : (hasPathnamePrefix(name, SRCDIR "/include/com/") || hasPathnamePrefix(name, SRCDIR "/include/cppu/") || hasPathnamePrefix(name, SRCDIR "/include/cppuhelper/") || hasPathnamePrefix(name, SRCDIR "/include/osl/") || hasPathnamePrefix(name, SRCDIR "/include/rtl/") || hasPathnamePrefix(name, SRCDIR "/include/sal/") || hasPathnamePrefix(name, SRCDIR "/include/salhelper/") || hasPathnamePrefix(name, SRCDIR "/include/systools/") || hasPathnamePrefix(name, SRCDIR "/include/typelib/") || hasPathnamePrefix(name, SRCDIR "/include/uno/")); } bool Plugin::isInUnoIncludeFile(const FunctionDecl* functionDecl) const { return isInUnoIncludeFile(compiler.getSourceManager().getSpellingLoc( functionDecl->getCanonicalDecl()->getNameInfo().getLoc())); } namespace { class ParentBuilder : public RecursiveASTVisitor< ParentBuilder > { public: bool VisitFunctionDecl( const FunctionDecl* function ); bool VisitObjCMethodDecl( const ObjCMethodDecl* method ); void walk( const Stmt* stmt ); bool shouldVisitTemplateInstantiations () const { return true; } std::unordered_map< const Stmt*, const Stmt* >* parents; }; bool ParentBuilder::VisitFunctionDecl( const FunctionDecl* function ) { // if( ignoreLocation( declaration )) // return true; ??? if( function->doesThisDeclarationHaveABody()) { const Stmt* body = function->getBody(); (*parents)[ body ] = NULL; // no parent walk( body ); } if( const CXXConstructorDecl* ctor = dyn_cast< CXXConstructorDecl >( function )) { for( CXXConstructorDecl::init_const_iterator it = ctor->init_begin(); it != ctor->init_end(); ++it ) { const Expr* init_expression = (*it)->getInit(); (*parents)[ init_expression ] = NULL; walk( init_expression ); } } return true; } bool ParentBuilder::VisitObjCMethodDecl( const ObjCMethodDecl* method ) { // if( ignoreLocation( declaration )) // return true; ??? if( method->hasBody()) { const Stmt* body = method->getBody(); (*parents)[ body ] = NULL; // no parent walk( body ); } return true; } void ParentBuilder::walk( const Stmt* stmt ) { for( ConstStmtIterator it = stmt->child_begin(); it != stmt->child_end(); ++it ) { if( *it != NULL ) { (*parents)[ *it ] = stmt; walk( *it ); } } } } // namespace void Plugin::buildParents( CompilerInstance& compiler ) { assert( parents.empty()); ParentBuilder builder; builder.parents = &parents; builder.TraverseDecl( compiler.getASTContext().getTranslationUnitDecl()); } SourceLocation Plugin::locationAfterToken( SourceLocation location ) { return Lexer::getLocForEndOfToken( location, 0, compiler.getSourceManager(), compiler.getLangOpts()); } bool Plugin::isUnitTestMode() { return PluginHandler::isUnitTestMode(); } bool Plugin::containsPreprocessingConditionalInclusion(SourceRange range) { auto const begin = compiler.getSourceManager().getExpansionLoc( range.getBegin()); auto const end = compiler.getSourceManager().getExpansionLoc( range.getEnd()); assert(begin.isFileID() && end.isFileID()); if (!(begin == end || compiler.getSourceManager().isBeforeInTranslationUnit( begin, end))) { // Conservatively assume "yes" if lexing fails (e.g., due to // macros): return true; } auto hash = false; for (auto loc = begin;;) { Token tok; if (Lexer::getRawToken( loc, tok, compiler.getSourceManager(), compiler.getLangOpts(), true)) { // Conservatively assume "yes" if lexing fails (e.g., due to // macros): return true; } if (hash && tok.is(tok::raw_identifier)) { auto const id = tok.getRawIdentifier(); if (id == "if" || id == "ifdef" || id == "ifndef" || id == "elif" || id == "else" || id == "endif") { return true; } } if (loc == range.getEnd()) { break; } hash = tok.is(tok::hash) && tok.isAtStartOfLine(); loc = loc.getLocWithOffset( std::max( Lexer::MeasureTokenLength( loc, compiler.getSourceManager(), compiler.getLangOpts()), 1)); } return false; } RewritePlugin::RewritePlugin( const InstantiationData& data ) : Plugin( data ) , rewriter( data.rewriter ) { } bool RewritePlugin::insertText( SourceLocation Loc, StringRef Str, bool InsertAfter, bool indentNewLines ) { assert( rewriter ); if (wouldRewriteWorkdir(Loc)) return false; if( rewriter->InsertText( Loc, Str, InsertAfter, indentNewLines )) return reportEditFailure( Loc ); return true; } bool RewritePlugin::insertTextAfter( SourceLocation Loc, StringRef Str ) { assert( rewriter ); if (wouldRewriteWorkdir(Loc)) return false; if( rewriter->InsertTextAfter( Loc, Str )) return reportEditFailure( Loc ); return true; } bool RewritePlugin::insertTextAfterToken( SourceLocation Loc, StringRef Str ) { assert( rewriter ); if (wouldRewriteWorkdir(Loc)) return false; if( rewriter->InsertTextAfterToken( Loc, Str )) return reportEditFailure( Loc ); return true; } bool RewritePlugin::insertTextBefore( SourceLocation Loc, StringRef Str ) { assert( rewriter ); if (wouldRewriteWorkdir(Loc)) return false; if( rewriter->InsertTextBefore( Loc, Str )) return reportEditFailure( Loc ); return true; } bool RewritePlugin::removeText( SourceLocation Start, unsigned Length, RewriteOptions opts ) { CharSourceRange range( SourceRange( Start, Start.getLocWithOffset( Length )), false ); return removeText( range, opts ); } bool RewritePlugin::removeText( SourceRange range, RewriteOptions opts ) { return removeText( CharSourceRange( range, true ), opts ); } bool RewritePlugin::removeText( CharSourceRange range, RewriteOptions opts ) { assert( rewriter ); if (wouldRewriteWorkdir(range.getBegin())) return false; if( rewriter->getRangeSize( range, opts ) == -1 ) return reportEditFailure( range.getBegin()); if( !handler.addRemoval( range.getBegin() ) ) { report( DiagnosticsEngine::Warning, "double code removal, possible plugin error", range.getBegin()); return true; } if( opts.flags & RemoveWholeStatement || opts.flags & RemoveAllWhitespace ) { if( !adjustRangeForOptions( &range, opts )) return reportEditFailure( range.getBegin()); } if( rewriter->RemoveText( range, opts )) return reportEditFailure( range.getBegin()); return true; } bool RewritePlugin::adjustRangeForOptions( CharSourceRange* range, RewriteOptions opts ) { assert( rewriter ); SourceManager& SM = rewriter->getSourceMgr(); SourceLocation fileStartLoc = SM.getLocForStartOfFile( SM.getFileID( range->getBegin())); if( fileStartLoc.isInvalid()) return false; bool isInvalid = false; const char* fileBuf = SM.getCharacterData( fileStartLoc, &isInvalid ); if( isInvalid ) return false; const char* startBuf = SM.getCharacterData( range->getBegin(), &isInvalid ); if( isInvalid ) return false; SourceLocation locationEnd = range->getEnd(); if( range->isTokenRange()) locationEnd = locationAfterToken( locationEnd ); const char* endBuf = SM.getCharacterData( locationEnd, &isInvalid ); if( isInvalid ) return false; const char* startPos = startBuf; --startPos; while( startPos >= fileBuf && ( *startPos == ' ' || *startPos == '\t' )) --startPos; if( startPos >= fileBuf && *startPos == '\n' ) startPos = startBuf - 1; // do not remove indentation whitespace (RemoveLineIfEmpty can do that) const char* endPos = endBuf; while( *endPos == ' ' || *endPos == '\t' ) ++endPos; if( opts.flags & RemoveWholeStatement ) { if( *endPos == ';' ) ++endPos; else return false; } *range = CharSourceRange( SourceRange( range->getBegin().getLocWithOffset( startPos - startBuf + 1 ), locationEnd.getLocWithOffset( endPos - endBuf )), false ); return true; } bool RewritePlugin::replaceText( SourceLocation Start, unsigned OrigLength, StringRef NewStr ) { assert( rewriter ); if (wouldRewriteWorkdir(Start)) return false; if( OrigLength != 0 && !handler.addRemoval( Start ) ) { report( DiagnosticsEngine::Warning, "double code replacement, possible plugin error", Start ); return true; } if( rewriter->ReplaceText( Start, OrigLength, NewStr )) return reportEditFailure( Start ); return true; } bool RewritePlugin::replaceText( SourceRange range, StringRef NewStr ) { assert( rewriter ); if (wouldRewriteWorkdir(range.getBegin())) return false; if( rewriter->getRangeSize( range ) == -1 ) return reportEditFailure( range.getBegin()); if( !handler.addRemoval( range.getBegin() ) ) { report( DiagnosticsEngine::Warning, "double code replacement, possible plugin error", range.getBegin()); return true; } if( rewriter->ReplaceText( range, NewStr )) return reportEditFailure( range.getBegin()); return true; } bool RewritePlugin::replaceText( SourceRange range, SourceRange replacementRange ) { assert( rewriter ); if (wouldRewriteWorkdir(range.getBegin())) return false; if( rewriter->getRangeSize( range ) == -1 ) return reportEditFailure( range.getBegin()); if( !handler.addRemoval( range.getBegin() ) ) { report( DiagnosticsEngine::Warning, "double code replacement, possible plugin error", range.getBegin()); return true; } if( rewriter->ReplaceText( range, replacementRange )) return reportEditFailure( range.getBegin()); return true; } bool RewritePlugin::wouldRewriteWorkdir(SourceLocation loc) { if (loc.isInvalid() || loc.isMacroID()) { return false; } return compiler.getSourceManager().getFilename( compiler.getSourceManager().getSpellingLoc(loc)) .startswith(WORKDIR "/"); } bool RewritePlugin::reportEditFailure( SourceLocation loc ) { report( DiagnosticsEngine::Warning, "cannot perform source modification (macro expansion involved?)", loc ); return false; } namespace { template bool checkPathname( StringRef pathname, StringRef against, Fn check) { if (check(pathname, against)) { return true; } #if defined _WIN32 for (std::size_t n = 0;;) { std::size_t n1 = pathname.find('\\', n); if (n1 >= against.size()) { return check(pathname.substr(n), against.substr(n)); } if (against[n1] != '/' || pathname.substr(n, n1 - n) != against.substr(n, n1 - n)) { break; } n = n1 + 1; } #endif return false; } } bool hasPathnamePrefix(StringRef pathname, StringRef prefix) { return checkPathname( pathname, prefix, [](StringRef p, StringRef a) { return p.startswith(a); }); } bool isSamePathname(StringRef pathname, StringRef other) { return checkPathname( pathname, other, [](StringRef p, StringRef a) { return p == a; }); } } // namespace /* vim:set shiftwidth=4 softtabstop=4 expandtab: */