1 //===---------- IssueHash.cpp - Generate identification hashes --*- C++ -*-===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 #include "clang/StaticAnalyzer/Core/IssueHash.h"
10 #include "clang/AST/ASTContext.h"
11 #include "clang/AST/Decl.h"
12 #include "clang/AST/DeclCXX.h"
13 #include "clang/Basic/SourceManager.h"
14 #include "clang/Basic/Specifiers.h"
15 #include "clang/Lex/Lexer.h"
16 #include "llvm/ADT/SmallVector.h"
17 #include "llvm/ADT/StringExtras.h"
18 #include "llvm/ADT/StringRef.h"
19 #include "llvm/ADT/Twine.h"
20 #include "llvm/Support/LineIterator.h"
21 #include "llvm/Support/MD5.h"
22 #include "llvm/Support/Path.h"
23
24 #include <functional>
25 #include <sstream>
26 #include <string>
27
28 using namespace clang;
29
30 // Get a string representation of the parts of the signature that can be
31 // overloaded on.
GetSignature(const FunctionDecl * Target)32 static std::string GetSignature(const FunctionDecl *Target) {
33 if (!Target)
34 return "";
35 std::string Signature;
36
37 if (!isa<CXXConstructorDecl>(Target) && !isa<CXXDestructorDecl>(Target) &&
38 !isa<CXXConversionDecl>(Target))
39 Signature.append(Target->getReturnType().getAsString()).append(" ");
40 Signature.append(Target->getQualifiedNameAsString()).append("(");
41
42 for (int i = 0, paramsCount = Target->getNumParams(); i < paramsCount; ++i) {
43 if (i)
44 Signature.append(", ");
45 Signature.append(Target->getParamDecl(i)->getType().getAsString());
46 }
47
48 if (Target->isVariadic())
49 Signature.append(", ...");
50 Signature.append(")");
51
52 const auto *TargetT =
53 llvm::dyn_cast_or_null<FunctionType>(Target->getType().getTypePtr());
54
55 if (!TargetT || !isa<CXXMethodDecl>(Target))
56 return Signature;
57
58 if (TargetT->isConst())
59 Signature.append(" const");
60 if (TargetT->isVolatile())
61 Signature.append(" volatile");
62 if (TargetT->isRestrict())
63 Signature.append(" restrict");
64
65 if (const auto *TargetPT =
66 dyn_cast_or_null<FunctionProtoType>(Target->getType().getTypePtr())) {
67 switch (TargetPT->getRefQualifier()) {
68 case RQ_LValue:
69 Signature.append(" &");
70 break;
71 case RQ_RValue:
72 Signature.append(" &&");
73 break;
74 default:
75 break;
76 }
77 }
78
79 return Signature;
80 }
81
GetEnclosingDeclContextSignature(const Decl * D)82 static std::string GetEnclosingDeclContextSignature(const Decl *D) {
83 if (!D)
84 return "";
85
86 if (const auto *ND = dyn_cast<NamedDecl>(D)) {
87 std::string DeclName;
88
89 switch (ND->getKind()) {
90 case Decl::Namespace:
91 case Decl::Record:
92 case Decl::CXXRecord:
93 case Decl::Enum:
94 DeclName = ND->getQualifiedNameAsString();
95 break;
96 case Decl::CXXConstructor:
97 case Decl::CXXDestructor:
98 case Decl::CXXConversion:
99 case Decl::CXXMethod:
100 case Decl::Function:
101 DeclName = GetSignature(dyn_cast_or_null<FunctionDecl>(ND));
102 break;
103 case Decl::ObjCMethod:
104 // ObjC Methods can not be overloaded, qualified name uniquely identifies
105 // the method.
106 DeclName = ND->getQualifiedNameAsString();
107 break;
108 default:
109 break;
110 }
111
112 return DeclName;
113 }
114
115 return "";
116 }
117
GetNthLineOfFile(llvm::MemoryBuffer * Buffer,int Line)118 static StringRef GetNthLineOfFile(llvm::MemoryBuffer *Buffer, int Line) {
119 if (!Buffer)
120 return "";
121
122 llvm::line_iterator LI(*Buffer, false);
123 for (; !LI.is_at_eof() && LI.line_number() != Line; ++LI)
124 ;
125
126 return *LI;
127 }
128
NormalizeLine(const SourceManager & SM,FullSourceLoc & L,const LangOptions & LangOpts)129 static std::string NormalizeLine(const SourceManager &SM, FullSourceLoc &L,
130 const LangOptions &LangOpts) {
131 static StringRef Whitespaces = " \t\n";
132
133 StringRef Str = GetNthLineOfFile(SM.getBuffer(L.getFileID(), L),
134 L.getExpansionLineNumber());
135 unsigned col = Str.find_first_not_of(Whitespaces);
136 col++;
137 SourceLocation StartOfLine =
138 SM.translateLineCol(SM.getFileID(L), L.getExpansionLineNumber(), col);
139 llvm::MemoryBuffer *Buffer =
140 SM.getBuffer(SM.getFileID(StartOfLine), StartOfLine);
141 if (!Buffer)
142 return {};
143
144 const char *BufferPos = SM.getCharacterData(StartOfLine);
145
146 Token Token;
147 Lexer Lexer(SM.getLocForStartOfFile(SM.getFileID(StartOfLine)), LangOpts,
148 Buffer->getBufferStart(), BufferPos, Buffer->getBufferEnd());
149
150 size_t NextStart = 0;
151 std::ostringstream LineBuff;
152 while (!Lexer.LexFromRawLexer(Token) && NextStart < 2) {
153 if (Token.isAtStartOfLine() && NextStart++ > 0)
154 continue;
155 LineBuff << std::string(SM.getCharacterData(Token.getLocation()),
156 Token.getLength());
157 }
158
159 return LineBuff.str();
160 }
161
GetHashOfContent(StringRef Content)162 static llvm::SmallString<32> GetHashOfContent(StringRef Content) {
163 llvm::MD5 Hash;
164 llvm::MD5::MD5Result MD5Res;
165 SmallString<32> Res;
166
167 Hash.update(Content);
168 Hash.final(MD5Res);
169 llvm::MD5::stringifyResult(MD5Res, Res);
170
171 return Res;
172 }
173
GetIssueString(const SourceManager & SM,FullSourceLoc & IssueLoc,StringRef CheckerName,StringRef BugType,const Decl * D,const LangOptions & LangOpts)174 std::string clang::GetIssueString(const SourceManager &SM,
175 FullSourceLoc &IssueLoc,
176 StringRef CheckerName, StringRef BugType,
177 const Decl *D,
178 const LangOptions &LangOpts) {
179 static StringRef Delimiter = "$";
180
181 return (llvm::Twine(CheckerName) + Delimiter +
182 GetEnclosingDeclContextSignature(D) + Delimiter +
183 llvm::utostr(IssueLoc.getExpansionColumnNumber()) + Delimiter +
184 NormalizeLine(SM, IssueLoc, LangOpts) + Delimiter + BugType)
185 .str();
186 }
187
GetIssueHash(const SourceManager & SM,FullSourceLoc & IssueLoc,StringRef CheckerName,StringRef BugType,const Decl * D,const LangOptions & LangOpts)188 SmallString<32> clang::GetIssueHash(const SourceManager &SM,
189 FullSourceLoc &IssueLoc,
190 StringRef CheckerName, StringRef BugType,
191 const Decl *D,
192 const LangOptions &LangOpts) {
193
194 return GetHashOfContent(
195 GetIssueString(SM, IssueLoc, CheckerName, BugType, D, LangOpts));
196 }
197