mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-10 02:36:06 +00:00
d04a8d4b33
Sooooo many of these had incorrect or strange main module includes. I have manually inspected all of these, and fixed the main module include to be the nearest plausible thing I could find. If you own or care about any of these source files, I encourage you to take some time and check that these edits were sensible. I can't have broken anything (I strictly added headers, and reordered them, never removed), but they may not be the headers you'd really like to identify as containing the API being implemented. Many forward declarations and missing includes were added to a header files to allow them to parse cleanly when included first. The main module rule does in fact have its merits. =] git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@169131 91177308-0d34-0410-b5e6-96231b3b80d8
337 lines
11 KiB
C++
337 lines
11 KiB
C++
//===-- MCJIT.cpp - MC-based Just-in-Time Compiler ------------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "MCJIT.h"
|
|
#include "llvm/DataLayout.h"
|
|
#include "llvm/DerivedTypes.h"
|
|
#include "llvm/ExecutionEngine/GenericValue.h"
|
|
#include "llvm/ExecutionEngine/JITEventListener.h"
|
|
#include "llvm/ExecutionEngine/JITMemoryManager.h"
|
|
#include "llvm/ExecutionEngine/MCJIT.h"
|
|
#include "llvm/ExecutionEngine/ObjectBuffer.h"
|
|
#include "llvm/ExecutionEngine/ObjectImage.h"
|
|
#include "llvm/Function.h"
|
|
#include "llvm/MC/MCAsmInfo.h"
|
|
#include "llvm/Support/DynamicLibrary.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/MutexGuard.h"
|
|
|
|
using namespace llvm;
|
|
|
|
namespace {
|
|
|
|
static struct RegisterJIT {
|
|
RegisterJIT() { MCJIT::Register(); }
|
|
} JITRegistrator;
|
|
|
|
}
|
|
|
|
extern "C" void LLVMLinkInMCJIT() {
|
|
}
|
|
|
|
ExecutionEngine *MCJIT::createJIT(Module *M,
|
|
std::string *ErrorStr,
|
|
JITMemoryManager *JMM,
|
|
bool GVsWithCode,
|
|
TargetMachine *TM) {
|
|
// Try to register the program as a source of symbols to resolve against.
|
|
//
|
|
// FIXME: Don't do this here.
|
|
sys::DynamicLibrary::LoadLibraryPermanently(0, NULL);
|
|
|
|
return new MCJIT(M, TM, JMM, GVsWithCode);
|
|
}
|
|
|
|
MCJIT::MCJIT(Module *m, TargetMachine *tm, RTDyldMemoryManager *MM,
|
|
bool AllocateGVsWithCode)
|
|
: ExecutionEngine(m), TM(tm), Ctx(0), MemMgr(MM), Dyld(MM),
|
|
isCompiled(false), M(m) {
|
|
|
|
setDataLayout(TM->getDataLayout());
|
|
}
|
|
|
|
MCJIT::~MCJIT() {
|
|
if (LoadedObject)
|
|
NotifyFreeingObject(*LoadedObject.get());
|
|
delete MemMgr;
|
|
delete TM;
|
|
}
|
|
|
|
void MCJIT::emitObject(Module *m) {
|
|
/// Currently, MCJIT only supports a single module and the module passed to
|
|
/// this function call is expected to be the contained module. The module
|
|
/// is passed as a parameter here to prepare for multiple module support in
|
|
/// the future.
|
|
assert(M == m);
|
|
|
|
// Get a thread lock to make sure we aren't trying to compile multiple times
|
|
MutexGuard locked(lock);
|
|
|
|
// FIXME: Track compilation state on a per-module basis when multiple modules
|
|
// are supported.
|
|
// Re-compilation is not supported
|
|
if (isCompiled)
|
|
return;
|
|
|
|
PassManager PM;
|
|
|
|
PM.add(new DataLayout(*TM->getDataLayout()));
|
|
|
|
// The RuntimeDyld will take ownership of this shortly
|
|
OwningPtr<ObjectBufferStream> Buffer(new ObjectBufferStream());
|
|
|
|
// Turn the machine code intermediate representation into bytes in memory
|
|
// that may be executed.
|
|
if (TM->addPassesToEmitMC(PM, Ctx, Buffer->getOStream(), false)) {
|
|
report_fatal_error("Target does not support MC emission!");
|
|
}
|
|
|
|
// Initialize passes.
|
|
PM.run(*m);
|
|
// Flush the output buffer to get the generated code into memory
|
|
Buffer->flush();
|
|
|
|
// Load the object into the dynamic linker.
|
|
// handing off ownership of the buffer
|
|
LoadedObject.reset(Dyld.loadObject(Buffer.take()));
|
|
if (!LoadedObject)
|
|
report_fatal_error(Dyld.getErrorString());
|
|
|
|
// Resolve any relocations.
|
|
Dyld.resolveRelocations();
|
|
|
|
// FIXME: Make this optional, maybe even move it to a JIT event listener
|
|
LoadedObject->registerWithDebugger();
|
|
|
|
NotifyObjectEmitted(*LoadedObject);
|
|
|
|
// FIXME: Add support for per-module compilation state
|
|
isCompiled = true;
|
|
}
|
|
|
|
// FIXME: Add a parameter to identify which object is being finalized when
|
|
// MCJIT supports multiple modules.
|
|
// FIXME: Provide a way to separate code emission, relocations and page
|
|
// protection in the interface.
|
|
void MCJIT::finalizeObject() {
|
|
// If the module hasn't been compiled, just do that.
|
|
if (!isCompiled) {
|
|
// If the call to Dyld.resolveRelocations() is removed from emitObject()
|
|
// we'll need to do that here.
|
|
emitObject(M);
|
|
|
|
// Set page permissions.
|
|
MemMgr->applyPermissions();
|
|
|
|
return;
|
|
}
|
|
|
|
// Resolve any relocations.
|
|
Dyld.resolveRelocations();
|
|
|
|
// Set page permissions.
|
|
MemMgr->applyPermissions();
|
|
}
|
|
|
|
void *MCJIT::getPointerToBasicBlock(BasicBlock *BB) {
|
|
report_fatal_error("not yet implemented");
|
|
}
|
|
|
|
void *MCJIT::getPointerToFunction(Function *F) {
|
|
// FIXME: This should really return a uint64_t since it's a pointer in the
|
|
// target address space, not our local address space. That's part of the
|
|
// ExecutionEngine interface, though. Fix that when the old JIT finally
|
|
// dies.
|
|
|
|
// FIXME: Add support for per-module compilation state
|
|
if (!isCompiled)
|
|
emitObject(M);
|
|
|
|
if (F->isDeclaration() || F->hasAvailableExternallyLinkage()) {
|
|
bool AbortOnFailure = !F->hasExternalWeakLinkage();
|
|
void *Addr = getPointerToNamedFunction(F->getName(), AbortOnFailure);
|
|
addGlobalMapping(F, Addr);
|
|
return Addr;
|
|
}
|
|
|
|
// FIXME: Should the Dyld be retaining module information? Probably not.
|
|
// FIXME: Should we be using the mangler for this? Probably.
|
|
//
|
|
// This is the accessor for the target address, so make sure to check the
|
|
// load address of the symbol, not the local address.
|
|
StringRef BaseName = F->getName();
|
|
if (BaseName[0] == '\1')
|
|
return (void*)Dyld.getSymbolLoadAddress(BaseName.substr(1));
|
|
return (void*)Dyld.getSymbolLoadAddress((TM->getMCAsmInfo()->getGlobalPrefix()
|
|
+ BaseName).str());
|
|
}
|
|
|
|
void *MCJIT::recompileAndRelinkFunction(Function *F) {
|
|
report_fatal_error("not yet implemented");
|
|
}
|
|
|
|
void MCJIT::freeMachineCodeForFunction(Function *F) {
|
|
report_fatal_error("not yet implemented");
|
|
}
|
|
|
|
GenericValue MCJIT::runFunction(Function *F,
|
|
const std::vector<GenericValue> &ArgValues) {
|
|
assert(F && "Function *F was null at entry to run()");
|
|
|
|
void *FPtr = getPointerToFunction(F);
|
|
assert(FPtr && "Pointer to fn's code was null after getPointerToFunction");
|
|
FunctionType *FTy = F->getFunctionType();
|
|
Type *RetTy = FTy->getReturnType();
|
|
|
|
assert((FTy->getNumParams() == ArgValues.size() ||
|
|
(FTy->isVarArg() && FTy->getNumParams() <= ArgValues.size())) &&
|
|
"Wrong number of arguments passed into function!");
|
|
assert(FTy->getNumParams() == ArgValues.size() &&
|
|
"This doesn't support passing arguments through varargs (yet)!");
|
|
|
|
// Handle some common cases first. These cases correspond to common `main'
|
|
// prototypes.
|
|
if (RetTy->isIntegerTy(32) || RetTy->isVoidTy()) {
|
|
switch (ArgValues.size()) {
|
|
case 3:
|
|
if (FTy->getParamType(0)->isIntegerTy(32) &&
|
|
FTy->getParamType(1)->isPointerTy() &&
|
|
FTy->getParamType(2)->isPointerTy()) {
|
|
int (*PF)(int, char **, const char **) =
|
|
(int(*)(int, char **, const char **))(intptr_t)FPtr;
|
|
|
|
// Call the function.
|
|
GenericValue rv;
|
|
rv.IntVal = APInt(32, PF(ArgValues[0].IntVal.getZExtValue(),
|
|
(char **)GVTOP(ArgValues[1]),
|
|
(const char **)GVTOP(ArgValues[2])));
|
|
return rv;
|
|
}
|
|
break;
|
|
case 2:
|
|
if (FTy->getParamType(0)->isIntegerTy(32) &&
|
|
FTy->getParamType(1)->isPointerTy()) {
|
|
int (*PF)(int, char **) = (int(*)(int, char **))(intptr_t)FPtr;
|
|
|
|
// Call the function.
|
|
GenericValue rv;
|
|
rv.IntVal = APInt(32, PF(ArgValues[0].IntVal.getZExtValue(),
|
|
(char **)GVTOP(ArgValues[1])));
|
|
return rv;
|
|
}
|
|
break;
|
|
case 1:
|
|
if (FTy->getNumParams() == 1 &&
|
|
FTy->getParamType(0)->isIntegerTy(32)) {
|
|
GenericValue rv;
|
|
int (*PF)(int) = (int(*)(int))(intptr_t)FPtr;
|
|
rv.IntVal = APInt(32, PF(ArgValues[0].IntVal.getZExtValue()));
|
|
return rv;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Handle cases where no arguments are passed first.
|
|
if (ArgValues.empty()) {
|
|
GenericValue rv;
|
|
switch (RetTy->getTypeID()) {
|
|
default: llvm_unreachable("Unknown return type for function call!");
|
|
case Type::IntegerTyID: {
|
|
unsigned BitWidth = cast<IntegerType>(RetTy)->getBitWidth();
|
|
if (BitWidth == 1)
|
|
rv.IntVal = APInt(BitWidth, ((bool(*)())(intptr_t)FPtr)());
|
|
else if (BitWidth <= 8)
|
|
rv.IntVal = APInt(BitWidth, ((char(*)())(intptr_t)FPtr)());
|
|
else if (BitWidth <= 16)
|
|
rv.IntVal = APInt(BitWidth, ((short(*)())(intptr_t)FPtr)());
|
|
else if (BitWidth <= 32)
|
|
rv.IntVal = APInt(BitWidth, ((int(*)())(intptr_t)FPtr)());
|
|
else if (BitWidth <= 64)
|
|
rv.IntVal = APInt(BitWidth, ((int64_t(*)())(intptr_t)FPtr)());
|
|
else
|
|
llvm_unreachable("Integer types > 64 bits not supported");
|
|
return rv;
|
|
}
|
|
case Type::VoidTyID:
|
|
rv.IntVal = APInt(32, ((int(*)())(intptr_t)FPtr)());
|
|
return rv;
|
|
case Type::FloatTyID:
|
|
rv.FloatVal = ((float(*)())(intptr_t)FPtr)();
|
|
return rv;
|
|
case Type::DoubleTyID:
|
|
rv.DoubleVal = ((double(*)())(intptr_t)FPtr)();
|
|
return rv;
|
|
case Type::X86_FP80TyID:
|
|
case Type::FP128TyID:
|
|
case Type::PPC_FP128TyID:
|
|
llvm_unreachable("long double not supported yet");
|
|
case Type::PointerTyID:
|
|
return PTOGV(((void*(*)())(intptr_t)FPtr)());
|
|
}
|
|
}
|
|
|
|
llvm_unreachable("Full-featured argument passing not supported yet!");
|
|
}
|
|
|
|
void *MCJIT::getPointerToNamedFunction(const std::string &Name,
|
|
bool AbortOnFailure) {
|
|
// FIXME: Add support for per-module compilation state
|
|
if (!isCompiled)
|
|
emitObject(M);
|
|
|
|
if (!isSymbolSearchingDisabled() && MemMgr) {
|
|
void *ptr = MemMgr->getPointerToNamedFunction(Name, false);
|
|
if (ptr)
|
|
return ptr;
|
|
}
|
|
|
|
/// If a LazyFunctionCreator is installed, use it to get/create the function.
|
|
if (LazyFunctionCreator)
|
|
if (void *RP = LazyFunctionCreator(Name))
|
|
return RP;
|
|
|
|
if (AbortOnFailure) {
|
|
report_fatal_error("Program used external function '"+Name+
|
|
"' which could not be resolved!");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void MCJIT::RegisterJITEventListener(JITEventListener *L) {
|
|
if (L == NULL)
|
|
return;
|
|
MutexGuard locked(lock);
|
|
EventListeners.push_back(L);
|
|
}
|
|
void MCJIT::UnregisterJITEventListener(JITEventListener *L) {
|
|
if (L == NULL)
|
|
return;
|
|
MutexGuard locked(lock);
|
|
SmallVector<JITEventListener*, 2>::reverse_iterator I=
|
|
std::find(EventListeners.rbegin(), EventListeners.rend(), L);
|
|
if (I != EventListeners.rend()) {
|
|
std::swap(*I, EventListeners.back());
|
|
EventListeners.pop_back();
|
|
}
|
|
}
|
|
void MCJIT::NotifyObjectEmitted(const ObjectImage& Obj) {
|
|
MutexGuard locked(lock);
|
|
for (unsigned I = 0, S = EventListeners.size(); I < S; ++I) {
|
|
EventListeners[I]->NotifyObjectEmitted(Obj);
|
|
}
|
|
}
|
|
void MCJIT::NotifyFreeingObject(const ObjectImage& Obj) {
|
|
MutexGuard locked(lock);
|
|
for (unsigned I = 0, S = EventListeners.size(); I < S; ++I) {
|
|
EventListeners[I]->NotifyFreeingObject(Obj);
|
|
}
|
|
}
|