Static Value-Flow Analysis
Loading...
Searching...
No Matches
ExtAPI.cpp
Go to the documentation of this file.
1//===- ExtAPI.cpp -- External functions -----------------------------------------//
2//
3// SVF: Static Value-Flow Analysis
4//
5// Copyright (C) <2013-2017> <Yulei Sui>
6//
7
8// This program is free software: you can redistribute it and/or modify
9// it under the terms of the GNU Affero General Public License as published by
10// the Free Software Foundation, either version 3 of the License, or
11// (at your option) any later version.
12
13// This program is distributed in the hope that it will be useful,
14// but WITHOUT ANY WARRANTY; without even the implied warranty of
15// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16// GNU Affero General Public License for more details.
17
18// You should have received a copy of the GNU Affero General Public License
19// along with this program. If not, see <http://www.gnu.org/licenses/>.
20//
21//===----------------------------------------------------------------------===//
22
23/*
24 * ExtAPI.cpp
25 *
26 * Created on: July 1, 2022
27 * Author: Shuangxiang Kan
28 */
29
30#include "Util/ExtAPI.h"
31#include "Util/SVFUtil.h"
32#include "Util/Options.h"
33#include "Util/config.h"
34#include <ostream>
35#include <sys/stat.h>
36#include "SVFIR/SVFVariables.h"
37
38using namespace SVF;
39
40ExtAPI* ExtAPI::extOp = nullptr;
41std::string ExtAPI::extBcPath = "";
42
44{
45 if (extOp == nullptr)
46 {
47 extOp = new ExtAPI;
48 }
49 return extOp;
50}
51
53{
54 if (extOp != nullptr)
55 {
56 delete extOp;
57 extOp = nullptr;
58 }
59}
60
61// Set extapi.bc file path
62bool ExtAPI::setExtBcPath(const std::string& path)
63{
64 struct stat statbuf;
65 if (!path.empty() && !stat(path.c_str(), &statbuf))
66 {
68 return true;
69 }
70 return false;
71}
72
73// Get environment variables $SVF_DIR and "npm root" through popen() method
74static std::string GetStdoutFromCommand(const std::string& command)
75{
76 char buffer[128];
77 std::string result = "";
78 // Open pipe to file
79 FILE* pipe = popen(command.c_str(), "r");
80 if (!pipe)
81 {
82 return "popen failed!";
83 }
84 // read till end of process:
85 while (!feof(pipe))
86 {
87 // use buffer to read and add to result
88 if (fgets(buffer, 128, pipe) != NULL)
89 result += buffer;
90 }
91 pclose(pipe);
92 // remove "\n"
93 result.erase(remove(result.begin(), result.end(), '\n'), result.end());
94 return result;
95}
96
97// Get extapi.bc file path in npm
98static std::string getFilePath(const std::string& path)
99{
100 std::string bcFilePath = "";
101 if (path.compare("SVF_DIR") == 0)
102 {
103 char const* svfdir = getenv("SVF_DIR");
104 if (svfdir)
106 }
107 else if (path.compare("npm root") == 0)
108 {
110 bcFilePath.append("/SVF");
111 }
112
113 if (!bcFilePath.empty() && bcFilePath.back() != '/')
114 bcFilePath.push_back('/');
115 bcFilePath.append(SVF_BUILD_TYPE "-build").append("/lib/extapi.bc");
116 return bcFilePath;
117}
118
119// Get extapi.bc path
121{
122 // Default ways of retrieving extapi.bc (in order of precedence):
123 // 1. Set `path/to/extapi.bc` through `setExtBcPath()`
124 // 2. Set `path/to/extapi.bc` through the command line argument `-extapi=path/to/extapi.bc`
125 // 3. Get location generated by CMakeLists.txt from `config.h` header file (if SVF was installed)
126 // 4. Get location in build tree based from `config.h` header file (if SVF was only built)
127 // 5. Get location based on environment variable $ENV{SVF_DIR}
128 // 6. Search for `extapi.bc` from root directory for npm installation (iff SVF installed through npm)
129
130 // 1. Set `path/to/extapi.bc` through `setExtBcPath()`
131 if (!extBcPath.empty())
132 return extBcPath;
133
134 // 2. Set `path/to/extapi.bc` through the command line argument `-extapi=path/to/extapi.bc`
136 return extBcPath;
137
138 // 3. Get location generated by CMakeLists.txt from `config.h` header file (if SVF was installed)
139 if (setExtBcPath(SVF_EXTAPI_BC)) // Full path is available (for custom file names)
140 return extBcPath;
141 if (setExtBcPath(SVF_EXTAPI_DIR "/extapi.bc")) // Based on directory & default filename
142 return extBcPath;
143
144 // 4. Get location in build tree based from `config.h` header file (if SVF was only built)
145 if (setExtBcPath(SVF_BUILD_DIR "/lib/extapi.bc"))
146 return extBcPath;
147
148 // 5. Get location based on environment variable $ENV{SVF_DIR}
149 if (setExtBcPath(getFilePath("SVF_DIR")))
150 return extBcPath;
151
152 // 6. Search for `extapi.bc` from root directory for npm installation (iff SVF installed through npm)
153 if (setExtBcPath(getFilePath("npm root")))
154 return extBcPath;
155
156 SVFUtil::errs() << "ERROR: Failed to find \"extapi.bc\" LLVM bitcode file in " << extBcPath << std::endl
157 << "To override the default locations for \"extapi.bc\", you can:" << std::endl
158 << "\t1. Use the command line argument \"-extapi=path/to/extapi.bc\"" << std::endl
159 << "\t2. Use the \"setExtBcPath()\" function *BEFORE* calling \"buildSVFModule()\"" << std::endl
160 << "\t3. Override the paths in \"svf/Util/config.h\" (WARNING: will be overwritten when "
161 << "rebuilding SVF (generated by CMakeLists.txt))" << std::endl;
162 abort();
163}
164
165
166void ExtAPI::setExtFuncAnnotations(const FunObjVar* fun, const std::vector<std::string>& funcAnnotations)
167{
168 assert(fun && "Null FunObjVar* pointer");
170}
171
172bool ExtAPI::hasExtFuncAnnotation(const FunObjVar *fun, const std::string &funcAnnotation)
173{
174 assert(fun && "Null FunObjVar* pointer");
175 auto it = funObjVar2Annotations.find(fun);
176 if (it != funObjVar2Annotations.end())
177 {
178 for (const std::string& annotation : it->second)
179 if (annotation.find(funcAnnotation) != std::string::npos)
180 return true;
181 }
182 return false;
183}
184
185
186std::string ExtAPI::getExtFuncAnnotation(const FunObjVar* fun, const std::string& funcAnnotation)
187{
188 assert(fun && "Null FunObjVar* pointer");
189 auto it = funObjVar2Annotations.find(fun);
190 if (it != funObjVar2Annotations.end())
191 {
192 for (const std::string& annotation : it->second)
193 if (annotation.find(funcAnnotation) != std::string::npos)
194 return annotation;
195 }
196 return "";
197}
198
199const std::vector<std::string>& ExtAPI::getExtFuncAnnotations(const FunObjVar* fun)
200{
201 assert(fun && "Null FunObjVar* pointer");
202 auto it = funObjVar2Annotations.find(fun);
203 if (it != funObjVar2Annotations.end())
204 return it->second;
205 return funObjVar2Annotations[fun];
206}
207
209{
210 return F &&
211 (hasExtFuncAnnotation(F, "MEMCPY") || hasExtFuncAnnotation(F, "STRCPY")
212 || hasExtFuncAnnotation(F, "STRCAT"));
213}
214
216{
217 return F && hasExtFuncAnnotation(F, "MEMSET");
218}
219
221{
222 return F && hasExtFuncAnnotation(F, "ALLOC_HEAP_RET");
223}
224
225// Does (F) allocate a new object and assign it to one of its arguments?
227{
228 return F && hasExtFuncAnnotation(F, "ALLOC_HEAP_ARG");
229}
230
232{
233 return F && hasExtFuncAnnotation(F, "ALLOC_STACK_RET");
234}
235
236// Get the position of argument which holds the new object
238{
239 std::string allocArg = getExtFuncAnnotation(F, "ALLOC_HEAP_ARG");
240 assert(!allocArg.empty() && "Not an alloc call via argument or incorrect extern function annotation!");
241
242 std::string number;
243 for (char c : allocArg)
244 {
245 if (isdigit(c))
246 number.push_back(c);
247 }
248 assert(!number.empty() && "Incorrect naming convention for svf external functions(ALLOC_HEAP_ARG + number)?");
249 return std::stoi(number);
250}
251
252// Does (F) reallocate a new object?
254{
255 return F && hasExtFuncAnnotation(F, "REALLOC_HEAP_RET");
256}
258{
259 assert(F && "Null FunObjVar* pointer");
260 if (F->isDeclaration() || F->isIntrinsic())
261 return true;
262 else if (hasExtFuncAnnotation(F, "OVERWRITE") && getExtFuncAnnotations(F).size() == 1)
263 return false;
264 else
265 return !getExtFuncAnnotations(F).empty();
266}
static std::string GetStdoutFromCommand(const std::string &command)
Definition ExtAPI.cpp:74
static std::string getFilePath(const std::string &path)
Definition ExtAPI.cpp:98
char * buffer
Definition cJSON.h:163
const char *const const double number
Definition cJSON.h:268
bool is_alloc_stack_ret(const FunObjVar *F)
Definition ExtAPI.cpp:231
bool is_arg_alloc(const FunObjVar *F)
Definition ExtAPI.cpp:226
bool is_memcpy(const FunObjVar *F)
Definition ExtAPI.cpp:208
std::string getExtBcPath()
Definition ExtAPI.cpp:120
static std::string extBcPath
Definition ExtAPI.h:59
static ExtAPI * extOp
Definition ExtAPI.h:53
s32_t get_alloc_arg_pos(const FunObjVar *F)
Definition ExtAPI.cpp:237
static ExtAPI * getExtAPI()
Definition ExtAPI.cpp:43
bool is_ext(const FunObjVar *funObjVar)
Definition ExtAPI.cpp:257
std::string getExtFuncAnnotation(const FunObjVar *fun, const std::string &funcAnnotation)
Definition ExtAPI.cpp:186
bool is_realloc(const FunObjVar *F)
Definition ExtAPI.cpp:253
Map< const FunObjVar *, std::vector< std::string > > funObjVar2Annotations
Definition ExtAPI.h:56
void setExtFuncAnnotations(const FunObjVar *fun, const std::vector< std::string > &funcAnnotations)
Definition ExtAPI.cpp:166
ExtAPI()=default
static void destory()
Definition ExtAPI.cpp:52
static bool setExtBcPath(const std::string &path)
Definition ExtAPI.cpp:62
bool hasExtFuncAnnotation(const FunObjVar *fun, const std::string &funcAnnotation)
Definition ExtAPI.cpp:172
bool is_alloc(const FunObjVar *F)
Definition ExtAPI.cpp:220
const std::vector< std::string > & getExtFuncAnnotations(const FunObjVar *fun)
Definition ExtAPI.cpp:199
bool is_memset(const FunObjVar *F)
Definition ExtAPI.cpp:215
bool isIntrinsic() const
bool isDeclaration() const
static const Option< std::string > ExtAPIPath
Definition Options.h:220
#define NULL
Definition extapi.c:2
char * fgets(char *str, int n, void *stream)
Definition extapi.c:746
int isdigit(int c)
Definition extapi.c:851
std::ostream & errs()
Overwrite llvm::errs()
Definition SVFUtil.h:58
for isBitcode
Definition BasicTypes.h:68
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
signed s32_t
Definition GeneralType.h:48