/* -*- 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 #include #include #include #include #include #include #include "compat.hxx" #include "plugin.hxx" template<> struct std::iterator_traits { typedef std::ptrdiff_t difference_type; typedef Expr * value_type; typedef Expr const ** pointer; typedef Expr const & reference; typedef std::random_access_iterator_tag iterator_category; }; template<> struct std::iterator_traits { typedef std::ptrdiff_t difference_type; typedef Expr const * value_type; typedef Expr const ** pointer; typedef Expr const & reference; typedef std::random_access_iterator_tag iterator_category; }; namespace { Expr const * ignoreParenAndTemporaryMaterialization(Expr const * expr) { for (;;) { expr = expr->IgnoreParens(); auto e = dyn_cast(expr); if (e == nullptr) { return expr; } expr = e->GetTemporaryExpr(); } } Expr const * ignoreParenImpCastAndComma(Expr const * expr) { for (;;) { expr = expr->IgnoreParenImpCasts(); BinaryOperator const * op = dyn_cast(expr); if (op == nullptr || op->getOpcode() != BO_Comma) { return expr; } expr = op->getRHS(); } } SubstTemplateTypeParmType const * getAsSubstTemplateTypeParmType(QualType type) { //TODO: unwrap all kinds of (non-SubstTemplateTypeParmType) sugar, not only // TypedefType sugar: for (;;) { TypedefType const * t = type->getAs(); if (t == nullptr) { return dyn_cast(type); } type = t->desugar(); } } bool isBool(QualType type, bool allowTypedefs = true) { if (type->isBooleanType()) { return true; } if (!allowTypedefs) { return false; } TypedefType const * t2 = type->getAs(); if (t2 == nullptr) { return false; } std::string name(t2->getDecl()->getNameAsString()); return name == "sal_Bool" || name == "BOOL" || name == "Boolean" || name == "FT_Bool" || name == "FcBool" || name == "GLboolean" || name == "NPBool" || name == "UBool" || name == "dbus_bool_t" || name == "gboolean" || name == "hb_bool_t" || name == "jboolean"; } bool isBool(Expr const * expr, bool allowTypedefs = true) { return isBool(expr->getType(), allowTypedefs); } bool isBoolExpr(Expr const * expr) { if (isBool(expr)) { return true; } expr = ignoreParenImpCastAndComma(expr); ConditionalOperator const * co = dyn_cast(expr); if (co != nullptr) { ImplicitCastExpr const * ic1 = dyn_cast( co->getTrueExpr()->IgnoreParens()); ImplicitCastExpr const * ic2 = dyn_cast( co->getFalseExpr()->IgnoreParens()); if (ic1 != nullptr && ic2 != nullptr && ic1->getType()->isSpecificBuiltinType(BuiltinType::Int) && isBoolExpr(ic1->getSubExpr()->IgnoreParens()) && ic2->getType()->isSpecificBuiltinType(BuiltinType::Int) && isBoolExpr(ic2->getSubExpr()->IgnoreParens())) { return true; } } std::stack stack; Expr const * e = expr; for (;;) { e = ignoreParenImpCastAndComma(e); MemberExpr const * me = dyn_cast(e); if (me == nullptr) { break; } stack.push(e); e = me->getBase(); } for (;;) { e = ignoreParenImpCastAndComma(e); CXXOperatorCallExpr const * op = dyn_cast(e); if (op == nullptr || op->getOperator() != OO_Subscript) { break; } stack.push(e); e = op->getArg(0); } if (!stack.empty()) { TemplateSpecializationType const * t = e->getType()->getAs(); for (;;) { if (t == nullptr) { break; } QualType ty; MemberExpr const * me = dyn_cast(stack.top()); if (me != nullptr) { TemplateDecl const * td = t->getTemplateName().getAsTemplateDecl(); if (td == nullptr) { break; } TemplateParameterList const * ps = td->getTemplateParameters(); SubstTemplateTypeParmType const * t2 = getAsSubstTemplateTypeParmType( me->getMemberDecl()->getType()); if (t2 == nullptr) { break; } auto i = std::find( ps->begin(), ps->end(), t2->getReplacedParameter()->getDecl()); if (i == ps->end()) { break; } if (ps->size() != t->getNumArgs()) { //TODO break; } TemplateArgument const & arg = t->getArg(i - ps->begin()); if (arg.getKind() != TemplateArgument::Type) { break; } ty = arg.getAsType(); } else { CXXOperatorCallExpr const * op = dyn_cast(stack.top()); assert(op != nullptr); TemplateDecl const * d = t->getTemplateName().getAsTemplateDecl(); if (d == nullptr || (d->getQualifiedNameAsString() != "com::sun::star::uno::Sequence") || t->getNumArgs() != 1 || t->getArg(0).getKind() != TemplateArgument::Type) { break; } ty = t->getArg(0).getAsType(); } stack.pop(); if (stack.empty()) { if (isBool(ty)) { return true; } break; } t = ty->getAs(); } } return false; } // It appears that, given a function declaration, there is no way to determine // the language linkage of the function's type, only of the function's name // (via FunctionDecl::isExternC); however, in a case like // // extern "C" { static void f(); } // // the function's name does not have C language linkage while the function's // type does (as clarified in C++11 [decl.link]); cf. // "Language linkage of function type": bool hasCLanguageLinkageType(FunctionDecl const * decl) { assert(decl != nullptr); if (decl->isExternC()) { return true; } #if (__clang_major__ == 3 && __clang_minor__ >= 3) || __clang_major__ > 3 if (decl->isInExternCContext()) { return true; } #else if (decl->getCanonicalDecl()->getDeclContext()->isExternCContext()) { return true; } #endif return false; } class ImplicitBoolConversion: public RecursiveASTVisitor, public loplugin::Plugin { public: explicit ImplicitBoolConversion(InstantiationData const & data): Plugin(data) {} virtual void run() override { TraverseDecl(compiler.getASTContext().getTranslationUnitDecl()); } bool TraverseCallExpr(CallExpr * expr); bool TraverseCXXMemberCallExpr(CXXMemberCallExpr * expr); bool TraverseCXXConstructExpr(CXXConstructExpr * expr); bool TraverseCXXTemporaryObjectExpr(CXXTemporaryObjectExpr * expr); bool TraverseCStyleCastExpr(CStyleCastExpr * expr); bool TraverseCXXStaticCastExpr(CXXStaticCastExpr * expr); bool TraverseCXXFunctionalCastExpr(CXXFunctionalCastExpr * expr); bool TraverseConditionalOperator(ConditionalOperator * expr); bool TraverseBinLT(BinaryOperator * expr); bool TraverseBinLE(BinaryOperator * expr); bool TraverseBinGT(BinaryOperator * expr); bool TraverseBinGE(BinaryOperator * expr); bool TraverseBinEQ(BinaryOperator * expr); bool TraverseBinNE(BinaryOperator * expr); bool TraverseBinAssign(BinaryOperator * expr); bool TraverseBinAndAssign(CompoundAssignOperator * expr); bool TraverseBinOrAssign(CompoundAssignOperator * expr); bool TraverseBinXorAssign(CompoundAssignOperator * expr); bool TraverseReturnStmt(ReturnStmt * stmt); bool TraverseFunctionDecl(FunctionDecl * decl); bool VisitImplicitCastExpr(ImplicitCastExpr const * expr); private: bool isExternCFunctionCall( CallExpr const * expr, FunctionProtoType const ** functionType); bool isExternCFunctionCallReturningInt(Expr const * expr); void checkCXXConstructExpr(CXXConstructExpr const * expr); void reportWarning(ImplicitCastExpr const * expr); std::stack> nested; std::stack calls; bool externCIntFunctionDefinition = false; }; bool ImplicitBoolConversion::TraverseCallExpr(CallExpr * expr) { nested.push(std::vector()); calls.push(expr); bool ret = RecursiveASTVisitor::TraverseCallExpr(expr); FunctionProtoType const * t; bool ext = isExternCFunctionCall(expr, &t); assert(!nested.empty()); for (auto i: nested.top()) { auto j = std::find_if( expr->arg_begin(), expr->arg_end(), [&i](Expr * e) { return i == ignoreParenAndTemporaryMaterialization(e); }); if (j == expr->arg_end()) { reportWarning(i); } else { std::ptrdiff_t n = j - expr->arg_begin(); assert(n >= 0); if (ext) { if (t != nullptr) { assert( static_cast(n) < compat::getNumParams(*t) || t->isVariadic()); if (static_cast(n) < compat::getNumParams(*t) && !(compat::getParamType(*t, n)->isSpecificBuiltinType( BuiltinType::Int) || (compat::getParamType(*t, n) ->isSpecificBuiltinType(BuiltinType::UInt)) || (compat::getParamType(*t, n) ->isSpecificBuiltinType(BuiltinType::Long)))) { reportWarning(i); } } else { reportWarning(i); } } else { // Filter out // // template void f(T); // f(true); // DeclRefExpr const * dr = dyn_cast( expr->getCallee()->IgnoreParenImpCasts()); if (dr != nullptr && dr->hasExplicitTemplateArgs()) { FunctionDecl const * fd = dyn_cast(dr->getDecl()); if (fd != nullptr && static_cast(n) < fd->getNumParams()) { SubstTemplateTypeParmType const * t2 = getAsSubstTemplateTypeParmType( fd->getParamDecl(n)->getType() .getNonReferenceType()); if (t2 != nullptr) { //TODO: fix this superficial nonsense check: ASTTemplateArgumentListInfo const & ai = dr->getExplicitTemplateArgs(); if (ai.NumTemplateArgs == 1 && (ai[0].getArgument().getKind() == TemplateArgument::Type) && isBool(ai[0].getTypeSourceInfo()->getType())) { continue; } } } } reportWarning(i); } } } calls.pop(); nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCXXMemberCallExpr(CXXMemberCallExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCXXMemberCallExpr(expr); assert(!nested.empty()); for (auto i: nested.top()) { auto j = std::find_if( expr->arg_begin(), expr->arg_end(), [&i](Expr * e) { return i == ignoreParenAndTemporaryMaterialization(e); }); if (j != expr->arg_end()) { // Filter out // // template struct S { void f(T); }; // S s; // s.f(true); // std::ptrdiff_t n = j - expr->arg_begin(); assert(n >= 0); QualType ty = ignoreParenImpCastAndComma(expr->getImplicitObjectArgument()) ->getType(); if (dyn_cast(expr->getCallee())->isArrow()) { ty = ty->getAs()->getPointeeType(); } TemplateSpecializationType const * ct = ty->getAs(); CXXMethodDecl const * d = expr->getMethodDecl(); if (ct != nullptr && static_cast(n) < d->getNumParams()) { SubstTemplateTypeParmType const * pt = getAsSubstTemplateTypeParmType( d->getParamDecl(n)->getType().getNonReferenceType()); if (pt != nullptr) { TemplateDecl const * td = ct->getTemplateName().getAsTemplateDecl(); if (td != nullptr) { //TODO: fix this superficial nonsense check: if (ct->getNumArgs() >= 1 && ct->getArg(0).getKind() == TemplateArgument::Type && isBool(ct->getArg(0).getAsType())) { continue; } } } } } reportWarning(i); } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCXXConstructExpr(CXXConstructExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCXXConstructExpr(expr); checkCXXConstructExpr(expr); nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCXXTemporaryObjectExpr( CXXTemporaryObjectExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCXXTemporaryObjectExpr(expr); checkCXXConstructExpr(expr); nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCStyleCastExpr(CStyleCastExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCStyleCastExpr(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getSubExpr()->IgnoreParens()) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCXXStaticCastExpr(CXXStaticCastExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCXXStaticCastExpr(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getSubExpr()->IgnoreParens()) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseCXXFunctionalCastExpr( CXXFunctionalCastExpr * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseCXXFunctionalCastExpr(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getSubExpr()->IgnoreParens()) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseConditionalOperator( ConditionalOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseConditionalOperator(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getTrueExpr()->IgnoreParens() && (isBoolExpr(expr->getFalseExpr()->IgnoreParenImpCasts()) || isExternCFunctionCallReturningInt(expr->getFalseExpr()))) || (i == expr->getFalseExpr()->IgnoreParens() && (isBoolExpr(expr->getTrueExpr()->IgnoreParenImpCasts()) || isExternCFunctionCallReturningInt( expr->getTrueExpr()))))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinLT(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinLT(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinLE(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinLE(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinGT(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinGT(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinGE(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinGE(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinEQ(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinEQ(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinNE(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinNE(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (!((i == expr->getLHS()->IgnoreParens() && isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) || (i == expr->getRHS()->IgnoreParens() && isBool(expr->getLHS()->IgnoreParenImpCasts(), false)))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinAssign(BinaryOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinAssign(expr); // /usr/include/gtk-2.0/gtk/gtktogglebutton.h: struct _GtkToggleButton: // guint GSEAL (active) : 1; // even though : // "active" gboolean : Read / Write bool ext = false; MemberExpr const * me = dyn_cast(expr->getLHS()); if (me != nullptr) { FieldDecl const * fd = dyn_cast(me->getMemberDecl()); if (fd != nullptr && fd->isBitField() && fd->getBitWidthValue(compiler.getASTContext()) == 1) { TypedefType const * t = fd->getType()->getAs(); ext = t != nullptr && t->getDecl()->getNameAsString() == "guint"; } } assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getRHS()->IgnoreParens() || !(ext || isBoolExpr(expr->getLHS()))) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseBinAndAssign(CompoundAssignOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinAndAssign(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getRHS()->IgnoreParens() || !isBool(expr->getLHS()->IgnoreParens(), false)) { reportWarning(i); } } nested.pop(); if (!ignoreLocation(expr) && isBool(expr->getLHS(), false) && !isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) { report( DiagnosticsEngine::Warning, "mix of %0 and %1 in operator &=", expr->getRHS()->getLocStart()) << expr->getLHS()->getType() << expr->getRHS()->IgnoreParenImpCasts()->getType() << expr->getSourceRange(); } return ret; } bool ImplicitBoolConversion::TraverseBinOrAssign(CompoundAssignOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinOrAssign(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getRHS()->IgnoreParens() || !isBool(expr->getLHS()->IgnoreParens(), false)) { reportWarning(i); } } nested.pop(); if (!ignoreLocation(expr) && isBool(expr->getLHS(), false) && !isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) { report( DiagnosticsEngine::Warning, "mix of %0 and %1 in operator |=", expr->getRHS()->getLocStart()) << expr->getLHS()->getType() << expr->getRHS()->IgnoreParenImpCasts()->getType() << expr->getSourceRange(); } return ret; } bool ImplicitBoolConversion::TraverseBinXorAssign(CompoundAssignOperator * expr) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseBinXorAssign(expr); assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr->getRHS()->IgnoreParens() || !isBool(expr->getLHS()->IgnoreParens(), false)) { reportWarning(i); } } nested.pop(); if (!ignoreLocation(expr) && isBool(expr->getLHS(), false) && !isBool(expr->getRHS()->IgnoreParenImpCasts(), false)) { report( DiagnosticsEngine::Warning, "mix of %0 and %1 in operator ^=", expr->getRHS()->getLocStart()) << expr->getLHS()->getType() << expr->getRHS()->IgnoreParenImpCasts()->getType() << expr->getSourceRange(); } return ret; } bool ImplicitBoolConversion::TraverseReturnStmt(ReturnStmt * stmt) { nested.push(std::vector()); bool ret = RecursiveASTVisitor::TraverseReturnStmt(stmt); Expr const * expr = stmt->getRetValue(); if (expr != nullptr) { ExprWithCleanups const * ec = dyn_cast(expr); if (ec != nullptr) { expr = ec->getSubExpr(); } expr = expr->IgnoreParens(); } assert(!nested.empty()); for (auto i: nested.top()) { if (i != expr || !externCIntFunctionDefinition) { reportWarning(i); } } nested.pop(); return ret; } bool ImplicitBoolConversion::TraverseFunctionDecl(FunctionDecl * decl) { bool ext = false; if (hasCLanguageLinkageType(decl) && decl->isThisDeclarationADefinition()) { QualType t { compat::getReturnType(*decl) }; if (t->isSpecificBuiltinType(BuiltinType::Int) || t->isSpecificBuiltinType(BuiltinType::UInt)) { ext = true; } else { TypedefType const * t2 = t->getAs(); // cf. rtl_locale_equals (and sal_Int32 can be long): if (t2 != nullptr && t2->getDecl()->getNameAsString() == "sal_Int32") { ext = true; } } } if (ext) { assert(!externCIntFunctionDefinition); externCIntFunctionDefinition = true; } bool ret = RecursiveASTVisitor::TraverseFunctionDecl(decl); if (ext) { externCIntFunctionDefinition = false; } return ret; } bool ImplicitBoolConversion::VisitImplicitCastExpr( ImplicitCastExpr const * expr) { if (ignoreLocation(expr)) { return true; } if (expr->getSubExprAsWritten()->getType()->isBooleanType() && !isBool(expr)) { if (nested.empty()) { reportWarning(expr); } else { nested.top().push_back(expr); } return true; } ExplicitCastExpr const * sub = dyn_cast( expr->getSubExpr()->IgnoreParenImpCasts()); if (sub != nullptr && (sub->getSubExpr()->IgnoreParenImpCasts()->getType().IgnoreParens() == expr->getType().IgnoreParens()) && isBool(sub->getSubExpr()->IgnoreParenImpCasts())) { report( DiagnosticsEngine::Warning, "explicit conversion (%0) from %1 to %2 implicitly cast back to %3", expr->getLocStart()) << sub->getCastKindName() << sub->getSubExpr()->IgnoreParenImpCasts()->getType() << sub->getType() << expr->getType() << expr->getSourceRange(); return true; } if (expr->getType()->isBooleanType() && !isBoolExpr(expr->getSubExpr()) && !calls.empty()) { CallExpr const * call = calls.top(); if (std::find_if( call->arg_begin(), call->arg_end(), [expr](Expr const * e) { return expr == e->IgnoreParens(); }) != call->arg_end()) { report( DiagnosticsEngine::Warning, "implicit conversion (%0) of call argument from %1 to %2", expr->getLocStart()) << expr->getCastKindName() << expr->getSubExpr()->getType() << expr->getType() << expr->getSourceRange(); return true; } } return true; } bool ImplicitBoolConversion::isExternCFunctionCall( CallExpr const * expr, FunctionProtoType const ** functionType) { assert(functionType != nullptr); *functionType = nullptr; Decl const * d = expr->getCalleeDecl(); if (d != nullptr) { FunctionDecl const * fd = dyn_cast(d); if (fd != nullptr && (fd->isExternC() || compiler.getSourceManager().isInExternCSystemHeader( fd->getLocation()))) { PointerType const * pt = fd->getType()->getAs(); QualType t2(pt == nullptr ? fd->getType() : pt->getPointeeType()); *functionType = t2->getAs(); assert( *functionType != nullptr || !compiler.getLangOpts().CPlusPlus || (fd->getBuiltinID() != Builtin::NotBuiltin && isa(t2))); // __builtin_*s have no proto type? return true; } VarDecl const * vd = dyn_cast(d); if (vd != nullptr && vd->isExternC()) { PointerType const * pt = vd->getType()->getAs(); *functionType = ((pt == nullptr ? vd->getType() : pt->getPointeeType()) ->castAs()); return true; } } return false; } bool ImplicitBoolConversion::isExternCFunctionCallReturningInt( Expr const * expr) { CallExpr const * e = dyn_cast(expr->IgnoreParenImpCasts()); FunctionProtoType const * t; return e != nullptr && e->getType()->isSpecificBuiltinType(BuiltinType::Int) && isExternCFunctionCall(e, &t); } void ImplicitBoolConversion::checkCXXConstructExpr( CXXConstructExpr const * expr) { assert(!nested.empty()); for (auto i: nested.top()) { auto j = std::find_if( expr->arg_begin(), expr->arg_end(), [&i](Expr const * e) { return i == ignoreParenAndTemporaryMaterialization(e); }); if (j != expr->arg_end()) { TemplateSpecializationType const * t1 = expr->getType()-> getAs(); SubstTemplateTypeParmType const * t2 = nullptr; CXXConstructorDecl const * d = expr->getConstructor(); if (d->getNumParams() == expr->getNumArgs()) { //TODO: better check t2 = getAsSubstTemplateTypeParmType( d->getParamDecl(j - expr->arg_begin())->getType() .getNonReferenceType()); } if (t1 != nullptr && t2 != nullptr) { TemplateDecl const * td = t1->getTemplateName().getAsTemplateDecl(); if (td != nullptr) { TemplateParameterList const * ps = td->getTemplateParameters(); auto i = std::find( ps->begin(), ps->end(), t2->getReplacedParameter()->getDecl()); if (i != ps->end()) { if (ps->size() == t1->getNumArgs()) { //TODO TemplateArgument const & arg = t1->getArg( i - ps->begin()); if (arg.getKind() == TemplateArgument::Type && isBool(arg.getAsType())) { continue; } } } } } } reportWarning(i); } } void ImplicitBoolConversion::reportWarning(ImplicitCastExpr const * expr) { report( DiagnosticsEngine::Warning, "implicit conversion (%0) from bool to %1", expr->getLocStart()) << expr->getCastKindName() << expr->getType() << expr->getSourceRange(); } loplugin::Plugin::Registration X( "implicitboolconversion"); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */