xref: /netbsd-src/external/apache2/llvm/dist/clang/lib/StaticAnalyzer/Checkers/CastValueChecker.cpp (revision e038c9c4676b0f19b1b7dd08a940c6ed64a6d5ae)
17330f729Sjoerg //===- CastValueChecker - Model implementation of custom RTTIs --*- C++ -*-===//
27330f729Sjoerg //
37330f729Sjoerg // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
47330f729Sjoerg // See https://llvm.org/LICENSE.txt for license information.
57330f729Sjoerg // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
67330f729Sjoerg //
77330f729Sjoerg //===----------------------------------------------------------------------===//
87330f729Sjoerg //
97330f729Sjoerg //  This defines CastValueChecker which models casts of custom RTTIs.
107330f729Sjoerg //
117330f729Sjoerg // TODO list:
127330f729Sjoerg // - It only allows one succesful cast between two types however in the wild
137330f729Sjoerg //   the object could be casted to multiple types.
147330f729Sjoerg // - It needs to check the most likely type information from the dynamic type
157330f729Sjoerg //   map to increase precision of dynamic casting.
167330f729Sjoerg //
177330f729Sjoerg //===----------------------------------------------------------------------===//
187330f729Sjoerg 
197330f729Sjoerg #include "clang/AST/DeclTemplate.h"
207330f729Sjoerg #include "clang/StaticAnalyzer/Checkers/BuiltinCheckerRegistration.h"
217330f729Sjoerg #include "clang/StaticAnalyzer/Core/Checker.h"
227330f729Sjoerg #include "clang/StaticAnalyzer/Core/CheckerManager.h"
237330f729Sjoerg #include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
247330f729Sjoerg #include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
257330f729Sjoerg #include "clang/StaticAnalyzer/Core/PathSensitive/DynamicType.h"
267330f729Sjoerg #include "llvm/ADT/Optional.h"
277330f729Sjoerg #include <utility>
287330f729Sjoerg 
297330f729Sjoerg using namespace clang;
307330f729Sjoerg using namespace ento;
317330f729Sjoerg 
327330f729Sjoerg namespace {
33*e038c9c4Sjoerg class CastValueChecker : public Checker<check::DeadSymbols, eval::Call> {
347330f729Sjoerg   enum class CallKind { Function, Method, InstanceOf };
357330f729Sjoerg 
367330f729Sjoerg   using CastCheck =
377330f729Sjoerg       std::function<void(const CastValueChecker *, const CallEvent &Call,
387330f729Sjoerg                          DefinedOrUnknownSVal, CheckerContext &)>;
397330f729Sjoerg 
407330f729Sjoerg public:
417330f729Sjoerg   // We have five cases to evaluate a cast:
427330f729Sjoerg   // 1) The parameter is non-null, the return value is non-null.
437330f729Sjoerg   // 2) The parameter is non-null, the return value is null.
447330f729Sjoerg   // 3) The parameter is null, the return value is null.
457330f729Sjoerg   // cast: 1;  dyn_cast: 1, 2;  cast_or_null: 1, 3;  dyn_cast_or_null: 1, 2, 3.
467330f729Sjoerg   //
477330f729Sjoerg   // 4) castAs: Has no parameter, the return value is non-null.
487330f729Sjoerg   // 5) getAs:  Has no parameter, the return value is null or non-null.
497330f729Sjoerg   //
507330f729Sjoerg   // We have two cases to check the parameter is an instance of the given type.
517330f729Sjoerg   // 1) isa:             The parameter is non-null, returns boolean.
527330f729Sjoerg   // 2) isa_and_nonnull: The parameter is null or non-null, returns boolean.
537330f729Sjoerg   bool evalCall(const CallEvent &Call, CheckerContext &C) const;
54*e038c9c4Sjoerg   void checkDeadSymbols(SymbolReaper &SR, CheckerContext &C) const;
557330f729Sjoerg 
567330f729Sjoerg private:
577330f729Sjoerg   // These are known in the LLVM project. The pairs are in the following form:
587330f729Sjoerg   // {{{namespace, call}, argument-count}, {callback, kind}}
597330f729Sjoerg   const CallDescriptionMap<std::pair<CastCheck, CallKind>> CDM = {
607330f729Sjoerg       {{{"llvm", "cast"}, 1},
617330f729Sjoerg        {&CastValueChecker::evalCast, CallKind::Function}},
627330f729Sjoerg       {{{"llvm", "dyn_cast"}, 1},
637330f729Sjoerg        {&CastValueChecker::evalDynCast, CallKind::Function}},
647330f729Sjoerg       {{{"llvm", "cast_or_null"}, 1},
657330f729Sjoerg        {&CastValueChecker::evalCastOrNull, CallKind::Function}},
667330f729Sjoerg       {{{"llvm", "dyn_cast_or_null"}, 1},
677330f729Sjoerg        {&CastValueChecker::evalDynCastOrNull, CallKind::Function}},
687330f729Sjoerg       {{{"clang", "castAs"}, 0},
697330f729Sjoerg        {&CastValueChecker::evalCastAs, CallKind::Method}},
707330f729Sjoerg       {{{"clang", "getAs"}, 0},
717330f729Sjoerg        {&CastValueChecker::evalGetAs, CallKind::Method}},
727330f729Sjoerg       {{{"llvm", "isa"}, 1},
737330f729Sjoerg        {&CastValueChecker::evalIsa, CallKind::InstanceOf}},
747330f729Sjoerg       {{{"llvm", "isa_and_nonnull"}, 1},
757330f729Sjoerg        {&CastValueChecker::evalIsaAndNonNull, CallKind::InstanceOf}}};
767330f729Sjoerg 
777330f729Sjoerg   void evalCast(const CallEvent &Call, DefinedOrUnknownSVal DV,
787330f729Sjoerg                 CheckerContext &C) const;
797330f729Sjoerg   void evalDynCast(const CallEvent &Call, DefinedOrUnknownSVal DV,
807330f729Sjoerg                    CheckerContext &C) const;
817330f729Sjoerg   void evalCastOrNull(const CallEvent &Call, DefinedOrUnknownSVal DV,
827330f729Sjoerg                       CheckerContext &C) const;
837330f729Sjoerg   void evalDynCastOrNull(const CallEvent &Call, DefinedOrUnknownSVal DV,
847330f729Sjoerg                          CheckerContext &C) const;
857330f729Sjoerg   void evalCastAs(const CallEvent &Call, DefinedOrUnknownSVal DV,
867330f729Sjoerg                   CheckerContext &C) const;
877330f729Sjoerg   void evalGetAs(const CallEvent &Call, DefinedOrUnknownSVal DV,
887330f729Sjoerg                  CheckerContext &C) const;
897330f729Sjoerg   void evalIsa(const CallEvent &Call, DefinedOrUnknownSVal DV,
907330f729Sjoerg                CheckerContext &C) const;
917330f729Sjoerg   void evalIsaAndNonNull(const CallEvent &Call, DefinedOrUnknownSVal DV,
927330f729Sjoerg                          CheckerContext &C) const;
937330f729Sjoerg };
947330f729Sjoerg } // namespace
957330f729Sjoerg 
isInfeasibleCast(const DynamicCastInfo * CastInfo,bool CastSucceeds)967330f729Sjoerg static bool isInfeasibleCast(const DynamicCastInfo *CastInfo,
977330f729Sjoerg                              bool CastSucceeds) {
987330f729Sjoerg   if (!CastInfo)
997330f729Sjoerg     return false;
1007330f729Sjoerg 
1017330f729Sjoerg   return CastSucceeds ? CastInfo->fails() : CastInfo->succeeds();
1027330f729Sjoerg }
1037330f729Sjoerg 
getNoteTag(CheckerContext & C,const DynamicCastInfo * CastInfo,QualType CastToTy,const Expr * Object,bool CastSucceeds,bool IsKnownCast)1047330f729Sjoerg static const NoteTag *getNoteTag(CheckerContext &C,
1057330f729Sjoerg                                  const DynamicCastInfo *CastInfo,
1067330f729Sjoerg                                  QualType CastToTy, const Expr *Object,
1077330f729Sjoerg                                  bool CastSucceeds, bool IsKnownCast) {
1087330f729Sjoerg   std::string CastToName =
109*e038c9c4Sjoerg       CastInfo ? CastInfo->to()->getAsCXXRecordDecl()->getNameAsString()
1107330f729Sjoerg                : CastToTy->getPointeeCXXRecordDecl()->getNameAsString();
1117330f729Sjoerg   Object = Object->IgnoreParenImpCasts();
1127330f729Sjoerg 
1137330f729Sjoerg   return C.getNoteTag(
1147330f729Sjoerg       [=]() -> std::string {
1157330f729Sjoerg         SmallString<128> Msg;
1167330f729Sjoerg         llvm::raw_svector_ostream Out(Msg);
1177330f729Sjoerg 
1187330f729Sjoerg         if (!IsKnownCast)
1197330f729Sjoerg           Out << "Assuming ";
1207330f729Sjoerg 
1217330f729Sjoerg         if (const auto *DRE = dyn_cast<DeclRefExpr>(Object)) {
122*e038c9c4Sjoerg           Out << '\'' << DRE->getDecl()->getDeclName() << '\'';
123*e038c9c4Sjoerg         } else if (const auto *ME = dyn_cast<MemberExpr>(Object)) {
124*e038c9c4Sjoerg           Out << (IsKnownCast ? "Field '" : "field '")
125*e038c9c4Sjoerg               << ME->getMemberDecl()->getDeclName() << '\'';
126*e038c9c4Sjoerg         } else {
127*e038c9c4Sjoerg           Out << (IsKnownCast ? "The object" : "the object");
128*e038c9c4Sjoerg         }
129*e038c9c4Sjoerg 
130*e038c9c4Sjoerg         Out << ' ' << (CastSucceeds ? "is a" : "is not a") << " '" << CastToName
131*e038c9c4Sjoerg             << '\'';
132*e038c9c4Sjoerg 
133*e038c9c4Sjoerg         return std::string(Out.str());
134*e038c9c4Sjoerg       },
135*e038c9c4Sjoerg       /*IsPrunable=*/true);
136*e038c9c4Sjoerg }
137*e038c9c4Sjoerg 
getNoteTag(CheckerContext & C,SmallVector<QualType,4> CastToTyVec,const Expr * Object,bool IsKnownCast)138*e038c9c4Sjoerg static const NoteTag *getNoteTag(CheckerContext &C,
139*e038c9c4Sjoerg                                  SmallVector<QualType, 4> CastToTyVec,
140*e038c9c4Sjoerg                                  const Expr *Object,
141*e038c9c4Sjoerg                                  bool IsKnownCast) {
142*e038c9c4Sjoerg   Object = Object->IgnoreParenImpCasts();
143*e038c9c4Sjoerg 
144*e038c9c4Sjoerg   return C.getNoteTag(
145*e038c9c4Sjoerg       [=]() -> std::string {
146*e038c9c4Sjoerg         SmallString<128> Msg;
147*e038c9c4Sjoerg         llvm::raw_svector_ostream Out(Msg);
148*e038c9c4Sjoerg 
149*e038c9c4Sjoerg         if (!IsKnownCast)
150*e038c9c4Sjoerg           Out << "Assuming ";
151*e038c9c4Sjoerg 
152*e038c9c4Sjoerg         if (const auto *DRE = dyn_cast<DeclRefExpr>(Object)) {
1537330f729Sjoerg           Out << '\'' << DRE->getDecl()->getNameAsString() << '\'';
1547330f729Sjoerg         } else if (const auto *ME = dyn_cast<MemberExpr>(Object)) {
1557330f729Sjoerg           Out << (IsKnownCast ? "Field '" : "field '")
1567330f729Sjoerg               << ME->getMemberDecl()->getNameAsString() << '\'';
1577330f729Sjoerg         } else {
1587330f729Sjoerg           Out << (IsKnownCast ? "The object" : "the object");
1597330f729Sjoerg         }
160*e038c9c4Sjoerg         Out << " is";
1617330f729Sjoerg 
162*e038c9c4Sjoerg         bool First = true;
163*e038c9c4Sjoerg         for (QualType CastToTy: CastToTyVec) {
164*e038c9c4Sjoerg           std::string CastToName =
165*e038c9c4Sjoerg             CastToTy->getAsCXXRecordDecl() ?
166*e038c9c4Sjoerg             CastToTy->getAsCXXRecordDecl()->getNameAsString() :
167*e038c9c4Sjoerg             CastToTy->getPointeeCXXRecordDecl()->getNameAsString();
168*e038c9c4Sjoerg           Out << ' ' << ((CastToTyVec.size() == 1) ? "not" :
169*e038c9c4Sjoerg                          (First ? "neither" : "nor")) << " a '" << CastToName
1707330f729Sjoerg               << '\'';
171*e038c9c4Sjoerg           First = false;
172*e038c9c4Sjoerg         }
1737330f729Sjoerg 
174*e038c9c4Sjoerg         return std::string(Out.str());
1757330f729Sjoerg       },
1767330f729Sjoerg       /*IsPrunable=*/true);
1777330f729Sjoerg }
1787330f729Sjoerg 
1797330f729Sjoerg //===----------------------------------------------------------------------===//
1807330f729Sjoerg // Main logic to evaluate a cast.
1817330f729Sjoerg //===----------------------------------------------------------------------===//
1827330f729Sjoerg 
alignReferenceTypes(QualType toAlign,QualType alignTowards,ASTContext & ACtx)1837330f729Sjoerg static QualType alignReferenceTypes(QualType toAlign, QualType alignTowards,
1847330f729Sjoerg                                     ASTContext &ACtx) {
1857330f729Sjoerg   if (alignTowards->isLValueReferenceType() &&
1867330f729Sjoerg       alignTowards.isConstQualified()) {
1877330f729Sjoerg     toAlign.addConst();
1887330f729Sjoerg     return ACtx.getLValueReferenceType(toAlign);
1897330f729Sjoerg   } else if (alignTowards->isLValueReferenceType())
1907330f729Sjoerg     return ACtx.getLValueReferenceType(toAlign);
1917330f729Sjoerg   else if (alignTowards->isRValueReferenceType())
1927330f729Sjoerg     return ACtx.getRValueReferenceType(toAlign);
1937330f729Sjoerg 
1947330f729Sjoerg   llvm_unreachable("Must align towards a reference type!");
1957330f729Sjoerg }
1967330f729Sjoerg 
addCastTransition(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C,bool IsNonNullParam,bool IsNonNullReturn,bool IsCheckedCast=false)1977330f729Sjoerg static void addCastTransition(const CallEvent &Call, DefinedOrUnknownSVal DV,
1987330f729Sjoerg                               CheckerContext &C, bool IsNonNullParam,
1997330f729Sjoerg                               bool IsNonNullReturn,
2007330f729Sjoerg                               bool IsCheckedCast = false) {
2017330f729Sjoerg   ProgramStateRef State = C.getState()->assume(DV, IsNonNullParam);
2027330f729Sjoerg   if (!State)
2037330f729Sjoerg     return;
2047330f729Sjoerg 
2057330f729Sjoerg   const Expr *Object;
2067330f729Sjoerg   QualType CastFromTy;
2077330f729Sjoerg   QualType CastToTy = Call.getResultType();
2087330f729Sjoerg 
2097330f729Sjoerg   if (Call.getNumArgs() > 0) {
2107330f729Sjoerg     Object = Call.getArgExpr(0);
2117330f729Sjoerg     CastFromTy = Call.parameters()[0]->getType();
2127330f729Sjoerg   } else {
2137330f729Sjoerg     Object = cast<CXXInstanceCall>(&Call)->getCXXThisExpr();
2147330f729Sjoerg     CastFromTy = Object->getType();
2157330f729Sjoerg     if (CastToTy->isPointerType()) {
2167330f729Sjoerg       if (!CastFromTy->isPointerType())
2177330f729Sjoerg         return;
2187330f729Sjoerg     } else {
2197330f729Sjoerg       if (!CastFromTy->isReferenceType())
2207330f729Sjoerg         return;
2217330f729Sjoerg 
2227330f729Sjoerg       CastFromTy = alignReferenceTypes(CastFromTy, CastToTy, C.getASTContext());
2237330f729Sjoerg     }
2247330f729Sjoerg   }
2257330f729Sjoerg 
2267330f729Sjoerg   const MemRegion *MR = DV.getAsRegion();
2277330f729Sjoerg   const DynamicCastInfo *CastInfo =
2287330f729Sjoerg       getDynamicCastInfo(State, MR, CastFromTy, CastToTy);
2297330f729Sjoerg 
2307330f729Sjoerg   // We assume that every checked cast succeeds.
2317330f729Sjoerg   bool CastSucceeds = IsCheckedCast || CastFromTy == CastToTy;
2327330f729Sjoerg   if (!CastSucceeds) {
2337330f729Sjoerg     if (CastInfo)
2347330f729Sjoerg       CastSucceeds = IsNonNullReturn && CastInfo->succeeds();
2357330f729Sjoerg     else
2367330f729Sjoerg       CastSucceeds = IsNonNullReturn;
2377330f729Sjoerg   }
2387330f729Sjoerg 
2397330f729Sjoerg   // Check for infeasible casts.
2407330f729Sjoerg   if (isInfeasibleCast(CastInfo, CastSucceeds)) {
2417330f729Sjoerg     C.generateSink(State, C.getPredecessor());
2427330f729Sjoerg     return;
2437330f729Sjoerg   }
2447330f729Sjoerg 
2457330f729Sjoerg   // Store the type and the cast information.
2467330f729Sjoerg   bool IsKnownCast = CastInfo || IsCheckedCast || CastFromTy == CastToTy;
2477330f729Sjoerg   if (!IsKnownCast || IsCheckedCast)
2487330f729Sjoerg     State = setDynamicTypeAndCastInfo(State, MR, CastFromTy, CastToTy,
2497330f729Sjoerg                                       CastSucceeds);
2507330f729Sjoerg 
2517330f729Sjoerg   SVal V = CastSucceeds ? C.getSValBuilder().evalCast(DV, CastToTy, CastFromTy)
2527330f729Sjoerg                         : C.getSValBuilder().makeNull();
2537330f729Sjoerg   C.addTransition(
2547330f729Sjoerg       State->BindExpr(Call.getOriginExpr(), C.getLocationContext(), V, false),
2557330f729Sjoerg       getNoteTag(C, CastInfo, CastToTy, Object, CastSucceeds, IsKnownCast));
2567330f729Sjoerg }
2577330f729Sjoerg 
addInstanceOfTransition(const CallEvent & Call,DefinedOrUnknownSVal DV,ProgramStateRef State,CheckerContext & C,bool IsInstanceOf)2587330f729Sjoerg static void addInstanceOfTransition(const CallEvent &Call,
2597330f729Sjoerg                                     DefinedOrUnknownSVal DV,
2607330f729Sjoerg                                     ProgramStateRef State, CheckerContext &C,
2617330f729Sjoerg                                     bool IsInstanceOf) {
2627330f729Sjoerg   const FunctionDecl *FD = Call.getDecl()->getAsFunction();
2637330f729Sjoerg   QualType CastFromTy = Call.parameters()[0]->getType();
264*e038c9c4Sjoerg   SmallVector<QualType, 4> CastToTyVec;
265*e038c9c4Sjoerg   for (unsigned idx = 0; idx < FD->getTemplateSpecializationArgs()->size() - 1;
266*e038c9c4Sjoerg        ++idx) {
267*e038c9c4Sjoerg     TemplateArgument CastToTempArg =
268*e038c9c4Sjoerg       FD->getTemplateSpecializationArgs()->get(idx);
269*e038c9c4Sjoerg     switch (CastToTempArg.getKind()) {
270*e038c9c4Sjoerg     default:
271*e038c9c4Sjoerg       return;
272*e038c9c4Sjoerg     case TemplateArgument::Type:
273*e038c9c4Sjoerg       CastToTyVec.push_back(CastToTempArg.getAsType());
274*e038c9c4Sjoerg       break;
275*e038c9c4Sjoerg     case TemplateArgument::Pack:
276*e038c9c4Sjoerg       for (TemplateArgument ArgInPack: CastToTempArg.pack_elements())
277*e038c9c4Sjoerg         CastToTyVec.push_back(ArgInPack.getAsType());
278*e038c9c4Sjoerg       break;
279*e038c9c4Sjoerg     }
280*e038c9c4Sjoerg   }
281*e038c9c4Sjoerg 
282*e038c9c4Sjoerg   const MemRegion *MR = DV.getAsRegion();
283*e038c9c4Sjoerg   if (MR && CastFromTy->isReferenceType())
284*e038c9c4Sjoerg     MR = State->getSVal(DV.castAs<Loc>()).getAsRegion();
285*e038c9c4Sjoerg 
286*e038c9c4Sjoerg   bool Success = false;
287*e038c9c4Sjoerg   bool IsAnyKnown = false;
288*e038c9c4Sjoerg   for (QualType CastToTy: CastToTyVec) {
2897330f729Sjoerg     if (CastFromTy->isPointerType())
2907330f729Sjoerg       CastToTy = C.getASTContext().getPointerType(CastToTy);
2917330f729Sjoerg     else if (CastFromTy->isReferenceType())
2927330f729Sjoerg       CastToTy = alignReferenceTypes(CastToTy, CastFromTy, C.getASTContext());
2937330f729Sjoerg     else
2947330f729Sjoerg       return;
2957330f729Sjoerg 
2967330f729Sjoerg     const DynamicCastInfo *CastInfo =
2977330f729Sjoerg       getDynamicCastInfo(State, MR, CastFromTy, CastToTy);
2987330f729Sjoerg 
2997330f729Sjoerg     bool CastSucceeds;
3007330f729Sjoerg     if (CastInfo)
3017330f729Sjoerg       CastSucceeds = IsInstanceOf && CastInfo->succeeds();
3027330f729Sjoerg     else
3037330f729Sjoerg       CastSucceeds = IsInstanceOf || CastFromTy == CastToTy;
3047330f729Sjoerg 
3057330f729Sjoerg     // Store the type and the cast information.
3067330f729Sjoerg     bool IsKnownCast = CastInfo || CastFromTy == CastToTy;
307*e038c9c4Sjoerg     IsAnyKnown = IsAnyKnown || IsKnownCast;
308*e038c9c4Sjoerg     ProgramStateRef NewState = State;
3097330f729Sjoerg     if (!IsKnownCast)
310*e038c9c4Sjoerg       NewState = setDynamicTypeAndCastInfo(State, MR, CastFromTy, CastToTy,
3117330f729Sjoerg                                            IsInstanceOf);
3127330f729Sjoerg 
313*e038c9c4Sjoerg     if (CastSucceeds) {
314*e038c9c4Sjoerg       Success = true;
315*e038c9c4Sjoerg       C.addTransition(
316*e038c9c4Sjoerg           NewState->BindExpr(Call.getOriginExpr(), C.getLocationContext(),
317*e038c9c4Sjoerg                              C.getSValBuilder().makeTruthVal(true)),
318*e038c9c4Sjoerg           getNoteTag(C, CastInfo, CastToTy, Call.getArgExpr(0), true,
319*e038c9c4Sjoerg                      IsKnownCast));
320*e038c9c4Sjoerg       if (IsKnownCast)
321*e038c9c4Sjoerg         return;
322*e038c9c4Sjoerg     } else if (CastInfo && CastInfo->succeeds()) {
323*e038c9c4Sjoerg       C.generateSink(NewState, C.getPredecessor());
324*e038c9c4Sjoerg       return;
325*e038c9c4Sjoerg     }
326*e038c9c4Sjoerg   }
327*e038c9c4Sjoerg 
328*e038c9c4Sjoerg   if (!Success) {
3297330f729Sjoerg     C.addTransition(
3307330f729Sjoerg         State->BindExpr(Call.getOriginExpr(), C.getLocationContext(),
331*e038c9c4Sjoerg                         C.getSValBuilder().makeTruthVal(false)),
332*e038c9c4Sjoerg         getNoteTag(C, CastToTyVec, Call.getArgExpr(0), IsAnyKnown));
333*e038c9c4Sjoerg   }
3347330f729Sjoerg }
3357330f729Sjoerg 
3367330f729Sjoerg //===----------------------------------------------------------------------===//
3377330f729Sjoerg // Evaluating cast, dyn_cast, cast_or_null, dyn_cast_or_null.
3387330f729Sjoerg //===----------------------------------------------------------------------===//
3397330f729Sjoerg 
evalNonNullParamNonNullReturn(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C,bool IsCheckedCast=false)3407330f729Sjoerg static void evalNonNullParamNonNullReturn(const CallEvent &Call,
3417330f729Sjoerg                                           DefinedOrUnknownSVal DV,
3427330f729Sjoerg                                           CheckerContext &C,
3437330f729Sjoerg                                           bool IsCheckedCast = false) {
3447330f729Sjoerg   addCastTransition(Call, DV, C, /*IsNonNullParam=*/true,
3457330f729Sjoerg                     /*IsNonNullReturn=*/true, IsCheckedCast);
3467330f729Sjoerg }
3477330f729Sjoerg 
evalNonNullParamNullReturn(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C)3487330f729Sjoerg static void evalNonNullParamNullReturn(const CallEvent &Call,
3497330f729Sjoerg                                        DefinedOrUnknownSVal DV,
3507330f729Sjoerg                                        CheckerContext &C) {
3517330f729Sjoerg   addCastTransition(Call, DV, C, /*IsNonNullParam=*/true,
3527330f729Sjoerg                     /*IsNonNullReturn=*/false);
3537330f729Sjoerg }
3547330f729Sjoerg 
evalNullParamNullReturn(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C)3557330f729Sjoerg static void evalNullParamNullReturn(const CallEvent &Call,
3567330f729Sjoerg                                     DefinedOrUnknownSVal DV,
3577330f729Sjoerg                                     CheckerContext &C) {
3587330f729Sjoerg   if (ProgramStateRef State = C.getState()->assume(DV, false))
3597330f729Sjoerg     C.addTransition(State->BindExpr(Call.getOriginExpr(),
3607330f729Sjoerg                                     C.getLocationContext(),
3617330f729Sjoerg                                     C.getSValBuilder().makeNull(), false),
3627330f729Sjoerg                     C.getNoteTag("Assuming null pointer is passed into cast",
3637330f729Sjoerg                                  /*IsPrunable=*/true));
3647330f729Sjoerg }
3657330f729Sjoerg 
evalCast(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const3667330f729Sjoerg void CastValueChecker::evalCast(const CallEvent &Call, DefinedOrUnknownSVal DV,
3677330f729Sjoerg                                 CheckerContext &C) const {
3687330f729Sjoerg   evalNonNullParamNonNullReturn(Call, DV, C, /*IsCheckedCast=*/true);
3697330f729Sjoerg }
3707330f729Sjoerg 
evalDynCast(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const3717330f729Sjoerg void CastValueChecker::evalDynCast(const CallEvent &Call,
3727330f729Sjoerg                                    DefinedOrUnknownSVal DV,
3737330f729Sjoerg                                    CheckerContext &C) const {
3747330f729Sjoerg   evalNonNullParamNonNullReturn(Call, DV, C);
3757330f729Sjoerg   evalNonNullParamNullReturn(Call, DV, C);
3767330f729Sjoerg }
3777330f729Sjoerg 
evalCastOrNull(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const3787330f729Sjoerg void CastValueChecker::evalCastOrNull(const CallEvent &Call,
3797330f729Sjoerg                                       DefinedOrUnknownSVal DV,
3807330f729Sjoerg                                       CheckerContext &C) const {
3817330f729Sjoerg   evalNonNullParamNonNullReturn(Call, DV, C);
3827330f729Sjoerg   evalNullParamNullReturn(Call, DV, C);
3837330f729Sjoerg }
3847330f729Sjoerg 
evalDynCastOrNull(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const3857330f729Sjoerg void CastValueChecker::evalDynCastOrNull(const CallEvent &Call,
3867330f729Sjoerg                                          DefinedOrUnknownSVal DV,
3877330f729Sjoerg                                          CheckerContext &C) const {
3887330f729Sjoerg   evalNonNullParamNonNullReturn(Call, DV, C);
3897330f729Sjoerg   evalNonNullParamNullReturn(Call, DV, C);
3907330f729Sjoerg   evalNullParamNullReturn(Call, DV, C);
3917330f729Sjoerg }
3927330f729Sjoerg 
3937330f729Sjoerg //===----------------------------------------------------------------------===//
3947330f729Sjoerg // Evaluating castAs, getAs.
3957330f729Sjoerg //===----------------------------------------------------------------------===//
3967330f729Sjoerg 
evalZeroParamNonNullReturn(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C,bool IsCheckedCast=false)3977330f729Sjoerg static void evalZeroParamNonNullReturn(const CallEvent &Call,
3987330f729Sjoerg                                        DefinedOrUnknownSVal DV,
3997330f729Sjoerg                                        CheckerContext &C,
4007330f729Sjoerg                                        bool IsCheckedCast = false) {
4017330f729Sjoerg   addCastTransition(Call, DV, C, /*IsNonNullParam=*/true,
4027330f729Sjoerg                     /*IsNonNullReturn=*/true, IsCheckedCast);
4037330f729Sjoerg }
4047330f729Sjoerg 
evalZeroParamNullReturn(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C)4057330f729Sjoerg static void evalZeroParamNullReturn(const CallEvent &Call,
4067330f729Sjoerg                                     DefinedOrUnknownSVal DV,
4077330f729Sjoerg                                     CheckerContext &C) {
4087330f729Sjoerg   addCastTransition(Call, DV, C, /*IsNonNullParam=*/true,
4097330f729Sjoerg                     /*IsNonNullReturn=*/false);
4107330f729Sjoerg }
4117330f729Sjoerg 
evalCastAs(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const4127330f729Sjoerg void CastValueChecker::evalCastAs(const CallEvent &Call,
4137330f729Sjoerg                                   DefinedOrUnknownSVal DV,
4147330f729Sjoerg                                   CheckerContext &C) const {
4157330f729Sjoerg   evalZeroParamNonNullReturn(Call, DV, C, /*IsCheckedCast=*/true);
4167330f729Sjoerg }
4177330f729Sjoerg 
evalGetAs(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const4187330f729Sjoerg void CastValueChecker::evalGetAs(const CallEvent &Call, DefinedOrUnknownSVal DV,
4197330f729Sjoerg                                  CheckerContext &C) const {
4207330f729Sjoerg   evalZeroParamNonNullReturn(Call, DV, C);
4217330f729Sjoerg   evalZeroParamNullReturn(Call, DV, C);
4227330f729Sjoerg }
4237330f729Sjoerg 
4247330f729Sjoerg //===----------------------------------------------------------------------===//
4257330f729Sjoerg // Evaluating isa, isa_and_nonnull.
4267330f729Sjoerg //===----------------------------------------------------------------------===//
4277330f729Sjoerg 
evalIsa(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const4287330f729Sjoerg void CastValueChecker::evalIsa(const CallEvent &Call, DefinedOrUnknownSVal DV,
4297330f729Sjoerg                                CheckerContext &C) const {
4307330f729Sjoerg   ProgramStateRef NonNullState, NullState;
4317330f729Sjoerg   std::tie(NonNullState, NullState) = C.getState()->assume(DV);
4327330f729Sjoerg 
4337330f729Sjoerg   if (NonNullState) {
4347330f729Sjoerg     addInstanceOfTransition(Call, DV, NonNullState, C, /*IsInstanceOf=*/true);
4357330f729Sjoerg     addInstanceOfTransition(Call, DV, NonNullState, C, /*IsInstanceOf=*/false);
4367330f729Sjoerg   }
4377330f729Sjoerg 
4387330f729Sjoerg   if (NullState) {
4397330f729Sjoerg     C.generateSink(NullState, C.getPredecessor());
4407330f729Sjoerg   }
4417330f729Sjoerg }
4427330f729Sjoerg 
evalIsaAndNonNull(const CallEvent & Call,DefinedOrUnknownSVal DV,CheckerContext & C) const4437330f729Sjoerg void CastValueChecker::evalIsaAndNonNull(const CallEvent &Call,
4447330f729Sjoerg                                          DefinedOrUnknownSVal DV,
4457330f729Sjoerg                                          CheckerContext &C) const {
4467330f729Sjoerg   ProgramStateRef NonNullState, NullState;
4477330f729Sjoerg   std::tie(NonNullState, NullState) = C.getState()->assume(DV);
4487330f729Sjoerg 
4497330f729Sjoerg   if (NonNullState) {
4507330f729Sjoerg     addInstanceOfTransition(Call, DV, NonNullState, C, /*IsInstanceOf=*/true);
4517330f729Sjoerg     addInstanceOfTransition(Call, DV, NonNullState, C, /*IsInstanceOf=*/false);
4527330f729Sjoerg   }
4537330f729Sjoerg 
4547330f729Sjoerg   if (NullState) {
4557330f729Sjoerg     addInstanceOfTransition(Call, DV, NullState, C, /*IsInstanceOf=*/false);
4567330f729Sjoerg   }
4577330f729Sjoerg }
4587330f729Sjoerg 
4597330f729Sjoerg //===----------------------------------------------------------------------===//
4607330f729Sjoerg // Main logic to evaluate a call.
4617330f729Sjoerg //===----------------------------------------------------------------------===//
4627330f729Sjoerg 
evalCall(const CallEvent & Call,CheckerContext & C) const4637330f729Sjoerg bool CastValueChecker::evalCall(const CallEvent &Call,
4647330f729Sjoerg                                 CheckerContext &C) const {
4657330f729Sjoerg   const auto *Lookup = CDM.lookup(Call);
4667330f729Sjoerg   if (!Lookup)
4677330f729Sjoerg     return false;
4687330f729Sjoerg 
4697330f729Sjoerg   const CastCheck &Check = Lookup->first;
4707330f729Sjoerg   CallKind Kind = Lookup->second;
4717330f729Sjoerg 
4727330f729Sjoerg   Optional<DefinedOrUnknownSVal> DV;
4737330f729Sjoerg 
4747330f729Sjoerg   switch (Kind) {
4757330f729Sjoerg   case CallKind::Function: {
4767330f729Sjoerg     // We only model casts from pointers to pointers or from references
4777330f729Sjoerg     // to references. Other casts are most likely specialized and we
4787330f729Sjoerg     // cannot model them.
4797330f729Sjoerg     QualType ParamT = Call.parameters()[0]->getType();
4807330f729Sjoerg     QualType ResultT = Call.getResultType();
4817330f729Sjoerg     if (!(ParamT->isPointerType() && ResultT->isPointerType()) &&
482*e038c9c4Sjoerg         !(ParamT->isReferenceType() && ResultT->isReferenceType())) {
4837330f729Sjoerg       return false;
484*e038c9c4Sjoerg     }
4857330f729Sjoerg 
4867330f729Sjoerg     DV = Call.getArgSVal(0).getAs<DefinedOrUnknownSVal>();
4877330f729Sjoerg     break;
4887330f729Sjoerg   }
4897330f729Sjoerg   case CallKind::InstanceOf: {
4907330f729Sjoerg     // We need to obtain the only template argument to determinte the type.
4917330f729Sjoerg     const FunctionDecl *FD = Call.getDecl()->getAsFunction();
4927330f729Sjoerg     if (!FD || !FD->getTemplateSpecializationArgs())
4937330f729Sjoerg       return false;
4947330f729Sjoerg 
4957330f729Sjoerg     DV = Call.getArgSVal(0).getAs<DefinedOrUnknownSVal>();
4967330f729Sjoerg     break;
4977330f729Sjoerg   }
4987330f729Sjoerg   case CallKind::Method:
4997330f729Sjoerg     const auto *InstanceCall = dyn_cast<CXXInstanceCall>(&Call);
5007330f729Sjoerg     if (!InstanceCall)
5017330f729Sjoerg       return false;
5027330f729Sjoerg 
5037330f729Sjoerg     DV = InstanceCall->getCXXThisVal().getAs<DefinedOrUnknownSVal>();
5047330f729Sjoerg     break;
5057330f729Sjoerg   }
5067330f729Sjoerg 
5077330f729Sjoerg   if (!DV)
5087330f729Sjoerg     return false;
5097330f729Sjoerg 
5107330f729Sjoerg   Check(this, Call, *DV, C);
5117330f729Sjoerg   return true;
5127330f729Sjoerg }
5137330f729Sjoerg 
checkDeadSymbols(SymbolReaper & SR,CheckerContext & C) const514*e038c9c4Sjoerg void CastValueChecker::checkDeadSymbols(SymbolReaper &SR,
515*e038c9c4Sjoerg                                         CheckerContext &C) const {
516*e038c9c4Sjoerg   C.addTransition(removeDeadCasts(C.getState(), SR));
517*e038c9c4Sjoerg }
518*e038c9c4Sjoerg 
registerCastValueChecker(CheckerManager & Mgr)5197330f729Sjoerg void ento::registerCastValueChecker(CheckerManager &Mgr) {
5207330f729Sjoerg   Mgr.registerChecker<CastValueChecker>();
5217330f729Sjoerg }
5227330f729Sjoerg 
shouldRegisterCastValueChecker(const CheckerManager & mgr)523*e038c9c4Sjoerg bool ento::shouldRegisterCastValueChecker(const CheckerManager &mgr) {
5247330f729Sjoerg   return true;
5257330f729Sjoerg }
526