forked from OSchip/llvm-project
171 lines
5.8 KiB
C++
171 lines
5.8 KiB
C++
//===---- llvm-jitlink.h - Session and format-specific decls ----*- C++ -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// llvm-jitlink Session class and tool utilities.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_TOOLS_LLVM_JITLINK_LLVM_JITLINK_H
|
|
#define LLVM_TOOLS_LLVM_JITLINK_LLVM_JITLINK_H
|
|
|
|
#include "llvm/ADT/Optional.h"
|
|
#include "llvm/ADT/StringSet.h"
|
|
#include "llvm/ADT/Triple.h"
|
|
#include "llvm/ExecutionEngine/Orc/Core.h"
|
|
#include "llvm/ExecutionEngine/Orc/ObjectLinkingLayer.h"
|
|
#include "llvm/ExecutionEngine/Orc/OrcRPCTargetProcessControl.h"
|
|
#include "llvm/ExecutionEngine/Orc/Shared/FDRawByteChannel.h"
|
|
#include "llvm/ExecutionEngine/Orc/Shared/RPCUtils.h"
|
|
#include "llvm/ExecutionEngine/Orc/TargetProcessControl.h"
|
|
#include "llvm/ExecutionEngine/RuntimeDyldChecker.h"
|
|
#include "llvm/Support/Error.h"
|
|
#include "llvm/Support/Regex.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <vector>
|
|
|
|
namespace llvm {
|
|
|
|
struct Session;
|
|
|
|
/// ObjectLinkingLayer with additional support for symbol promotion.
|
|
class LLVMJITLinkObjectLinkingLayer : public orc::ObjectLinkingLayer {
|
|
public:
|
|
using orc::ObjectLinkingLayer::add;
|
|
|
|
LLVMJITLinkObjectLinkingLayer(Session &S,
|
|
jitlink::JITLinkMemoryManager &MemMgr);
|
|
|
|
Error add(orc::ResourceTrackerSP RT,
|
|
std::unique_ptr<MemoryBuffer> O) override;
|
|
|
|
private:
|
|
Session &S;
|
|
};
|
|
|
|
using LLVMJITLinkChannel = orc::shared::FDRawByteChannel;
|
|
using LLVMJITLinkRPCEndpoint =
|
|
orc::shared::MultiThreadedRPCEndpoint<LLVMJITLinkChannel>;
|
|
using LLVMJITLinkRemoteMemoryAccess =
|
|
orc::OrcRPCTPCMemoryAccess<LLVMJITLinkRPCEndpoint>;
|
|
|
|
class LLVMJITLinkRemoteTargetProcessControl
|
|
: public orc::OrcRPCTargetProcessControlBase<LLVMJITLinkRPCEndpoint> {
|
|
public:
|
|
using BaseT = orc::OrcRPCTargetProcessControlBase<LLVMJITLinkRPCEndpoint>;
|
|
static Expected<std::unique_ptr<TargetProcessControl>> LaunchExecutor();
|
|
|
|
static Expected<std::unique_ptr<TargetProcessControl>> ConnectToExecutor();
|
|
|
|
Error disconnect() override;
|
|
|
|
private:
|
|
using LLVMJITLinkRemoteMemoryAccess =
|
|
orc::OrcRPCTPCMemoryAccess<LLVMJITLinkRemoteTargetProcessControl>;
|
|
|
|
using LLVMJITLinkRemoteMemoryManager =
|
|
orc::OrcRPCTPCJITLinkMemoryManager<LLVMJITLinkRemoteTargetProcessControl>;
|
|
|
|
LLVMJITLinkRemoteTargetProcessControl(
|
|
std::shared_ptr<orc::SymbolStringPool> SSP,
|
|
std::unique_ptr<LLVMJITLinkChannel> Channel,
|
|
std::unique_ptr<LLVMJITLinkRPCEndpoint> Endpoint,
|
|
ErrorReporter ReportError, Error &Err)
|
|
: BaseT(std::move(SSP), *Endpoint, std::move(ReportError)),
|
|
Channel(std::move(Channel)), Endpoint(std::move(Endpoint)) {
|
|
ErrorAsOutParameter _(&Err);
|
|
|
|
ListenerThread = std::thread([&]() {
|
|
while (!Finished) {
|
|
if (auto Err = this->Endpoint->handleOne()) {
|
|
reportError(std::move(Err));
|
|
return;
|
|
}
|
|
}
|
|
});
|
|
|
|
if (auto Err2 = initializeORCRPCTPCBase()) {
|
|
Err = joinErrors(std::move(Err2), disconnect());
|
|
return;
|
|
}
|
|
|
|
OwnedMemAccess = std::make_unique<LLVMJITLinkRemoteMemoryAccess>(*this);
|
|
MemAccess = OwnedMemAccess.get();
|
|
OwnedMemMgr = std::make_unique<LLVMJITLinkRemoteMemoryManager>(*this);
|
|
MemMgr = OwnedMemMgr.get();
|
|
}
|
|
|
|
std::unique_ptr<LLVMJITLinkChannel> Channel;
|
|
std::unique_ptr<LLVMJITLinkRPCEndpoint> Endpoint;
|
|
std::unique_ptr<TargetProcessControl::MemoryAccess> OwnedMemAccess;
|
|
std::unique_ptr<jitlink::JITLinkMemoryManager> OwnedMemMgr;
|
|
std::atomic<bool> Finished{false};
|
|
std::thread ListenerThread;
|
|
};
|
|
|
|
struct Session {
|
|
std::unique_ptr<orc::TargetProcessControl> TPC;
|
|
orc::ExecutionSession ES;
|
|
orc::JITDylib *MainJD;
|
|
LLVMJITLinkObjectLinkingLayer ObjLayer;
|
|
std::vector<orc::JITDylib *> JDSearchOrder;
|
|
|
|
~Session();
|
|
|
|
static Expected<std::unique_ptr<Session>> Create(Triple TT);
|
|
void dumpSessionInfo(raw_ostream &OS);
|
|
void modifyPassConfig(const Triple &FTT,
|
|
jitlink::PassConfiguration &PassConfig);
|
|
|
|
using MemoryRegionInfo = RuntimeDyldChecker::MemoryRegionInfo;
|
|
|
|
struct FileInfo {
|
|
StringMap<MemoryRegionInfo> SectionInfos;
|
|
StringMap<MemoryRegionInfo> StubInfos;
|
|
StringMap<MemoryRegionInfo> GOTEntryInfos;
|
|
};
|
|
|
|
using SymbolInfoMap = StringMap<MemoryRegionInfo>;
|
|
using FileInfoMap = StringMap<FileInfo>;
|
|
|
|
Expected<FileInfo &> findFileInfo(StringRef FileName);
|
|
Expected<MemoryRegionInfo &> findSectionInfo(StringRef FileName,
|
|
StringRef SectionName);
|
|
Expected<MemoryRegionInfo &> findStubInfo(StringRef FileName,
|
|
StringRef TargetName);
|
|
Expected<MemoryRegionInfo &> findGOTEntryInfo(StringRef FileName,
|
|
StringRef TargetName);
|
|
|
|
bool isSymbolRegistered(StringRef Name);
|
|
Expected<MemoryRegionInfo &> findSymbolInfo(StringRef SymbolName,
|
|
Twine ErrorMsgStem);
|
|
|
|
SymbolInfoMap SymbolInfos;
|
|
FileInfoMap FileInfos;
|
|
uint64_t SizeBeforePruning = 0;
|
|
uint64_t SizeAfterFixups = 0;
|
|
|
|
StringSet<> HarnessFiles;
|
|
StringSet<> HarnessExternals;
|
|
StringSet<> HarnessDefinitions;
|
|
DenseMap<StringRef, StringRef> CanonicalWeakDefs;
|
|
|
|
private:
|
|
Session(std::unique_ptr<orc::TargetProcessControl> TPC, Error &Err);
|
|
};
|
|
|
|
/// Record symbols, GOT entries, stubs, and sections for ELF file.
|
|
Error registerELFGraphInfo(Session &S, jitlink::LinkGraph &G);
|
|
|
|
/// Record symbols, GOT entries, stubs, and sections for MachO file.
|
|
Error registerMachOGraphInfo(Session &S, jitlink::LinkGraph &G);
|
|
|
|
} // end namespace llvm
|
|
|
|
#endif // LLVM_TOOLS_LLVM_JITLINK_LLVM_JITLINK_H
|