/* * Copyright (C) 2015 Andrei Karas * * This file is part of Paranoid null checker. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "analysis/statement.h" #include "command.h" #include "logger.h" #include "analysis/analysis.h" #include "analysis/reports.h" #include "analysis/walkitem.h" #include "nodes/expr/eq_expr.h" #include "nodes/expr/modify_expr.h" #include "nodes/expr/ne_expr.h" #include "nodes/expr/pointerplus_expr.h" #include "nodes/ref/indirect_ref.h" #include "nodes/stmt/if_stmt.h" #include #include "localconsts.h" namespace Analysis { void analyseCondition(Node *node, Node *condNode, Node *thenNode, Node *elseNode, const WalkItem &wi, WalkItem &wo) { condNode = skipNop(condNode); thenNode = skipNop(thenNode); elseNode = skipNop(elseNode); WalkItem wci = wi; WalkItem wco = wo; WalkItem wo2 = wo; WalkItem wo3 = wo; walkTree(condNode, wci, wco); Log::dumpWI(node, "wco ", wco); WalkItem wi2 = wi; removeNeedCheckNullVarsSetAll(wi2, wco.checkedNonNullVars); wi2.needCheckNullVars.insert(wco.checkedNullVars.begin(), wco.checkedNullVars.end()); wi2.knownNonNullVars.insert(wco.knownNonNullVars.begin(), wco.knownNonNullVars.end()); wi2.knownNullVars.insert(wco.knownNullVars.begin(), wco.knownNullVars.end()); Log::dumpWI(node, "wi2 then ", wi2); reportParmDeclNullPointer(node, thenNode, wi2); walkTree(thenNode, wi2, wo2); Log::dumpWI(node, "wo2 then ", wo2); WalkItem wi3 = wi; removeNeedCheckNullVarsSetAll(wi3, wco.checkedNullVars); wi3.needCheckNullVars.insert(wco.checkedNonNullVars.begin(), wco.checkedNonNullVars.end()); wi3.knownNullVars.insert(wco.knownNonNullVars.begin(), wco.knownNonNullVars.end()); wi3.knownNonNullVars.insert(wco.knownNullVars.begin(), wco.knownNullVars.end()); Log::dumpWI(node, "wi3 else ", wi3); reportParmDeclNullPointer(node, elseNode, wi3); walkTree(elseNode, wi3, wo3); Log::dumpWI(node, "wo3 else ", wo3); // probably condition wrong if (wo2.cleanExpr) mergeNullChecked(wo, wo2); // probably condition wrong if (wo3.cleanExpr) mergeNullChecked(wo, wo3); // need check for cleanExpr? intersectNonNullChecked(wo, wo2, wo3); if (wo2.isReturned) { // add variable for ignore for all parent nodes except special like IF_STMT FOR_EACH (it, wco.checkedNullVars) { wo.removeNullVarsAll.insert(it); removeNeedCheckNullVar(wo, it); } } if (wo3.isReturned) { // add variable for ignore for all parent nodes except special like IF_STMT FOR_EACH (it, wco.checkedNonNullVars) { wo.removeNullVarsAll.insert(it); removeNeedCheckNullVar(wo, it); } } if (wo2.isReturned && wo3.isReturned) { // add variable for ignore for all parent nodes except special like IF_STMT FOR_EACH (it, wo.needCheckNullVars) { wo.removeNullVarsAll.insert(it); } } wo.isReturned = false; wo.cleanExpr = true; wo.stopWalking = true; wo.uselessExpr = false; } void analyseIfStmt(IfStmtNode *node, const WalkItem &wi, WalkItem &wo) { // need condition node if (!node->condition || checkCommand(FindArgs)) return; analyseCondition(node, node->condition, node->thenNode, node->elseNode, wi, wo); } }