/* * Copyright (C) 2015 Andrei Karas * * This file is part of AstDumper. * * 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/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; removeCheckNullVarsSet(wi2, wco.checkedNonNullVars); wi2.checkNullVars.insert(wco.checkedNullVars.begin(), wco.checkedNullVars.end()); Log::dumpWI(node, "wi2 then ", wi2); reportParmDeclNullPointer(node, thenNode, wi2); walkTree(thenNode, wi2, wo2); Log::dumpWI(node, "wo2 then ", wo2); WalkItem wi3 = wi; removeCheckNullVarsSet(wi3, wco.checkedNullVars); wi3.checkNullVars.insert(wco.checkedNonNullVars.begin(), wco.checkedNonNullVars.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 (std::set::const_iterator, it, wco.checkedNullVars) { wo.removeNullVars.insert(*it); wo.checkNullVars.erase(*it); wo.addNullVars.erase(*it); } } if (wo3.isReturned) { // add variable for ignore for all parent nodes except special like IF_STMT FOR_EACH (std::set::const_iterator, it, wco.checkedNonNullVars) { wo.removeNullVars.insert(*it); wo.checkNullVars.erase(*it); wo.addNullVars.erase(*it); } } if (wo2.isReturned && wo3.isReturned) { // add variable for ignore for all parent nodes except special like IF_STMT FOR_EACH (std::set::const_iterator, it, wo.checkNullVars) { wo.removeNullVars.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 || command == FindArgs) return; analyseCondition(node, node->condition, node->thenNode, node->elseNode, wi, wo); } }