forked from microsoft/llvm-mctoll
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathExternalFunctions.cpp
93 lines (87 loc) · 3.54 KB
/
ExternalFunctions.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
//===-- ExternalFunctions.h - Binary raiser utility llvm-mctoll -------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file contains the implementation of the ExternalFunction class
// and the table of known external functions.
//
//===----------------------------------------------------------------------===//
#include "ExternalFunctions.h"
const std::map<StringRef, ExternalFunctions::RetAndArgs>
ExternalFunctions::GlibcFunctions = {
{"printf", {"i32", {"i8*"}, true}},
{"malloc", {"i8*", {"i64"}, false}},
{"memcpy", {"i8*", {"i8*", "i8*", "i64"}, false}},
{"strcpy", {"i8*", {"i8*", "i8*"}, false}},
{"__isoc99_scanf", {"i32", {"i8*"}, true}},
{"time", {"i64", {"i64*"}, false}},
{"puts", {"i32", {"i8*"}, false}},
{"atoi", {"i32", {"i8*"}, false}}};
// Given the primitive type's string representation, return the Type*
// corresponding to it.
Type *ExternalFunctions::getPrimitiveType(const StringRef &TypeStr,
LLVMContext &llvmCtx) {
Type *retType = nullptr;
if (TypeStr.equals("void")) {
retType = Type::getVoidTy(llvmCtx);
} else if (TypeStr.equals("i8")) {
retType = Type::getInt8Ty(llvmCtx);
} else if (TypeStr.equals("i16")) {
retType = Type::getInt16Ty(llvmCtx);
} else if (TypeStr.equals("i32")) {
retType = Type::getInt32Ty(llvmCtx);
} else if (TypeStr.equals("i64")) {
retType = Type::getInt64Ty(llvmCtx);
} else if (TypeStr.equals("i8*")) {
retType = Type::getInt8PtrTy(llvmCtx);
} else if (TypeStr.equals("i16*")) {
retType = Type::getInt16PtrTy(llvmCtx);
} else if (TypeStr.equals("i32*")) {
retType = Type::getInt32PtrTy(llvmCtx);
} else if (TypeStr.equals("i64*")) {
retType = Type::getInt64PtrTy(llvmCtx);
}
assert((retType != nullptr) &&
"Unsupported primitive type specified in known function prototype");
return retType;
}
// Construct and return a Function* corresponding to a known glibc function.
Function *ExternalFunctions::Create(StringRef &CFuncName, Module &module) {
Function *Func = nullptr;
llvm::LLVMContext &llvmContext(module.getContext());
FunctionType *FuncType = nullptr;
auto iter = ExternalFunctions::GlibcFunctions.find(CFuncName);
if (iter == ExternalFunctions::GlibcFunctions.end()) {
errs() << CFuncName.data() << "\n";
assert(false && "Unspported undefined function");
}
Func = module.getFunction(CFuncName);
if (Func == nullptr) {
const ExternalFunctions::RetAndArgs &retAndArgs = iter->second;
Type *RetType =
ExternalFunctions::getPrimitiveType(retAndArgs.ReturnType, llvmContext);
std::vector<Type *> ArgVec;
for (StringRef arg : retAndArgs.Arguments) {
Type *argType = ExternalFunctions::getPrimitiveType(arg, llvmContext);
ArgVec.push_back(argType);
}
ArrayRef<Type *> Args(ArgVec);
FuncType = FunctionType::get(RetType, Args, retAndArgs.isVariadic);
if (FuncType == nullptr) {
errs() << CFuncName.data() << "\n";
assert(false &&
"Failed to construct function type for external function");
}
Constant *FC = module.getOrInsertFunction(CFuncName, FuncType);
assert(isa<Function>(FC) && "Expect Function");
Func = reinterpret_cast<Function *>(FC);
Func->setCallingConv(CallingConv::C);
Func->setDSOLocal(true);
}
return Func;
}