blob: 4e224261cbffdfe2bafc677ec2d0f6d5215acf78 [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
43static const char *ProgramName;
44
45static void Message(const char *Type, const Twine &Msg) {
46 errs() << ProgramName << ": " << Type << ": " << Msg << "\n";
47}
48
49static int Error(const Twine &Msg) {
50 Message("error", Msg);
51 return 1;
52}
53
54/* *** */
55
Jim Grosbach82c25b42011-03-18 17:24:21 +000056static int executeInput() {
Jim Grosbach1cb19a42011-03-18 17:11:39 +000057 // Load the input memory buffer.
58 OwningPtr<MemoryBuffer> InputBuffer;
59 if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFile, InputBuffer))
60 return Error("unable to read input: '" + ec.message() + "'");
61
Jim Grosbach6e563312011-03-21 22:15:52 +000062 // Instantiate a dynamic linker.
63 RuntimeDyld Dyld;
Jim Grosbach1cb19a42011-03-18 17:11:39 +000064
Jim Grosbach6e563312011-03-21 22:15:52 +000065 // Load the object file into it.
66 if (Dyld.loadObject(InputBuffer.take()))
67 return true;
Jim Grosbach1cb19a42011-03-18 17:11:39 +000068
69 // Get the address of "_main".
Jim Grosbach6e563312011-03-21 22:15:52 +000070 void *MainAddress = Dyld.getSymbolAddress("_main");
71 if (MainAddress == 0)
Jim Grosbach1cb19a42011-03-18 17:11:39 +000072 return Error("no definition for '_main'");
73
74 // Invalidate the instruction cache.
Jim Grosbach6e563312011-03-21 22:15:52 +000075 sys::MemoryBlock Data = Dyld.getMemoryBlock();
Jim Grosbach1cb19a42011-03-18 17:11:39 +000076 sys::Memory::InvalidateInstructionCache(Data.base(), Data.size());
77
78 // Make sure the memory is executable.
Jim Grosbach6e563312011-03-21 22:15:52 +000079 std::string ErrorStr;
Jim Grosbach1cb19a42011-03-18 17:11:39 +000080 if (!sys::Memory::setExecutable(Data, &ErrorStr))
81 return Error("unable to mark function executable: '" + ErrorStr + "'");
82
83 // Dispatch to _main().
Jim Grosbach1cb19a42011-03-18 17:11:39 +000084 errs() << "loaded '_main' at: " << MainAddress << "\n";
85
86 int (*Main)(int, const char**) =
87 (int(*)(int,const char**)) uintptr_t(MainAddress);
88 const char **Argv = new const char*[2];
89 Argv[0] = InputFile.c_str();
90 Argv[1] = 0;
91 return Main(1, Argv);
92}
93
94int main(int argc, char **argv) {
95 ProgramName = argv[0];
96 llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
97
98 cl::ParseCommandLineOptions(argc, argv, "llvm MC-JIT tool\n");
99
100 switch (Action) {
101 default:
102 case AC_Execute:
Jim Grosbach82c25b42011-03-18 17:24:21 +0000103 return executeInput();
Jim Grosbach1cb19a42011-03-18 17:11:39 +0000104 }
105
106 return 0;
107}