blob: c6748fa1dc7e3e50f4cbb3b4e3e27e2ee2d7d5f8 [file] [log] [blame]
Chris Lattner6701a862003-05-14 13:26:47 +00001//===-- VM.cpp - LLVM Just in Time Compiler -------------------------------===//
Chris Lattnerbd199fb2002-12-24 00:01:05 +00002//
3// This tool implements a just-in-time compiler for LLVM, allowing direct
4// execution of LLVM bytecode in an efficient manner.
5//
6//===----------------------------------------------------------------------===//
7
8#include "VM.h"
9#include "llvm/Target/TargetMachine.h"
10#include "llvm/CodeGen/MachineCodeEmitter.h"
11#include "llvm/Function.h"
Chris Lattnerbd199fb2002-12-24 00:01:05 +000012
13VM::~VM() {
14 delete MCE;
15 delete &TM;
16}
17
18/// setupPassManager - Initialize the VM PassManager object with all of the
19/// passes needed for the target to generate code.
20///
21void VM::setupPassManager() {
22 // Compile LLVM Code down to machine code in the intermediate representation
23 if (TM.addPassesToJITCompile(PM)) {
24 std::cerr << "lli: target '" << TM.getName()
25 << "' doesn't support JIT compilation!\n";
26 abort();
27 }
28
29 // Turn the machine code intermediate representation into bytes in memory that
30 // may be executed.
31 //
32 if (TM.addPassesToEmitMachineCode(PM, *MCE)) {
33 std::cerr << "lli: target '" << TM.getName()
34 << "' doesn't support machine code emission!\n";
35 abort();
36 }
37}
38
39void *VM::resolveFunctionReference(void *RefAddr) {
40 Function *F = FunctionRefs[RefAddr];
41 assert(F && "Reference address not known!");
42
43 void *Addr = getPointerToFunction(F);
44 assert(Addr && "Pointer to function unknown!");
45
46 FunctionRefs.erase(RefAddr);
47 return Addr;
48}
49
50const std::string &VM::getFunctionReferencedName(void *RefAddr) {
Chris Lattner6701a862003-05-14 13:26:47 +000051 assert(FunctionRefs[RefAddr] && "Function address unknown!");
Chris Lattnerbd199fb2002-12-24 00:01:05 +000052 return FunctionRefs[RefAddr]->getName();
53}
54
Chris Lattnerbd199fb2002-12-24 00:01:05 +000055/// getPointerToFunction - This method is used to get the address of the
56/// specified function, compiling it if neccesary.
57///
58void *VM::getPointerToFunction(const Function *F) {
59 void *&Addr = GlobalAddress[F]; // Function already code gen'd
60 if (Addr) return Addr;
61
Chris Lattner0d448c02003-01-13 01:00:48 +000062 if (F->isExternal())
63 return Addr = getPointerToNamedFunction(F->getName());
Chris Lattnerbd199fb2002-12-24 00:01:05 +000064
Chris Lattner66a84942003-05-08 21:08:43 +000065 static bool isAlreadyCodeGenerating = false;
66 if (isAlreadyCodeGenerating) {
Chris Lattner6125fdd2003-05-09 03:30:07 +000067 // Generate a function stub instead of reentering...
68 void *SAddr = emitStubForFunction(*F);
69 assert(SAddr && "Target machine doesn't support function stub generation!");
70 return SAddr;
Chris Lattner66a84942003-05-08 21:08:43 +000071 }
72
73 // FIXME: JIT all of the functions in the module. Eventually this will JIT
74 // functions on demand. This has the effect of populating all of the
75 // non-external functions into the GlobalAddress table.
76 isAlreadyCodeGenerating = true;
Chris Lattnerbd199fb2002-12-24 00:01:05 +000077 PM.run(getModule());
Chris Lattner66a84942003-05-08 21:08:43 +000078 isAlreadyCodeGenerating = false;
Chris Lattnerbd199fb2002-12-24 00:01:05 +000079
80 assert(Addr && "Code generation didn't add function to GlobalAddress table!");
81 return Addr;
82}