1f22ef01cSRoman Divacky //===-- ExternalFunctions.cpp - Implement External Functions --------------===//
2f22ef01cSRoman Divacky //
3f22ef01cSRoman Divacky // The LLVM Compiler Infrastructure
4f22ef01cSRoman Divacky //
5f22ef01cSRoman Divacky // This file is distributed under the University of Illinois Open Source
6f22ef01cSRoman Divacky // License. See LICENSE.TXT for details.
7f22ef01cSRoman Divacky //
8f22ef01cSRoman Divacky //===----------------------------------------------------------------------===//
9f22ef01cSRoman Divacky //
10f22ef01cSRoman Divacky // This file contains both code to deal with invoking "external" functions, but
11f22ef01cSRoman Divacky // also contains code that implements "exported" external functions.
12f22ef01cSRoman Divacky //
13f22ef01cSRoman Divacky // There are currently two mechanisms for handling external functions in the
14f22ef01cSRoman Divacky // Interpreter. The first is to implement lle_* wrapper functions that are
15f22ef01cSRoman Divacky // specific to well-known library functions which manually translate the
16f22ef01cSRoman Divacky // arguments from GenericValues and make the call. If such a wrapper does
17f22ef01cSRoman Divacky // not exist, and libffi is available, then the Interpreter will attempt to
18f22ef01cSRoman Divacky // invoke the function using libffi, after finding its address.
19f22ef01cSRoman Divacky //
20f22ef01cSRoman Divacky //===----------------------------------------------------------------------===//
21f22ef01cSRoman Divacky
22f22ef01cSRoman Divacky #include "Interpreter.h"
23d88c1a5aSDimitry Andric #include "llvm/ADT/APInt.h"
24d88c1a5aSDimitry Andric #include "llvm/ADT/ArrayRef.h"
25f22ef01cSRoman Divacky #include "llvm/Config/config.h" // Detect libffi
26d88c1a5aSDimitry Andric #include "llvm/ExecutionEngine/GenericValue.h"
27139f7f9bSDimitry Andric #include "llvm/IR/DataLayout.h"
28139f7f9bSDimitry Andric #include "llvm/IR/DerivedTypes.h"
29d88c1a5aSDimitry Andric #include "llvm/IR/Function.h"
30d88c1a5aSDimitry Andric #include "llvm/IR/Type.h"
31d88c1a5aSDimitry Andric #include "llvm/Support/Casting.h"
322754fe60SDimitry Andric #include "llvm/Support/DynamicLibrary.h"
33139f7f9bSDimitry Andric #include "llvm/Support/ErrorHandling.h"
34f22ef01cSRoman Divacky #include "llvm/Support/ManagedStatic.h"
352754fe60SDimitry Andric #include "llvm/Support/Mutex.h"
3639d628a0SDimitry Andric #include "llvm/Support/UniqueLock.h"
37db17bf38SDimitry Andric #include "llvm/Support/raw_ostream.h"
38d88c1a5aSDimitry Andric #include <cassert>
39139f7f9bSDimitry Andric #include <cmath>
40f22ef01cSRoman Divacky #include <csignal>
41d88c1a5aSDimitry Andric #include <cstdint>
42f22ef01cSRoman Divacky #include <cstdio>
43f22ef01cSRoman Divacky #include <cstring>
44139f7f9bSDimitry Andric #include <map>
45d88c1a5aSDimitry Andric #include <string>
46d88c1a5aSDimitry Andric #include <utility>
47d88c1a5aSDimitry Andric #include <vector>
48f22ef01cSRoman Divacky
49f22ef01cSRoman Divacky #ifdef HAVE_FFI_CALL
50f22ef01cSRoman Divacky #ifdef HAVE_FFI_H
51f22ef01cSRoman Divacky #include <ffi.h>
52f22ef01cSRoman Divacky #define USE_LIBFFI
53f22ef01cSRoman Divacky #elif HAVE_FFI_FFI_H
54f22ef01cSRoman Divacky #include <ffi/ffi.h>
55f22ef01cSRoman Divacky #define USE_LIBFFI
56f22ef01cSRoman Divacky #endif
57f22ef01cSRoman Divacky #endif
58f22ef01cSRoman Divacky
59f22ef01cSRoman Divacky using namespace llvm;
60f22ef01cSRoman Divacky
61f22ef01cSRoman Divacky static ManagedStatic<sys::Mutex> FunctionsLock;
62f22ef01cSRoman Divacky
638f0fd8f6SDimitry Andric typedef GenericValue (*ExFunc)(FunctionType *, ArrayRef<GenericValue>);
64f22ef01cSRoman Divacky static ManagedStatic<std::map<const Function *, ExFunc> > ExportedFunctions;
6539d628a0SDimitry Andric static ManagedStatic<std::map<std::string, ExFunc> > FuncNames;
66f22ef01cSRoman Divacky
67f22ef01cSRoman Divacky #ifdef USE_LIBFFI
68f22ef01cSRoman Divacky typedef void (*RawFunc)();
69f22ef01cSRoman Divacky static ManagedStatic<std::map<const Function *, RawFunc> > RawFunctions;
70f22ef01cSRoman Divacky #endif
71f22ef01cSRoman Divacky
72f22ef01cSRoman Divacky static Interpreter *TheInterpreter;
73f22ef01cSRoman Divacky
getTypeID(Type * Ty)746122f3e6SDimitry Andric static char getTypeID(Type *Ty) {
75f22ef01cSRoman Divacky switch (Ty->getTypeID()) {
76f22ef01cSRoman Divacky case Type::VoidTyID: return 'V';
77f22ef01cSRoman Divacky case Type::IntegerTyID:
78f22ef01cSRoman Divacky switch (cast<IntegerType>(Ty)->getBitWidth()) {
79f22ef01cSRoman Divacky case 1: return 'o';
80f22ef01cSRoman Divacky case 8: return 'B';
81f22ef01cSRoman Divacky case 16: return 'S';
82f22ef01cSRoman Divacky case 32: return 'I';
83f22ef01cSRoman Divacky case 64: return 'L';
84f22ef01cSRoman Divacky default: return 'N';
85f22ef01cSRoman Divacky }
86f22ef01cSRoman Divacky case Type::FloatTyID: return 'F';
87f22ef01cSRoman Divacky case Type::DoubleTyID: return 'D';
88f22ef01cSRoman Divacky case Type::PointerTyID: return 'P';
89f22ef01cSRoman Divacky case Type::FunctionTyID:return 'M';
90f22ef01cSRoman Divacky case Type::StructTyID: return 'T';
91f22ef01cSRoman Divacky case Type::ArrayTyID: return 'A';
92f22ef01cSRoman Divacky default: return 'U';
93f22ef01cSRoman Divacky }
94f22ef01cSRoman Divacky }
95f22ef01cSRoman Divacky
96f22ef01cSRoman Divacky // Try to find address of external function given a Function object.
97f22ef01cSRoman Divacky // Please note, that interpreter doesn't know how to assemble a
98f22ef01cSRoman Divacky // real call in general case (this is JIT job), that's why it assumes,
99f22ef01cSRoman Divacky // that all external functions has the same (and pretty "general") signature.
100f22ef01cSRoman Divacky // The typical example of such functions are "lle_X_" ones.
lookupFunction(const Function * F)101f22ef01cSRoman Divacky static ExFunc lookupFunction(const Function *F) {
102f22ef01cSRoman Divacky // Function not found, look it up... start by figuring out what the
103f22ef01cSRoman Divacky // composite function name should be.
104f22ef01cSRoman Divacky std::string ExtName = "lle_";
1056122f3e6SDimitry Andric FunctionType *FT = F->getFunctionType();
106*b5893f02SDimitry Andric ExtName += getTypeID(FT->getReturnType());
107*b5893f02SDimitry Andric for (Type *T : FT->params())
108*b5893f02SDimitry Andric ExtName += getTypeID(T);
109ff0cc061SDimitry Andric ExtName += ("_" + F->getName()).str();
110f22ef01cSRoman Divacky
111f22ef01cSRoman Divacky sys::ScopedLock Writer(*FunctionsLock);
11239d628a0SDimitry Andric ExFunc FnPtr = (*FuncNames)[ExtName];
11391bc56edSDimitry Andric if (!FnPtr)
114ff0cc061SDimitry Andric FnPtr = (*FuncNames)[("lle_X_" + F->getName()).str()];
11591bc56edSDimitry Andric if (!FnPtr) // Try calling a generic function... if it exists...
116ff0cc061SDimitry Andric FnPtr = (ExFunc)(intptr_t)sys::DynamicLibrary::SearchForAddressOfSymbol(
117ff0cc061SDimitry Andric ("lle_X_" + F->getName()).str());
11891bc56edSDimitry Andric if (FnPtr)
119f22ef01cSRoman Divacky ExportedFunctions->insert(std::make_pair(F, FnPtr)); // Cache for later
120f22ef01cSRoman Divacky return FnPtr;
121f22ef01cSRoman Divacky }
122f22ef01cSRoman Divacky
123f22ef01cSRoman Divacky #ifdef USE_LIBFFI
ffiTypeFor(Type * Ty)1246122f3e6SDimitry Andric static ffi_type *ffiTypeFor(Type *Ty) {
125f22ef01cSRoman Divacky switch (Ty->getTypeID()) {
126f22ef01cSRoman Divacky case Type::VoidTyID: return &ffi_type_void;
127f22ef01cSRoman Divacky case Type::IntegerTyID:
128f22ef01cSRoman Divacky switch (cast<IntegerType>(Ty)->getBitWidth()) {
129f22ef01cSRoman Divacky case 8: return &ffi_type_sint8;
130f22ef01cSRoman Divacky case 16: return &ffi_type_sint16;
131f22ef01cSRoman Divacky case 32: return &ffi_type_sint32;
132f22ef01cSRoman Divacky case 64: return &ffi_type_sint64;
133f22ef01cSRoman Divacky }
134f22ef01cSRoman Divacky case Type::FloatTyID: return &ffi_type_float;
135f22ef01cSRoman Divacky case Type::DoubleTyID: return &ffi_type_double;
136f22ef01cSRoman Divacky case Type::PointerTyID: return &ffi_type_pointer;
137f22ef01cSRoman Divacky default: break;
138f22ef01cSRoman Divacky }
139f22ef01cSRoman Divacky // TODO: Support other types such as StructTyID, ArrayTyID, OpaqueTyID, etc.
140f22ef01cSRoman Divacky report_fatal_error("Type could not be mapped for use with libffi.");
141f22ef01cSRoman Divacky return NULL;
142f22ef01cSRoman Divacky }
143f22ef01cSRoman Divacky
ffiValueFor(Type * Ty,const GenericValue & AV,void * ArgDataPtr)1446122f3e6SDimitry Andric static void *ffiValueFor(Type *Ty, const GenericValue &AV,
145f22ef01cSRoman Divacky void *ArgDataPtr) {
146f22ef01cSRoman Divacky switch (Ty->getTypeID()) {
147f22ef01cSRoman Divacky case Type::IntegerTyID:
148f22ef01cSRoman Divacky switch (cast<IntegerType>(Ty)->getBitWidth()) {
149f22ef01cSRoman Divacky case 8: {
150f22ef01cSRoman Divacky int8_t *I8Ptr = (int8_t *) ArgDataPtr;
151f22ef01cSRoman Divacky *I8Ptr = (int8_t) AV.IntVal.getZExtValue();
152f22ef01cSRoman Divacky return ArgDataPtr;
153f22ef01cSRoman Divacky }
154f22ef01cSRoman Divacky case 16: {
155f22ef01cSRoman Divacky int16_t *I16Ptr = (int16_t *) ArgDataPtr;
156f22ef01cSRoman Divacky *I16Ptr = (int16_t) AV.IntVal.getZExtValue();
157f22ef01cSRoman Divacky return ArgDataPtr;
158f22ef01cSRoman Divacky }
159f22ef01cSRoman Divacky case 32: {
160f22ef01cSRoman Divacky int32_t *I32Ptr = (int32_t *) ArgDataPtr;
161f22ef01cSRoman Divacky *I32Ptr = (int32_t) AV.IntVal.getZExtValue();
162f22ef01cSRoman Divacky return ArgDataPtr;
163f22ef01cSRoman Divacky }
164f22ef01cSRoman Divacky case 64: {
165f22ef01cSRoman Divacky int64_t *I64Ptr = (int64_t *) ArgDataPtr;
166f22ef01cSRoman Divacky *I64Ptr = (int64_t) AV.IntVal.getZExtValue();
167f22ef01cSRoman Divacky return ArgDataPtr;
168f22ef01cSRoman Divacky }
169f22ef01cSRoman Divacky }
170f22ef01cSRoman Divacky case Type::FloatTyID: {
171f22ef01cSRoman Divacky float *FloatPtr = (float *) ArgDataPtr;
172f22ef01cSRoman Divacky *FloatPtr = AV.FloatVal;
173f22ef01cSRoman Divacky return ArgDataPtr;
174f22ef01cSRoman Divacky }
175f22ef01cSRoman Divacky case Type::DoubleTyID: {
176f22ef01cSRoman Divacky double *DoublePtr = (double *) ArgDataPtr;
177f22ef01cSRoman Divacky *DoublePtr = AV.DoubleVal;
178f22ef01cSRoman Divacky return ArgDataPtr;
179f22ef01cSRoman Divacky }
180f22ef01cSRoman Divacky case Type::PointerTyID: {
181f22ef01cSRoman Divacky void **PtrPtr = (void **) ArgDataPtr;
182f22ef01cSRoman Divacky *PtrPtr = GVTOP(AV);
183f22ef01cSRoman Divacky return ArgDataPtr;
184f22ef01cSRoman Divacky }
185f22ef01cSRoman Divacky default: break;
186f22ef01cSRoman Divacky }
187f22ef01cSRoman Divacky // TODO: Support other types such as StructTyID, ArrayTyID, OpaqueTyID, etc.
188f22ef01cSRoman Divacky report_fatal_error("Type value could not be mapped for use with libffi.");
189f22ef01cSRoman Divacky return NULL;
190f22ef01cSRoman Divacky }
191f22ef01cSRoman Divacky
ffiInvoke(RawFunc Fn,Function * F,ArrayRef<GenericValue> ArgVals,const DataLayout & TD,GenericValue & Result)1928f0fd8f6SDimitry Andric static bool ffiInvoke(RawFunc Fn, Function *F, ArrayRef<GenericValue> ArgVals,
1937d523365SDimitry Andric const DataLayout &TD, GenericValue &Result) {
194f22ef01cSRoman Divacky ffi_cif cif;
1956122f3e6SDimitry Andric FunctionType *FTy = F->getFunctionType();
196f22ef01cSRoman Divacky const unsigned NumArgs = F->arg_size();
197f22ef01cSRoman Divacky
198f22ef01cSRoman Divacky // TODO: We don't have type information about the remaining arguments, because
199f22ef01cSRoman Divacky // this information is never passed into ExecutionEngine::runFunction().
200f22ef01cSRoman Divacky if (ArgVals.size() > NumArgs && F->isVarArg()) {
201f22ef01cSRoman Divacky report_fatal_error("Calling external var arg function '" + F->getName()
202f22ef01cSRoman Divacky + "' is not supported by the Interpreter.");
203f22ef01cSRoman Divacky }
204f22ef01cSRoman Divacky
205f22ef01cSRoman Divacky unsigned ArgBytes = 0;
206f22ef01cSRoman Divacky
207f22ef01cSRoman Divacky std::vector<ffi_type*> args(NumArgs);
208f22ef01cSRoman Divacky for (Function::const_arg_iterator A = F->arg_begin(), E = F->arg_end();
209f22ef01cSRoman Divacky A != E; ++A) {
210f22ef01cSRoman Divacky const unsigned ArgNo = A->getArgNo();
2116122f3e6SDimitry Andric Type *ArgTy = FTy->getParamType(ArgNo);
212f22ef01cSRoman Divacky args[ArgNo] = ffiTypeFor(ArgTy);
2137d523365SDimitry Andric ArgBytes += TD.getTypeStoreSize(ArgTy);
214f22ef01cSRoman Divacky }
215f22ef01cSRoman Divacky
216f22ef01cSRoman Divacky SmallVector<uint8_t, 128> ArgData;
217f22ef01cSRoman Divacky ArgData.resize(ArgBytes);
218f22ef01cSRoman Divacky uint8_t *ArgDataPtr = ArgData.data();
219f22ef01cSRoman Divacky SmallVector<void*, 16> values(NumArgs);
220f22ef01cSRoman Divacky for (Function::const_arg_iterator A = F->arg_begin(), E = F->arg_end();
221f22ef01cSRoman Divacky A != E; ++A) {
222f22ef01cSRoman Divacky const unsigned ArgNo = A->getArgNo();
2236122f3e6SDimitry Andric Type *ArgTy = FTy->getParamType(ArgNo);
224f22ef01cSRoman Divacky values[ArgNo] = ffiValueFor(ArgTy, ArgVals[ArgNo], ArgDataPtr);
2257d523365SDimitry Andric ArgDataPtr += TD.getTypeStoreSize(ArgTy);
226f22ef01cSRoman Divacky }
227f22ef01cSRoman Divacky
2286122f3e6SDimitry Andric Type *RetTy = FTy->getReturnType();
229f22ef01cSRoman Divacky ffi_type *rtype = ffiTypeFor(RetTy);
230f22ef01cSRoman Divacky
231*b5893f02SDimitry Andric if (ffi_prep_cif(&cif, FFI_DEFAULT_ABI, NumArgs, rtype, args.data()) ==
232*b5893f02SDimitry Andric FFI_OK) {
233f22ef01cSRoman Divacky SmallVector<uint8_t, 128> ret;
234f22ef01cSRoman Divacky if (RetTy->getTypeID() != Type::VoidTyID)
2357d523365SDimitry Andric ret.resize(TD.getTypeStoreSize(RetTy));
236f22ef01cSRoman Divacky ffi_call(&cif, Fn, ret.data(), values.data());
237f22ef01cSRoman Divacky switch (RetTy->getTypeID()) {
238f22ef01cSRoman Divacky case Type::IntegerTyID:
239f22ef01cSRoman Divacky switch (cast<IntegerType>(RetTy)->getBitWidth()) {
240f22ef01cSRoman Divacky case 8: Result.IntVal = APInt(8 , *(int8_t *) ret.data()); break;
241f22ef01cSRoman Divacky case 16: Result.IntVal = APInt(16, *(int16_t*) ret.data()); break;
242f22ef01cSRoman Divacky case 32: Result.IntVal = APInt(32, *(int32_t*) ret.data()); break;
243f22ef01cSRoman Divacky case 64: Result.IntVal = APInt(64, *(int64_t*) ret.data()); break;
244f22ef01cSRoman Divacky }
245f22ef01cSRoman Divacky break;
246f22ef01cSRoman Divacky case Type::FloatTyID: Result.FloatVal = *(float *) ret.data(); break;
247f22ef01cSRoman Divacky case Type::DoubleTyID: Result.DoubleVal = *(double*) ret.data(); break;
248f22ef01cSRoman Divacky case Type::PointerTyID: Result.PointerVal = *(void **) ret.data(); break;
249f22ef01cSRoman Divacky default: break;
250f22ef01cSRoman Divacky }
251f22ef01cSRoman Divacky return true;
252f22ef01cSRoman Divacky }
253f22ef01cSRoman Divacky
254f22ef01cSRoman Divacky return false;
255f22ef01cSRoman Divacky }
256f22ef01cSRoman Divacky #endif // USE_LIBFFI
257f22ef01cSRoman Divacky
callExternalFunction(Function * F,ArrayRef<GenericValue> ArgVals)258f22ef01cSRoman Divacky GenericValue Interpreter::callExternalFunction(Function *F,
2598f0fd8f6SDimitry Andric ArrayRef<GenericValue> ArgVals) {
260f22ef01cSRoman Divacky TheInterpreter = this;
261f22ef01cSRoman Divacky
26239d628a0SDimitry Andric unique_lock<sys::Mutex> Guard(*FunctionsLock);
263f22ef01cSRoman Divacky
264f22ef01cSRoman Divacky // Do a lookup to see if the function is in our cache... this should just be a
265f22ef01cSRoman Divacky // deferred annotation!
266f22ef01cSRoman Divacky std::map<const Function *, ExFunc>::iterator FI = ExportedFunctions->find(F);
267f22ef01cSRoman Divacky if (ExFunc Fn = (FI == ExportedFunctions->end()) ? lookupFunction(F)
268f22ef01cSRoman Divacky : FI->second) {
26939d628a0SDimitry Andric Guard.unlock();
270f22ef01cSRoman Divacky return Fn(F->getFunctionType(), ArgVals);
271f22ef01cSRoman Divacky }
272f22ef01cSRoman Divacky
273f22ef01cSRoman Divacky #ifdef USE_LIBFFI
274f22ef01cSRoman Divacky std::map<const Function *, RawFunc>::iterator RF = RawFunctions->find(F);
275f22ef01cSRoman Divacky RawFunc RawFn;
276f22ef01cSRoman Divacky if (RF == RawFunctions->end()) {
277f22ef01cSRoman Divacky RawFn = (RawFunc)(intptr_t)
278f22ef01cSRoman Divacky sys::DynamicLibrary::SearchForAddressOfSymbol(F->getName());
279f22ef01cSRoman Divacky if (!RawFn)
280f22ef01cSRoman Divacky RawFn = (RawFunc)(intptr_t)getPointerToGlobalIfAvailable(F);
281f22ef01cSRoman Divacky if (RawFn != 0)
282f22ef01cSRoman Divacky RawFunctions->insert(std::make_pair(F, RawFn)); // Cache for later
283f22ef01cSRoman Divacky } else {
284f22ef01cSRoman Divacky RawFn = RF->second;
285f22ef01cSRoman Divacky }
286f22ef01cSRoman Divacky
28739d628a0SDimitry Andric Guard.unlock();
288f22ef01cSRoman Divacky
289f22ef01cSRoman Divacky GenericValue Result;
2903861d79fSDimitry Andric if (RawFn != 0 && ffiInvoke(RawFn, F, ArgVals, getDataLayout(), Result))
291f22ef01cSRoman Divacky return Result;
292f22ef01cSRoman Divacky #endif // USE_LIBFFI
293f22ef01cSRoman Divacky
294f22ef01cSRoman Divacky if (F->getName() == "__main")
295f22ef01cSRoman Divacky errs() << "Tried to execute an unknown external function: "
29617a519f9SDimitry Andric << *F->getType() << " __main\n";
297f22ef01cSRoman Divacky else
298f22ef01cSRoman Divacky report_fatal_error("Tried to execute an unknown external function: " +
29917a519f9SDimitry Andric F->getName());
300f22ef01cSRoman Divacky #ifndef USE_LIBFFI
301f22ef01cSRoman Divacky errs() << "Recompiling LLVM with --enable-libffi might help.\n";
302f22ef01cSRoman Divacky #endif
303f22ef01cSRoman Divacky return GenericValue();
304f22ef01cSRoman Divacky }
305f22ef01cSRoman Divacky
306f22ef01cSRoman Divacky //===----------------------------------------------------------------------===//
307f22ef01cSRoman Divacky // Functions "exported" to the running application...
308f22ef01cSRoman Divacky //
309f22ef01cSRoman Divacky
310f22ef01cSRoman Divacky // void atexit(Function*)
lle_X_atexit(FunctionType * FT,ArrayRef<GenericValue> Args)3118f0fd8f6SDimitry Andric static GenericValue lle_X_atexit(FunctionType *FT,
3128f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
313f22ef01cSRoman Divacky assert(Args.size() == 1);
314f22ef01cSRoman Divacky TheInterpreter->addAtExitHandler((Function*)GVTOP(Args[0]));
315f22ef01cSRoman Divacky GenericValue GV;
316f22ef01cSRoman Divacky GV.IntVal = 0;
317f22ef01cSRoman Divacky return GV;
318f22ef01cSRoman Divacky }
319f22ef01cSRoman Divacky
320f22ef01cSRoman Divacky // void exit(int)
lle_X_exit(FunctionType * FT,ArrayRef<GenericValue> Args)3218f0fd8f6SDimitry Andric static GenericValue lle_X_exit(FunctionType *FT, ArrayRef<GenericValue> Args) {
322f22ef01cSRoman Divacky TheInterpreter->exitCalled(Args[0]);
323f22ef01cSRoman Divacky return GenericValue();
324f22ef01cSRoman Divacky }
325f22ef01cSRoman Divacky
326f22ef01cSRoman Divacky // void abort(void)
lle_X_abort(FunctionType * FT,ArrayRef<GenericValue> Args)3278f0fd8f6SDimitry Andric static GenericValue lle_X_abort(FunctionType *FT, ArrayRef<GenericValue> Args) {
328f22ef01cSRoman Divacky //FIXME: should we report or raise here?
329f22ef01cSRoman Divacky //report_fatal_error("Interpreted program raised SIGABRT");
330f22ef01cSRoman Divacky raise (SIGABRT);
331f22ef01cSRoman Divacky return GenericValue();
332f22ef01cSRoman Divacky }
333f22ef01cSRoman Divacky
334f22ef01cSRoman Divacky // int sprintf(char *, const char *, ...) - a very rough implementation to make
335f22ef01cSRoman Divacky // output useful.
lle_X_sprintf(FunctionType * FT,ArrayRef<GenericValue> Args)3368f0fd8f6SDimitry Andric static GenericValue lle_X_sprintf(FunctionType *FT,
3378f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
338f22ef01cSRoman Divacky char *OutputBuffer = (char *)GVTOP(Args[0]);
339f22ef01cSRoman Divacky const char *FmtStr = (const char *)GVTOP(Args[1]);
340f22ef01cSRoman Divacky unsigned ArgNo = 2;
341f22ef01cSRoman Divacky
342f22ef01cSRoman Divacky // printf should return # chars printed. This is completely incorrect, but
343f22ef01cSRoman Divacky // close enough for now.
344f22ef01cSRoman Divacky GenericValue GV;
345f22ef01cSRoman Divacky GV.IntVal = APInt(32, strlen(FmtStr));
346d88c1a5aSDimitry Andric while (true) {
347f22ef01cSRoman Divacky switch (*FmtStr) {
348f22ef01cSRoman Divacky case 0: return GV; // Null terminator...
349f22ef01cSRoman Divacky default: // Normal nonspecial character
350f22ef01cSRoman Divacky sprintf(OutputBuffer++, "%c", *FmtStr++);
351f22ef01cSRoman Divacky break;
352f22ef01cSRoman Divacky case '\\': { // Handle escape codes
353f22ef01cSRoman Divacky sprintf(OutputBuffer, "%c%c", *FmtStr, *(FmtStr+1));
354f22ef01cSRoman Divacky FmtStr += 2; OutputBuffer += 2;
355f22ef01cSRoman Divacky break;
356f22ef01cSRoman Divacky }
357f22ef01cSRoman Divacky case '%': { // Handle format specifiers
358f22ef01cSRoman Divacky char FmtBuf[100] = "", Buffer[1000] = "";
359f22ef01cSRoman Divacky char *FB = FmtBuf;
360f22ef01cSRoman Divacky *FB++ = *FmtStr++;
361f22ef01cSRoman Divacky char Last = *FB++ = *FmtStr++;
362f22ef01cSRoman Divacky unsigned HowLong = 0;
363f22ef01cSRoman Divacky while (Last != 'c' && Last != 'd' && Last != 'i' && Last != 'u' &&
364f22ef01cSRoman Divacky Last != 'o' && Last != 'x' && Last != 'X' && Last != 'e' &&
365f22ef01cSRoman Divacky Last != 'E' && Last != 'g' && Last != 'G' && Last != 'f' &&
366f22ef01cSRoman Divacky Last != 'p' && Last != 's' && Last != '%') {
367f22ef01cSRoman Divacky if (Last == 'l' || Last == 'L') HowLong++; // Keep track of l's
368f22ef01cSRoman Divacky Last = *FB++ = *FmtStr++;
369f22ef01cSRoman Divacky }
370f22ef01cSRoman Divacky *FB = 0;
371f22ef01cSRoman Divacky
372f22ef01cSRoman Divacky switch (Last) {
373f22ef01cSRoman Divacky case '%':
374f22ef01cSRoman Divacky memcpy(Buffer, "%", 2); break;
375f22ef01cSRoman Divacky case 'c':
376f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf, uint32_t(Args[ArgNo++].IntVal.getZExtValue()));
377f22ef01cSRoman Divacky break;
378f22ef01cSRoman Divacky case 'd': case 'i':
379f22ef01cSRoman Divacky case 'u': case 'o':
380f22ef01cSRoman Divacky case 'x': case 'X':
381f22ef01cSRoman Divacky if (HowLong >= 1) {
382f22ef01cSRoman Divacky if (HowLong == 1 &&
3837d523365SDimitry Andric TheInterpreter->getDataLayout().getPointerSizeInBits() == 64 &&
384f22ef01cSRoman Divacky sizeof(long) < sizeof(int64_t)) {
385f22ef01cSRoman Divacky // Make sure we use %lld with a 64 bit argument because we might be
386f22ef01cSRoman Divacky // compiling LLI on a 32 bit compiler.
387f22ef01cSRoman Divacky unsigned Size = strlen(FmtBuf);
388f22ef01cSRoman Divacky FmtBuf[Size] = FmtBuf[Size-1];
389f22ef01cSRoman Divacky FmtBuf[Size+1] = 0;
390f22ef01cSRoman Divacky FmtBuf[Size-1] = 'l';
391f22ef01cSRoman Divacky }
392f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf, Args[ArgNo++].IntVal.getZExtValue());
393f22ef01cSRoman Divacky } else
394f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf,uint32_t(Args[ArgNo++].IntVal.getZExtValue()));
395f22ef01cSRoman Divacky break;
396f22ef01cSRoman Divacky case 'e': case 'E': case 'g': case 'G': case 'f':
397f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf, Args[ArgNo++].DoubleVal); break;
398f22ef01cSRoman Divacky case 'p':
399f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf, (void*)GVTOP(Args[ArgNo++])); break;
400f22ef01cSRoman Divacky case 's':
401f22ef01cSRoman Divacky sprintf(Buffer, FmtBuf, (char*)GVTOP(Args[ArgNo++])); break;
402f22ef01cSRoman Divacky default:
403f22ef01cSRoman Divacky errs() << "<unknown printf code '" << *FmtStr << "'!>";
404f22ef01cSRoman Divacky ArgNo++; break;
405f22ef01cSRoman Divacky }
406f22ef01cSRoman Divacky size_t Len = strlen(Buffer);
407f22ef01cSRoman Divacky memcpy(OutputBuffer, Buffer, Len + 1);
408f22ef01cSRoman Divacky OutputBuffer += Len;
409f22ef01cSRoman Divacky }
410f22ef01cSRoman Divacky break;
411f22ef01cSRoman Divacky }
412f22ef01cSRoman Divacky }
413f785676fSDimitry Andric return GV;
414f22ef01cSRoman Divacky }
415f22ef01cSRoman Divacky
416f22ef01cSRoman Divacky // int printf(const char *, ...) - a very rough implementation to make output
417f22ef01cSRoman Divacky // useful.
lle_X_printf(FunctionType * FT,ArrayRef<GenericValue> Args)4188f0fd8f6SDimitry Andric static GenericValue lle_X_printf(FunctionType *FT,
4198f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
420f22ef01cSRoman Divacky char Buffer[10000];
421f22ef01cSRoman Divacky std::vector<GenericValue> NewArgs;
422f22ef01cSRoman Divacky NewArgs.push_back(PTOGV((void*)&Buffer[0]));
423f22ef01cSRoman Divacky NewArgs.insert(NewArgs.end(), Args.begin(), Args.end());
424f22ef01cSRoman Divacky GenericValue GV = lle_X_sprintf(FT, NewArgs);
425f22ef01cSRoman Divacky outs() << Buffer;
426f22ef01cSRoman Divacky return GV;
427f22ef01cSRoman Divacky }
428f22ef01cSRoman Divacky
429f22ef01cSRoman Divacky // int sscanf(const char *format, ...);
lle_X_sscanf(FunctionType * FT,ArrayRef<GenericValue> args)4308f0fd8f6SDimitry Andric static GenericValue lle_X_sscanf(FunctionType *FT,
4318f0fd8f6SDimitry Andric ArrayRef<GenericValue> args) {
432f22ef01cSRoman Divacky assert(args.size() < 10 && "Only handle up to 10 args to sscanf right now!");
433f22ef01cSRoman Divacky
434f22ef01cSRoman Divacky char *Args[10];
435f22ef01cSRoman Divacky for (unsigned i = 0; i < args.size(); ++i)
436f22ef01cSRoman Divacky Args[i] = (char*)GVTOP(args[i]);
437f22ef01cSRoman Divacky
438f22ef01cSRoman Divacky GenericValue GV;
439f22ef01cSRoman Divacky GV.IntVal = APInt(32, sscanf(Args[0], Args[1], Args[2], Args[3], Args[4],
440f22ef01cSRoman Divacky Args[5], Args[6], Args[7], Args[8], Args[9]));
441f22ef01cSRoman Divacky return GV;
442f22ef01cSRoman Divacky }
443f22ef01cSRoman Divacky
444f22ef01cSRoman Divacky // int scanf(const char *format, ...);
lle_X_scanf(FunctionType * FT,ArrayRef<GenericValue> args)4458f0fd8f6SDimitry Andric static GenericValue lle_X_scanf(FunctionType *FT, ArrayRef<GenericValue> args) {
446f22ef01cSRoman Divacky assert(args.size() < 10 && "Only handle up to 10 args to scanf right now!");
447f22ef01cSRoman Divacky
448f22ef01cSRoman Divacky char *Args[10];
449f22ef01cSRoman Divacky for (unsigned i = 0; i < args.size(); ++i)
450f22ef01cSRoman Divacky Args[i] = (char*)GVTOP(args[i]);
451f22ef01cSRoman Divacky
452f22ef01cSRoman Divacky GenericValue GV;
453f22ef01cSRoman Divacky GV.IntVal = APInt(32, scanf( Args[0], Args[1], Args[2], Args[3], Args[4],
454f22ef01cSRoman Divacky Args[5], Args[6], Args[7], Args[8], Args[9]));
455f22ef01cSRoman Divacky return GV;
456f22ef01cSRoman Divacky }
457f22ef01cSRoman Divacky
458f22ef01cSRoman Divacky // int fprintf(FILE *, const char *, ...) - a very rough implementation to make
459f22ef01cSRoman Divacky // output useful.
lle_X_fprintf(FunctionType * FT,ArrayRef<GenericValue> Args)4608f0fd8f6SDimitry Andric static GenericValue lle_X_fprintf(FunctionType *FT,
4618f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
462f22ef01cSRoman Divacky assert(Args.size() >= 2);
463f22ef01cSRoman Divacky char Buffer[10000];
464f22ef01cSRoman Divacky std::vector<GenericValue> NewArgs;
465f22ef01cSRoman Divacky NewArgs.push_back(PTOGV(Buffer));
466f22ef01cSRoman Divacky NewArgs.insert(NewArgs.end(), Args.begin()+1, Args.end());
467f22ef01cSRoman Divacky GenericValue GV = lle_X_sprintf(FT, NewArgs);
468f22ef01cSRoman Divacky
469f22ef01cSRoman Divacky fputs(Buffer, (FILE *) GVTOP(Args[0]));
470f22ef01cSRoman Divacky return GV;
471f22ef01cSRoman Divacky }
472f22ef01cSRoman Divacky
lle_X_memset(FunctionType * FT,ArrayRef<GenericValue> Args)473f785676fSDimitry Andric static GenericValue lle_X_memset(FunctionType *FT,
4748f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
475f785676fSDimitry Andric int val = (int)Args[1].IntVal.getSExtValue();
476f785676fSDimitry Andric size_t len = (size_t)Args[2].IntVal.getZExtValue();
477f785676fSDimitry Andric memset((void *)GVTOP(Args[0]), val, len);
478f785676fSDimitry Andric // llvm.memset.* returns void, lle_X_* returns GenericValue,
479f785676fSDimitry Andric // so here we return GenericValue with IntVal set to zero
480f785676fSDimitry Andric GenericValue GV;
481f785676fSDimitry Andric GV.IntVal = 0;
482f785676fSDimitry Andric return GV;
483f785676fSDimitry Andric }
484f785676fSDimitry Andric
lle_X_memcpy(FunctionType * FT,ArrayRef<GenericValue> Args)485f785676fSDimitry Andric static GenericValue lle_X_memcpy(FunctionType *FT,
4868f0fd8f6SDimitry Andric ArrayRef<GenericValue> Args) {
487f785676fSDimitry Andric memcpy(GVTOP(Args[0]), GVTOP(Args[1]),
488f785676fSDimitry Andric (size_t)(Args[2].IntVal.getLimitedValue()));
489f785676fSDimitry Andric
490f785676fSDimitry Andric // llvm.memcpy* returns void, lle_X_* returns GenericValue,
491f785676fSDimitry Andric // so here we return GenericValue with IntVal set to zero
492f785676fSDimitry Andric GenericValue GV;
493f785676fSDimitry Andric GV.IntVal = 0;
494f785676fSDimitry Andric return GV;
495f785676fSDimitry Andric }
496f785676fSDimitry Andric
initializeExternalFunctions()497f22ef01cSRoman Divacky void Interpreter::initializeExternalFunctions() {
498f22ef01cSRoman Divacky sys::ScopedLock Writer(*FunctionsLock);
49939d628a0SDimitry Andric (*FuncNames)["lle_X_atexit"] = lle_X_atexit;
50039d628a0SDimitry Andric (*FuncNames)["lle_X_exit"] = lle_X_exit;
50139d628a0SDimitry Andric (*FuncNames)["lle_X_abort"] = lle_X_abort;
502f22ef01cSRoman Divacky
50339d628a0SDimitry Andric (*FuncNames)["lle_X_printf"] = lle_X_printf;
50439d628a0SDimitry Andric (*FuncNames)["lle_X_sprintf"] = lle_X_sprintf;
50539d628a0SDimitry Andric (*FuncNames)["lle_X_sscanf"] = lle_X_sscanf;
50639d628a0SDimitry Andric (*FuncNames)["lle_X_scanf"] = lle_X_scanf;
50739d628a0SDimitry Andric (*FuncNames)["lle_X_fprintf"] = lle_X_fprintf;
50839d628a0SDimitry Andric (*FuncNames)["lle_X_memset"] = lle_X_memset;
50939d628a0SDimitry Andric (*FuncNames)["lle_X_memcpy"] = lle_X_memcpy;
510f22ef01cSRoman Divacky }
511