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