2012-09-05 16:50:34 +00:00
|
|
|
//===- RemoteTarget.cpp - LLVM Remote process JIT execution --------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Implementation of the RemoteTarget class which executes JITed code in a
|
|
|
|
// separate address range from where it was built.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "RemoteTarget.h"
|
2013-10-02 17:12:36 +00:00
|
|
|
#include "RemoteTargetExternal.h"
|
2012-09-15 18:45:38 +00:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/Support/DataTypes.h"
|
|
|
|
#include "llvm/Support/Memory.h"
|
2012-09-05 16:50:34 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string>
|
2013-10-02 17:12:36 +00:00
|
|
|
|
2012-09-05 16:50:34 +00:00
|
|
|
using namespace llvm;
|
|
|
|
|
2013-10-02 17:12:36 +00:00
|
|
|
// Static methods
|
|
|
|
RemoteTarget *RemoteTarget::createRemoteTarget() {
|
|
|
|
return new RemoteTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
RemoteTarget *RemoteTarget::createExternalRemoteTarget(std::string &ChildName) {
|
|
|
|
#ifdef LLVM_ON_UNIX
|
|
|
|
return new RemoteTargetExternal(ChildName);
|
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RemoteTarget::hostSupportsExternalRemoteTarget() {
|
|
|
|
#ifdef LLVM_ON_UNIX
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Simulated remote execution
|
|
|
|
//
|
|
|
|
// This implementation will simply move generated code and data to a new memory
|
|
|
|
// location in the current executable and let it run from there.
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2012-09-05 16:50:34 +00:00
|
|
|
bool RemoteTarget::allocateSpace(size_t Size, unsigned Alignment,
|
|
|
|
uint64_t &Address) {
|
|
|
|
sys::MemoryBlock *Prev = Allocations.size() ? &Allocations.back() : NULL;
|
|
|
|
sys::MemoryBlock Mem = sys::Memory::AllocateRWX(Size, Prev, &ErrorMsg);
|
|
|
|
if (Mem.base() == NULL)
|
|
|
|
return true;
|
|
|
|
if ((uintptr_t)Mem.base() % Alignment) {
|
|
|
|
ErrorMsg = "unable to allocate sufficiently aligned memory";
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
Address = reinterpret_cast<uint64_t>(Mem.base());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RemoteTarget::loadData(uint64_t Address, const void *Data, size_t Size) {
|
|
|
|
memcpy ((void*)Address, Data, Size);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RemoteTarget::loadCode(uint64_t Address, const void *Data, size_t Size) {
|
|
|
|
memcpy ((void*)Address, Data, Size);
|
2012-10-31 20:37:14 +00:00
|
|
|
sys::MemoryBlock Mem((void*)Address, Size);
|
|
|
|
sys::Memory::setExecutable(Mem, &ErrorMsg);
|
2012-09-05 16:50:34 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RemoteTarget::executeCode(uint64_t Address, int &RetVal) {
|
|
|
|
int (*fn)(void) = (int(*)(void))Address;
|
|
|
|
RetVal = fn();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoteTarget::create() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoteTarget::stop() {
|
|
|
|
for (unsigned i = 0, e = Allocations.size(); i != e; ++i)
|
|
|
|
sys::Memory::ReleaseRWX(Allocations[i]);
|
|
|
|
}
|