in src/checks/manuallevel/assert-with-side-effects.cpp [70:130]
void AssertWithSideEffects::VisitStmt(Stmt *stm)
{
const SourceLocation stmStart = stm->getBeginLoc();
if (!clazy::isInMacro(&m_astContext, stmStart, "Q_ASSERT")) {
return;
}
bool warn = false;
const bool checkfunctions = m_aggressiveness & AlsoCheckFunctionCallsAggressiveness;
auto *memberCall = dyn_cast<CXXMemberCallExpr>(stm);
if (memberCall) {
if (checkfunctions) {
CXXMethodDecl *method = memberCall->getMethodDecl();
if (!method->isConst() && !methodIsOK(clazy::qualifiedMethodName(method)) && !functionIsOk(clazy::name(method))) {
// llvm::errs() << "reason1 " << clazy::qualifiedMethodName(method) << "\n";
warn = true;
}
}
} else if (auto *call = dyn_cast<CallExpr>(stm)) {
// Non member function calls not allowed
FunctionDecl *func = call->getDirectCallee();
if (func && checkfunctions) {
if (isa<CXXMethodDecl>(func)) { // This will be visited next, so ignore it now
return;
}
if (functionIsOk(clazy::name(func))) {
return;
}
warn = true;
}
} else if (auto *op = dyn_cast<BinaryOperator>(stm)) {
if (op->isAssignmentOp()) {
if (auto *declRef = dyn_cast<DeclRefExpr>(op->getLHS())) {
ValueDecl *valueDecl = declRef->getDecl();
if (valueDecl && sm().isBeforeInSLocAddrSpace(valueDecl->getBeginLoc(), stmStart)) {
// llvm::errs() << "reason3\n";
warn = true;
}
}
}
} else if (auto *op = dyn_cast<UnaryOperator>(stm)) {
if (auto *declRef = dyn_cast<DeclRefExpr>(op->getSubExpr())) {
ValueDecl *valueDecl = declRef->getDecl();
auto type = op->getOpcode();
if (type != UnaryOperatorKind::UO_Deref && type != UnaryOperatorKind::UO_AddrOf) {
if (valueDecl && sm().isBeforeInSLocAddrSpace(valueDecl->getBeginLoc(), stmStart)) {
// llvm::errs() << "reason5 " << op->getOpcodeStr() << "\n";
warn = true;
}
}
}
}
if (warn) {
emitWarning(stmStart, "Code inside Q_ASSERT has side-effects but won't be built in release mode");
}
}