blob: 2e7236b5d500692b73d9497d774393ee8a331441 [file] [log] [blame]
Chris Lattnerbd199fb2002-12-24 00:01:05 +00001//===-- JIT.cpp - LLVM Just in Time Compiler ------------------------------===//
2//
3// This file implements the top-level support for creating a Just-In-Time
4// compiler for the current architecture.
5//
6//===----------------------------------------------------------------------===//
7
8#include "VM.h"
9#include "llvm/Target/TargetMachine.h"
10#include "llvm/Target/TargetMachineImpls.h"
11#include "llvm/Module.h"
Misha Brukmanabb027c2003-05-27 21:40:39 +000012#include "Support/CommandLine.h"
13
14namespace {
15 cl::opt<std::string>
16 Arch("march", cl::desc("Architecture: `x86' or `sparc'"), cl::Prefix,
17 cl::value_desc("machine architecture"));
18
19 static std::string DefaultArch =
20#if defined(i386) || defined(__i386__) || defined(__x86__)
21 "x86";
22#elif defined(sparc) || defined(__sparc__) || defined(__sparcv9)
23 "sparc";
24#else
25 "";
26#endif
27
28}
Chris Lattnerbd199fb2002-12-24 00:01:05 +000029
30
31/// createJIT - Create an return a new JIT compiler if there is one available
32/// for the current target. Otherwise it returns null.
33///
34ExecutionEngine *ExecutionEngine::createJIT(Module *M, unsigned Config) {
Misha Brukmanabb027c2003-05-27 21:40:39 +000035
36 TargetMachine* (*TargetMachineAllocator)(unsigned) = 0;
37 if (Arch == "")
38 Arch = DefaultArch;
Chris Lattnerbd199fb2002-12-24 00:01:05 +000039
Misha Brukmanabb027c2003-05-27 21:40:39 +000040 // Allow a command-line switch to override what *should* be the default target
41 // machine for this platform. This allows for debugging a Sparc JIT on X86 --
42 // our X86 machines are much faster at recompiling LLVM and linking lli.
43 if (Arch == "x86") {
44 TargetMachineAllocator = allocateX86TargetMachine;
45 } else if (Arch == "sparc") {
Misha Brukman906f5fa2003-06-02 03:23:16 +000046 TargetMachineAllocator = allocateSparcTargetMachine;
Misha Brukmanabb027c2003-05-27 21:40:39 +000047 }
48
49 if (TargetMachineAllocator) {
50 // Allocate a target...
51 TargetMachine *Target = (*TargetMachineAllocator)(Config);
52 assert(Target && "Could not allocate target machine!");
53
54 // Create the virtual machine object...
55 return new VM(M, Target);
56 } else {
57 return 0;
58 }
Chris Lattnerbd199fb2002-12-24 00:01:05 +000059}
60
61VM::VM(Module *M, TargetMachine *tm) : ExecutionEngine(M), TM(*tm) {
62 setTargetData(TM.getTargetData());
Misha Brukmanabb027c2003-05-27 21:40:39 +000063
64 // Initialize MCE
Misha Brukman906f5fa2003-06-02 03:23:16 +000065 MCE = createEmitter(*this);
Misha Brukmanabb027c2003-05-27 21:40:39 +000066
Chris Lattnerbd199fb2002-12-24 00:01:05 +000067 setupPassManager();
Chris Lattner56adf152003-05-12 02:14:34 +000068 emitGlobals();
Chris Lattnerbd199fb2002-12-24 00:01:05 +000069}
70
71int VM::run(const std::string &FnName, const std::vector<std::string> &Args) {
72 Function *F = getModule().getNamedFunction(FnName);
73 if (F == 0) {
74 std::cerr << "Could not find function '" << FnName <<"' in module!\n";
75 return 1;
76 }
77
78 int(*PF)(int, char**) = (int(*)(int, char**))getPointerToFunction(F);
79 assert(PF != 0 && "Null pointer to function?");
80
81 // Build an argv vector...
82 char **Argv = (char**)CreateArgv(Args);
83
84 // Call the main function...
Chris Lattner22080f92003-05-14 13:53:40 +000085 int Result = PF(Args.size(), Argv);
86
87 // Run any atexit handlers now!
88 runAtExitHandlers();
89 return Result;
Chris Lattnerbd199fb2002-12-24 00:01:05 +000090}