15#define DEBUG_TYPE "orc"
19namespace rt_bootstrap {
22 assert(Slabs.empty() &&
"shutdown not called?");
31 std::lock_guard<std::mutex> Lock(M);
32 assert(!Slabs.count(MB.base()) &&
"Duplicate allocation addr");
33 Slabs[MB.base()].Size =
Size;
45 "finalization request",
51 std::vector<sys::MemoryBlock> MBsToReset;
53 for (
auto &MB : MBsToReset)
63 RR.
End = std::max(RR.
End, Seg.Addr + Seg.Size);
68 formatv(
"Segment {0:x} content size ({1:x} bytes) "
69 "exceeds segment size ({2:x} bytes)",
70 Seg.Addr.getValue(), Seg.Content.size(), Seg.Size),
75 formatv(
"Segment {0:x} -- {1:x} crosses boundary of "
76 "allocation {2:x} -- {3:x}",
80 char *Mem = Seg.Addr.toPtr<
char *>();
81 if (!Seg.Content.empty())
82 memcpy(Mem, Seg.Content.data(), Seg.Content.size());
83 memset(Mem + Seg.Content.size(), 0, Seg.Size - Seg.Content.size());
84 assert(Seg.Size <= std::numeric_limits<size_t>::max());
91 MBsToReset.push_back(MB);
97 auto DeallocActions = runFinalizeActions(FR.
Actions);
99 return DeallocActions.takeError();
102 std::lock_guard<std::mutex> Lock(M);
103 auto Region = createRegionInfo(RR,
"In initialize");
105 return Region.takeError();
106 Region->DeallocActions = std::move(*DeallocActions);
116 const std::vector<ExecutorAddr> &InitKeys) {
120 std::vector<shared::WrapperFunctionCall> DeallocActions;
122 std::scoped_lock<std::mutex> Lock(M);
123 auto Slab = getSlabInfo(KeyAddr,
"In deinitialize");
125 Err =
joinErrors(std::move(Err), Slab.takeError());
129 auto RI = getRegionInfo(*Slab, KeyAddr,
"In deinitialize");
131 Err =
joinErrors(std::move(Err), RI.takeError());
135 DeallocActions = std::move(RI->DeallocActions);
139 runDeallocActions(std::move(DeallocActions)));
146 const std::vector<ExecutorAddr> &Bases) {
151 std::vector<shared::WrapperFunctionCall> DeallocActions;
155 std::scoped_lock<std::mutex> Lock(M);
157 auto SlabI = Slabs.find(
Base.toPtr<
void *>());
158 if (SlabI == Slabs.end()) {
162 " is not part of any reserved "
168 auto &Slab = SlabI->second;
170 for (
auto &[Addr,
Region] : Slab.Regions)
173 MB = {
Base.toPtr<
void *>(), Slab.Size};
178 Err =
joinErrors(std::move(Err), runDeallocActions(DeallocActions));
189 std::vector<ExecutorAddr> Bases;
191 std::scoped_lock<std::mutex> Lock(M);
192 for (
auto &[
Base, Slab] : Slabs)
214 auto MakeBadSlabError = [&]() {
216 Context +
", address " +
formatv(
"{0:x}",
A) +
217 " is not part of any reserved address range",
221 auto I = Slabs.upper_bound(
A.toPtr<
void *>());
222 if (
I == Slabs.begin())
223 return MakeBadSlabError();
227 return MakeBadSlabError();
233SimpleExecutorMemoryManager::getSlabInfo(ExecutorAddrRange R,
235 auto MakeBadSlabError = [&]() {
237 Context +
", range " +
formatv(
"{0:x}", R) +
238 " is not part of any reserved address range",
242 auto I = Slabs.upper_bound(R.Start.toPtr<
void *>());
243 if (
I == Slabs.begin())
244 return MakeBadSlabError();
248 return MakeBadSlabError();
253Expected<SimpleExecutorMemoryManager::RegionInfo &>
254SimpleExecutorMemoryManager::createRegionInfo(ExecutorAddrRange R,
257 auto Slab = getSlabInfo(R,
Context);
259 return Slab.takeError();
261 auto MakeBadRegionError = [&](ExecutorAddrRange
Other,
bool Prev) {
264 (Prev ?
"previous" :
"following") +
269 auto I = Slab->Regions.upper_bound(
R.Start);
270 if (
I != Slab->Regions.begin()) {
271 auto J = std::prev(
I);
272 ExecutorAddrRange PrevRange(J->first, J->second.Size);
273 if (PrevRange.overlaps(R))
274 return MakeBadRegionError(PrevRange,
true);
276 if (
I != Slab->Regions.end()) {
277 ExecutorAddrRange NextRange(
I->first,
I->second.Size);
278 if (NextRange.overlaps(R))
279 return MakeBadRegionError(NextRange,
false);
282 auto &RInfo = Slab->Regions[
R.Start];
283 RInfo.Size =
R.size();
287Expected<SimpleExecutorMemoryManager::RegionInfo &>
288SimpleExecutorMemoryManager::getRegionInfo(SlabInfo &Slab, ExecutorAddr
A,
290 auto I = Slab.Regions.find(
A);
291 if (
I == Slab.Regions.end())
294 " does not correspond to the start of any initialized region",
300Expected<SimpleExecutorMemoryManager::RegionInfo &>
301SimpleExecutorMemoryManager::getRegionInfo(ExecutorAddr
A, StringRef
Context) {
302 auto Slab = getSlabInfo(
A,
Context);
304 return Slab.takeError();
306 return getRegionInfo(*Slab,
A,
Context);
309llvm::orc::shared::CWrapperFunctionResult
310SimpleExecutorMemoryManager::reserveWrapper(
const char *ArgData,
312 return shared::WrapperFunction<rt::SPSSimpleRemoteMemoryMapReserveSignature>::
313 handle(ArgData, ArgSize,
319llvm::orc::shared::CWrapperFunctionResult
320SimpleExecutorMemoryManager::initializeWrapper(
const char *ArgData,
323 WrapperFunction<rt::SPSSimpleRemoteMemoryMapInitializeSignature>::handle(
330llvm::orc::shared::CWrapperFunctionResult
331SimpleExecutorMemoryManager::deinitializeWrapper(
const char *ArgData,
333 return shared::WrapperFunction<
335 handle(ArgData, ArgSize,
341llvm::orc::shared::CWrapperFunctionResult
342SimpleExecutorMemoryManager::releaseWrapper(
const char *ArgData,
344 return shared::WrapperFunction<rt::SPSSimpleRemoteMemoryMapReleaseSignature>::
345 handle(ArgData, ArgSize,
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
#define LLVM_UNLIKELY(EXPR)
This file defines the make_scope_exit function, which executes user-defined cleanup logic at scope ex...
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
StringRef - Represent a constant reference to a string, i.e.
Represents an address in the executor process.
static ExecutorAddr fromPtr(T *Ptr, UnwrapFn &&Unwrap=UnwrapFn())
Create an ExecutorAddr from the given pointer.
std::enable_if_t< std::is_pointer< T >::value, T > toPtr(WrapFn &&Wrap=WrapFn()) const
Cast this ExecutorAddr to a pointer of the given type.
Error deinitialize(const std::vector< ExecutorAddr > &InitKeys)
Error release(const std::vector< ExecutorAddr > &Bases)
~SimpleExecutorMemoryManager() override
Error shutdown() override
Expected< ExecutorAddr > reserve(uint64_t Size)
Expected< ExecutorAddr > initialize(tpctypes::FinalizeRequest &FR)
void addBootstrapSymbols(StringMap< ExecutorAddr > &M) override
This class encapsulates the notion of a memory block which has an address and a size.
static LLVM_ABI std::error_code protectMappedMemory(const MemoryBlock &Block, unsigned Flags)
This method sets the protection flags for a block of memory to the state specified by /p Flags.
static LLVM_ABI std::error_code releaseMappedMemory(MemoryBlock &Block)
This method releases a block of memory that was allocated with the allocateMappedMemory method.
static LLVM_ABI void InvalidateInstructionCache(const void *Addr, size_t Len)
InvalidateInstructionCache - Before the JIT can run a block of code that has been emitted it must inv...
static LLVM_ABI MemoryBlock allocateMappedMemory(size_t NumBytes, const MemoryBlock *const NearBlock, unsigned Flags, std::error_code &EC)
This method allocates a block of memory that is suitable for loading dynamically generated code (e....
LLVM_ABI const char * SimpleExecutorMemoryManagerInitializeWrapperName
LLVM_ABI const char * SimpleExecutorMemoryManagerReserveWrapperName
LLVM_ABI const char * SimpleExecutorMemoryManagerReleaseWrapperName
LLVM_ABI const char * SimpleExecutorMemoryManagerDeinitializeWrapperName
shared::SPSError( shared::SPSExecutorAddr, shared::SPSSequence< shared::SPSExecutorAddr >) SPSSimpleRemoteMemoryMapDeinitializeSignature
LLVM_ABI const char * SimpleExecutorMemoryManagerInstanceName
MethodWrapperHandler< RetT, ClassT, ArgTs... > makeMethodWrapperHandler(RetT(ClassT::*Method)(ArgTs...))
Create a MethodWrapperHandler object from the given method pointer.
uint64_t ExecutorAddrDiff
sys::Memory::ProtectionFlags toSysMemoryProtectionFlags(MemProt MP)
Convert a MemProt value to a corresponding sys::Memory::ProtectionFlags value.
This is an optimization pass for GlobalISel generic memory operations.
detail::scope_exit< std::decay_t< Callable > > make_scope_exit(Callable &&F)
LLVM_ABI std::error_code inconvertibleErrorCode()
The value returned by this function can be returned from convertToErrorCode for Error values where no...
auto formatv(bool Validate, const char *Fmt, Ts &&...Vals)
auto reverse(ContainerTy &&C)
Error joinErrors(Error E1, Error E2)
Concatenate errors.
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
OutputIt copy(R &&Range, OutputIt Out)
LLVM_ABI Error errorCodeToError(std::error_code EC)
Helper for converting an std::error_code to a Error.
Represents an address range in the exceutor process.
ExecutorAddrDiff size() const
bool contains(ExecutorAddr Addr) const
std::vector< SegFinalizeRequest > Segments
shared::AllocActions Actions