CFGBlock* WalkAST_VisitChildren(Stmt* S);
CFGBlock* WalkAST_VisitVarDecl(VarDecl* D);
CFGBlock* WalkAST_VisitStmtExpr(StmtExpr* S);
+ CFGBlock* WalkAST_VisitCallExpr(CallExpr* C);
void FinishBlock(CFGBlock* B);
};
Succ = ConfluenceBlock;
Block = NULL;
CFGBlock* LHSBlock = Visit(C->getLHS());
+ FinishBlock(LHSBlock);
Succ = ConfluenceBlock;
Block = NULL;
CFGBlock* RHSBlock = Visit(C->getRHS());
+ FinishBlock(RHSBlock);
Block = createBlock(false);
Block->addSuccessor(LHSBlock);
Succ = ConfluenceBlock;
Block = NULL;
CFGBlock* LHSBlock = Visit(C->getLHS());
-
+ FinishBlock(LHSBlock);
+
Succ = ConfluenceBlock;
Block = NULL;
CFGBlock* RHSBlock = Visit(C->getRHS());
+ FinishBlock(RHSBlock);
Block = createBlock(false);
Block->addSuccessor(LHSBlock);
if (AlwaysAddStmt) Block->appendStmt(S);
return Block;
}
+
+ case Stmt::CallExprClass:
+ return WalkAST_VisitCallExpr(cast<CallExpr>(S));
case Stmt::StmtExprClass:
return WalkAST_VisitStmtExpr(cast<StmtExpr>(S));
return VisitCompoundStmt(S->getSubStmt());
}
+/// WalkAST_VisitCallExpr - Utility method to handle function calls that
+/// are nested in expressions. The idea is that each function call should
+/// appear as a distinct statement in the CFGBlock.
+CFGBlock* CFGBuilder::WalkAST_VisitCallExpr(CallExpr* C) {
+ Block->appendStmt(C);
+ return WalkAST_VisitChildren(C);
+}
+
/// VisitStmt - Handle statements with no branching control flow.
CFGBlock* CFGBuilder::VisitStmt(Stmt* Statement) {
// We cannot assume that we are in the middle of a basic block, since
// newly created blocks will be pointed to be "Block".
return addStmt(I->getCond());
}
+
CFGBlock* CFGBuilder::VisitReturnStmt(ReturnStmt* R) {
// If we were in the middle of a block we stop processing that block