xref: /llvm-project/clang/lib/StaticAnalyzer/Checkers/RetainCountChecker/RetainCountChecker.cpp (revision 41dc8de6ae9f5ac1d39e5b8a86c2990a5c1feb9c)
1 //==-- RetainCountChecker.cpp - Checks for leaks and other issues -*- C++ -*--//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 //  This file defines the methods for RetainCountChecker, which implements
11 //  a reference count checker for Core Foundation and Cocoa on (Mac OS X).
12 //
13 //===----------------------------------------------------------------------===//
14 
15 #include "RetainCountChecker.h"
16 
17 using namespace clang;
18 using namespace ento;
19 using namespace retaincountchecker;
20 using llvm::StrInStrNoCase;
21 
22 REGISTER_MAP_WITH_PROGRAMSTATE(RefBindings, SymbolRef, RefVal)
23 
24 namespace clang {
25 namespace ento {
26 namespace retaincountchecker {
27 
28 const RefVal *getRefBinding(ProgramStateRef State, SymbolRef Sym) {
29   return State->get<RefBindings>(Sym);
30 }
31 
32 ProgramStateRef setRefBinding(ProgramStateRef State, SymbolRef Sym,
33                                      RefVal Val) {
34   assert(Sym != nullptr);
35   return State->set<RefBindings>(Sym, Val);
36 }
37 
38 ProgramStateRef removeRefBinding(ProgramStateRef State, SymbolRef Sym) {
39   return State->remove<RefBindings>(Sym);
40 }
41 
42 } // end namespace retaincountchecker
43 } // end namespace ento
44 } // end namespace clang
45 
46 void RefVal::print(raw_ostream &Out) const {
47   if (!T.isNull())
48     Out << "Tracked " << T.getAsString() << '/';
49 
50   switch (getKind()) {
51     default: llvm_unreachable("Invalid RefVal kind");
52     case Owned: {
53       Out << "Owned";
54       unsigned cnt = getCount();
55       if (cnt) Out << " (+ " << cnt << ")";
56       break;
57     }
58 
59     case NotOwned: {
60       Out << "NotOwned";
61       unsigned cnt = getCount();
62       if (cnt) Out << " (+ " << cnt << ")";
63       break;
64     }
65 
66     case ReturnedOwned: {
67       Out << "ReturnedOwned";
68       unsigned cnt = getCount();
69       if (cnt) Out << " (+ " << cnt << ")";
70       break;
71     }
72 
73     case ReturnedNotOwned: {
74       Out << "ReturnedNotOwned";
75       unsigned cnt = getCount();
76       if (cnt) Out << " (+ " << cnt << ")";
77       break;
78     }
79 
80     case Released:
81       Out << "Released";
82       break;
83 
84     case ErrorDeallocNotOwned:
85       Out << "-dealloc (not-owned)";
86       break;
87 
88     case ErrorLeak:
89       Out << "Leaked";
90       break;
91 
92     case ErrorLeakReturned:
93       Out << "Leaked (Bad naming)";
94       break;
95 
96     case ErrorUseAfterRelease:
97       Out << "Use-After-Release [ERROR]";
98       break;
99 
100     case ErrorReleaseNotOwned:
101       Out << "Release of Not-Owned [ERROR]";
102       break;
103 
104     case RefVal::ErrorOverAutorelease:
105       Out << "Over-autoreleased";
106       break;
107 
108     case RefVal::ErrorReturnedNotOwned:
109       Out << "Non-owned object returned instead of owned";
110       break;
111   }
112 
113   switch (getIvarAccessHistory()) {
114   case IvarAccessHistory::None:
115     break;
116   case IvarAccessHistory::AccessedDirectly:
117     Out << " [direct ivar access]";
118     break;
119   case IvarAccessHistory::ReleasedAfterDirectAccess:
120     Out << " [released after direct ivar access]";
121   }
122 
123   if (ACnt) {
124     Out << " [autorelease -" << ACnt << ']';
125   }
126 }
127 
128 namespace {
129 class StopTrackingCallback final : public SymbolVisitor {
130   ProgramStateRef state;
131 public:
132   StopTrackingCallback(ProgramStateRef st) : state(std::move(st)) {}
133   ProgramStateRef getState() const { return state; }
134 
135   bool VisitSymbol(SymbolRef sym) override {
136     state = state->remove<RefBindings>(sym);
137     return true;
138   }
139 };
140 } // end anonymous namespace
141 
142 //===----------------------------------------------------------------------===//
143 // Handle statements that may have an effect on refcounts.
144 //===----------------------------------------------------------------------===//
145 
146 void RetainCountChecker::checkPostStmt(const BlockExpr *BE,
147                                        CheckerContext &C) const {
148 
149   // Scan the BlockDecRefExprs for any object the retain count checker
150   // may be tracking.
151   if (!BE->getBlockDecl()->hasCaptures())
152     return;
153 
154   ProgramStateRef state = C.getState();
155   auto *R = cast<BlockDataRegion>(C.getSVal(BE).getAsRegion());
156 
157   BlockDataRegion::referenced_vars_iterator I = R->referenced_vars_begin(),
158                                             E = R->referenced_vars_end();
159 
160   if (I == E)
161     return;
162 
163   // FIXME: For now we invalidate the tracking of all symbols passed to blocks
164   // via captured variables, even though captured variables result in a copy
165   // and in implicit increment/decrement of a retain count.
166   SmallVector<const MemRegion*, 10> Regions;
167   const LocationContext *LC = C.getLocationContext();
168   MemRegionManager &MemMgr = C.getSValBuilder().getRegionManager();
169 
170   for ( ; I != E; ++I) {
171     const VarRegion *VR = I.getCapturedRegion();
172     if (VR->getSuperRegion() == R) {
173       VR = MemMgr.getVarRegion(VR->getDecl(), LC);
174     }
175     Regions.push_back(VR);
176   }
177 
178   state =
179     state->scanReachableSymbols<StopTrackingCallback>(Regions.data(),
180                                     Regions.data() + Regions.size()).getState();
181   C.addTransition(state);
182 }
183 
184 void RetainCountChecker::checkPostStmt(const CastExpr *CE,
185                                        CheckerContext &C) const {
186   const ObjCBridgedCastExpr *BE = dyn_cast<ObjCBridgedCastExpr>(CE);
187   if (!BE)
188     return;
189 
190   ArgEffect AE = IncRef;
191 
192   switch (BE->getBridgeKind()) {
193     case OBC_Bridge:
194       // Do nothing.
195       return;
196     case OBC_BridgeRetained:
197       AE = IncRef;
198       break;
199     case OBC_BridgeTransfer:
200       AE = DecRefBridgedTransferred;
201       break;
202   }
203 
204   ProgramStateRef state = C.getState();
205   SymbolRef Sym = C.getSVal(CE).getAsLocSymbol();
206   if (!Sym)
207     return;
208   const RefVal* T = getRefBinding(state, Sym);
209   if (!T)
210     return;
211 
212   RefVal::Kind hasErr = (RefVal::Kind) 0;
213   state = updateSymbol(state, Sym, *T, AE, hasErr, C);
214 
215   if (hasErr) {
216     // FIXME: If we get an error during a bridge cast, should we report it?
217     return;
218   }
219 
220   C.addTransition(state);
221 }
222 
223 void RetainCountChecker::processObjCLiterals(CheckerContext &C,
224                                              const Expr *Ex) const {
225   ProgramStateRef state = C.getState();
226   const ExplodedNode *pred = C.getPredecessor();
227   for (const Stmt *Child : Ex->children()) {
228     SVal V = pred->getSVal(Child);
229     if (SymbolRef sym = V.getAsSymbol())
230       if (const RefVal* T = getRefBinding(state, sym)) {
231         RefVal::Kind hasErr = (RefVal::Kind) 0;
232         state = updateSymbol(state, sym, *T, MayEscape, hasErr, C);
233         if (hasErr) {
234           processNonLeakError(state, Child->getSourceRange(), hasErr, sym, C);
235           return;
236         }
237       }
238   }
239 
240   // Return the object as autoreleased.
241   //  RetEffect RE = RetEffect::MakeNotOwned(RetEffect::ObjC);
242   if (SymbolRef sym =
243         state->getSVal(Ex, pred->getLocationContext()).getAsSymbol()) {
244     QualType ResultTy = Ex->getType();
245     state = setRefBinding(state, sym,
246                           RefVal::makeNotOwned(RetEffect::ObjC, ResultTy));
247   }
248 
249   C.addTransition(state);
250 }
251 
252 void RetainCountChecker::checkPostStmt(const ObjCArrayLiteral *AL,
253                                        CheckerContext &C) const {
254   // Apply the 'MayEscape' to all values.
255   processObjCLiterals(C, AL);
256 }
257 
258 void RetainCountChecker::checkPostStmt(const ObjCDictionaryLiteral *DL,
259                                        CheckerContext &C) const {
260   // Apply the 'MayEscape' to all keys and values.
261   processObjCLiterals(C, DL);
262 }
263 
264 void RetainCountChecker::checkPostStmt(const ObjCBoxedExpr *Ex,
265                                        CheckerContext &C) const {
266   const ExplodedNode *Pred = C.getPredecessor();
267   ProgramStateRef State = Pred->getState();
268 
269   if (SymbolRef Sym = Pred->getSVal(Ex).getAsSymbol()) {
270     QualType ResultTy = Ex->getType();
271     State = setRefBinding(State, Sym,
272                           RefVal::makeNotOwned(RetEffect::ObjC, ResultTy));
273   }
274 
275   C.addTransition(State);
276 }
277 
278 void RetainCountChecker::checkPostStmt(const ObjCIvarRefExpr *IRE,
279                                        CheckerContext &C) const {
280   Optional<Loc> IVarLoc = C.getSVal(IRE).getAs<Loc>();
281   if (!IVarLoc)
282     return;
283 
284   ProgramStateRef State = C.getState();
285   SymbolRef Sym = State->getSVal(*IVarLoc).getAsSymbol();
286   if (!Sym || !dyn_cast_or_null<ObjCIvarRegion>(Sym->getOriginRegion()))
287     return;
288 
289   // Accessing an ivar directly is unusual. If we've done that, be more
290   // forgiving about what the surrounding code is allowed to do.
291 
292   QualType Ty = Sym->getType();
293   RetEffect::ObjKind Kind;
294   if (Ty->isObjCRetainableType())
295     Kind = RetEffect::ObjC;
296   else if (coreFoundation::isCFObjectRef(Ty))
297     Kind = RetEffect::CF;
298   else
299     return;
300 
301   // If the value is already known to be nil, don't bother tracking it.
302   ConstraintManager &CMgr = State->getConstraintManager();
303   if (CMgr.isNull(State, Sym).isConstrainedTrue())
304     return;
305 
306   if (const RefVal *RV = getRefBinding(State, Sym)) {
307     // If we've seen this symbol before, or we're only seeing it now because
308     // of something the analyzer has synthesized, don't do anything.
309     if (RV->getIvarAccessHistory() != RefVal::IvarAccessHistory::None ||
310         isSynthesizedAccessor(C.getStackFrame())) {
311       return;
312     }
313 
314     // Note that this value has been loaded from an ivar.
315     C.addTransition(setRefBinding(State, Sym, RV->withIvarAccess()));
316     return;
317   }
318 
319   RefVal PlusZero = RefVal::makeNotOwned(Kind, Ty);
320 
321   // In a synthesized accessor, the effective retain count is +0.
322   if (isSynthesizedAccessor(C.getStackFrame())) {
323     C.addTransition(setRefBinding(State, Sym, PlusZero));
324     return;
325   }
326 
327   State = setRefBinding(State, Sym, PlusZero.withIvarAccess());
328   C.addTransition(State);
329 }
330 
331 void RetainCountChecker::checkPostCall(const CallEvent &Call,
332                                        CheckerContext &C) const {
333   RetainSummaryManager &Summaries = getSummaryManager(C);
334 
335   // Leave null if no receiver.
336   QualType ReceiverType;
337   if (const auto *MC = dyn_cast<ObjCMethodCall>(&Call)) {
338     if (MC->isInstanceMessage()) {
339       SVal ReceiverV = MC->getReceiverSVal();
340       if (SymbolRef Sym = ReceiverV.getAsLocSymbol())
341         if (const RefVal *T = getRefBinding(C.getState(), Sym))
342           ReceiverType = T->getType();
343     }
344   }
345 
346   const RetainSummary *Summ = Summaries.getSummary(Call, ReceiverType);
347 
348   if (C.wasInlined) {
349     processSummaryOfInlined(*Summ, Call, C);
350     return;
351   }
352   checkSummary(*Summ, Call, C);
353 }
354 
355 /// GetReturnType - Used to get the return type of a message expression or
356 ///  function call with the intention of affixing that type to a tracked symbol.
357 ///  While the return type can be queried directly from RetEx, when
358 ///  invoking class methods we augment to the return type to be that of
359 ///  a pointer to the class (as opposed it just being id).
360 // FIXME: We may be able to do this with related result types instead.
361 // This function is probably overestimating.
362 static QualType GetReturnType(const Expr *RetE, ASTContext &Ctx) {
363   QualType RetTy = RetE->getType();
364   // If RetE is not a message expression just return its type.
365   // If RetE is a message expression, return its types if it is something
366   /// more specific than id.
367   if (const ObjCMessageExpr *ME = dyn_cast<ObjCMessageExpr>(RetE))
368     if (const ObjCObjectPointerType *PT = RetTy->getAs<ObjCObjectPointerType>())
369       if (PT->isObjCQualifiedIdType() || PT->isObjCIdType() ||
370           PT->isObjCClassType()) {
371         // At this point we know the return type of the message expression is
372         // id, id<...>, or Class. If we have an ObjCInterfaceDecl, we know this
373         // is a call to a class method whose type we can resolve.  In such
374         // cases, promote the return type to XXX* (where XXX is the class).
375         const ObjCInterfaceDecl *D = ME->getReceiverInterface();
376         return !D ? RetTy :
377                     Ctx.getObjCObjectPointerType(Ctx.getObjCInterfaceType(D));
378       }
379 
380   return RetTy;
381 }
382 
383 static Optional<RefVal> refValFromRetEffect(RetEffect RE,
384                                             QualType ResultTy) {
385   if (RE.isOwned()) {
386     return RefVal::makeOwned(RE.getObjKind(), ResultTy);
387   } else if (RE.notOwned()) {
388     return RefVal::makeNotOwned(RE.getObjKind(), ResultTy);
389   }
390 
391   return None;
392 }
393 
394 // We don't always get the exact modeling of the function with regards to the
395 // retain count checker even when the function is inlined. For example, we need
396 // to stop tracking the symbols which were marked with StopTrackingHard.
397 void RetainCountChecker::processSummaryOfInlined(const RetainSummary &Summ,
398                                                  const CallEvent &CallOrMsg,
399                                                  CheckerContext &C) const {
400   ProgramStateRef state = C.getState();
401 
402   // Evaluate the effect of the arguments.
403   for (unsigned idx = 0, e = CallOrMsg.getNumArgs(); idx != e; ++idx) {
404     if (Summ.getArg(idx) == StopTrackingHard) {
405       SVal V = CallOrMsg.getArgSVal(idx);
406       if (SymbolRef Sym = V.getAsLocSymbol()) {
407         state = removeRefBinding(state, Sym);
408       }
409     }
410   }
411 
412   // Evaluate the effect on the message receiver.
413   if (const auto *MsgInvocation = dyn_cast<ObjCMethodCall>(&CallOrMsg)) {
414     if (SymbolRef Sym = MsgInvocation->getReceiverSVal().getAsLocSymbol()) {
415       if (Summ.getReceiverEffect() == StopTrackingHard) {
416         state = removeRefBinding(state, Sym);
417       }
418     }
419   }
420 
421   // Consult the summary for the return value.
422   RetEffect RE = Summ.getRetEffect();
423 
424   if (SymbolRef Sym = CallOrMsg.getReturnValue().getAsSymbol()) {
425     if (const auto *MCall = dyn_cast<CXXMemberCall>(&CallOrMsg)) {
426       if (Optional<RefVal> updatedRefVal =
427               refValFromRetEffect(RE, MCall->getResultType())) {
428         state = setRefBinding(state, Sym, *updatedRefVal);
429       }
430     }
431 
432     if (RE.getKind() == RetEffect::NoRetHard)
433       state = removeRefBinding(state, Sym);
434   }
435 
436   C.addTransition(state);
437 }
438 
439 static ProgramStateRef updateOutParameter(ProgramStateRef State,
440                                           SVal ArgVal,
441                                           ArgEffect Effect) {
442   auto *ArgRegion = dyn_cast_or_null<TypedValueRegion>(ArgVal.getAsRegion());
443   if (!ArgRegion)
444     return State;
445 
446   QualType PointeeTy = ArgRegion->getValueType();
447   if (!coreFoundation::isCFObjectRef(PointeeTy))
448     return State;
449 
450   SVal PointeeVal = State->getSVal(ArgRegion);
451   SymbolRef Pointee = PointeeVal.getAsLocSymbol();
452   if (!Pointee)
453     return State;
454 
455   switch (Effect) {
456   case UnretainedOutParameter:
457     State = setRefBinding(State, Pointee,
458                           RefVal::makeNotOwned(RetEffect::CF, PointeeTy));
459     break;
460   case RetainedOutParameter:
461     // Do nothing. Retained out parameters will either point to a +1 reference
462     // or NULL, but the way you check for failure differs depending on the API.
463     // Consequently, we don't have a good way to track them yet.
464     break;
465 
466   default:
467     llvm_unreachable("only for out parameters");
468   }
469 
470   return State;
471 }
472 
473 void RetainCountChecker::checkSummary(const RetainSummary &Summ,
474                                       const CallEvent &CallOrMsg,
475                                       CheckerContext &C) const {
476   ProgramStateRef state = C.getState();
477 
478   // Evaluate the effect of the arguments.
479   RefVal::Kind hasErr = (RefVal::Kind) 0;
480   SourceRange ErrorRange;
481   SymbolRef ErrorSym = nullptr;
482 
483   for (unsigned idx = 0, e = CallOrMsg.getNumArgs(); idx != e; ++idx) {
484     SVal V = CallOrMsg.getArgSVal(idx);
485 
486     ArgEffect Effect = Summ.getArg(idx);
487     if (Effect == RetainedOutParameter || Effect == UnretainedOutParameter) {
488       state = updateOutParameter(state, V, Effect);
489     } else if (SymbolRef Sym = V.getAsLocSymbol()) {
490       if (const RefVal *T = getRefBinding(state, Sym)) {
491         state = updateSymbol(state, Sym, *T, Effect, hasErr, C);
492         if (hasErr) {
493           ErrorRange = CallOrMsg.getArgSourceRange(idx);
494           ErrorSym = Sym;
495           break;
496         }
497       }
498     }
499   }
500 
501   // Evaluate the effect on the message receiver / `this` argument.
502   bool ReceiverIsTracked = false;
503   if (!hasErr) {
504     if (const auto *MsgInvocation = dyn_cast<ObjCMethodCall>(&CallOrMsg)) {
505       if (SymbolRef Sym = MsgInvocation->getReceiverSVal().getAsLocSymbol()) {
506         if (const RefVal *T = getRefBinding(state, Sym)) {
507           ReceiverIsTracked = true;
508           state = updateSymbol(state, Sym, *T, Summ.getReceiverEffect(),
509                                  hasErr, C);
510           if (hasErr) {
511             ErrorRange = MsgInvocation->getOriginExpr()->getReceiverRange();
512             ErrorSym = Sym;
513           }
514         }
515       }
516     } else if (const auto *MCall = dyn_cast<CXXMemberCall>(&CallOrMsg)) {
517       if (SymbolRef Sym = MCall->getCXXThisVal().getAsLocSymbol()) {
518         if (const RefVal *T = getRefBinding(state, Sym)) {
519           state = updateSymbol(state, Sym, *T, Summ.getThisEffect(),
520                                hasErr, C);
521           if (hasErr) {
522             ErrorRange = MCall->getOriginExpr()->getSourceRange();
523             ErrorSym = Sym;
524           }
525         }
526       }
527     }
528   }
529 
530   // Process any errors.
531   if (hasErr) {
532     processNonLeakError(state, ErrorRange, hasErr, ErrorSym, C);
533     return;
534   }
535 
536   // Consult the summary for the return value.
537   RetEffect RE = Summ.getRetEffect();
538 
539   if (RE.getKind() == RetEffect::OwnedWhenTrackedReceiver) {
540     if (ReceiverIsTracked)
541       RE = getSummaryManager(C).getObjAllocRetEffect();
542     else
543       RE = RetEffect::MakeNoRet();
544   }
545 
546   if (SymbolRef Sym = CallOrMsg.getReturnValue().getAsSymbol()) {
547     QualType ResultTy = CallOrMsg.getResultType();
548     if (RE.notOwned()) {
549       const Expr *Ex = CallOrMsg.getOriginExpr();
550       assert(Ex);
551       ResultTy = GetReturnType(Ex, C.getASTContext());
552     }
553     if (Optional<RefVal> updatedRefVal = refValFromRetEffect(RE, ResultTy))
554       state = setRefBinding(state, Sym, *updatedRefVal);
555   }
556 
557   // This check is actually necessary; otherwise the statement builder thinks
558   // we've hit a previously-found path.
559   // Normally addTransition takes care of this, but we want the node pointer.
560   ExplodedNode *NewNode;
561   if (state == C.getState()) {
562     NewNode = C.getPredecessor();
563   } else {
564     NewNode = C.addTransition(state);
565   }
566 
567   // Annotate the node with summary we used.
568   if (NewNode) {
569     // FIXME: This is ugly. See checkEndAnalysis for why it's necessary.
570     if (ShouldResetSummaryLog) {
571       SummaryLog.clear();
572       ShouldResetSummaryLog = false;
573     }
574     SummaryLog[NewNode] = &Summ;
575   }
576 }
577 
578 ProgramStateRef
579 RetainCountChecker::updateSymbol(ProgramStateRef state, SymbolRef sym,
580                                  RefVal V, ArgEffect E, RefVal::Kind &hasErr,
581                                  CheckerContext &C) const {
582   bool IgnoreRetainMsg = (bool)C.getASTContext().getLangOpts().ObjCAutoRefCount;
583   switch (E) {
584   default:
585     break;
586   case IncRefMsg:
587     E = IgnoreRetainMsg ? DoNothing : IncRef;
588     break;
589   case DecRefMsg:
590     E = IgnoreRetainMsg ? DoNothing: DecRef;
591     break;
592   case DecRefMsgAndStopTrackingHard:
593     E = IgnoreRetainMsg ? StopTracking : DecRefAndStopTrackingHard;
594     break;
595   case MakeCollectable:
596     E = DoNothing;
597   }
598 
599   // Handle all use-after-releases.
600   if (V.getKind() == RefVal::Released) {
601     V = V ^ RefVal::ErrorUseAfterRelease;
602     hasErr = V.getKind();
603     return setRefBinding(state, sym, V);
604   }
605 
606   switch (E) {
607     case DecRefMsg:
608     case IncRefMsg:
609     case MakeCollectable:
610     case DecRefMsgAndStopTrackingHard:
611       llvm_unreachable("DecRefMsg/IncRefMsg/MakeCollectable already converted");
612 
613     case UnretainedOutParameter:
614     case RetainedOutParameter:
615       llvm_unreachable("Applies to pointer-to-pointer parameters, which should "
616                        "not have ref state.");
617 
618     case Dealloc:
619       switch (V.getKind()) {
620         default:
621           llvm_unreachable("Invalid RefVal state for an explicit dealloc.");
622         case RefVal::Owned:
623           // The object immediately transitions to the released state.
624           V = V ^ RefVal::Released;
625           V.clearCounts();
626           return setRefBinding(state, sym, V);
627         case RefVal::NotOwned:
628           V = V ^ RefVal::ErrorDeallocNotOwned;
629           hasErr = V.getKind();
630           break;
631       }
632       break;
633 
634     case MayEscape:
635       if (V.getKind() == RefVal::Owned) {
636         V = V ^ RefVal::NotOwned;
637         break;
638       }
639 
640       // Fall-through.
641 
642     case DoNothing:
643       return state;
644 
645     case Autorelease:
646       // Update the autorelease counts.
647       V = V.autorelease();
648       break;
649 
650     case StopTracking:
651     case StopTrackingHard:
652       return removeRefBinding(state, sym);
653 
654     case IncRef:
655       switch (V.getKind()) {
656         default:
657           llvm_unreachable("Invalid RefVal state for a retain.");
658         case RefVal::Owned:
659         case RefVal::NotOwned:
660           V = V + 1;
661           break;
662       }
663       break;
664 
665     case DecRef:
666     case DecRefBridgedTransferred:
667     case DecRefAndStopTrackingHard:
668       switch (V.getKind()) {
669         default:
670           // case 'RefVal::Released' handled above.
671           llvm_unreachable("Invalid RefVal state for a release.");
672 
673         case RefVal::Owned:
674           assert(V.getCount() > 0);
675           if (V.getCount() == 1) {
676             if (E == DecRefBridgedTransferred ||
677                 V.getIvarAccessHistory() ==
678                   RefVal::IvarAccessHistory::AccessedDirectly)
679               V = V ^ RefVal::NotOwned;
680             else
681               V = V ^ RefVal::Released;
682           } else if (E == DecRefAndStopTrackingHard) {
683             return removeRefBinding(state, sym);
684           }
685 
686           V = V - 1;
687           break;
688 
689         case RefVal::NotOwned:
690           if (V.getCount() > 0) {
691             if (E == DecRefAndStopTrackingHard)
692               return removeRefBinding(state, sym);
693             V = V - 1;
694           } else if (V.getIvarAccessHistory() ==
695                        RefVal::IvarAccessHistory::AccessedDirectly) {
696             // Assume that the instance variable was holding on the object at
697             // +1, and we just didn't know.
698             if (E == DecRefAndStopTrackingHard)
699               return removeRefBinding(state, sym);
700             V = V.releaseViaIvar() ^ RefVal::Released;
701           } else {
702             V = V ^ RefVal::ErrorReleaseNotOwned;
703             hasErr = V.getKind();
704           }
705           break;
706       }
707       break;
708   }
709   return setRefBinding(state, sym, V);
710 }
711 
712 void RetainCountChecker::processNonLeakError(ProgramStateRef St,
713                                              SourceRange ErrorRange,
714                                              RefVal::Kind ErrorKind,
715                                              SymbolRef Sym,
716                                              CheckerContext &C) const {
717   // HACK: Ignore retain-count issues on values accessed through ivars,
718   // because of cases like this:
719   //   [_contentView retain];
720   //   [_contentView removeFromSuperview];
721   //   [self addSubview:_contentView]; // invalidates 'self'
722   //   [_contentView release];
723   if (const RefVal *RV = getRefBinding(St, Sym))
724     if (RV->getIvarAccessHistory() != RefVal::IvarAccessHistory::None)
725       return;
726 
727   ExplodedNode *N = C.generateErrorNode(St);
728   if (!N)
729     return;
730 
731   CFRefBug *BT;
732   switch (ErrorKind) {
733     default:
734       llvm_unreachable("Unhandled error.");
735     case RefVal::ErrorUseAfterRelease:
736       if (!useAfterRelease)
737         useAfterRelease.reset(new UseAfterRelease(this));
738       BT = useAfterRelease.get();
739       break;
740     case RefVal::ErrorReleaseNotOwned:
741       if (!releaseNotOwned)
742         releaseNotOwned.reset(new BadRelease(this));
743       BT = releaseNotOwned.get();
744       break;
745     case RefVal::ErrorDeallocNotOwned:
746       if (!deallocNotOwned)
747         deallocNotOwned.reset(new DeallocNotOwned(this));
748       BT = deallocNotOwned.get();
749       break;
750   }
751 
752   assert(BT);
753   auto report = llvm::make_unique<CFRefReport>(
754       *BT, C.getASTContext().getLangOpts(), SummaryLog, N, Sym);
755   report->addRange(ErrorRange);
756   C.emitReport(std::move(report));
757 }
758 
759 //===----------------------------------------------------------------------===//
760 // Handle the return values of retain-count-related functions.
761 //===----------------------------------------------------------------------===//
762 
763 bool RetainCountChecker::evalCall(const CallExpr *CE, CheckerContext &C) const {
764   // Get the callee. We're only interested in simple C functions.
765   ProgramStateRef state = C.getState();
766   const FunctionDecl *FD = C.getCalleeDecl(CE);
767   if (!FD)
768     return false;
769 
770   RetainSummaryManager &SmrMgr = getSummaryManager(C);
771   QualType ResultTy = CE->getCallReturnType(C.getASTContext());
772 
773   // See if the function has 'rc_ownership_trusted_implementation'
774   // annotate attribute. If it does, we will not inline it.
775   bool hasTrustedImplementationAnnotation = false;
776 
777   const LocationContext *LCtx = C.getLocationContext();
778 
779   // Process OSDynamicCast: should just return the first argument.
780   // For now, tresting the cast as a no-op, and disregarding the case where
781   // the output becomes null due to the type mismatch.
782   if (FD->getNameAsString() == "safeMetaCast") {
783     state = state->BindExpr(CE, LCtx,
784                             state->getSVal(CE->getArg(0), LCtx));
785     C.addTransition(state);
786     return true;
787   }
788 
789   // See if it's one of the specific functions we know how to eval.
790   if (!SmrMgr.canEval(CE, FD, hasTrustedImplementationAnnotation))
791     return false;
792 
793   // Bind the return value.
794   SVal RetVal = state->getSVal(CE->getArg(0), LCtx);
795   if (RetVal.isUnknown() ||
796       (hasTrustedImplementationAnnotation && !ResultTy.isNull())) {
797     // If the receiver is unknown or the function has
798     // 'rc_ownership_trusted_implementation' annotate attribute, conjure a
799     // return value.
800     SValBuilder &SVB = C.getSValBuilder();
801     RetVal = SVB.conjureSymbolVal(nullptr, CE, LCtx, ResultTy, C.blockCount());
802   }
803   state = state->BindExpr(CE, LCtx, RetVal, false);
804 
805   // FIXME: This should not be necessary, but otherwise the argument seems to be
806   // considered alive during the next statement.
807   if (const MemRegion *ArgRegion = RetVal.getAsRegion()) {
808     // Save the refcount status of the argument.
809     SymbolRef Sym = RetVal.getAsLocSymbol();
810     const RefVal *Binding = nullptr;
811     if (Sym)
812       Binding = getRefBinding(state, Sym);
813 
814     // Invalidate the argument region.
815     state = state->invalidateRegions(
816         ArgRegion, CE, C.blockCount(), LCtx,
817         /*CausesPointerEscape*/ hasTrustedImplementationAnnotation);
818 
819     // Restore the refcount status of the argument.
820     if (Binding)
821       state = setRefBinding(state, Sym, *Binding);
822   }
823 
824   C.addTransition(state);
825   return true;
826 }
827 
828 ExplodedNode * RetainCountChecker::processReturn(const ReturnStmt *S,
829                                                  CheckerContext &C) const {
830   ExplodedNode *Pred = C.getPredecessor();
831 
832   // Only adjust the reference count if this is the top-level call frame,
833   // and not the result of inlining.  In the future, we should do
834   // better checking even for inlined calls, and see if they match
835   // with their expected semantics (e.g., the method should return a retained
836   // object, etc.).
837   if (!C.inTopFrame())
838     return Pred;
839 
840   if (!S)
841     return Pred;
842 
843   const Expr *RetE = S->getRetValue();
844   if (!RetE)
845     return Pred;
846 
847   ProgramStateRef state = C.getState();
848   SymbolRef Sym =
849     state->getSValAsScalarOrLoc(RetE, C.getLocationContext()).getAsLocSymbol();
850   if (!Sym)
851     return Pred;
852 
853   // Get the reference count binding (if any).
854   const RefVal *T = getRefBinding(state, Sym);
855   if (!T)
856     return Pred;
857 
858   // Change the reference count.
859   RefVal X = *T;
860 
861   switch (X.getKind()) {
862     case RefVal::Owned: {
863       unsigned cnt = X.getCount();
864       assert(cnt > 0);
865       X.setCount(cnt - 1);
866       X = X ^ RefVal::ReturnedOwned;
867       break;
868     }
869 
870     case RefVal::NotOwned: {
871       unsigned cnt = X.getCount();
872       if (cnt) {
873         X.setCount(cnt - 1);
874         X = X ^ RefVal::ReturnedOwned;
875       } else {
876         X = X ^ RefVal::ReturnedNotOwned;
877       }
878       break;
879     }
880 
881     default:
882       return Pred;
883   }
884 
885   // Update the binding.
886   state = setRefBinding(state, Sym, X);
887   Pred = C.addTransition(state);
888 
889   // At this point we have updated the state properly.
890   // Everything after this is merely checking to see if the return value has
891   // been over- or under-retained.
892 
893   // Did we cache out?
894   if (!Pred)
895     return nullptr;
896 
897   // Update the autorelease counts.
898   static CheckerProgramPointTag AutoreleaseTag(this, "Autorelease");
899   state = handleAutoreleaseCounts(state, Pred, &AutoreleaseTag, C, Sym, X, S);
900 
901   // Have we generated a sink node?
902   if (!state)
903     return nullptr;
904 
905   // Get the updated binding.
906   T = getRefBinding(state, Sym);
907   assert(T);
908   X = *T;
909 
910   // Consult the summary of the enclosing method.
911   RetainSummaryManager &Summaries = getSummaryManager(C);
912   const Decl *CD = &Pred->getCodeDecl();
913   RetEffect RE = RetEffect::MakeNoRet();
914 
915   // FIXME: What is the convention for blocks? Is there one?
916   if (const ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(CD)) {
917     const RetainSummary *Summ = Summaries.getMethodSummary(MD);
918     RE = Summ->getRetEffect();
919   } else if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(CD)) {
920     if (!isa<CXXMethodDecl>(FD)) {
921       const RetainSummary *Summ = Summaries.getFunctionSummary(FD);
922       RE = Summ->getRetEffect();
923     }
924   }
925 
926   return checkReturnWithRetEffect(S, C, Pred, RE, X, Sym, state);
927 }
928 
929 ExplodedNode * RetainCountChecker::checkReturnWithRetEffect(const ReturnStmt *S,
930                                                   CheckerContext &C,
931                                                   ExplodedNode *Pred,
932                                                   RetEffect RE, RefVal X,
933                                                   SymbolRef Sym,
934                                                   ProgramStateRef state) const {
935   // HACK: Ignore retain-count issues on values accessed through ivars,
936   // because of cases like this:
937   //   [_contentView retain];
938   //   [_contentView removeFromSuperview];
939   //   [self addSubview:_contentView]; // invalidates 'self'
940   //   [_contentView release];
941   if (X.getIvarAccessHistory() != RefVal::IvarAccessHistory::None)
942     return Pred;
943 
944   // Any leaks or other errors?
945   if (X.isReturnedOwned() && X.getCount() == 0) {
946     if (RE.getKind() != RetEffect::NoRet) {
947       if (!RE.isOwned()) {
948 
949         // The returning type is a CF, we expect the enclosing method should
950         // return ownership.
951         X = X ^ RefVal::ErrorLeakReturned;
952 
953         // Generate an error node.
954         state = setRefBinding(state, Sym, X);
955 
956         static CheckerProgramPointTag ReturnOwnLeakTag(this, "ReturnsOwnLeak");
957         ExplodedNode *N = C.addTransition(state, Pred, &ReturnOwnLeakTag);
958         if (N) {
959           const LangOptions &LOpts = C.getASTContext().getLangOpts();
960           auto R = llvm::make_unique<CFRefLeakReport>(
961               *getLeakAtReturnBug(LOpts), LOpts, SummaryLog, N, Sym, C,
962               IncludeAllocationLine);
963           C.emitReport(std::move(R));
964         }
965         return N;
966       }
967     }
968   } else if (X.isReturnedNotOwned()) {
969     if (RE.isOwned()) {
970       if (X.getIvarAccessHistory() ==
971             RefVal::IvarAccessHistory::AccessedDirectly) {
972         // Assume the method was trying to transfer a +1 reference from a
973         // strong ivar to the caller.
974         state = setRefBinding(state, Sym,
975                               X.releaseViaIvar() ^ RefVal::ReturnedOwned);
976       } else {
977         // Trying to return a not owned object to a caller expecting an
978         // owned object.
979         state = setRefBinding(state, Sym, X ^ RefVal::ErrorReturnedNotOwned);
980 
981         static CheckerProgramPointTag
982             ReturnNotOwnedTag(this, "ReturnNotOwnedForOwned");
983 
984         ExplodedNode *N = C.addTransition(state, Pred, &ReturnNotOwnedTag);
985         if (N) {
986           if (!returnNotOwnedForOwned)
987             returnNotOwnedForOwned.reset(new ReturnedNotOwnedForOwned(this));
988 
989           auto R = llvm::make_unique<CFRefReport>(
990               *returnNotOwnedForOwned, C.getASTContext().getLangOpts(),
991               SummaryLog, N, Sym);
992           C.emitReport(std::move(R));
993         }
994         return N;
995       }
996     }
997   }
998   return Pred;
999 }
1000 
1001 //===----------------------------------------------------------------------===//
1002 // Check various ways a symbol can be invalidated.
1003 //===----------------------------------------------------------------------===//
1004 
1005 void RetainCountChecker::checkBind(SVal loc, SVal val, const Stmt *S,
1006                                    CheckerContext &C) const {
1007   // Are we storing to something that causes the value to "escape"?
1008   bool escapes = true;
1009 
1010   // A value escapes in three possible cases (this may change):
1011   //
1012   // (1) we are binding to something that is not a memory region.
1013   // (2) we are binding to a memregion that does not have stack storage
1014   // (3) we are binding to a memregion with stack storage that the store
1015   //     does not understand.
1016   ProgramStateRef state = C.getState();
1017 
1018   if (auto regionLoc = loc.getAs<loc::MemRegionVal>()) {
1019     escapes = !regionLoc->getRegion()->hasStackStorage();
1020 
1021     if (!escapes) {
1022       // To test (3), generate a new state with the binding added.  If it is
1023       // the same state, then it escapes (since the store cannot represent
1024       // the binding).
1025       // Do this only if we know that the store is not supposed to generate the
1026       // same state.
1027       SVal StoredVal = state->getSVal(regionLoc->getRegion());
1028       if (StoredVal != val)
1029         escapes = (state == (state->bindLoc(*regionLoc, val, C.getLocationContext())));
1030     }
1031     if (!escapes) {
1032       // Case 4: We do not currently model what happens when a symbol is
1033       // assigned to a struct field, so be conservative here and let the symbol
1034       // go. TODO: This could definitely be improved upon.
1035       escapes = !isa<VarRegion>(regionLoc->getRegion());
1036     }
1037   }
1038 
1039   // If we are storing the value into an auto function scope variable annotated
1040   // with (__attribute__((cleanup))), stop tracking the value to avoid leak
1041   // false positives.
1042   if (const auto *LVR = dyn_cast_or_null<VarRegion>(loc.getAsRegion())) {
1043     const VarDecl *VD = LVR->getDecl();
1044     if (VD->hasAttr<CleanupAttr>()) {
1045       escapes = true;
1046     }
1047   }
1048 
1049   // If our store can represent the binding and we aren't storing to something
1050   // that doesn't have local storage then just return and have the simulation
1051   // state continue as is.
1052   if (!escapes)
1053       return;
1054 
1055   // Otherwise, find all symbols referenced by 'val' that we are tracking
1056   // and stop tracking them.
1057   state = state->scanReachableSymbols<StopTrackingCallback>(val).getState();
1058   C.addTransition(state);
1059 }
1060 
1061 ProgramStateRef RetainCountChecker::evalAssume(ProgramStateRef state,
1062                                                SVal Cond,
1063                                                bool Assumption) const {
1064   // FIXME: We may add to the interface of evalAssume the list of symbols
1065   //  whose assumptions have changed.  For now we just iterate through the
1066   //  bindings and check if any of the tracked symbols are NULL.  This isn't
1067   //  too bad since the number of symbols we will track in practice are
1068   //  probably small and evalAssume is only called at branches and a few
1069   //  other places.
1070   RefBindingsTy B = state->get<RefBindings>();
1071 
1072   if (B.isEmpty())
1073     return state;
1074 
1075   bool changed = false;
1076   RefBindingsTy::Factory &RefBFactory = state->get_context<RefBindings>();
1077 
1078   for (RefBindingsTy::iterator I = B.begin(), E = B.end(); I != E; ++I) {
1079     // Check if the symbol is null stop tracking the symbol.
1080     ConstraintManager &CMgr = state->getConstraintManager();
1081     ConditionTruthVal AllocFailed = CMgr.isNull(state, I.getKey());
1082     if (AllocFailed.isConstrainedTrue()) {
1083       changed = true;
1084       B = RefBFactory.remove(B, I.getKey());
1085     }
1086   }
1087 
1088   if (changed)
1089     state = state->set<RefBindings>(B);
1090 
1091   return state;
1092 }
1093 
1094 ProgramStateRef
1095 RetainCountChecker::checkRegionChanges(ProgramStateRef state,
1096                                        const InvalidatedSymbols *invalidated,
1097                                        ArrayRef<const MemRegion *> ExplicitRegions,
1098                                        ArrayRef<const MemRegion *> Regions,
1099                                        const LocationContext *LCtx,
1100                                        const CallEvent *Call) const {
1101   if (!invalidated)
1102     return state;
1103 
1104   llvm::SmallPtrSet<SymbolRef, 8> WhitelistedSymbols;
1105   for (ArrayRef<const MemRegion *>::iterator I = ExplicitRegions.begin(),
1106        E = ExplicitRegions.end(); I != E; ++I) {
1107     if (const SymbolicRegion *SR = (*I)->StripCasts()->getAs<SymbolicRegion>())
1108       WhitelistedSymbols.insert(SR->getSymbol());
1109   }
1110 
1111   for (InvalidatedSymbols::const_iterator I=invalidated->begin(),
1112        E = invalidated->end(); I!=E; ++I) {
1113     SymbolRef sym = *I;
1114     if (WhitelistedSymbols.count(sym))
1115       continue;
1116     // Remove any existing reference-count binding.
1117     state = removeRefBinding(state, sym);
1118   }
1119   return state;
1120 }
1121 
1122 ProgramStateRef
1123 RetainCountChecker::handleAutoreleaseCounts(ProgramStateRef state,
1124                                             ExplodedNode *Pred,
1125                                             const ProgramPointTag *Tag,
1126                                             CheckerContext &Ctx,
1127                                             SymbolRef Sym,
1128                                             RefVal V,
1129                                             const ReturnStmt *S) const {
1130   unsigned ACnt = V.getAutoreleaseCount();
1131 
1132   // No autorelease counts?  Nothing to be done.
1133   if (!ACnt)
1134     return state;
1135 
1136   unsigned Cnt = V.getCount();
1137 
1138   // FIXME: Handle sending 'autorelease' to already released object.
1139 
1140   if (V.getKind() == RefVal::ReturnedOwned)
1141     ++Cnt;
1142 
1143   // If we would over-release here, but we know the value came from an ivar,
1144   // assume it was a strong ivar that's just been relinquished.
1145   if (ACnt > Cnt &&
1146       V.getIvarAccessHistory() == RefVal::IvarAccessHistory::AccessedDirectly) {
1147     V = V.releaseViaIvar();
1148     --ACnt;
1149   }
1150 
1151   if (ACnt <= Cnt) {
1152     if (ACnt == Cnt) {
1153       V.clearCounts();
1154       if (V.getKind() == RefVal::ReturnedOwned) {
1155         V = V ^ RefVal::ReturnedNotOwned;
1156       } else {
1157         V = V ^ RefVal::NotOwned;
1158       }
1159     } else {
1160       V.setCount(V.getCount() - ACnt);
1161       V.setAutoreleaseCount(0);
1162     }
1163     return setRefBinding(state, Sym, V);
1164   }
1165 
1166   // HACK: Ignore retain-count issues on values accessed through ivars,
1167   // because of cases like this:
1168   //   [_contentView retain];
1169   //   [_contentView removeFromSuperview];
1170   //   [self addSubview:_contentView]; // invalidates 'self'
1171   //   [_contentView release];
1172   if (V.getIvarAccessHistory() != RefVal::IvarAccessHistory::None)
1173     return state;
1174 
1175   // Woah!  More autorelease counts then retain counts left.
1176   // Emit hard error.
1177   V = V ^ RefVal::ErrorOverAutorelease;
1178   state = setRefBinding(state, Sym, V);
1179 
1180   ExplodedNode *N = Ctx.generateSink(state, Pred, Tag);
1181   if (N) {
1182     SmallString<128> sbuf;
1183     llvm::raw_svector_ostream os(sbuf);
1184     os << "Object was autoreleased ";
1185     if (V.getAutoreleaseCount() > 1)
1186       os << V.getAutoreleaseCount() << " times but the object ";
1187     else
1188       os << "but ";
1189     os << "has a +" << V.getCount() << " retain count";
1190 
1191     if (!overAutorelease)
1192       overAutorelease.reset(new OverAutorelease(this));
1193 
1194     const LangOptions &LOpts = Ctx.getASTContext().getLangOpts();
1195     auto R = llvm::make_unique<CFRefReport>(*overAutorelease, LOpts, SummaryLog,
1196                                             N, Sym, os.str());
1197     Ctx.emitReport(std::move(R));
1198   }
1199 
1200   return nullptr;
1201 }
1202 
1203 ProgramStateRef
1204 RetainCountChecker::handleSymbolDeath(ProgramStateRef state,
1205                                       SymbolRef sid, RefVal V,
1206                                     SmallVectorImpl<SymbolRef> &Leaked) const {
1207   bool hasLeak;
1208 
1209   // HACK: Ignore retain-count issues on values accessed through ivars,
1210   // because of cases like this:
1211   //   [_contentView retain];
1212   //   [_contentView removeFromSuperview];
1213   //   [self addSubview:_contentView]; // invalidates 'self'
1214   //   [_contentView release];
1215   if (V.getIvarAccessHistory() != RefVal::IvarAccessHistory::None)
1216     hasLeak = false;
1217   else if (V.isOwned())
1218     hasLeak = true;
1219   else if (V.isNotOwned() || V.isReturnedOwned())
1220     hasLeak = (V.getCount() > 0);
1221   else
1222     hasLeak = false;
1223 
1224   if (!hasLeak)
1225     return removeRefBinding(state, sid);
1226 
1227   Leaked.push_back(sid);
1228   return setRefBinding(state, sid, V ^ RefVal::ErrorLeak);
1229 }
1230 
1231 ExplodedNode *
1232 RetainCountChecker::processLeaks(ProgramStateRef state,
1233                                  SmallVectorImpl<SymbolRef> &Leaked,
1234                                  CheckerContext &Ctx,
1235                                  ExplodedNode *Pred) const {
1236   // Generate an intermediate node representing the leak point.
1237   ExplodedNode *N = Ctx.addTransition(state, Pred);
1238 
1239   if (N) {
1240     for (SmallVectorImpl<SymbolRef>::iterator
1241          I = Leaked.begin(), E = Leaked.end(); I != E; ++I) {
1242 
1243       const LangOptions &LOpts = Ctx.getASTContext().getLangOpts();
1244       CFRefBug *BT = Pred ? getLeakWithinFunctionBug(LOpts)
1245                           : getLeakAtReturnBug(LOpts);
1246       assert(BT && "BugType not initialized.");
1247 
1248       Ctx.emitReport(llvm::make_unique<CFRefLeakReport>(
1249           *BT, LOpts, SummaryLog, N, *I, Ctx, IncludeAllocationLine));
1250     }
1251   }
1252 
1253   return N;
1254 }
1255 
1256 static bool isISLObjectRef(QualType Ty) {
1257   return StringRef(Ty.getAsString()).startswith("isl_");
1258 }
1259 
1260 void RetainCountChecker::checkBeginFunction(CheckerContext &Ctx) const {
1261   if (!Ctx.inTopFrame())
1262     return;
1263 
1264   RetainSummaryManager &SmrMgr = getSummaryManager(Ctx);
1265   const LocationContext *LCtx = Ctx.getLocationContext();
1266   const FunctionDecl *FD = dyn_cast<FunctionDecl>(LCtx->getDecl());
1267 
1268   if (!FD || SmrMgr.isTrustedReferenceCountImplementation(FD))
1269     return;
1270 
1271   ProgramStateRef state = Ctx.getState();
1272   const RetainSummary *FunctionSummary = SmrMgr.getFunctionSummary(FD);
1273   ArgEffects CalleeSideArgEffects = FunctionSummary->getArgEffects();
1274 
1275   for (unsigned idx = 0, e = FD->getNumParams(); idx != e; ++idx) {
1276     const ParmVarDecl *Param = FD->getParamDecl(idx);
1277     SymbolRef Sym = state->getSVal(state->getRegion(Param, LCtx)).getAsSymbol();
1278 
1279     QualType Ty = Param->getType();
1280     const ArgEffect *AE = CalleeSideArgEffects.lookup(idx);
1281     if (AE && *AE == DecRef && isISLObjectRef(Ty)) {
1282       state = setRefBinding(
1283           state, Sym, RefVal::makeOwned(RetEffect::ObjKind::Generalized, Ty));
1284     } else if (isISLObjectRef(Ty)) {
1285       state = setRefBinding(
1286           state, Sym,
1287           RefVal::makeNotOwned(RetEffect::ObjKind::Generalized, Ty));
1288     }
1289   }
1290 
1291   Ctx.addTransition(state);
1292 }
1293 
1294 void RetainCountChecker::checkEndFunction(const ReturnStmt *RS,
1295                                           CheckerContext &Ctx) const {
1296   ExplodedNode *Pred = processReturn(RS, Ctx);
1297 
1298   // Created state cached out.
1299   if (!Pred) {
1300     return;
1301   }
1302 
1303   ProgramStateRef state = Pred->getState();
1304   RefBindingsTy B = state->get<RefBindings>();
1305 
1306   // Don't process anything within synthesized bodies.
1307   const LocationContext *LCtx = Pred->getLocationContext();
1308   if (LCtx->getAnalysisDeclContext()->isBodyAutosynthesized()) {
1309     assert(!LCtx->inTopFrame());
1310     return;
1311   }
1312 
1313   for (RefBindingsTy::iterator I = B.begin(), E = B.end(); I != E; ++I) {
1314     state = handleAutoreleaseCounts(state, Pred, /*Tag=*/nullptr, Ctx,
1315                                     I->first, I->second);
1316     if (!state)
1317       return;
1318   }
1319 
1320   // If the current LocationContext has a parent, don't check for leaks.
1321   // We will do that later.
1322   // FIXME: we should instead check for imbalances of the retain/releases,
1323   // and suggest annotations.
1324   if (LCtx->getParent())
1325     return;
1326 
1327   B = state->get<RefBindings>();
1328   SmallVector<SymbolRef, 10> Leaked;
1329 
1330   for (RefBindingsTy::iterator I = B.begin(), E = B.end(); I != E; ++I)
1331     state = handleSymbolDeath(state, I->first, I->second, Leaked);
1332 
1333   processLeaks(state, Leaked, Ctx, Pred);
1334 }
1335 
1336 const ProgramPointTag *
1337 RetainCountChecker::getDeadSymbolTag(SymbolRef sym) const {
1338   const CheckerProgramPointTag *&tag = DeadSymbolTags[sym];
1339   if (!tag) {
1340     SmallString<64> buf;
1341     llvm::raw_svector_ostream out(buf);
1342     out << "Dead Symbol : ";
1343     sym->dumpToStream(out);
1344     tag = new CheckerProgramPointTag(this, out.str());
1345   }
1346   return tag;
1347 }
1348 
1349 void RetainCountChecker::checkDeadSymbols(SymbolReaper &SymReaper,
1350                                           CheckerContext &C) const {
1351   ExplodedNode *Pred = C.getPredecessor();
1352 
1353   ProgramStateRef state = C.getState();
1354   RefBindingsTy B = state->get<RefBindings>();
1355   SmallVector<SymbolRef, 10> Leaked;
1356 
1357   // Update counts from autorelease pools
1358   for (SymbolReaper::dead_iterator I = SymReaper.dead_begin(),
1359        E = SymReaper.dead_end(); I != E; ++I) {
1360     SymbolRef Sym = *I;
1361     if (const RefVal *T = B.lookup(Sym)){
1362       // Use the symbol as the tag.
1363       // FIXME: This might not be as unique as we would like.
1364       const ProgramPointTag *Tag = getDeadSymbolTag(Sym);
1365       state = handleAutoreleaseCounts(state, Pred, Tag, C, Sym, *T);
1366       if (!state)
1367         return;
1368 
1369       // Fetch the new reference count from the state, and use it to handle
1370       // this symbol.
1371       state = handleSymbolDeath(state, *I, *getRefBinding(state, Sym), Leaked);
1372     }
1373   }
1374 
1375   if (Leaked.empty()) {
1376     C.addTransition(state);
1377     return;
1378   }
1379 
1380   Pred = processLeaks(state, Leaked, C, Pred);
1381 
1382   // Did we cache out?
1383   if (!Pred)
1384     return;
1385 
1386   // Now generate a new node that nukes the old bindings.
1387   // The only bindings left at this point are the leaked symbols.
1388   RefBindingsTy::Factory &F = state->get_context<RefBindings>();
1389   B = state->get<RefBindings>();
1390 
1391   for (SmallVectorImpl<SymbolRef>::iterator I = Leaked.begin(),
1392                                             E = Leaked.end();
1393        I != E; ++I)
1394     B = F.remove(B, *I);
1395 
1396   state = state->set<RefBindings>(B);
1397   C.addTransition(state, Pred);
1398 }
1399 
1400 void RetainCountChecker::printState(raw_ostream &Out, ProgramStateRef State,
1401                                     const char *NL, const char *Sep) const {
1402 
1403   RefBindingsTy B = State->get<RefBindings>();
1404 
1405   if (B.isEmpty())
1406     return;
1407 
1408   Out << Sep << NL;
1409 
1410   for (RefBindingsTy::iterator I = B.begin(), E = B.end(); I != E; ++I) {
1411     Out << I->first << " : ";
1412     I->second.print(Out);
1413     Out << NL;
1414   }
1415 }
1416 
1417 //===----------------------------------------------------------------------===//
1418 // Checker registration.
1419 //===----------------------------------------------------------------------===//
1420 
1421 void ento::registerRetainCountChecker(CheckerManager &Mgr) {
1422   Mgr.registerChecker<RetainCountChecker>(Mgr.getAnalyzerOptions());
1423 }
1424