<html><head></head><body style="word-wrap: break-word; -webkit-nbsp-mode: space; -webkit-line-break: after-white-space; ">Hi Lei,<div><br></div><div>I think this is a good first step. Comments inline.</div><div><br><div><div>On Dec 6, 2010, at 5:59 PM, 章磊 wrote:</div><br class="Apple-interchange-newline"><blockquote type="cite">Hi clang,<br><br>I'm trying to find out how to implement this statistic checker. IMO, there could be several steps.<br><div style="margin-left: 40px;"><ol><li>Add a ConjuredSymbol to all the CallExpr in PostVisitCallExpr, set the state to Unchecked.</li></ol></div></blockquote><div>CallExprs are almost always guaranteed to evaluate to a symbolic value, so you don't need to create a new one. For example:</div><div><br></div><div><div>+void UncheckedReturnChecker::PostVisitCallExpr(CheckerContext &C, </div><div>+ const CallExpr *CE) {</div><div>+ unsigned Count = C.getNodeBuilder().getCurrentBlockCount();</div><div>+ SValBuilder &svalBuilder = C.getSValBuilder();</div><div>+ SymbolRef Sym = svalBuilder.getConjuredSymbolVal(0, CE, Count).getAsSymbol();</div><div>+</div><div>+ const GRState *state = C.getState();</div><div>+</div><div>+ // Once we encounter a call expr, set the initial state to Unchecked.</div><div>+ state = </div><div>+ state->set<CheckedReturnState>(Sym, CheckedReturnState::getUnchecked(CE));</div><div>+</div><div>+ C.addTransition(state);</div><div>+ return;</div><div>+}</div><div><br></div><div>Instead of conjuring a new symbol, try querying the SVal for the CallExpr. If it is a symbol, you can just add it to the GDM. For example:</div><div><br></div><div>const GRState *state = C.getState();</div><div>if (SymExpr *sym = state->getSVal(CE)->getAsSymbol()) {</div><div> state = state->set<CheckedReturnState>(sym, CheckReturnState::getUnchecked(CE));</div><div> C.addTransition(state);</div><div>}</div></div><div><br></div><div>The nice thing about this approach is that you associate your tracking state with the actual symbol returned by the CallExpr. That value is then tracked through assignments, etc.</div><br><blockquote type="cite"><div style="margin-left: 40px;"><ol start="2">
<li>In VisitBranchCondition, if a callexpr is checked, set the state to checked.</li></ol></div></blockquote><div>I think this logic looks alright, although it takes a highly syntactic approach instead of reasoning about symbolic values. One shortcoming of this approach is that it won't handle:</div><div><br></div><div> p = foo();</div><div> bool b = p == NULL;</div><div> if (b) { ... }</div><div><br></div><div>It is possible to handle this with a more sophisticated (albeit more complicated analysis) that reasons about symbolic values. Essentially, 'p == NULL' would evaluate to an SVal for which you could attach tracking state. Then, in VisitBranchCondition() you only need to reason about the SVal for the condition instead of its raw syntactic form.</div><br><blockquote type="cite"><div style="margin-left: 40px;"><ol start="3"><li>After all the decl was analyzed, do the first part(in a translationunit) of statistic count (and/or check) in AnalysisConsumer::HandleTranslationUnit.</li></ol></div></blockquote><div>Seems reasonable.</div><blockquote type="cite"><div style="margin-left: 40px;"><ol start="4">
<li>Implement a script to do the whole project's statistic check.</li></ol></div></blockquote>Seems reasonable.</div><div><br><blockquote type="cite">This patch try to do the first two steps. <br><br>I register the checker in RegisterExperimentalChecks, but a new action seems more appropriate. Statistic checkers are some different from the checkers in RegisterExperimentalChecks.<br></blockquote><div><br></div><div>Experimental checks is just a catch-all bucket for checks that are in development, but I agree with you that I think it makes sense to add a new flag for this case.</div><br><blockquote type="cite">
<br>In order to generate ExplodedNode but not a blockedge, i add a new "generateNode" function in GRBranchNodeBuilder. I think this function is not quite right. I use poststmt as the ProgramPoint, maybe we need a new ProgramPoint here?<br></blockquote><div><br></div><div>Yes, we'll likely need a new ProgramPoint. Since the condition is a Stmt*, it will have already been evaluated (with ExplodedNodes with a PostStmt program point). You might get unintended caching in the ExplodedGraph.</div><br><blockquote type="cite">
<br>Beacause this is a really experimental checker and not all the steps are finished, i can't give a testcase to show what i did. <br><br>More comments inline.<br></blockquote></div><br></div><div>One other nit is that the checker should implement 'evalDeadSymbols()' so that it clears out tracked symbols from the state. Otherwise we'll get artificial path explosion. Eventually we'll probably engineer the GDM so that it does this automatically for common cases.</div></body></html>