1d99bd55aSTed Kremenek //==- CheckSecuritySyntaxOnly.cpp - Basic security checks --------*- C++ -*-==//
2d99bd55aSTed Kremenek //
3d99bd55aSTed Kremenek //                     The LLVM Compiler Infrastructure
4d99bd55aSTed Kremenek //
5d99bd55aSTed Kremenek // This file is distributed under the University of Illinois Open Source
6d99bd55aSTed Kremenek // License. See LICENSE.TXT for details.
7d99bd55aSTed Kremenek //
8d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
9d99bd55aSTed Kremenek //
10d99bd55aSTed Kremenek //  This file defines a set of flow-insensitive security checks.
11d99bd55aSTed Kremenek //
12d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
13d99bd55aSTed Kremenek 
14d99bd55aSTed Kremenek #include "clang/Basic/TargetInfo.h"
15d99bd55aSTed Kremenek #include "clang/StaticAnalyzer/BugReporter/BugReporter.h"
16d99bd55aSTed Kremenek #include "clang/StaticAnalyzer/Checkers/LocalCheckers.h"
17d99bd55aSTed Kremenek #include "clang/AST/StmtVisitor.h"
18d99bd55aSTed Kremenek #include "llvm/Support/raw_ostream.h"
19d99bd55aSTed Kremenek 
20d99bd55aSTed Kremenek using namespace clang;
21d99bd55aSTed Kremenek using namespace ento;
22d99bd55aSTed Kremenek 
23d99bd55aSTed Kremenek static bool isArc4RandomAvailable(const ASTContext &Ctx) {
24d99bd55aSTed Kremenek   const llvm::Triple &T = Ctx.Target.getTriple();
25d99bd55aSTed Kremenek   return T.getVendor() == llvm::Triple::Apple ||
26*45e84b00SDouglas Gregor          T.getOS() == llvm::Triple::FreeBSD ||
27*45e84b00SDouglas Gregor          T.getOS() == llvm::Triple::NetBSD ||
28*45e84b00SDouglas Gregor          T.getOS() == llvm::Triple::OpenBSD ||
29*45e84b00SDouglas Gregor          T.getOS() == llvm::Triple::DragonFly;
30d99bd55aSTed Kremenek }
31d99bd55aSTed Kremenek 
32d99bd55aSTed Kremenek namespace {
33d99bd55aSTed Kremenek class WalkAST : public StmtVisitor<WalkAST> {
34d99bd55aSTed Kremenek   BugReporter &BR;
35d99bd55aSTed Kremenek   IdentifierInfo *II_gets;
36d99bd55aSTed Kremenek   IdentifierInfo *II_getpw;
37d99bd55aSTed Kremenek   IdentifierInfo *II_mktemp;
38d99bd55aSTed Kremenek   enum { num_rands = 9 };
39d99bd55aSTed Kremenek   IdentifierInfo *II_rand[num_rands];
40d99bd55aSTed Kremenek   IdentifierInfo *II_random;
41d99bd55aSTed Kremenek   enum { num_setids = 6 };
42d99bd55aSTed Kremenek   IdentifierInfo *II_setid[num_setids];
43d99bd55aSTed Kremenek 
44d99bd55aSTed Kremenek   const bool CheckRand;
45d99bd55aSTed Kremenek 
46d99bd55aSTed Kremenek public:
47d99bd55aSTed Kremenek   WalkAST(BugReporter &br) : BR(br),
48d99bd55aSTed Kremenek                              II_gets(0), II_getpw(0), II_mktemp(0),
49d99bd55aSTed Kremenek                              II_rand(), II_random(0), II_setid(),
50d99bd55aSTed Kremenek                  CheckRand(isArc4RandomAvailable(BR.getContext())) {}
51d99bd55aSTed Kremenek 
52d99bd55aSTed Kremenek   // Statement visitor methods.
53d99bd55aSTed Kremenek   void VisitCallExpr(CallExpr *CE);
54d99bd55aSTed Kremenek   void VisitForStmt(ForStmt *S);
55d99bd55aSTed Kremenek   void VisitCompoundStmt (CompoundStmt *S);
56d99bd55aSTed Kremenek   void VisitStmt(Stmt *S) { VisitChildren(S); }
57d99bd55aSTed Kremenek 
58d99bd55aSTed Kremenek   void VisitChildren(Stmt *S);
59d99bd55aSTed Kremenek 
60d99bd55aSTed Kremenek   // Helpers.
61d99bd55aSTed Kremenek   IdentifierInfo *GetIdentifier(IdentifierInfo *& II, const char *str);
62d99bd55aSTed Kremenek 
63d99bd55aSTed Kremenek   // Checker-specific methods.
64d99bd55aSTed Kremenek   void CheckLoopConditionForFloat(const ForStmt *FS);
65d99bd55aSTed Kremenek   void CheckCall_gets(const CallExpr *CE, const FunctionDecl *FD);
66d99bd55aSTed Kremenek   void CheckCall_getpw(const CallExpr *CE, const FunctionDecl *FD);
67d99bd55aSTed Kremenek   void CheckCall_mktemp(const CallExpr *CE, const FunctionDecl *FD);
68d99bd55aSTed Kremenek   void CheckCall_rand(const CallExpr *CE, const FunctionDecl *FD);
69d99bd55aSTed Kremenek   void CheckCall_random(const CallExpr *CE, const FunctionDecl *FD);
70d99bd55aSTed Kremenek   void CheckUncheckedReturnValue(CallExpr *CE);
71d99bd55aSTed Kremenek };
72d99bd55aSTed Kremenek } // end anonymous namespace
73d99bd55aSTed Kremenek 
74d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
75d99bd55aSTed Kremenek // Helper methods.
76d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
77d99bd55aSTed Kremenek 
78d99bd55aSTed Kremenek IdentifierInfo *WalkAST::GetIdentifier(IdentifierInfo *& II, const char *str) {
79d99bd55aSTed Kremenek   if (!II)
80d99bd55aSTed Kremenek     II = &BR.getContext().Idents.get(str);
81d99bd55aSTed Kremenek 
82d99bd55aSTed Kremenek   return II;
83d99bd55aSTed Kremenek }
84d99bd55aSTed Kremenek 
85d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
86d99bd55aSTed Kremenek // AST walking.
87d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
88d99bd55aSTed Kremenek 
89d99bd55aSTed Kremenek void WalkAST::VisitChildren(Stmt *S) {
90d99bd55aSTed Kremenek   for (Stmt::child_iterator I = S->child_begin(), E = S->child_end(); I!=E; ++I)
91d99bd55aSTed Kremenek     if (Stmt *child = *I)
92d99bd55aSTed Kremenek       Visit(child);
93d99bd55aSTed Kremenek }
94d99bd55aSTed Kremenek 
95d99bd55aSTed Kremenek void WalkAST::VisitCallExpr(CallExpr *CE) {
96d99bd55aSTed Kremenek   if (const FunctionDecl *FD = CE->getDirectCallee()) {
97d99bd55aSTed Kremenek     CheckCall_gets(CE, FD);
98d99bd55aSTed Kremenek     CheckCall_getpw(CE, FD);
99d99bd55aSTed Kremenek     CheckCall_mktemp(CE, FD);
100d99bd55aSTed Kremenek     if (CheckRand) {
101d99bd55aSTed Kremenek       CheckCall_rand(CE, FD);
102d99bd55aSTed Kremenek       CheckCall_random(CE, FD);
103d99bd55aSTed Kremenek     }
104d99bd55aSTed Kremenek   }
105d99bd55aSTed Kremenek 
106d99bd55aSTed Kremenek   // Recurse and check children.
107d99bd55aSTed Kremenek   VisitChildren(CE);
108d99bd55aSTed Kremenek }
109d99bd55aSTed Kremenek 
110d99bd55aSTed Kremenek void WalkAST::VisitCompoundStmt(CompoundStmt *S) {
111d99bd55aSTed Kremenek   for (Stmt::child_iterator I = S->child_begin(), E = S->child_end(); I!=E; ++I)
112d99bd55aSTed Kremenek     if (Stmt *child = *I) {
113d99bd55aSTed Kremenek       if (CallExpr *CE = dyn_cast<CallExpr>(child))
114d99bd55aSTed Kremenek         CheckUncheckedReturnValue(CE);
115d99bd55aSTed Kremenek       Visit(child);
116d99bd55aSTed Kremenek     }
117d99bd55aSTed Kremenek }
118d99bd55aSTed Kremenek 
119d99bd55aSTed Kremenek void WalkAST::VisitForStmt(ForStmt *FS) {
120d99bd55aSTed Kremenek   CheckLoopConditionForFloat(FS);
121d99bd55aSTed Kremenek 
122d99bd55aSTed Kremenek   // Recurse and check children.
123d99bd55aSTed Kremenek   VisitChildren(FS);
124d99bd55aSTed Kremenek }
125d99bd55aSTed Kremenek 
126d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
127d99bd55aSTed Kremenek // Check: floating poing variable used as loop counter.
128d99bd55aSTed Kremenek // Originally: <rdar://problem/6336718>
129d99bd55aSTed Kremenek // Implements: CERT security coding advisory FLP-30.
130d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
131d99bd55aSTed Kremenek 
132d99bd55aSTed Kremenek static const DeclRefExpr*
133d99bd55aSTed Kremenek GetIncrementedVar(const Expr *expr, const VarDecl *x, const VarDecl *y) {
134d99bd55aSTed Kremenek   expr = expr->IgnoreParenCasts();
135d99bd55aSTed Kremenek 
136d99bd55aSTed Kremenek   if (const BinaryOperator *B = dyn_cast<BinaryOperator>(expr)) {
137d99bd55aSTed Kremenek     if (!(B->isAssignmentOp() || B->isCompoundAssignmentOp() ||
138d99bd55aSTed Kremenek           B->getOpcode() == BO_Comma))
139d99bd55aSTed Kremenek       return NULL;
140d99bd55aSTed Kremenek 
141d99bd55aSTed Kremenek     if (const DeclRefExpr *lhs = GetIncrementedVar(B->getLHS(), x, y))
142d99bd55aSTed Kremenek       return lhs;
143d99bd55aSTed Kremenek 
144d99bd55aSTed Kremenek     if (const DeclRefExpr *rhs = GetIncrementedVar(B->getRHS(), x, y))
145d99bd55aSTed Kremenek       return rhs;
146d99bd55aSTed Kremenek 
147d99bd55aSTed Kremenek     return NULL;
148d99bd55aSTed Kremenek   }
149d99bd55aSTed Kremenek 
150d99bd55aSTed Kremenek   if (const DeclRefExpr *DR = dyn_cast<DeclRefExpr>(expr)) {
151d99bd55aSTed Kremenek     const NamedDecl *ND = DR->getDecl();
152d99bd55aSTed Kremenek     return ND == x || ND == y ? DR : NULL;
153d99bd55aSTed Kremenek   }
154d99bd55aSTed Kremenek 
155d99bd55aSTed Kremenek   if (const UnaryOperator *U = dyn_cast<UnaryOperator>(expr))
156d99bd55aSTed Kremenek     return U->isIncrementDecrementOp()
157d99bd55aSTed Kremenek       ? GetIncrementedVar(U->getSubExpr(), x, y) : NULL;
158d99bd55aSTed Kremenek 
159d99bd55aSTed Kremenek   return NULL;
160d99bd55aSTed Kremenek }
161d99bd55aSTed Kremenek 
162d99bd55aSTed Kremenek /// CheckLoopConditionForFloat - This check looks for 'for' statements that
163d99bd55aSTed Kremenek ///  use a floating point variable as a loop counter.
164d99bd55aSTed Kremenek ///  CERT: FLP30-C, FLP30-CPP.
165d99bd55aSTed Kremenek ///
166d99bd55aSTed Kremenek void WalkAST::CheckLoopConditionForFloat(const ForStmt *FS) {
167d99bd55aSTed Kremenek   // Does the loop have a condition?
168d99bd55aSTed Kremenek   const Expr *condition = FS->getCond();
169d99bd55aSTed Kremenek 
170d99bd55aSTed Kremenek   if (!condition)
171d99bd55aSTed Kremenek     return;
172d99bd55aSTed Kremenek 
173d99bd55aSTed Kremenek   // Does the loop have an increment?
174d99bd55aSTed Kremenek   const Expr *increment = FS->getInc();
175d99bd55aSTed Kremenek 
176d99bd55aSTed Kremenek   if (!increment)
177d99bd55aSTed Kremenek     return;
178d99bd55aSTed Kremenek 
179d99bd55aSTed Kremenek   // Strip away '()' and casts.
180d99bd55aSTed Kremenek   condition = condition->IgnoreParenCasts();
181d99bd55aSTed Kremenek   increment = increment->IgnoreParenCasts();
182d99bd55aSTed Kremenek 
183d99bd55aSTed Kremenek   // Is the loop condition a comparison?
184d99bd55aSTed Kremenek   const BinaryOperator *B = dyn_cast<BinaryOperator>(condition);
185d99bd55aSTed Kremenek 
186d99bd55aSTed Kremenek   if (!B)
187d99bd55aSTed Kremenek     return;
188d99bd55aSTed Kremenek 
189d99bd55aSTed Kremenek   // Is this a comparison?
190d99bd55aSTed Kremenek   if (!(B->isRelationalOp() || B->isEqualityOp()))
191d99bd55aSTed Kremenek     return;
192d99bd55aSTed Kremenek 
193d99bd55aSTed Kremenek   // Are we comparing variables?
194d99bd55aSTed Kremenek   const DeclRefExpr *drLHS =
195d99bd55aSTed Kremenek     dyn_cast<DeclRefExpr>(B->getLHS()->IgnoreParenLValueCasts());
196d99bd55aSTed Kremenek   const DeclRefExpr *drRHS =
197d99bd55aSTed Kremenek     dyn_cast<DeclRefExpr>(B->getRHS()->IgnoreParenLValueCasts());
198d99bd55aSTed Kremenek 
199d99bd55aSTed Kremenek   // Does at least one of the variables have a floating point type?
200d99bd55aSTed Kremenek   drLHS = drLHS && drLHS->getType()->isRealFloatingType() ? drLHS : NULL;
201d99bd55aSTed Kremenek   drRHS = drRHS && drRHS->getType()->isRealFloatingType() ? drRHS : NULL;
202d99bd55aSTed Kremenek 
203d99bd55aSTed Kremenek   if (!drLHS && !drRHS)
204d99bd55aSTed Kremenek     return;
205d99bd55aSTed Kremenek 
206d99bd55aSTed Kremenek   const VarDecl *vdLHS = drLHS ? dyn_cast<VarDecl>(drLHS->getDecl()) : NULL;
207d99bd55aSTed Kremenek   const VarDecl *vdRHS = drRHS ? dyn_cast<VarDecl>(drRHS->getDecl()) : NULL;
208d99bd55aSTed Kremenek 
209d99bd55aSTed Kremenek   if (!vdLHS && !vdRHS)
210d99bd55aSTed Kremenek     return;
211d99bd55aSTed Kremenek 
212d99bd55aSTed Kremenek   // Does either variable appear in increment?
213d99bd55aSTed Kremenek   const DeclRefExpr *drInc = GetIncrementedVar(increment, vdLHS, vdRHS);
214d99bd55aSTed Kremenek 
215d99bd55aSTed Kremenek   if (!drInc)
216d99bd55aSTed Kremenek     return;
217d99bd55aSTed Kremenek 
218d99bd55aSTed Kremenek   // Emit the error.  First figure out which DeclRefExpr in the condition
219d99bd55aSTed Kremenek   // referenced the compared variable.
220d99bd55aSTed Kremenek   const DeclRefExpr *drCond = vdLHS == drInc->getDecl() ? drLHS : drRHS;
221d99bd55aSTed Kremenek 
222d99bd55aSTed Kremenek   llvm::SmallVector<SourceRange, 2> ranges;
223d99bd55aSTed Kremenek   llvm::SmallString<256> sbuf;
224d99bd55aSTed Kremenek   llvm::raw_svector_ostream os(sbuf);
225d99bd55aSTed Kremenek 
226d99bd55aSTed Kremenek   os << "Variable '" << drCond->getDecl()->getName()
227d99bd55aSTed Kremenek      << "' with floating point type '" << drCond->getType().getAsString()
228d99bd55aSTed Kremenek      << "' should not be used as a loop counter";
229d99bd55aSTed Kremenek 
230d99bd55aSTed Kremenek   ranges.push_back(drCond->getSourceRange());
231d99bd55aSTed Kremenek   ranges.push_back(drInc->getSourceRange());
232d99bd55aSTed Kremenek 
233d99bd55aSTed Kremenek   const char *bugType = "Floating point variable used as loop counter";
234d99bd55aSTed Kremenek   BR.EmitBasicReport(bugType, "Security", os.str(),
235d99bd55aSTed Kremenek                      FS->getLocStart(), ranges.data(), ranges.size());
236d99bd55aSTed Kremenek }
237d99bd55aSTed Kremenek 
238d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
239d99bd55aSTed Kremenek // Check: Any use of 'gets' is insecure.
240d99bd55aSTed Kremenek // Originally: <rdar://problem/6335715>
241d99bd55aSTed Kremenek // Implements (part of): 300-BSI (buildsecurityin.us-cert.gov)
242d99bd55aSTed Kremenek // CWE-242: Use of Inherently Dangerous Function
243d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
244d99bd55aSTed Kremenek 
245d99bd55aSTed Kremenek void WalkAST::CheckCall_gets(const CallExpr *CE, const FunctionDecl *FD) {
246d99bd55aSTed Kremenek   if (FD->getIdentifier() != GetIdentifier(II_gets, "gets"))
247d99bd55aSTed Kremenek     return;
248d99bd55aSTed Kremenek 
249d99bd55aSTed Kremenek   const FunctionProtoType *FPT
250d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
251d99bd55aSTed Kremenek   if (!FPT)
252d99bd55aSTed Kremenek     return;
253d99bd55aSTed Kremenek 
254d99bd55aSTed Kremenek   // Verify that the function takes a single argument.
255d99bd55aSTed Kremenek   if (FPT->getNumArgs() != 1)
256d99bd55aSTed Kremenek     return;
257d99bd55aSTed Kremenek 
258d99bd55aSTed Kremenek   // Is the argument a 'char*'?
259d99bd55aSTed Kremenek   const PointerType *PT = dyn_cast<PointerType>(FPT->getArgType(0));
260d99bd55aSTed Kremenek   if (!PT)
261d99bd55aSTed Kremenek     return;
262d99bd55aSTed Kremenek 
263d99bd55aSTed Kremenek   if (PT->getPointeeType().getUnqualifiedType() != BR.getContext().CharTy)
264d99bd55aSTed Kremenek     return;
265d99bd55aSTed Kremenek 
266d99bd55aSTed Kremenek   // Issue a warning.
267d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
268d99bd55aSTed Kremenek   BR.EmitBasicReport("Potential buffer overflow in call to 'gets'",
269d99bd55aSTed Kremenek                      "Security",
270d99bd55aSTed Kremenek                      "Call to function 'gets' is extremely insecure as it can "
271d99bd55aSTed Kremenek                      "always result in a buffer overflow",
272d99bd55aSTed Kremenek                      CE->getLocStart(), &R, 1);
273d99bd55aSTed Kremenek }
274d99bd55aSTed Kremenek 
275d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
276d99bd55aSTed Kremenek // Check: Any use of 'getpwd' is insecure.
277d99bd55aSTed Kremenek // CWE-477: Use of Obsolete Functions
278d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
279d99bd55aSTed Kremenek 
280d99bd55aSTed Kremenek void WalkAST::CheckCall_getpw(const CallExpr *CE, const FunctionDecl *FD) {
281d99bd55aSTed Kremenek   if (FD->getIdentifier() != GetIdentifier(II_getpw, "getpw"))
282d99bd55aSTed Kremenek     return;
283d99bd55aSTed Kremenek 
284d99bd55aSTed Kremenek   const FunctionProtoType *FPT
285d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
286d99bd55aSTed Kremenek   if (!FPT)
287d99bd55aSTed Kremenek     return;
288d99bd55aSTed Kremenek 
289d99bd55aSTed Kremenek   // Verify that the function takes two arguments.
290d99bd55aSTed Kremenek   if (FPT->getNumArgs() != 2)
291d99bd55aSTed Kremenek     return;
292d99bd55aSTed Kremenek 
293d99bd55aSTed Kremenek   // Verify the first argument type is integer.
294d99bd55aSTed Kremenek   if (!FPT->getArgType(0)->isIntegerType())
295d99bd55aSTed Kremenek     return;
296d99bd55aSTed Kremenek 
297d99bd55aSTed Kremenek   // Verify the second argument type is char*.
298d99bd55aSTed Kremenek   const PointerType *PT = dyn_cast<PointerType>(FPT->getArgType(1));
299d99bd55aSTed Kremenek   if (!PT)
300d99bd55aSTed Kremenek     return;
301d99bd55aSTed Kremenek 
302d99bd55aSTed Kremenek   if (PT->getPointeeType().getUnqualifiedType() != BR.getContext().CharTy)
303d99bd55aSTed Kremenek     return;
304d99bd55aSTed Kremenek 
305d99bd55aSTed Kremenek   // Issue a warning.
306d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
307d99bd55aSTed Kremenek   BR.EmitBasicReport("Potential buffer overflow in call to 'getpw'",
308d99bd55aSTed Kremenek                      "Security",
309d99bd55aSTed Kremenek                      "The getpw() function is dangerous as it may overflow the "
310d99bd55aSTed Kremenek                      "provided buffer. It is obsoleted by getpwuid().",
311d99bd55aSTed Kremenek                      CE->getLocStart(), &R, 1);
312d99bd55aSTed Kremenek }
313d99bd55aSTed Kremenek 
314d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
315d99bd55aSTed Kremenek // Check: Any use of 'mktemp' is insecure.It is obsoleted by mkstemp().
316d99bd55aSTed Kremenek // CWE-377: Insecure Temporary File
317d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
318d99bd55aSTed Kremenek 
319d99bd55aSTed Kremenek void WalkAST::CheckCall_mktemp(const CallExpr *CE, const FunctionDecl *FD) {
320d99bd55aSTed Kremenek   if (FD->getIdentifier() != GetIdentifier(II_mktemp, "mktemp"))
321d99bd55aSTed Kremenek     return;
322d99bd55aSTed Kremenek 
323d99bd55aSTed Kremenek   const FunctionProtoType *FPT
324d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
325d99bd55aSTed Kremenek   if(!FPT)
326d99bd55aSTed Kremenek     return;
327d99bd55aSTed Kremenek 
328d99bd55aSTed Kremenek   // Verify that the funcion takes a single argument.
329d99bd55aSTed Kremenek   if (FPT->getNumArgs() != 1)
330d99bd55aSTed Kremenek     return;
331d99bd55aSTed Kremenek 
332d99bd55aSTed Kremenek   // Verify that the argument is Pointer Type.
333d99bd55aSTed Kremenek   const PointerType *PT = dyn_cast<PointerType>(FPT->getArgType(0));
334d99bd55aSTed Kremenek   if (!PT)
335d99bd55aSTed Kremenek     return;
336d99bd55aSTed Kremenek 
337d99bd55aSTed Kremenek   // Verify that the argument is a 'char*'.
338d99bd55aSTed Kremenek   if (PT->getPointeeType().getUnqualifiedType() != BR.getContext().CharTy)
339d99bd55aSTed Kremenek     return;
340d99bd55aSTed Kremenek 
341d99bd55aSTed Kremenek   // Issue a waring.
342d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
343d99bd55aSTed Kremenek   BR.EmitBasicReport("Potential insecure temporary file in call 'mktemp'",
344d99bd55aSTed Kremenek                      "Security",
345d99bd55aSTed Kremenek                      "Call to function 'mktemp' is insecure as it always "
346d99bd55aSTed Kremenek                      "creates or uses insecure temporary file.  Use 'mkstemp' instead",
347d99bd55aSTed Kremenek                      CE->getLocStart(), &R, 1);
348d99bd55aSTed Kremenek }
349d99bd55aSTed Kremenek 
350d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
351d99bd55aSTed Kremenek // Check: Linear congruent random number generators should not be used
352d99bd55aSTed Kremenek // Originally: <rdar://problem/63371000>
353d99bd55aSTed Kremenek // CWE-338: Use of cryptographically weak prng
354d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
355d99bd55aSTed Kremenek 
356d99bd55aSTed Kremenek void WalkAST::CheckCall_rand(const CallExpr *CE, const FunctionDecl *FD) {
357d99bd55aSTed Kremenek   if (II_rand[0] == NULL) {
358d99bd55aSTed Kremenek     // This check applies to these functions
359d99bd55aSTed Kremenek     static const char * const identifiers[num_rands] = {
360d99bd55aSTed Kremenek       "drand48", "erand48", "jrand48", "lrand48", "mrand48", "nrand48",
361d99bd55aSTed Kremenek       "lcong48",
362d99bd55aSTed Kremenek       "rand", "rand_r"
363d99bd55aSTed Kremenek     };
364d99bd55aSTed Kremenek 
365d99bd55aSTed Kremenek     for (size_t i = 0; i < num_rands; i++)
366d99bd55aSTed Kremenek       II_rand[i] = &BR.getContext().Idents.get(identifiers[i]);
367d99bd55aSTed Kremenek   }
368d99bd55aSTed Kremenek 
369d99bd55aSTed Kremenek   const IdentifierInfo *id = FD->getIdentifier();
370d99bd55aSTed Kremenek   size_t identifierid;
371d99bd55aSTed Kremenek 
372d99bd55aSTed Kremenek   for (identifierid = 0; identifierid < num_rands; identifierid++)
373d99bd55aSTed Kremenek     if (id == II_rand[identifierid])
374d99bd55aSTed Kremenek       break;
375d99bd55aSTed Kremenek 
376d99bd55aSTed Kremenek   if (identifierid >= num_rands)
377d99bd55aSTed Kremenek     return;
378d99bd55aSTed Kremenek 
379d99bd55aSTed Kremenek   const FunctionProtoType *FTP
380d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
381d99bd55aSTed Kremenek   if (!FTP)
382d99bd55aSTed Kremenek     return;
383d99bd55aSTed Kremenek 
384d99bd55aSTed Kremenek   if (FTP->getNumArgs() == 1) {
385d99bd55aSTed Kremenek     // Is the argument an 'unsigned short *'?
386d99bd55aSTed Kremenek     // (Actually any integer type is allowed.)
387d99bd55aSTed Kremenek     const PointerType *PT = dyn_cast<PointerType>(FTP->getArgType(0));
388d99bd55aSTed Kremenek     if (!PT)
389d99bd55aSTed Kremenek       return;
390d99bd55aSTed Kremenek 
391d99bd55aSTed Kremenek     if (! PT->getPointeeType()->isIntegerType())
392d99bd55aSTed Kremenek       return;
393d99bd55aSTed Kremenek   }
394d99bd55aSTed Kremenek   else if (FTP->getNumArgs() != 0)
395d99bd55aSTed Kremenek     return;
396d99bd55aSTed Kremenek 
397d99bd55aSTed Kremenek   // Issue a warning.
398d99bd55aSTed Kremenek   llvm::SmallString<256> buf1;
399d99bd55aSTed Kremenek   llvm::raw_svector_ostream os1(buf1);
400d99bd55aSTed Kremenek   os1 << '\'' << FD << "' is a poor random number generator";
401d99bd55aSTed Kremenek 
402d99bd55aSTed Kremenek   llvm::SmallString<256> buf2;
403d99bd55aSTed Kremenek   llvm::raw_svector_ostream os2(buf2);
404d99bd55aSTed Kremenek   os2 << "Function '" << FD
405d99bd55aSTed Kremenek       << "' is obsolete because it implements a poor random number generator."
406d99bd55aSTed Kremenek       << "  Use 'arc4random' instead";
407d99bd55aSTed Kremenek 
408d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
409d99bd55aSTed Kremenek   BR.EmitBasicReport(os1.str(), "Security", os2.str(),CE->getLocStart(), &R, 1);
410d99bd55aSTed Kremenek }
411d99bd55aSTed Kremenek 
412d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
413d99bd55aSTed Kremenek // Check: 'random' should not be used
414d99bd55aSTed Kremenek // Originally: <rdar://problem/63371000>
415d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
416d99bd55aSTed Kremenek 
417d99bd55aSTed Kremenek void WalkAST::CheckCall_random(const CallExpr *CE, const FunctionDecl *FD) {
418d99bd55aSTed Kremenek   if (FD->getIdentifier() != GetIdentifier(II_random, "random"))
419d99bd55aSTed Kremenek     return;
420d99bd55aSTed Kremenek 
421d99bd55aSTed Kremenek   const FunctionProtoType *FTP
422d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
423d99bd55aSTed Kremenek   if (!FTP)
424d99bd55aSTed Kremenek     return;
425d99bd55aSTed Kremenek 
426d99bd55aSTed Kremenek   // Verify that the function takes no argument.
427d99bd55aSTed Kremenek   if (FTP->getNumArgs() != 0)
428d99bd55aSTed Kremenek     return;
429d99bd55aSTed Kremenek 
430d99bd55aSTed Kremenek   // Issue a warning.
431d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
432d99bd55aSTed Kremenek   BR.EmitBasicReport("'random' is not a secure random number generator",
433d99bd55aSTed Kremenek                      "Security",
434d99bd55aSTed Kremenek                      "The 'random' function produces a sequence of values that "
435d99bd55aSTed Kremenek                      "an adversary may be able to predict.  Use 'arc4random' "
436d99bd55aSTed Kremenek                      "instead", CE->getLocStart(), &R, 1);
437d99bd55aSTed Kremenek }
438d99bd55aSTed Kremenek 
439d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
440d99bd55aSTed Kremenek // Check: Should check whether privileges are dropped successfully.
441d99bd55aSTed Kremenek // Originally: <rdar://problem/6337132>
442d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
443d99bd55aSTed Kremenek 
444d99bd55aSTed Kremenek void WalkAST::CheckUncheckedReturnValue(CallExpr *CE) {
445d99bd55aSTed Kremenek   const FunctionDecl *FD = CE->getDirectCallee();
446d99bd55aSTed Kremenek   if (!FD)
447d99bd55aSTed Kremenek     return;
448d99bd55aSTed Kremenek 
449d99bd55aSTed Kremenek   if (II_setid[0] == NULL) {
450d99bd55aSTed Kremenek     static const char * const identifiers[num_setids] = {
451d99bd55aSTed Kremenek       "setuid", "setgid", "seteuid", "setegid",
452d99bd55aSTed Kremenek       "setreuid", "setregid"
453d99bd55aSTed Kremenek     };
454d99bd55aSTed Kremenek 
455d99bd55aSTed Kremenek     for (size_t i = 0; i < num_setids; i++)
456d99bd55aSTed Kremenek       II_setid[i] = &BR.getContext().Idents.get(identifiers[i]);
457d99bd55aSTed Kremenek   }
458d99bd55aSTed Kremenek 
459d99bd55aSTed Kremenek   const IdentifierInfo *id = FD->getIdentifier();
460d99bd55aSTed Kremenek   size_t identifierid;
461d99bd55aSTed Kremenek 
462d99bd55aSTed Kremenek   for (identifierid = 0; identifierid < num_setids; identifierid++)
463d99bd55aSTed Kremenek     if (id == II_setid[identifierid])
464d99bd55aSTed Kremenek       break;
465d99bd55aSTed Kremenek 
466d99bd55aSTed Kremenek   if (identifierid >= num_setids)
467d99bd55aSTed Kremenek     return;
468d99bd55aSTed Kremenek 
469d99bd55aSTed Kremenek   const FunctionProtoType *FTP
470d99bd55aSTed Kremenek     = dyn_cast<FunctionProtoType>(FD->getType().IgnoreParens());
471d99bd55aSTed Kremenek   if (!FTP)
472d99bd55aSTed Kremenek     return;
473d99bd55aSTed Kremenek 
474d99bd55aSTed Kremenek   // Verify that the function takes one or two arguments (depending on
475d99bd55aSTed Kremenek   //   the function).
476d99bd55aSTed Kremenek   if (FTP->getNumArgs() != (identifierid < 4 ? 1 : 2))
477d99bd55aSTed Kremenek     return;
478d99bd55aSTed Kremenek 
479d99bd55aSTed Kremenek   // The arguments must be integers.
480d99bd55aSTed Kremenek   for (unsigned i = 0; i < FTP->getNumArgs(); i++)
481d99bd55aSTed Kremenek     if (! FTP->getArgType(i)->isIntegerType())
482d99bd55aSTed Kremenek       return;
483d99bd55aSTed Kremenek 
484d99bd55aSTed Kremenek   // Issue a warning.
485d99bd55aSTed Kremenek   llvm::SmallString<256> buf1;
486d99bd55aSTed Kremenek   llvm::raw_svector_ostream os1(buf1);
487d99bd55aSTed Kremenek   os1 << "Return value is not checked in call to '" << FD << '\'';
488d99bd55aSTed Kremenek 
489d99bd55aSTed Kremenek   llvm::SmallString<256> buf2;
490d99bd55aSTed Kremenek   llvm::raw_svector_ostream os2(buf2);
491d99bd55aSTed Kremenek   os2 << "The return value from the call to '" << FD
492d99bd55aSTed Kremenek       << "' is not checked.  If an error occurs in '" << FD
493d99bd55aSTed Kremenek       << "', the following code may execute with unexpected privileges";
494d99bd55aSTed Kremenek 
495d99bd55aSTed Kremenek   SourceRange R = CE->getCallee()->getSourceRange();
496d99bd55aSTed Kremenek   BR.EmitBasicReport(os1.str(), "Security", os2.str(),CE->getLocStart(), &R, 1);
497d99bd55aSTed Kremenek }
498d99bd55aSTed Kremenek 
499d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
500d99bd55aSTed Kremenek // Entry point for check.
501d99bd55aSTed Kremenek //===----------------------------------------------------------------------===//
502d99bd55aSTed Kremenek 
503d99bd55aSTed Kremenek void ento::CheckSecuritySyntaxOnly(const Decl *D, BugReporter &BR) {
504d99bd55aSTed Kremenek   WalkAST walker(BR);
505d99bd55aSTed Kremenek   walker.Visit(D->getBody());
506d99bd55aSTed Kremenek }
507