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