summaryrefslogtreecommitdiffstats
path: root/compilerplugins/clang/stringloop.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'compilerplugins/clang/stringloop.cxx')
-rw-r--r--compilerplugins/clang/stringloop.cxx291
1 files changed, 0 insertions, 291 deletions
diff --git a/compilerplugins/clang/stringloop.cxx b/compilerplugins/clang/stringloop.cxx
deleted file mode 100644
index cb3ad327738f..000000000000
--- a/compilerplugins/clang/stringloop.cxx
+++ /dev/null
@@ -1,291 +0,0 @@
-/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-/*
- * This file is part of the LibreOffice project.
- *
- * This Source Code Form is subject to the terms of the Mozilla Public
- * License, v. 2.0. If a copy of the MPL was not distributed with this
- * file, You can obtain one at http://mozilla.org/MPL/2.0/.
- */
-
-#include "check.hxx"
-#include "plugin.hxx"
-#include <vector>
-
-/** Look for OUString/OString being appended to inside a loop, where OUStringBuffer/OStringBuffer would be a better idea
- */
-namespace
-{
-class StringLoop : public clang::RecursiveASTVisitor<StringLoop>, public loplugin::Plugin
-{
-public:
- explicit StringLoop(loplugin::InstantiationData const& rData)
- : Plugin(rData)
- {
- }
-
- void run() override;
- bool TraverseForStmt(ForStmt*);
- bool TraverseCXXForRangeStmt(CXXForRangeStmt*);
- bool TraverseDoStmt(DoStmt*);
- bool TraverseWhileStmt(WhileStmt*);
- bool VisitVarDecl(VarDecl const*);
- bool VisitCallExpr(CallExpr const*);
-
-private:
- int m_insideLoop = 0;
- using VarDeclList = std::vector<VarDecl const*>;
- std::vector<VarDeclList> m_varsPerLoopLevel;
-};
-
-void StringLoop::run()
-{
- // Various places are not worth changing, the code becomes too awkward
- // Just exclude stuff as I go
- StringRef fn(handler.getMainFileName());
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/bridges/"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/cppuhelper/source/shlib.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/registry/source/regimpl.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/l10ntools/source/lngmerge.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/tools/qa/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/jvmfwk/"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/svl/source/passwordcontainer/passwordcontainer.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/svl/source/numbers/zformat.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/svl/source/numbers/zforscan.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/vcl/source/control/combobox.cxx"))
- return;
- if (loplugin::isSamePathname(fn, SRCDIR "/vcl/source/gdi/pdfwriter_impl.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/svtools/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/idl/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/framework/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/basic/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sfx2/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/avmedia/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/connectivity/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/editeng/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/svx/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/basctl/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/filter/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/chart2/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/cui/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/dbaccess/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/oox/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/writerfilter/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/desktop/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/extensions/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/dtrans/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/i18npool/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/embeddedobj/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sd/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/xmloff/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/xmlhelp/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/forms/"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/core/tool/address.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/core/tool/compiler.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/ui/docshell/impex.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/ui/miscdlgs/acredlin.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/ui/pagedlg/areasdlg.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/ui/view/gridwin2.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sc/source/filter/html/htmlpars.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/doc/doctxm.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/edit/edattr.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/layout/dbg_lay.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/ascii/ascatr.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/html/htmlforw.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/unocore/unosect.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/unocore/unochart.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/core/unocore/unoobj.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/html/parcss1.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/html/svxcss1.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/html/swhtml.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/uibase/utlui/gloslst.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/uibase/utlui/content.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/uibase/docvw/edtwin.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/ww8/ww8atr.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/ww8/ww8scan.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/filter/ww8/ww8par5.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/fldui/fldfunc.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/misc/bookmark.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/dbui/mmlayoutpage.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/dbui/dbinsdlg.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/dbui/mmresultdialogs.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/sw/source/ui/index/cnttab.cxx"))
- return;
- if (loplugin::hasPathnamePrefix(fn, SRCDIR "/ucb/source/ucp/file/bc.cxx"))
- return;
-
- TraverseDecl(compiler.getASTContext().getTranslationUnitDecl());
-}
-
-bool StringLoop::TraverseForStmt(ForStmt* stmt)
-{
- ++m_insideLoop;
- m_varsPerLoopLevel.push_back({});
- auto const ret = RecursiveASTVisitor::TraverseForStmt(stmt);
- m_varsPerLoopLevel.pop_back();
- --m_insideLoop;
- return ret;
-}
-
-bool StringLoop::TraverseCXXForRangeStmt(CXXForRangeStmt* stmt)
-{
- ++m_insideLoop;
- m_varsPerLoopLevel.push_back({});
- auto const ret = RecursiveASTVisitor::TraverseCXXForRangeStmt(stmt);
- m_varsPerLoopLevel.pop_back();
- --m_insideLoop;
- return ret;
-}
-
-bool StringLoop::TraverseDoStmt(DoStmt* stmt)
-{
- ++m_insideLoop;
- m_varsPerLoopLevel.push_back({});
- auto const ret = RecursiveASTVisitor::TraverseDoStmt(stmt);
- m_varsPerLoopLevel.pop_back();
- --m_insideLoop;
- return ret;
-}
-
-bool StringLoop::TraverseWhileStmt(WhileStmt* stmt)
-{
- ++m_insideLoop;
- m_varsPerLoopLevel.push_back({});
- auto const ret = RecursiveASTVisitor::TraverseWhileStmt(stmt);
- m_varsPerLoopLevel.pop_back();
- --m_insideLoop;
- return ret;
-}
-
-bool StringLoop::VisitVarDecl(VarDecl const* varDecl)
-{
- if (ignoreLocation(varDecl))
- return true;
- if (!m_insideLoop)
- return true;
- m_varsPerLoopLevel.back().push_back(varDecl);
- return true;
-}
-
-bool StringLoop::VisitCallExpr(CallExpr const* callExpr)
-{
- if (ignoreLocation(callExpr))
- return true;
- if (!m_insideLoop)
- return true;
- auto operatorCallExpr = dyn_cast<CXXOperatorCallExpr>(callExpr);
- if (!operatorCallExpr)
- return true;
- if (operatorCallExpr->getOperator() != OO_PlusEqual)
- return true;
-
- if (auto memberExpr = dyn_cast<MemberExpr>(callExpr->getArg(0)))
- {
- auto tc = loplugin::TypeCheck(memberExpr->getType());
- if (!tc.Class("OUString").Namespace("rtl").GlobalNamespace()
- && !tc.Class("OString").Namespace("rtl").GlobalNamespace())
- return true;
- auto fieldDecl = dyn_cast<FieldDecl>(memberExpr->getMemberDecl());
- if (isInUnoIncludeFile(
- compiler.getSourceManager().getSpellingLoc(fieldDecl->getLocation())))
- return true;
- if (ignoreLocation(compiler.getSourceManager().getSpellingLoc(fieldDecl->getLocation())))
- return true;
- report(DiagnosticsEngine::Warning,
- "appending to OUString in loop, rather use OUStringBuffer",
- compat::getBeginLoc(operatorCallExpr))
- << operatorCallExpr->getSourceRange();
- report(DiagnosticsEngine::Note, "field here", compat::getBeginLoc(fieldDecl))
- << fieldDecl->getSourceRange();
- }
- else if (auto declRefExpr = dyn_cast<DeclRefExpr>(callExpr->getArg(0)))
- {
- if (auto varDecl = dyn_cast<VarDecl>(declRefExpr->getDecl()))
- {
- auto tc = loplugin::TypeCheck(varDecl->getType());
- if (!tc.Class("OUString").Namespace("rtl").GlobalNamespace()
- && !tc.Class("OString").Namespace("rtl").GlobalNamespace())
- return true;
- // if the var is at the same block scope as the +=, not interesting
- auto vars = m_varsPerLoopLevel.back();
- if (std::find(vars.begin(), vars.end(), varDecl) != vars.end())
- return true;
- report(DiagnosticsEngine::Warning,
- "appending to OUString in loop, rather use OUStringBuffer",
- compat::getBeginLoc(operatorCallExpr))
- << operatorCallExpr->getSourceRange();
- report(DiagnosticsEngine::Note, "var here", compat::getBeginLoc(varDecl))
- << varDecl->getSourceRange();
- }
- }
- return true;
-}
-
-loplugin::Plugin::Registration<StringLoop> X("stringloop", false);
-
-} // namespace
-
-/* vim:set shiftwidth=4 softtabstop=4 expandtab: */