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