153 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			153 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
//===-- RemoteJITUtils.h - Utilities for remote-JITing with LLI -*- C++ -*-===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// Utilities for remote-JITing with LLI.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#ifndef LLVM_TOOLS_LLI_REMOTEJITUTILS_H
 | 
						|
#define LLVM_TOOLS_LLI_REMOTEJITUTILS_H
 | 
						|
 | 
						|
#include "llvm/ExecutionEngine/Orc/RPCChannel.h"
 | 
						|
#include "llvm/ExecutionEngine/RTDyldMemoryManager.h"
 | 
						|
#include <mutex>
 | 
						|
 | 
						|
#if !defined(_MSC_VER) && !defined(__MINGW32__)
 | 
						|
#include <unistd.h>
 | 
						|
#else
 | 
						|
#include <io.h>
 | 
						|
#endif
 | 
						|
 | 
						|
/// RPC channel that reads from and writes from file descriptors.
 | 
						|
class FDRPCChannel final : public llvm::orc::remote::RPCChannel {
 | 
						|
public:
 | 
						|
  FDRPCChannel(int InFD, int OutFD) : InFD(InFD), OutFD(OutFD) {}
 | 
						|
 | 
						|
  llvm::Error readBytes(char *Dst, unsigned Size) override {
 | 
						|
    assert(Dst && "Attempt to read into null.");
 | 
						|
    ssize_t Completed = 0;
 | 
						|
    while (Completed < static_cast<ssize_t>(Size)) {
 | 
						|
      ssize_t Read = ::read(InFD, Dst + Completed, Size - Completed);
 | 
						|
      if (Read <= 0) {
 | 
						|
        auto ErrNo = errno;
 | 
						|
        if (ErrNo == EAGAIN || ErrNo == EINTR)
 | 
						|
          continue;
 | 
						|
        else
 | 
						|
          return llvm::errorCodeToError(
 | 
						|
                   std::error_code(errno, std::generic_category()));
 | 
						|
      }
 | 
						|
      Completed += Read;
 | 
						|
    }
 | 
						|
    return llvm::Error::success();
 | 
						|
  }
 | 
						|
 | 
						|
  llvm::Error appendBytes(const char *Src, unsigned Size) override {
 | 
						|
    assert(Src && "Attempt to append from null.");
 | 
						|
    ssize_t Completed = 0;
 | 
						|
    while (Completed < static_cast<ssize_t>(Size)) {
 | 
						|
      ssize_t Written = ::write(OutFD, Src + Completed, Size - Completed);
 | 
						|
      if (Written < 0) {
 | 
						|
        auto ErrNo = errno;
 | 
						|
        if (ErrNo == EAGAIN || ErrNo == EINTR)
 | 
						|
          continue;
 | 
						|
        else
 | 
						|
          return llvm::errorCodeToError(
 | 
						|
                   std::error_code(errno, std::generic_category()));
 | 
						|
      }
 | 
						|
      Completed += Written;
 | 
						|
    }
 | 
						|
    return llvm::Error::success();
 | 
						|
  }
 | 
						|
 | 
						|
  llvm::Error send() override { return llvm::Error::success(); }
 | 
						|
 | 
						|
private:
 | 
						|
  int InFD, OutFD;
 | 
						|
};
 | 
						|
 | 
						|
// launch the remote process (see lli.cpp) and return a channel to it.
 | 
						|
std::unique_ptr<FDRPCChannel> launchRemote();
 | 
						|
 | 
						|
namespace llvm {
 | 
						|
 | 
						|
// ForwardingMM - Adapter to connect MCJIT to Orc's Remote memory manager.
 | 
						|
class ForwardingMemoryManager : public llvm::RTDyldMemoryManager {
 | 
						|
public:
 | 
						|
  void setMemMgr(std::unique_ptr<RuntimeDyld::MemoryManager> MemMgr) {
 | 
						|
    this->MemMgr = std::move(MemMgr);
 | 
						|
  }
 | 
						|
 | 
						|
  void setResolver(std::unique_ptr<RuntimeDyld::SymbolResolver> Resolver) {
 | 
						|
    this->Resolver = std::move(Resolver);
 | 
						|
  }
 | 
						|
 | 
						|
  uint8_t *allocateCodeSection(uintptr_t Size, unsigned Alignment,
 | 
						|
                               unsigned SectionID,
 | 
						|
                               StringRef SectionName) override {
 | 
						|
    return MemMgr->allocateCodeSection(Size, Alignment, SectionID, SectionName);
 | 
						|
  }
 | 
						|
 | 
						|
  uint8_t *allocateDataSection(uintptr_t Size, unsigned Alignment,
 | 
						|
                               unsigned SectionID, StringRef SectionName,
 | 
						|
                               bool IsReadOnly) override {
 | 
						|
    return MemMgr->allocateDataSection(Size, Alignment, SectionID, SectionName,
 | 
						|
                                       IsReadOnly);
 | 
						|
  }
 | 
						|
 | 
						|
  void reserveAllocationSpace(uintptr_t CodeSize, uint32_t CodeAlign,
 | 
						|
                              uintptr_t RODataSize, uint32_t RODataAlign,
 | 
						|
                              uintptr_t RWDataSize,
 | 
						|
                              uint32_t RWDataAlign) override {
 | 
						|
    MemMgr->reserveAllocationSpace(CodeSize, CodeAlign, RODataSize, RODataAlign,
 | 
						|
                                   RWDataSize, RWDataAlign);
 | 
						|
  }
 | 
						|
 | 
						|
  bool needsToReserveAllocationSpace() override {
 | 
						|
    return MemMgr->needsToReserveAllocationSpace();
 | 
						|
  }
 | 
						|
 | 
						|
  void registerEHFrames(uint8_t *Addr, uint64_t LoadAddr,
 | 
						|
                        size_t Size) override {
 | 
						|
    MemMgr->registerEHFrames(Addr, LoadAddr, Size);
 | 
						|
  }
 | 
						|
 | 
						|
  void deregisterEHFrames(uint8_t *Addr, uint64_t LoadAddr,
 | 
						|
                          size_t Size) override {
 | 
						|
    MemMgr->deregisterEHFrames(Addr, LoadAddr, Size);
 | 
						|
  }
 | 
						|
 | 
						|
  bool finalizeMemory(std::string *ErrMsg = nullptr) override {
 | 
						|
    return MemMgr->finalizeMemory(ErrMsg);
 | 
						|
  }
 | 
						|
 | 
						|
  void notifyObjectLoaded(RuntimeDyld &RTDyld,
 | 
						|
                          const object::ObjectFile &Obj) override {
 | 
						|
    MemMgr->notifyObjectLoaded(RTDyld, Obj);
 | 
						|
  }
 | 
						|
 | 
						|
  // Don't hide the sibling notifyObjectLoaded from RTDyldMemoryManager.
 | 
						|
  using RTDyldMemoryManager::notifyObjectLoaded;
 | 
						|
 | 
						|
  RuntimeDyld::SymbolInfo findSymbol(const std::string &Name) override {
 | 
						|
    return Resolver->findSymbol(Name);
 | 
						|
  }
 | 
						|
 | 
						|
  RuntimeDyld::SymbolInfo
 | 
						|
  findSymbolInLogicalDylib(const std::string &Name) override {
 | 
						|
    return Resolver->findSymbolInLogicalDylib(Name);
 | 
						|
  }
 | 
						|
 | 
						|
private:
 | 
						|
  std::unique_ptr<RuntimeDyld::MemoryManager> MemMgr;
 | 
						|
  std::unique_ptr<RuntimeDyld::SymbolResolver> Resolver;
 | 
						|
};
 | 
						|
}
 | 
						|
 | 
						|
#endif
 |