blob: 9bdaa57017914a722336123c2e81d8f02f815110 [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"
Brian Gaeke97222942003-09-05 19:39:22 +00009#include "llvm/ExecutionEngine/GenericValue.h"
Chris Lattnerbd199fb2002-12-24 00:01:05 +000010#include "llvm/Target/TargetMachine.h"
11#include "llvm/Target/TargetMachineImpls.h"
12#include "llvm/Module.h"
Misha Brukmanabb027c2003-05-27 21:40:39 +000013#include "Support/CommandLine.h"
14
Misha Brukman4e8c9992003-06-06 06:59:55 +000015// FIXME: REMOVE THIS
16#include "llvm/PassManager.h"
17
Misha Brukman82742912003-07-02 17:53:19 +000018#if !defined(ENABLE_X86_JIT) && !defined(ENABLE_SPARC_JIT)
19#define NO_JITS_ENABLED
20#endif
Chris Lattner97ac14f2003-06-17 15:43:13 +000021
Misha Brukman82742912003-07-02 17:53:19 +000022namespace {
23 enum ArchName { x86, Sparc };
24
25#ifndef NO_JITS_ENABLED
Chris Lattner97ac14f2003-06-17 15:43:13 +000026 cl::opt<ArchName>
27 Arch("march", cl::desc("Architecture to JIT to:"), cl::Prefix,
Misha Brukman82742912003-07-02 17:53:19 +000028 cl::values(
29#ifdef ENABLE_X86_JIT
30 clEnumVal(x86, " IA-32 (Pentium and above)"),
31#endif
32#ifdef ENABLE_SPARC_JIT
Chris Lattnerde3209b2003-06-17 15:54:02 +000033 clEnumValN(Sparc, "sparc", " Sparc-V9"),
Misha Brukmanabb027c2003-05-27 21:40:39 +000034#endif
Chris Lattner97ac14f2003-06-17 15:43:13 +000035 0),
Misha Brukman82742912003-07-02 17:53:19 +000036#if defined(ENABLE_X86_JIT)
Chris Lattner97ac14f2003-06-17 15:43:13 +000037 cl::init(x86)
Misha Brukman82742912003-07-02 17:53:19 +000038#elif defined(ENABLE_SPARC_JIT)
Chris Lattnerde3209b2003-06-17 15:54:02 +000039 cl::init(Sparc)
Chris Lattner97ac14f2003-06-17 15:43:13 +000040#endif
41 );
Misha Brukman82742912003-07-02 17:53:19 +000042#endif /* NO_JITS_ENABLED */
Misha Brukmanabb027c2003-05-27 21:40:39 +000043}
Chris Lattnerbd199fb2002-12-24 00:01:05 +000044
Brian Gaeke82d82772003-09-03 20:34:19 +000045/// create - Create an return a new JIT compiler if there is one available
46/// for the current target. Otherwise, return null.
Chris Lattnerbd199fb2002-12-24 00:01:05 +000047///
Misha Brukman005e5e92003-10-14 21:37:41 +000048ExecutionEngine *VM::create(ModuleProvider *MP) {
Chris Lattner39c07262003-08-24 19:50:53 +000049 TargetMachine* (*TargetMachineAllocator)(const Module &) = 0;
Chris Lattnerbd199fb2002-12-24 00:01:05 +000050
Misha Brukmanabb027c2003-05-27 21:40:39 +000051 // Allow a command-line switch to override what *should* be the default target
52 // machine for this platform. This allows for debugging a Sparc JIT on X86 --
Misha Brukman82742912003-07-02 17:53:19 +000053 // our X86 machines are much faster at recompiling LLVM and linking LLI.
54#ifdef NO_JITS_ENABLED
55 return 0;
56#endif
57
Chris Lattner97ac14f2003-06-17 15:43:13 +000058 switch (Arch) {
Misha Brukman82742912003-07-02 17:53:19 +000059#ifdef ENABLE_X86_JIT
Chris Lattner97ac14f2003-06-17 15:43:13 +000060 case x86:
Misha Brukmanabb027c2003-05-27 21:40:39 +000061 TargetMachineAllocator = allocateX86TargetMachine;
Chris Lattner97ac14f2003-06-17 15:43:13 +000062 break;
Misha Brukman82742912003-07-02 17:53:19 +000063#endif
64#ifdef ENABLE_SPARC_JIT
Chris Lattnerde3209b2003-06-17 15:54:02 +000065 case Sparc:
Misha Brukman906f5fa2003-06-02 03:23:16 +000066 TargetMachineAllocator = allocateSparcTargetMachine;
Chris Lattner97ac14f2003-06-17 15:43:13 +000067 break;
Chris Lattner7aefa962003-06-17 15:32:38 +000068#endif
Chris Lattner97ac14f2003-06-17 15:43:13 +000069 default:
70 assert(0 && "-march flag not supported on this host!");
Misha Brukmanabb027c2003-05-27 21:40:39 +000071 }
Chris Lattner97ac14f2003-06-17 15:43:13 +000072
73 // Allocate a target...
Misha Brukman005e5e92003-10-14 21:37:41 +000074 TargetMachine *Target = TargetMachineAllocator(*(MP->getModule()));
Chris Lattner97ac14f2003-06-17 15:43:13 +000075 assert(Target && "Could not allocate target machine!");
76
77 // Create the virtual machine object...
Misha Brukman005e5e92003-10-14 21:37:41 +000078 return new VM(MP, Target);
Chris Lattnerbd199fb2002-12-24 00:01:05 +000079}
80
Misha Brukman005e5e92003-10-14 21:37:41 +000081VM::VM(ModuleProvider *MP, TargetMachine *tm) : ExecutionEngine(MP), TM(*tm),
82 PM(MP)
83{
Chris Lattnerbd199fb2002-12-24 00:01:05 +000084 setTargetData(TM.getTargetData());
Misha Brukmanabb027c2003-05-27 21:40:39 +000085
86 // Initialize MCE
Misha Brukman906f5fa2003-06-02 03:23:16 +000087 MCE = createEmitter(*this);
Misha Brukmanabb027c2003-05-27 21:40:39 +000088
Chris Lattnerbd199fb2002-12-24 00:01:05 +000089 setupPassManager();
Misha Brukman4e8c9992003-06-06 06:59:55 +000090
Misha Brukman82742912003-07-02 17:53:19 +000091#ifdef ENABLE_SPARC_JIT
Misha Brukman4e8c9992003-06-06 06:59:55 +000092 // THIS GOES BEYOND UGLY HACKS
93 if (TM.getName() == "UltraSparc-Native") {
94 extern Pass *createPreSelectionPass(TargetMachine &TM);
95 PassManager PM;
96 // Specialize LLVM code for this target machine and then
97 // run basic dataflow optimizations on LLVM code.
98 PM.add(createPreSelectionPass(TM));
Misha Brukman005e5e92003-10-14 21:37:41 +000099 // We cannot utilize function-at-a-time loading here because PreSelection
100 // is a ModulePass.
101 MP->materializeModule();
102 PM.run(*(MP->getModule()));
Misha Brukman4e8c9992003-06-06 06:59:55 +0000103 }
Chris Lattner7aefa962003-06-17 15:32:38 +0000104#endif
Misha Brukman4e8c9992003-06-06 06:59:55 +0000105
Chris Lattner56adf152003-05-12 02:14:34 +0000106 emitGlobals();
Chris Lattnerbd199fb2002-12-24 00:01:05 +0000107}
108
Brian Gaeke70975ee2003-09-05 18:42:01 +0000109/// run - Start execution with the specified function and arguments.
Chris Lattner05a1a302003-08-21 21:32:12 +0000110///
Brian Gaeke70975ee2003-09-05 18:42:01 +0000111GenericValue VM::run(Function *F, const std::vector<GenericValue> &ArgValues)
112{
113 assert (F && "Function *F was null at entry to run()");
Chris Lattnerbd199fb2002-12-24 00:01:05 +0000114
Brian Gaeke70975ee2003-09-05 18:42:01 +0000115 int (*PF)(int, char **, const char **) =
116 (int(*)(int, char **, const char **))getPointerToFunction(F);
117 assert(PF != 0 && "Pointer to fn's code was null after getPointerToFunction");
Chris Lattnerbd199fb2002-12-24 00:01:05 +0000118
Brian Gaeke70975ee2003-09-05 18:42:01 +0000119 // Call the function.
120 int ExitCode = PF(ArgValues[0].IntVal, (char **) GVTOP (ArgValues[1]),
121 (const char **) GVTOP (ArgValues[2]));
Chris Lattner22080f92003-05-14 13:53:40 +0000122
123 // Run any atexit handlers now!
124 runAtExitHandlers();
Brian Gaeke70975ee2003-09-05 18:42:01 +0000125
126 GenericValue rv;
127 rv.IntVal = ExitCode;
128 return rv;
Chris Lattnerbd199fb2002-12-24 00:01:05 +0000129}