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