blob: c2f1c3c48ddf9d044889b18e8d32cb361b694ead [file] [log] [blame]
Jim Grosbach1cb19a42011-03-18 17:11:39 +00001//===-- llvm-rtdyld.cpp - MCJIT Testing Tool ------------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This is a testing tool for use with the MC-JIT LLVM components.
11//
12//===----------------------------------------------------------------------===//
13
14#include "llvm/ADT/StringMap.h"
15#include "llvm/ADT/OwningPtr.h"
Jim Grosbach6e563312011-03-21 22:15:52 +000016#include "llvm/ExecutionEngine/RuntimeDyld.h"
Jim Grosbach1cb19a42011-03-18 17:11:39 +000017#include "llvm/Object/MachOObject.h"
18#include "llvm/Support/CommandLine.h"
19#include "llvm/Support/ManagedStatic.h"
20#include "llvm/Support/Memory.h"
21#include "llvm/Support/MemoryBuffer.h"
22#include "llvm/Support/raw_ostream.h"
23#include "llvm/Support/system_error.h"
24using namespace llvm;
25using namespace llvm::object;
26
27static cl::opt<std::string>
28InputFile(cl::Positional, cl::desc("<input file>"), cl::init("-"));
29
30enum ActionType {
31 AC_Execute
32};
33
34static cl::opt<ActionType>
35Action(cl::desc("Action to perform:"),
36 cl::init(AC_Execute),
37 cl::values(clEnumValN(AC_Execute, "execute",
38 "Load, link, and execute the inputs."),
39 clEnumValEnd));
40
41/* *** */
42
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000043// A trivial memory manager that doesn't do anything fancy, just uses the
44// support library allocation routines directly.
45class TrivialMemoryManager : public RTDyldMemoryManager {
46public:
Jim Grosbachc41ab782011-04-06 01:11:05 +000047 uint8_t *startFunctionBody(const char *Name, uintptr_t &Size);
48 void endFunctionBody(const char *Name, uint8_t *FunctionStart,
49 uint8_t *FunctionEnd) {}
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000050};
51
Jim Grosbachc41ab782011-04-06 01:11:05 +000052uint8_t *TrivialMemoryManager::startFunctionBody(const char *Name,
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000053 uintptr_t &Size) {
Jim Grosbachc41ab782011-04-06 01:11:05 +000054 return (uint8_t*)sys::Memory::AllocateRWX(Size, 0, 0).base();
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000055}
56
Jim Grosbach1cb19a42011-03-18 17:11:39 +000057static const char *ProgramName;
58
59static void Message(const char *Type, const Twine &Msg) {
60 errs() << ProgramName << ": " << Type << ": " << Msg << "\n";
61}
62
63static int Error(const Twine &Msg) {
64 Message("error", Msg);
65 return 1;
66}
67
68/* *** */
69
Jim Grosbach82c25b42011-03-18 17:24:21 +000070static int executeInput() {
Jim Grosbach1cb19a42011-03-18 17:11:39 +000071 // Load the input memory buffer.
72 OwningPtr<MemoryBuffer> InputBuffer;
73 if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFile, InputBuffer))
74 return Error("unable to read input: '" + ec.message() + "'");
75
Jim Grosbach6e563312011-03-21 22:15:52 +000076 // Instantiate a dynamic linker.
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000077 RuntimeDyld Dyld(new TrivialMemoryManager);
Jim Grosbach1cb19a42011-03-18 17:11:39 +000078
Jim Grosbach6e563312011-03-21 22:15:52 +000079 // Load the object file into it.
Jim Grosbachb3eecaf2011-03-22 18:19:42 +000080 if (Dyld.loadObject(InputBuffer.take())) {
81 return Error(Dyld.getErrorString());
82 }
Jim Grosbach1cb19a42011-03-18 17:11:39 +000083
84 // Get the address of "_main".
Jim Grosbachb0271052011-04-08 17:31:24 +000085 void *MainAddress = Dyld.getSymbolAddress("_main");
Jim Grosbach6e563312011-03-21 22:15:52 +000086 if (MainAddress == 0)
Jim Grosbach1cb19a42011-03-18 17:11:39 +000087 return Error("no definition for '_main'");
88
89 // Invalidate the instruction cache.
Jim Grosbach6e563312011-03-21 22:15:52 +000090 sys::MemoryBlock Data = Dyld.getMemoryBlock();
Jim Grosbach1cb19a42011-03-18 17:11:39 +000091 sys::Memory::InvalidateInstructionCache(Data.base(), Data.size());
92
93 // Make sure the memory is executable.
Jim Grosbach6e563312011-03-21 22:15:52 +000094 std::string ErrorStr;
Jim Grosbach1cb19a42011-03-18 17:11:39 +000095 if (!sys::Memory::setExecutable(Data, &ErrorStr))
96 return Error("unable to mark function executable: '" + ErrorStr + "'");
97
98 // Dispatch to _main().
Jim Grosbachfcbe5b72011-04-04 23:04:39 +000099 errs() << "loaded '_main' at: " << (void*)MainAddress << "\n";
Jim Grosbach1cb19a42011-03-18 17:11:39 +0000100
101 int (*Main)(int, const char**) =
102 (int(*)(int,const char**)) uintptr_t(MainAddress);
103 const char **Argv = new const char*[2];
104 Argv[0] = InputFile.c_str();
105 Argv[1] = 0;
106 return Main(1, Argv);
107}
108
109int main(int argc, char **argv) {
110 ProgramName = argv[0];
111 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
112
113 cl::ParseCommandLineOptions(argc, argv, "llvm MC-JIT tool\n");
114
115 switch (Action) {
116 default:
117 case AC_Execute:
Jim Grosbach82c25b42011-03-18 17:24:21 +0000118 return executeInput();
Jim Grosbach1cb19a42011-03-18 17:11:39 +0000119 }
120
121 return 0;
122}