//===--- UndefinedArgChecker.h - Undefined arguments checker ----*- C++ -*--==// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This defines BadCallChecker, a builtin check in GRExprEngine that performs // checks for undefined arguments. // //===----------------------------------------------------------------------===// #include "clang/Analysis/PathSensitive/CheckerVisitor.h" #include "clang/Analysis/PathSensitive/BugReporter.h" #include "GRExprEngineInternalChecks.h" using namespace clang; namespace { class VISIBILITY_HIDDEN UndefinedArgChecker : public CheckerVisitor { BugType *BT_call; BugType *BT_msg; public: UndefinedArgChecker() : BT_call(0), BT_msg(0) {} static void *getTag() { static int x = 0; return &x; } void PreVisitCallExpr(CheckerContext &C, const CallExpr *CE); void PreVisitObjCMessageExpr(CheckerContext &C, const ObjCMessageExpr *ME); }; } // end anonymous namespace void clang::RegisterUndefinedArgChecker(GRExprEngine &Eng) { Eng.registerCheck(new UndefinedArgChecker()); } void UndefinedArgChecker::PreVisitCallExpr(CheckerContext &C, const CallExpr *CE){ for (CallExpr::const_arg_iterator I = CE->arg_begin(), E = CE->arg_end(); I != E; ++I) { if (C.getState()->getSVal(*I).isUndef()) { if (ExplodedNode *N = C.GenerateNode(CE, true)) { if (!BT_call) BT_call = new BuiltinBug("Pass-by-value argument in function call is " "undefined"); // Generate a report for this bug. EnhancedBugReport *R = new EnhancedBugReport(*BT_call, BT_call->getName(), N); R->addRange((*I)->getSourceRange()); R->addVisitorCreator(bugreporter::registerTrackNullOrUndefValue, *I); C.EmitReport(R); return; } } } } void UndefinedArgChecker::PreVisitObjCMessageExpr(CheckerContext &C, const ObjCMessageExpr *ME) { // Check for any arguments that are uninitialized/undefined. const GRState *state = C.getState(); for (ObjCMessageExpr::const_arg_iterator I = ME->arg_begin(), E = ME->arg_end(); I != E; ++I) { if (state->getSVal(*I).isUndef()) { if (ExplodedNode *N = C.GenerateNode(ME, true)) { if (!BT_msg) BT_msg = new BuiltinBug("Pass-by-value argument in message expression" " is undefined"); // Generate a report for this bug. EnhancedBugReport *R = new EnhancedBugReport(*BT_msg, BT_msg->getName(), N); R->addRange((*I)->getSourceRange()); R->addVisitorCreator(bugreporter::registerTrackNullOrUndefValue, *I); C.EmitReport(R); return; } } } }