blob: b8ef6701c0df12c9199ef42c9fab19271e72c3b0 [file] [log] [blame]
Anna Zaks08be9b92011-08-04 17:28:06 +00001//==--- MacOSKeychainAPIChecker.cpp ------------------------------*- C++ -*-==//
Anna Zaks15f496c2011-08-01 22:40:01 +00002//
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// This checker flags misuses of KeyChainAPI. In particular, the password data
10// allocated/returned by SecKeychainItemCopyContent,
11// SecKeychainFindGenericPassword, SecKeychainFindInternetPassword functions has
12// to be freed using a call to SecKeychainItemFreeContent.
13//===----------------------------------------------------------------------===//
14
15#include "ClangSACheckers.h"
Chandler Carruth3a022472012-12-04 09:13:33 +000016#include "clang/StaticAnalyzer/Core/BugReporter/BugType.h"
Anna Zaks15f496c2011-08-01 22:40:01 +000017#include "clang/StaticAnalyzer/Core/Checker.h"
18#include "clang/StaticAnalyzer/Core/CheckerManager.h"
19#include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
Ted Kremenek001fd5b2011-08-15 22:09:50 +000020#include "clang/StaticAnalyzer/Core/PathSensitive/ProgramState.h"
21#include "clang/StaticAnalyzer/Core/PathSensitive/ProgramStateTrait.h"
Benjamin Kramer49038022012-02-04 13:45:25 +000022#include "llvm/ADT/SmallString.h"
Benjamin Kramer444a1302012-12-01 17:12:56 +000023#include "llvm/Support/raw_ostream.h"
Anna Zaks15f496c2011-08-01 22:40:01 +000024
25using namespace clang;
26using namespace ento;
27
28namespace {
29class MacOSKeychainAPIChecker : public Checker<check::PreStmt<CallExpr>,
Anna Zaks15f496c2011-08-01 22:40:01 +000030 check::PostStmt<CallExpr>,
Anna Zaks14b1af52017-01-13 00:50:41 +000031 check::DeadSymbols,
32 eval::Assume> {
Ahmed Charlesb8984322014-03-07 20:03:18 +000033 mutable std::unique_ptr<BugType> BT;
Anna Zaks388c18e2011-08-04 00:26:57 +000034
Anna Zaks15f496c2011-08-01 22:40:01 +000035public:
Anna Zaksc94894f2011-08-12 21:14:26 +000036 /// AllocationState is a part of the checker specific state together with the
37 /// MemRegion corresponding to the allocated data.
38 struct AllocationState {
Anna Zaksc94894f2011-08-12 21:14:26 +000039 /// The index of the allocator function.
40 unsigned int AllocatorIdx;
Anna Zaks5be8a4d2011-08-25 00:59:06 +000041 SymbolRef Region;
Anna Zaksc94894f2011-08-12 21:14:26 +000042
43 AllocationState(const Expr *E, unsigned int Idx, SymbolRef R) :
Anna Zaksc94894f2011-08-12 21:14:26 +000044 AllocatorIdx(Idx),
Anna Zaks5be8a4d2011-08-25 00:59:06 +000045 Region(R) {}
Anna Zaksc94894f2011-08-12 21:14:26 +000046
47 bool operator==(const AllocationState &X) const {
Anna Zaks5be8a4d2011-08-25 00:59:06 +000048 return (AllocatorIdx == X.AllocatorIdx &&
49 Region == X.Region);
Anna Zaksc94894f2011-08-12 21:14:26 +000050 }
Anna Zaks5be8a4d2011-08-25 00:59:06 +000051
Anna Zaksc94894f2011-08-12 21:14:26 +000052 void Profile(llvm::FoldingSetNodeID &ID) const {
Anna Zaksc94894f2011-08-12 21:14:26 +000053 ID.AddInteger(AllocatorIdx);
Anna Zaks5be8a4d2011-08-25 00:59:06 +000054 ID.AddPointer(Region);
Anna Zaksc94894f2011-08-12 21:14:26 +000055 }
56 };
57
Anna Zaks15f496c2011-08-01 22:40:01 +000058 void checkPreStmt(const CallExpr *S, CheckerContext &C) const;
Anna Zaks15f496c2011-08-01 22:40:01 +000059 void checkPostStmt(const CallExpr *S, CheckerContext &C) const;
Anna Zaksfdd0aca2011-08-12 21:56:43 +000060 void checkDeadSymbols(SymbolReaper &SR, CheckerContext &C) const;
Anna Zaks14b1af52017-01-13 00:50:41 +000061 ProgramStateRef evalAssume(ProgramStateRef state, SVal Cond,
62 bool Assumption) const;
63 void printState(raw_ostream &Out, ProgramStateRef State,
64 const char *NL, const char *Sep) const;
Anna Zaks15f496c2011-08-01 22:40:01 +000065
66private:
Anna Zaksf880cff2011-08-24 21:58:55 +000067 typedef std::pair<SymbolRef, const AllocationState*> AllocationPair;
Dmitri Gribenkof8579502013-01-12 19:30:44 +000068 typedef SmallVector<AllocationPair, 2> AllocationPairVec;
Anna Zaksbe460892011-08-24 20:52:46 +000069
70 enum APIKind {
Anna Zaks0897a232011-08-24 00:06:27 +000071 /// Denotes functions tracked by this checker.
72 ValidAPI = 0,
73 /// The functions commonly/mistakenly used in place of the given API.
74 ErrorAPI = 1,
75 /// The functions which may allocate the data. These are tracked to reduce
76 /// the false alarm rate.
77 PossibleAPI = 2
78 };
Anna Zaks08be9b92011-08-04 17:28:06 +000079 /// Stores the information about the allocator and deallocator functions -
80 /// these are the functions the checker is tracking.
81 struct ADFunctionInfo {
82 const char* Name;
83 unsigned int Param;
84 unsigned int DeallocatorIdx;
Anna Zaks0897a232011-08-24 00:06:27 +000085 APIKind Kind;
Anna Zaks08be9b92011-08-04 17:28:06 +000086 };
87 static const unsigned InvalidIdx = 100000;
Anna Zaks0897a232011-08-24 00:06:27 +000088 static const unsigned FunctionsToTrackSize = 8;
Anna Zaks08be9b92011-08-04 17:28:06 +000089 static const ADFunctionInfo FunctionsToTrack[FunctionsToTrackSize];
Anna Zaks177ecfa2011-08-05 23:52:45 +000090 /// The value, which represents no error return value for allocator functions.
91 static const unsigned NoErr = 0;
Anna Zaks15f496c2011-08-01 22:40:01 +000092
Anna Zaks08be9b92011-08-04 17:28:06 +000093 /// Given the function name, returns the index of the allocator/deallocator
94 /// function.
Anna Zaksbe460892011-08-24 20:52:46 +000095 static unsigned getTrackedFunctionIndex(StringRef Name, bool IsAllocator);
Anna Zaks388c18e2011-08-04 00:26:57 +000096
97 inline void initBugType() const {
98 if (!BT)
Alexander Kornienko4aca9b12014-02-11 21:49:21 +000099 BT.reset(new BugType(this, "Improper use of SecKeychain API",
Anna Zaks8ef07e52013-04-03 19:28:22 +0000100 "API Misuse (Apple)"));
Anna Zaks388c18e2011-08-04 00:26:57 +0000101 }
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000102
Anna Zaks85913db2011-08-25 00:32:42 +0000103 void generateDeallocatorMismatchReport(const AllocationPair &AP,
Anna Zaksbb167012011-08-23 23:47:36 +0000104 const Expr *ArgExpr,
Anna Zaks85913db2011-08-25 00:32:42 +0000105 CheckerContext &C) const;
Anna Zaksbb167012011-08-23 23:47:36 +0000106
Anna Zaks4b062cb2012-02-23 22:53:29 +0000107 /// Find the allocation site for Sym on the path leading to the node N.
Anna Zaksa043d0c2013-01-08 00:25:29 +0000108 const ExplodedNode *getAllocationNode(const ExplodedNode *N, SymbolRef Sym,
109 CheckerContext &C) const;
Anna Zaks4b062cb2012-02-23 22:53:29 +0000110
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000111 std::unique_ptr<BugReport> generateAllocatedDataNotReleasedReport(
112 const AllocationPair &AP, ExplodedNode *N, CheckerContext &C) const;
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000113
Ted Kremenek1e809b42012-03-09 01:13:14 +0000114 /// Mark an AllocationPair interesting for diagnostic reporting.
115 void markInteresting(BugReport *R, const AllocationPair &AP) const {
116 R->markInteresting(AP.first);
117 R->markInteresting(AP.second->Region);
118 }
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000119
Anna Zaksbe460892011-08-24 20:52:46 +0000120 /// The bug visitor which allows us to print extra diagnostics along the
121 /// BugReport path. For example, showing the allocation site of the leaked
122 /// region.
George Karpenkov70ec1dd2018-06-26 21:12:08 +0000123 class SecKeychainBugVisitor : public BugReporterVisitor {
Anna Zaksbe460892011-08-24 20:52:46 +0000124 protected:
125 // The allocated region symbol tracked by the main analysis.
126 SymbolRef Sym;
127
128 public:
129 SecKeychainBugVisitor(SymbolRef S) : Sym(S) {}
Anna Zaksbe460892011-08-24 20:52:46 +0000130
Craig Topperfb6b25b2014-03-15 04:29:04 +0000131 void Profile(llvm::FoldingSetNodeID &ID) const override {
Anna Zaksbe460892011-08-24 20:52:46 +0000132 static int X = 0;
133 ID.AddPointer(&X);
134 ID.AddPointer(Sym);
135 }
136
David Blaikie0a0c2752017-01-05 17:26:53 +0000137 std::shared_ptr<PathDiagnosticPiece> VisitNode(const ExplodedNode *N,
138 const ExplodedNode *PrevN,
139 BugReporterContext &BRC,
140 BugReport &BR) override;
Anna Zaksbe460892011-08-24 20:52:46 +0000141 };
Anna Zaks15f496c2011-08-01 22:40:01 +0000142};
Alexander Kornienkoab9db512015-06-22 23:07:51 +0000143}
Anna Zaks15f496c2011-08-01 22:40:01 +0000144
Anna Zaks5443a642011-08-15 23:23:15 +0000145/// ProgramState traits to store the currently allocated (and not yet freed)
146/// symbols. This is a map from the allocated content symbol to the
147/// corresponding AllocationState.
Jordan Rose0c153cb2012-11-02 01:54:06 +0000148REGISTER_MAP_WITH_PROGRAMSTATE(AllocatedData,
149 SymbolRef,
150 MacOSKeychainAPIChecker::AllocationState)
Anna Zaks15f496c2011-08-01 22:40:01 +0000151
Anna Zaks388c18e2011-08-04 00:26:57 +0000152static bool isEnclosingFunctionParam(const Expr *E) {
153 E = E->IgnoreParenCasts();
154 if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(E)) {
155 const ValueDecl *VD = DRE->getDecl();
156 if (isa<ImplicitParamDecl>(VD) || isa<ParmVarDecl>(VD))
157 return true;
158 }
159 return false;
160}
161
Anna Zaks08be9b92011-08-04 17:28:06 +0000162const MacOSKeychainAPIChecker::ADFunctionInfo
163 MacOSKeychainAPIChecker::FunctionsToTrack[FunctionsToTrackSize] = {
Anna Zaks0897a232011-08-24 00:06:27 +0000164 {"SecKeychainItemCopyContent", 4, 3, ValidAPI}, // 0
165 {"SecKeychainFindGenericPassword", 6, 3, ValidAPI}, // 1
166 {"SecKeychainFindInternetPassword", 13, 3, ValidAPI}, // 2
167 {"SecKeychainItemFreeContent", 1, InvalidIdx, ValidAPI}, // 3
168 {"SecKeychainItemCopyAttributesAndData", 5, 5, ValidAPI}, // 4
169 {"SecKeychainItemFreeAttributesAndData", 1, InvalidIdx, ValidAPI}, // 5
170 {"free", 0, InvalidIdx, ErrorAPI}, // 6
171 {"CFStringCreateWithBytesNoCopy", 1, InvalidIdx, PossibleAPI}, // 7
Anna Zaks08be9b92011-08-04 17:28:06 +0000172};
173
174unsigned MacOSKeychainAPIChecker::getTrackedFunctionIndex(StringRef Name,
Anna Zaksbe460892011-08-24 20:52:46 +0000175 bool IsAllocator) {
Anna Zaks08be9b92011-08-04 17:28:06 +0000176 for (unsigned I = 0; I < FunctionsToTrackSize; ++I) {
177 ADFunctionInfo FI = FunctionsToTrack[I];
178 if (FI.Name != Name)
179 continue;
180 // Make sure the function is of the right type (allocator vs deallocator).
181 if (IsAllocator && (FI.DeallocatorIdx == InvalidIdx))
182 return InvalidIdx;
183 if (!IsAllocator && (FI.DeallocatorIdx != InvalidIdx))
184 return InvalidIdx;
185
186 return I;
187 }
188 // The function is not tracked.
189 return InvalidIdx;
190}
191
Anna Zaksc94894f2011-08-12 21:14:26 +0000192static bool isBadDeallocationArgument(const MemRegion *Arg) {
Jordy Rosef80b2cc2012-03-11 00:08:24 +0000193 if (!Arg)
194 return false;
Alexander Kornienko9c104902015-12-28 13:06:58 +0000195 return isa<AllocaRegion>(Arg) || isa<BlockDataRegion>(Arg) ||
196 isa<TypedRegion>(Arg);
Anna Zaksc94894f2011-08-12 21:14:26 +0000197}
Jordy Rosef80b2cc2012-03-11 00:08:24 +0000198
Anna Zaksc52bed12011-08-05 00:37:00 +0000199/// Given the address expression, retrieve the value it's pointing to. Assume
Anna Zaksc94894f2011-08-12 21:14:26 +0000200/// that value is itself an address, and return the corresponding symbol.
201static SymbolRef getAsPointeeSymbol(const Expr *Expr,
202 CheckerContext &C) {
Ted Kremenek49b1e382012-01-26 21:29:00 +0000203 ProgramStateRef State = C.getState();
George Karpenkovd703ec92018-01-17 20:27:29 +0000204 SVal ArgV = C.getSVal(Expr);
Anna Zaks177ecfa2011-08-05 23:52:45 +0000205
David Blaikie05785d12013-02-20 22:23:23 +0000206 if (Optional<loc::MemRegionVal> X = ArgV.getAs<loc::MemRegionVal>()) {
Anna Zaksc52bed12011-08-05 00:37:00 +0000207 StoreManager& SM = C.getStoreManager();
Jordy Rosef80b2cc2012-03-11 00:08:24 +0000208 SymbolRef sym = SM.getBinding(State->getStore(), *X).getAsLocSymbol();
209 if (sym)
210 return sym;
Anna Zaksc52bed12011-08-05 00:37:00 +0000211 }
Craig Topper0dbb7832014-05-27 02:45:47 +0000212 return nullptr;
Anna Zaksc52bed12011-08-05 00:37:00 +0000213}
214
Anna Zaksbb167012011-08-23 23:47:36 +0000215// Report deallocator mismatch. Remove the region from tracking - reporting a
216// missing free error after this one is redundant.
217void MacOSKeychainAPIChecker::
Anna Zaks85913db2011-08-25 00:32:42 +0000218 generateDeallocatorMismatchReport(const AllocationPair &AP,
Anna Zaksbb167012011-08-23 23:47:36 +0000219 const Expr *ArgExpr,
Anna Zaks85913db2011-08-25 00:32:42 +0000220 CheckerContext &C) const {
Ted Kremenek49b1e382012-01-26 21:29:00 +0000221 ProgramStateRef State = C.getState();
Anna Zaks85913db2011-08-25 00:32:42 +0000222 State = State->remove<AllocatedData>(AP.first);
Devin Coughline39bd402015-09-16 22:03:05 +0000223 ExplodedNode *N = C.generateNonFatalErrorNode(State);
Anna Zaksbb167012011-08-23 23:47:36 +0000224
225 if (!N)
226 return;
227 initBugType();
Dylan Noblesmith2c1dd272012-02-05 02:13:05 +0000228 SmallString<80> sbuf;
Anna Zaksbb167012011-08-23 23:47:36 +0000229 llvm::raw_svector_ostream os(sbuf);
Anna Zaks85913db2011-08-25 00:32:42 +0000230 unsigned int PDeallocIdx =
231 FunctionsToTrack[AP.second->AllocatorIdx].DeallocatorIdx;
Anna Zaksbb167012011-08-23 23:47:36 +0000232
233 os << "Deallocator doesn't match the allocator: '"
234 << FunctionsToTrack[PDeallocIdx].Name << "' should be used.";
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000235 auto Report = llvm::make_unique<BugReport>(*BT, os.str(), N);
David Blaikie91e79022014-09-04 23:54:33 +0000236 Report->addVisitor(llvm::make_unique<SecKeychainBugVisitor>(AP.first));
Anna Zaksbb167012011-08-23 23:47:36 +0000237 Report->addRange(ArgExpr->getSourceRange());
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000238 markInteresting(Report.get(), AP);
239 C.emitReport(std::move(Report));
Anna Zaksbb167012011-08-23 23:47:36 +0000240}
241
Anna Zaks15f496c2011-08-01 22:40:01 +0000242void MacOSKeychainAPIChecker::checkPreStmt(const CallExpr *CE,
243 CheckerContext &C) const {
Anna Zaksc52bed12011-08-05 00:37:00 +0000244 unsigned idx = InvalidIdx;
Ted Kremenek49b1e382012-01-26 21:29:00 +0000245 ProgramStateRef State = C.getState();
Anna Zaks15f496c2011-08-01 22:40:01 +0000246
Jordan Rose6cd16c52012-07-10 23:13:01 +0000247 const FunctionDecl *FD = C.getCalleeDecl(CE);
248 if (!FD || FD->getKind() != Decl::Function)
249 return;
Ted Kremenek3a0678e2015-09-08 03:50:52 +0000250
Jordan Rose6cd16c52012-07-10 23:13:01 +0000251 StringRef funName = C.getCalleeName(FD);
Anna Zaksc6aa5312011-12-01 05:57:37 +0000252 if (funName.empty())
Anna Zaks15f496c2011-08-01 22:40:01 +0000253 return;
Anna Zaks15f496c2011-08-01 22:40:01 +0000254
Anna Zaksc52bed12011-08-05 00:37:00 +0000255 // If it is a call to an allocator function, it could be a double allocation.
256 idx = getTrackedFunctionIndex(funName, true);
257 if (idx != InvalidIdx) {
Anna Zaks33f06322015-02-05 01:02:56 +0000258 unsigned paramIdx = FunctionsToTrack[idx].Param;
259 if (CE->getNumArgs() <= paramIdx)
260 return;
261
262 const Expr *ArgExpr = CE->getArg(paramIdx);
Anna Zaksc94894f2011-08-12 21:14:26 +0000263 if (SymbolRef V = getAsPointeeSymbol(ArgExpr, C))
Anna Zaksc52bed12011-08-05 00:37:00 +0000264 if (const AllocationState *AS = State->get<AllocatedData>(V)) {
Anna Zaks14b1af52017-01-13 00:50:41 +0000265 // Remove the value from the state. The new symbol will be added for
266 // tracking when the second allocator is processed in checkPostStmt().
267 State = State->remove<AllocatedData>(V);
268 ExplodedNode *N = C.generateNonFatalErrorNode(State);
269 if (!N)
270 return;
271 initBugType();
272 SmallString<128> sbuf;
273 llvm::raw_svector_ostream os(sbuf);
274 unsigned int DIdx = FunctionsToTrack[AS->AllocatorIdx].DeallocatorIdx;
275 os << "Allocated data should be released before another call to "
276 << "the allocator: missing a call to '"
277 << FunctionsToTrack[DIdx].Name
278 << "'.";
279 auto Report = llvm::make_unique<BugReport>(*BT, os.str(), N);
280 Report->addVisitor(llvm::make_unique<SecKeychainBugVisitor>(V));
281 Report->addRange(ArgExpr->getSourceRange());
282 Report->markInteresting(AS->Region);
283 C.emitReport(std::move(Report));
Anna Zaksc52bed12011-08-05 00:37:00 +0000284 }
285 return;
286 }
287
288 // Is it a call to one of deallocator functions?
289 idx = getTrackedFunctionIndex(funName, false);
Anna Zaks08be9b92011-08-04 17:28:06 +0000290 if (idx == InvalidIdx)
Anna Zaks4aa34a532011-08-04 00:31:38 +0000291 return;
292
Anna Zaks33f06322015-02-05 01:02:56 +0000293 unsigned paramIdx = FunctionsToTrack[idx].Param;
294 if (CE->getNumArgs() <= paramIdx)
295 return;
296
Anna Zaksc94894f2011-08-12 21:14:26 +0000297 // Check the argument to the deallocator.
Anna Zaks33f06322015-02-05 01:02:56 +0000298 const Expr *ArgExpr = CE->getArg(paramIdx);
George Karpenkovd703ec92018-01-17 20:27:29 +0000299 SVal ArgSVal = C.getSVal(ArgExpr);
Anna Zaksc94894f2011-08-12 21:14:26 +0000300
301 // Undef is reported by another checker.
302 if (ArgSVal.isUndef())
303 return;
304
Jordy Rosef80b2cc2012-03-11 00:08:24 +0000305 SymbolRef ArgSM = ArgSVal.getAsLocSymbol();
Anna Zaksc94894f2011-08-12 21:14:26 +0000306
Anna Zaksc94894f2011-08-12 21:14:26 +0000307 // If the argument is coming from the heap, globals, or unknown, do not
308 // report it.
Jordy Rosef80b2cc2012-03-11 00:08:24 +0000309 bool RegionArgIsBad = false;
310 if (!ArgSM) {
311 if (!isBadDeallocationArgument(ArgSVal.getAsRegion()))
312 return;
313 RegionArgIsBad = true;
314 }
Anna Zaks4aa34a532011-08-04 00:31:38 +0000315
Anna Zaks0897a232011-08-24 00:06:27 +0000316 // Is the argument to the call being tracked?
317 const AllocationState *AS = State->get<AllocatedData>(ArgSM);
Anna Zaks14b1af52017-01-13 00:50:41 +0000318 if (!AS)
Anna Zaks0897a232011-08-24 00:06:27 +0000319 return;
Anna Zaks14b1af52017-01-13 00:50:41 +0000320
321 // TODO: We might want to report double free here.
Anna Zaks5443a642011-08-15 23:23:15 +0000322 // (that would involve tracking all the freed symbols in the checker state).
Anna Zaks14b1af52017-01-13 00:50:41 +0000323 if (RegionArgIsBad) {
Anna Zaks4aa34a532011-08-04 00:31:38 +0000324 // It is possible that this is a false positive - the argument might
325 // have entered as an enclosing function parameter.
326 if (isEnclosingFunctionParam(ArgExpr))
Anna Zaks15f496c2011-08-01 22:40:01 +0000327 return;
Anna Zaks388c18e2011-08-04 00:26:57 +0000328
Devin Coughline39bd402015-09-16 22:03:05 +0000329 ExplodedNode *N = C.generateNonFatalErrorNode(State);
Anna Zaks4aa34a532011-08-04 00:31:38 +0000330 if (!N)
331 return;
332 initBugType();
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000333 auto Report = llvm::make_unique<BugReport>(
334 *BT, "Trying to free data which has not been allocated.", N);
Anna Zaks4aa34a532011-08-04 00:31:38 +0000335 Report->addRange(ArgExpr->getSourceRange());
Ted Kremenek1e809b42012-03-09 01:13:14 +0000336 if (AS)
337 Report->markInteresting(AS->Region);
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000338 C.emitReport(std::move(Report));
Anna Zaks08be9b92011-08-04 17:28:06 +0000339 return;
Anna Zaks15f496c2011-08-01 22:40:01 +0000340 }
Anna Zaks4aa34a532011-08-04 00:31:38 +0000341
Anna Zaks0897a232011-08-24 00:06:27 +0000342 // Process functions which might deallocate.
343 if (FunctionsToTrack[idx].Kind == PossibleAPI) {
344
345 if (funName == "CFStringCreateWithBytesNoCopy") {
346 const Expr *DeallocatorExpr = CE->getArg(5)->IgnoreParenCasts();
347 // NULL ~ default deallocator, so warn.
348 if (DeallocatorExpr->isNullPointerConstant(C.getASTContext(),
349 Expr::NPC_ValueDependentIsNotNull)) {
Anna Zaks85913db2011-08-25 00:32:42 +0000350 const AllocationPair AP = std::make_pair(ArgSM, AS);
351 generateDeallocatorMismatchReport(AP, ArgExpr, C);
Anna Zaks0897a232011-08-24 00:06:27 +0000352 return;
353 }
354 // One of the default allocators, so warn.
355 if (const DeclRefExpr *DE = dyn_cast<DeclRefExpr>(DeallocatorExpr)) {
356 StringRef DeallocatorName = DE->getFoundDecl()->getName();
357 if (DeallocatorName == "kCFAllocatorDefault" ||
358 DeallocatorName == "kCFAllocatorSystemDefault" ||
359 DeallocatorName == "kCFAllocatorMalloc") {
Anna Zaks85913db2011-08-25 00:32:42 +0000360 const AllocationPair AP = std::make_pair(ArgSM, AS);
361 generateDeallocatorMismatchReport(AP, ArgExpr, C);
Anna Zaks0897a232011-08-24 00:06:27 +0000362 return;
363 }
364 // If kCFAllocatorNull, which does not deallocate, we still have to
Anna Zaks030e65d2013-01-07 19:13:00 +0000365 // find the deallocator.
366 if (DE->getFoundDecl()->getName() == "kCFAllocatorNull")
Anna Zaks0897a232011-08-24 00:06:27 +0000367 return;
Anna Zaks0897a232011-08-24 00:06:27 +0000368 }
Anna Zaks030e65d2013-01-07 19:13:00 +0000369 // In all other cases, assume the user supplied a correct deallocator
370 // that will free memory so stop tracking.
371 State = State->remove<AllocatedData>(ArgSM);
372 C.addTransition(State);
373 return;
Anna Zaks0897a232011-08-24 00:06:27 +0000374 }
Anna Zaks030e65d2013-01-07 19:13:00 +0000375
376 llvm_unreachable("We know of no other possible APIs.");
Anna Zaks0897a232011-08-24 00:06:27 +0000377 }
378
Anna Zaks5443a642011-08-15 23:23:15 +0000379 // The call is deallocating a value we previously allocated, so remove it
380 // from the next state.
381 State = State->remove<AllocatedData>(ArgSM);
382
Anna Zaksbb167012011-08-23 23:47:36 +0000383 // Check if the proper deallocator is used.
Anna Zaksc6861772011-08-04 21:53:01 +0000384 unsigned int PDeallocIdx = FunctionsToTrack[AS->AllocatorIdx].DeallocatorIdx;
Anna Zaks0897a232011-08-24 00:06:27 +0000385 if (PDeallocIdx != idx || (FunctionsToTrack[idx].Kind == ErrorAPI)) {
Anna Zaks85913db2011-08-25 00:32:42 +0000386 const AllocationPair AP = std::make_pair(ArgSM, AS);
387 generateDeallocatorMismatchReport(AP, ArgExpr, C);
Anna Zaksc6861772011-08-04 21:53:01 +0000388 return;
389 }
390
Anna Zaksda4c8d62011-10-26 21:06:34 +0000391 C.addTransition(State);
Anna Zaks15f496c2011-08-01 22:40:01 +0000392}
393
394void MacOSKeychainAPIChecker::checkPostStmt(const CallExpr *CE,
395 CheckerContext &C) const {
Ted Kremenek49b1e382012-01-26 21:29:00 +0000396 ProgramStateRef State = C.getState();
Jordan Rose6cd16c52012-07-10 23:13:01 +0000397 const FunctionDecl *FD = C.getCalleeDecl(CE);
398 if (!FD || FD->getKind() != Decl::Function)
399 return;
400
401 StringRef funName = C.getCalleeName(FD);
Anna Zaks15f496c2011-08-01 22:40:01 +0000402
403 // If a value has been allocated, add it to the set for tracking.
Anna Zaks08be9b92011-08-04 17:28:06 +0000404 unsigned idx = getTrackedFunctionIndex(funName, true);
405 if (idx == InvalidIdx)
Anna Zaks4aa34a532011-08-04 00:31:38 +0000406 return;
Anna Zaks388c18e2011-08-04 00:26:57 +0000407
Anna Zaks08be9b92011-08-04 17:28:06 +0000408 const Expr *ArgExpr = CE->getArg(FunctionsToTrack[idx].Param);
Anna Zaks59d741f2011-08-12 22:47:22 +0000409 // If the argument entered as an enclosing function parameter, skip it to
410 // avoid false positives.
Anna Zaks19a66672012-02-21 00:00:44 +0000411 if (isEnclosingFunctionParam(ArgExpr) &&
Craig Topper0dbb7832014-05-27 02:45:47 +0000412 C.getLocationContext()->getParent() == nullptr)
Anna Zaks59d741f2011-08-12 22:47:22 +0000413 return;
414
Anna Zaksc94894f2011-08-12 21:14:26 +0000415 if (SymbolRef V = getAsPointeeSymbol(ArgExpr, C)) {
416 // If the argument points to something that's not a symbolic region, it
417 // can be:
Anna Zaks4aa34a532011-08-04 00:31:38 +0000418 // - unknown (cannot reason about it)
419 // - undefined (already reported by other checker)
Anna Zaks08be9b92011-08-04 17:28:06 +0000420 // - constant (null - should not be tracked,
421 // other constant will generate a compiler warning)
Anna Zaks4aa34a532011-08-04 00:31:38 +0000422 // - goto (should be reported by other checker)
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000423
424 // The call return value symbol should stay alive for as long as the
425 // allocated value symbol, since our diagnostics depend on the value
426 // returned by the call. Ex: Data should only be freed if noErr was
427 // returned during allocation.)
George Karpenkovd703ec92018-01-17 20:27:29 +0000428 SymbolRef RetStatusSymbol = C.getSVal(CE).getAsSymbol();
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000429 C.getSymbolManager().addSymbolDependency(V, RetStatusSymbol);
430
431 // Track the allocated value in the checker state.
432 State = State->set<AllocatedData>(V, AllocationState(ArgExpr, idx,
Anna Zaksc94894f2011-08-12 21:14:26 +0000433 RetStatusSymbol));
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000434 assert(State);
Anna Zaksda4c8d62011-10-26 21:06:34 +0000435 C.addTransition(State);
Anna Zaks15f496c2011-08-01 22:40:01 +0000436 }
437}
438
Anna Zaks6ca4fd52012-02-28 03:07:06 +0000439// TODO: This logic is the same as in Malloc checker.
Anna Zaksa043d0c2013-01-08 00:25:29 +0000440const ExplodedNode *
441MacOSKeychainAPIChecker::getAllocationNode(const ExplodedNode *N,
Anna Zaks4b062cb2012-02-23 22:53:29 +0000442 SymbolRef Sym,
443 CheckerContext &C) const {
Anna Zaks6ca4fd52012-02-28 03:07:06 +0000444 const LocationContext *LeakContext = N->getLocationContext();
Anna Zaks4b062cb2012-02-23 22:53:29 +0000445 // Walk the ExplodedGraph backwards and find the first node that referred to
446 // the tracked symbol.
447 const ExplodedNode *AllocNode = N;
448
449 while (N) {
450 if (!N->getState()->get<AllocatedData>(Sym))
451 break;
Anna Zaks486a0ff2015-02-05 01:02:53 +0000452 // Allocation node, is the last node in the current or parent context in
453 // which the symbol was tracked.
454 const LocationContext *NContext = N->getLocationContext();
455 if (NContext == LeakContext ||
456 NContext->isParentOf(LeakContext))
Anna Zaks6ca4fd52012-02-28 03:07:06 +0000457 AllocNode = N;
Craig Topper0dbb7832014-05-27 02:45:47 +0000458 N = N->pred_empty() ? nullptr : *(N->pred_begin());
Anna Zaks4b062cb2012-02-23 22:53:29 +0000459 }
460
Anna Zaksa043d0c2013-01-08 00:25:29 +0000461 return AllocNode;
Anna Zaks4b062cb2012-02-23 22:53:29 +0000462}
463
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000464std::unique_ptr<BugReport>
465MacOSKeychainAPIChecker::generateAllocatedDataNotReleasedReport(
466 const AllocationPair &AP, ExplodedNode *N, CheckerContext &C) const {
Anna Zaksf880cff2011-08-24 21:58:55 +0000467 const ADFunctionInfo &FI = FunctionsToTrack[AP.second->AllocatorIdx];
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000468 initBugType();
Dylan Noblesmith2c1dd272012-02-05 02:13:05 +0000469 SmallString<70> sbuf;
Anna Zaks29f9b7a2011-08-15 18:42:00 +0000470 llvm::raw_svector_ostream os(sbuf);
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000471 os << "Allocated data is not released: missing a call to '"
472 << FunctionsToTrack[FI.DeallocatorIdx].Name << "'.";
Anna Zaks4b062cb2012-02-23 22:53:29 +0000473
474 // Most bug reports are cached at the location where they occurred.
475 // With leaks, we want to unique them by the location where they were
476 // allocated, and only report a single path.
Anna Zaks6ca4fd52012-02-28 03:07:06 +0000477 PathDiagnosticLocation LocUsedForUniqueing;
Anna Zaksa043d0c2013-01-08 00:25:29 +0000478 const ExplodedNode *AllocNode = getAllocationNode(N, AP.first, C);
Gabor Horvath6ee4f902016-08-18 07:54:50 +0000479 const Stmt *AllocStmt = PathDiagnosticLocation::getStmt(AllocNode);
Anna Zaks4b062cb2012-02-23 22:53:29 +0000480
Anna Zaksa043d0c2013-01-08 00:25:29 +0000481 if (AllocStmt)
482 LocUsedForUniqueing = PathDiagnosticLocation::createBegin(AllocStmt,
483 C.getSourceManager(),
484 AllocNode->getLocationContext());
485
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000486 auto Report =
487 llvm::make_unique<BugReport>(*BT, os.str(), N, LocUsedForUniqueing,
488 AllocNode->getLocationContext()->getDecl());
Anna Zaksa043d0c2013-01-08 00:25:29 +0000489
David Blaikie91e79022014-09-04 23:54:33 +0000490 Report->addVisitor(llvm::make_unique<SecKeychainBugVisitor>(AP.first));
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000491 markInteresting(Report.get(), AP);
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000492 return Report;
493}
494
Anna Zaks14b1af52017-01-13 00:50:41 +0000495/// If the return symbol is assumed to be error, remove the allocated info
496/// from consideration.
497ProgramStateRef MacOSKeychainAPIChecker::evalAssume(ProgramStateRef State,
498 SVal Cond,
499 bool Assumption) const {
500 AllocatedDataTy AMap = State->get<AllocatedData>();
501 if (AMap.isEmpty())
502 return State;
503
504 auto *CondBSE = dyn_cast_or_null<BinarySymExpr>(Cond.getAsSymExpr());
505 if (!CondBSE)
506 return State;
507 BinaryOperator::Opcode OpCode = CondBSE->getOpcode();
508 if (OpCode != BO_EQ && OpCode != BO_NE)
509 return State;
510
511 // Match for a restricted set of patterns for cmparison of error codes.
512 // Note, the comparisons of type '0 == st' are transformed into SymIntExpr.
513 SymbolRef ReturnSymbol = nullptr;
514 if (auto *SIE = dyn_cast<SymIntExpr>(CondBSE)) {
515 const llvm::APInt &RHS = SIE->getRHS();
516 bool ErrorIsReturned = (OpCode == BO_EQ && RHS != NoErr) ||
517 (OpCode == BO_NE && RHS == NoErr);
518 if (!Assumption)
519 ErrorIsReturned = !ErrorIsReturned;
520 if (ErrorIsReturned)
521 ReturnSymbol = SIE->getLHS();
522 }
523
524 if (ReturnSymbol)
525 for (auto I = AMap.begin(), E = AMap.end(); I != E; ++I) {
526 if (ReturnSymbol == I->second.Region)
527 State = State->remove<AllocatedData>(I->first);
528 }
529
530 return State;
531}
532
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000533void MacOSKeychainAPIChecker::checkDeadSymbols(SymbolReaper &SR,
534 CheckerContext &C) const {
Ted Kremenek49b1e382012-01-26 21:29:00 +0000535 ProgramStateRef State = C.getState();
Anna Zaks14b1af52017-01-13 00:50:41 +0000536 AllocatedDataTy AMap = State->get<AllocatedData>();
537 if (AMap.isEmpty())
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000538 return;
539
540 bool Changed = false;
Anna Zaksbe460892011-08-24 20:52:46 +0000541 AllocationPairVec Errors;
Anna Zaks14b1af52017-01-13 00:50:41 +0000542 for (auto I = AMap.begin(), E = AMap.end(); I != E; ++I) {
543 if (!SR.isDead(I->first))
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000544 continue;
545
546 Changed = true;
547 State = State->remove<AllocatedData>(I->first);
Anna Zaks14b1af52017-01-13 00:50:41 +0000548 // If the allocated symbol is null do not report.
Jordan Rose14fe9f32012-11-01 00:18:27 +0000549 ConstraintManager &CMgr = State->getConstraintManager();
550 ConditionTruthVal AllocFailed = CMgr.isNull(State, I.getKey());
Anna Zaks14b1af52017-01-13 00:50:41 +0000551 if (AllocFailed.isConstrainedTrue())
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000552 continue;
Anna Zaksf880cff2011-08-24 21:58:55 +0000553 Errors.push_back(std::make_pair(I->first, &I->second));
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000554 }
Anna Zaks4b062cb2012-02-23 22:53:29 +0000555 if (!Changed) {
556 // Generate the new, cleaned up state.
557 C.addTransition(State);
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000558 return;
Anna Zaks4b062cb2012-02-23 22:53:29 +0000559 }
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000560
Anton Yartsev6a619222014-02-17 18:25:34 +0000561 static CheckerProgramPointTag Tag(this, "DeadSymbolsLeak");
Devin Coughline39bd402015-09-16 22:03:05 +0000562 ExplodedNode *N = C.generateNonFatalErrorNode(C.getState(), &Tag);
563 if (!N)
564 return;
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000565
566 // Generate the error reports.
Benjamin Kramer72e64312015-09-24 14:48:49 +0000567 for (const auto &P : Errors)
Aaron Ballman8d3a7a52015-06-23 13:15:32 +0000568 C.emitReport(generateAllocatedDataNotReleasedReport(P, N, C));
Anna Zaks4b062cb2012-02-23 22:53:29 +0000569
570 // Generate the new, cleaned up state.
571 C.addTransition(State, N);
Anna Zaksfdd0aca2011-08-12 21:56:43 +0000572}
573
David Blaikie0a0c2752017-01-05 17:26:53 +0000574std::shared_ptr<PathDiagnosticPiece>
575MacOSKeychainAPIChecker::SecKeychainBugVisitor::VisitNode(
576 const ExplodedNode *N, const ExplodedNode *PrevN, BugReporterContext &BRC,
577 BugReport &BR) {
Anna Zaksbe460892011-08-24 20:52:46 +0000578 const AllocationState *AS = N->getState()->get<AllocatedData>(Sym);
579 if (!AS)
Craig Topper0dbb7832014-05-27 02:45:47 +0000580 return nullptr;
Anna Zaksbe460892011-08-24 20:52:46 +0000581 const AllocationState *ASPrev = PrevN->getState()->get<AllocatedData>(Sym);
582 if (ASPrev)
Craig Topper0dbb7832014-05-27 02:45:47 +0000583 return nullptr;
Anna Zaksbe460892011-08-24 20:52:46 +0000584
585 // (!ASPrev && AS) ~ We started tracking symbol in node N, it must be the
586 // allocation site.
David Blaikie87396b92013-02-21 22:23:56 +0000587 const CallExpr *CE =
588 cast<CallExpr>(N->getLocation().castAs<StmtPoint>().getStmt());
Anna Zaksbe460892011-08-24 20:52:46 +0000589 const FunctionDecl *funDecl = CE->getDirectCallee();
590 assert(funDecl && "We do not support indirect function calls as of now.");
591 StringRef funName = funDecl->getName();
592
593 // Get the expression of the corresponding argument.
594 unsigned Idx = getTrackedFunctionIndex(funName, true);
595 assert(Idx != InvalidIdx && "This should be a call to an allocator.");
596 const Expr *ArgExpr = CE->getArg(FunctionsToTrack[Idx].Param);
Anna Zaks3a769bd2011-09-15 01:08:34 +0000597 PathDiagnosticLocation Pos(ArgExpr, BRC.getSourceManager(),
598 N->getLocationContext());
David Blaikie0a0c2752017-01-05 17:26:53 +0000599 return std::make_shared<PathDiagnosticEventPiece>(Pos,
600 "Data is allocated here.");
Anna Zaks15f496c2011-08-01 22:40:01 +0000601}
602
Anna Zaks14b1af52017-01-13 00:50:41 +0000603void MacOSKeychainAPIChecker::printState(raw_ostream &Out,
604 ProgramStateRef State,
605 const char *NL,
606 const char *Sep) const {
607
608 AllocatedDataTy AMap = State->get<AllocatedData>();
609
610 if (!AMap.isEmpty()) {
611 Out << Sep << "KeychainAPIChecker :" << NL;
612 for (auto I = AMap.begin(), E = AMap.end(); I != E; ++I) {
613 I.getKey()->dumpToStream(Out);
614 }
615 }
616}
617
618
Anna Zaks15f496c2011-08-01 22:40:01 +0000619void ento::registerMacOSKeychainAPIChecker(CheckerManager &mgr) {
620 mgr.registerChecker<MacOSKeychainAPIChecker>();
621}