bump product version to 7.2.5.1
[LibreOffice.git] / compilerplugins / clang / shouldreturnbool.cxx
blob1372d81033740d02025c1b0cdf65638a412a5fec
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 */
10 #include <string>
11 #include <set>
12 #include <iostream>
14 #include "check.hxx"
15 #include "plugin.hxx"
16 #include "functionaddress.hxx"
19 Look for functions that only return 1 and/or 0, which sometimes indicates that the
20 function should be returning bool.
22 Note that is partly a question of taste and code style, which is why this plugin is off by default.
25 namespace
27 class ShouldReturnBool
28 : public loplugin::FunctionAddress<loplugin::FilteringPlugin<ShouldReturnBool>>
30 public:
31 explicit ShouldReturnBool(loplugin::InstantiationData const& data)
32 : FunctionAddress(data)
36 virtual void run() override
38 if (!compiler.getLangOpts().CPlusPlus)
39 return;
40 StringRef fn(handler.getMainFileName());
41 // functions used as function pointers
42 if (loplugin::isSamePathname(fn, SRCDIR "/sal/rtl/alloc_cache.cxx"))
43 return;
44 // false +, slightly odd usage, but not wrong
45 if (loplugin::isSamePathname(fn, SRCDIR "/libreofficekit/qa/tilebench/tilebench.cxx"))
46 return;
47 // uses the Unix convention of "non-zero return indicates error"
48 if (loplugin::isSamePathname(fn, SRCDIR "/idlc/source/idlcproduce.cxx"))
49 return;
50 // template magic
51 if (loplugin::isSamePathname(fn, SRCDIR "/vcl/source/gdi/bmpfast.cxx"))
52 return;
53 // fine
54 if (loplugin::isSamePathname(fn, SRCDIR "/svl/unx/source/svdde/ddedummy.cxx"))
55 return;
56 if (loplugin::isSamePathname(fn, SRCDIR "/vcl/source/opengl/OpenGLHelper.cxx"))
57 return;
58 if (loplugin::isSamePathname(fn, SRCDIR "/svtools/source/misc/imap2.cxx"))
59 return;
60 if (loplugin::isSamePathname(fn, SRCDIR "/svx/source/dialog/docrecovery.cxx"))
61 return;
62 if (loplugin::isSamePathname(fn, SRCDIR "/hwpfilter/source/lexer.cxx"))
63 return;
64 if (loplugin::isSamePathname(fn, SRCDIR "/hwpfilter/source/grammar.cxx"))
65 return;
66 if (loplugin::isSamePathname(
67 fn, SRCDIR "/connectivity/source/drivers/odbc/ODatabaseMetaDataResultSet.cxx"))
68 return;
69 if (loplugin::isSamePathname(fn, SRCDIR "/dbaccess/source/ui/browser/dsEntriesNoExp.cxx"))
70 return;
71 if (loplugin::isSamePathname(fn, SRCDIR "/lotuswordpro/source/filter/explode.cxx"))
72 return;
73 if (loplugin::isSamePathname(fn, SRCDIR "/filter/source/graphicfilter/ipict/ipict.cxx"))
74 return;
75 if (loplugin::isSamePathname(fn, SRCDIR "/sc/source/core/data/dptabsrc.cxx"))
76 return;
77 if (loplugin::isSamePathname(fn, SRCDIR "/sc/source/ui/docshell/docsh3.cxx"))
78 return;
79 if (loplugin::isSamePathname(fn, SRCDIR "/sd/source/ui/dlg/masterlayoutdlg.cxx"))
80 return;
81 if (loplugin::isSamePathname(fn, SRCDIR "/sd/source/filter/ppt/pptinanimations.cxx"))
82 return;
83 if (loplugin::isSamePathname(fn, SRCDIR "/vcl/unx/generic/app/i18n_im.cxx"))
84 return;
86 // callback
87 if (loplugin::isSamePathname(fn, SRCDIR "/sax/source/expatwrap/sax_expat.cxx"))
88 return;
89 if (loplugin::isSamePathname(fn, SRCDIR "/xmlsecurity/source/xmlsec/xmlstreamio.cxx"))
90 return;
91 if (loplugin::isSamePathname(fn, SRCDIR "/sw/source/filter/ww8/ww8par.cxx"))
92 return;
93 if (loplugin::isSamePathname(fn, SRCDIR "/sw/source/filter/ww8/ww8par2.cxx"))
94 return;
95 if (loplugin::isSamePathname(fn, SRCDIR "/sw/source/filter/ww8/ww8par5.cxx"))
96 return;
97 // SaxWriterHelper::writeSequence a little weird
98 if (loplugin::isSamePathname(fn, SRCDIR "/sax/source/expatwrap/saxwriter.cxx"))
99 return;
100 // main function
101 if (loplugin::isSamePathname(fn, SRCDIR "/xmlsecurity/workben/pdfverify.cxx"))
102 return;
104 TraverseDecl(compiler.getASTContext().getTranslationUnitDecl());
106 for (auto functionDecl : problemFunctions)
108 auto canonicalDecl = functionDecl->getCanonicalDecl();
109 if (getFunctionsWithAddressTaken().find(canonicalDecl)
110 != getFunctionsWithAddressTaken().end())
111 continue;
112 report(DiagnosticsEngine::Warning,
113 "only returning one or zero is an indication you want to return bool",
114 compat::getBeginLoc(functionDecl))
115 << functionDecl->getSourceRange();
116 if (canonicalDecl->getLocation() != functionDecl->getLocation())
118 report(DiagnosticsEngine::Note, "canonical function declaration here",
119 compat::getBeginLoc(canonicalDecl))
120 << canonicalDecl->getSourceRange();
125 bool TraverseFunctionDecl(FunctionDecl*);
126 bool TraverseCXXMethodDecl(CXXMethodDecl*);
127 bool VisitReturnStmt(ReturnStmt const*);
129 private:
130 bool mbInsideFunction = false;
131 bool mbFunctionOnlyReturningOneOrZero = false;
132 std::unordered_set<FunctionDecl const*> problemFunctions;
134 bool IsInteresting(FunctionDecl const*);
135 void Report(FunctionDecl const*) const;
136 bool isExprOneOrZero(Expr const*) const;
139 bool ShouldReturnBool::TraverseFunctionDecl(FunctionDecl* functionDecl)
141 bool ret;
142 if (IsInteresting(functionDecl))
144 mbInsideFunction = true;
145 mbFunctionOnlyReturningOneOrZero = true;
146 ret = FunctionAddress::TraverseFunctionDecl(functionDecl);
147 mbInsideFunction = false;
148 if (mbFunctionOnlyReturningOneOrZero)
149 problemFunctions.insert(functionDecl);
151 else
152 ret = FunctionAddress::TraverseFunctionDecl(functionDecl);
153 return ret;
156 bool ShouldReturnBool::TraverseCXXMethodDecl(CXXMethodDecl* methodDecl)
158 bool ret;
159 if (IsInteresting(methodDecl))
161 mbInsideFunction = true;
162 mbFunctionOnlyReturningOneOrZero = true;
163 ret = FunctionAddress::TraverseCXXMethodDecl(methodDecl);
164 mbInsideFunction = false;
165 if (mbFunctionOnlyReturningOneOrZero)
166 problemFunctions.insert(methodDecl);
168 else
169 ret = FunctionAddress::TraverseCXXMethodDecl(methodDecl);
170 return ret;
173 bool ShouldReturnBool::IsInteresting(FunctionDecl const* functionDecl)
175 if (ignoreLocation(functionDecl))
176 return false;
177 // ignore stuff that forms part of the stable URE interface
178 if (isInUnoIncludeFile(functionDecl))
179 return false;
180 if (functionDecl->getTemplatedKind() != FunctionDecl::TK_NonTemplate)
181 return false;
182 if (!functionDecl->isThisDeclarationADefinition())
183 return false;
184 if (functionDecl->isMain())
185 return false;
186 if (functionDecl->isExternC() || functionDecl->isInExternCContext())
187 return false;
188 auto methodDecl = dyn_cast<CXXMethodDecl>(functionDecl);
189 if (methodDecl && methodDecl->isVirtual())
190 return false;
191 auto tc = loplugin::TypeCheck(functionDecl->getReturnType());
192 if (tc.AnyBoolean() || tc.Void())
193 return false;
194 auto returnType = functionDecl->getReturnType();
195 if (returnType->isEnumeralType() || !returnType->getUnqualifiedDesugaredType()->isIntegerType())
196 return false;
197 // Ignore functions that contains #ifdef-ery
198 if (containsPreprocessingConditionalInclusion(functionDecl->getSourceRange()))
199 return false;
201 // not sure what basegfx is doing here
202 StringRef fileName{ getFilenameOfLocation(functionDecl->getLocation()) };
203 if (loplugin::isSamePathname(fileName, SRCDIR "/include/basegfx/range/basicrange.hxx"))
204 return false;
205 // false +
206 if (loplugin::isSamePathname(fileName, SRCDIR "/include/svl/macitem.hxx"))
207 return false;
208 if (loplugin::isSamePathname(fileName, SRCDIR "/lotuswordpro/source/filter/lwpcharsetmgr.hxx"))
209 return false;
210 if (loplugin::isSamePathname(fileName, SRCDIR "/sc/inc/dptabsrc.hxx"))
211 return false;
213 return true;
216 bool ShouldReturnBool::VisitReturnStmt(const ReturnStmt* returnStmt)
218 if (!mbInsideFunction)
219 return true;
220 if (!returnStmt->getRetValue())
221 return true;
222 if (loplugin::TypeCheck(returnStmt->getRetValue()->getType()).AnyBoolean())
223 return true;
224 if (!isExprOneOrZero(returnStmt->getRetValue()))
225 mbFunctionOnlyReturningOneOrZero = false;
226 return true;
229 bool ShouldReturnBool::isExprOneOrZero(const Expr* arg) const
231 arg = arg->IgnoreParenCasts();
232 // ignore this, it seems to trigger an infinite recursion
233 if (isa<UnaryExprOrTypeTraitExpr>(arg))
235 return false;
237 APSInt x1;
238 if (compat::EvaluateAsInt(arg, x1, compiler.getASTContext()))
240 return x1 == 1 || x1 == 0;
242 return false;
245 loplugin::Plugin::Registration<ShouldReturnBool> X("shouldreturnbool", false);
248 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */