diff options
Diffstat (limited to 'lib/Checker/VLASizeChecker.cpp')
| -rw-r--r-- | lib/Checker/VLASizeChecker.cpp | 49 | 
1 files changed, 44 insertions, 5 deletions
diff --git a/lib/Checker/VLASizeChecker.cpp b/lib/Checker/VLASizeChecker.cpp index cea9d191aa77..936991d6133c 100644 --- a/lib/Checker/VLASizeChecker.cpp +++ b/lib/Checker/VLASizeChecker.cpp @@ -9,10 +9,13 @@  //  // This defines VLASizeChecker, a builtin check in GRExprEngine that   // performs checks for declaration of VLA of undefined or zero size. +// In addition, VLASizeChecker is responsible for defining the extent +// of the MemRegion that represents a VLA.  //  //===----------------------------------------------------------------------===//  #include "GRExprEngineInternalChecks.h" +#include "clang/AST/CharUnits.h"  #include "clang/Checker/BugReporter/BugType.h"  #include "clang/Checker/PathSensitive/CheckerVisitor.h"  #include "clang/Checker/PathSensitive/GRExprEngine.h" @@ -42,9 +45,9 @@ void VLASizeChecker::PreVisitDeclStmt(CheckerContext &C, const DeclStmt *DS) {    const VarDecl *VD = dyn_cast<VarDecl>(DS->getSingleDecl());    if (!VD)      return; -   -  const VariableArrayType *VLA -    = C.getASTContext().getAsVariableArrayType(VD->getType()); + +  ASTContext &Ctx = C.getASTContext(); +  const VariableArrayType *VLA = Ctx.getAsVariableArrayType(VD->getType());    if (!VLA)      return; @@ -70,9 +73,14 @@ void VLASizeChecker::PreVisitDeclStmt(CheckerContext &C, const DeclStmt *DS) {      C.EmitReport(report);      return;    } + +  // See if the size value is known. It can't be undefined because we would have +  // warned about that already. +  if (sizeV.isUnknown()) +    return;    // Check if the size is zero. -  DefinedOrUnknownSVal sizeD = cast<DefinedOrUnknownSVal>(sizeV); +  DefinedSVal sizeD = cast<DefinedSVal>(sizeV);    const GRState *stateNotZero, *stateZero;    llvm::tie(stateNotZero, stateZero) = state->Assume(sizeD); @@ -92,5 +100,36 @@ void VLASizeChecker::PreVisitDeclStmt(CheckerContext &C, const DeclStmt *DS) {    }    // From this point on, assume that the size is not zero. -  C.addTransition(stateNotZero); +  state = stateNotZero; + +  // VLASizeChecker is responsible for defining the extent of the array being +  // declared. We do this by multiplying the array length by the element size, +  // then matching that with the array region's extent symbol. + +  // Convert the array length to size_t. +  ValueManager &ValMgr = C.getValueManager(); +  SValuator &SV = ValMgr.getSValuator(); +  QualType SizeTy = Ctx.getSizeType(); +  NonLoc ArrayLength = cast<NonLoc>(SV.EvalCast(sizeD, SizeTy, SE->getType())); + +  // Get the element size. +  CharUnits EleSize = Ctx.getTypeSizeInChars(VLA->getElementType()); +  SVal EleSizeVal = ValMgr.makeIntVal(EleSize.getQuantity(), SizeTy); + +  // Multiply the array length by the element size. +  SVal ArraySizeVal = SV.EvalBinOpNN(state, BinaryOperator::Mul, ArrayLength, +                                     cast<NonLoc>(EleSizeVal), SizeTy); + +  // Finally, Assume that the array's extent matches the given size. +  const LocationContext *LC = C.getPredecessor()->getLocationContext(); +  DefinedOrUnknownSVal Extent = state->getRegion(VD, LC)->getExtent(ValMgr); +  DefinedOrUnknownSVal ArraySize = cast<DefinedOrUnknownSVal>(ArraySizeVal); +  DefinedOrUnknownSVal SizeIsKnown = SV.EvalEQ(state, Extent, ArraySize); +  state = state->Assume(SizeIsKnown, true); + +  // Assume should not fail at this point. +  assert(state); + +  // Remember our assumptions! +  C.addTransition(state);  }  | 
