Lang Hames f93df5ebd9
[ORC] Add read operations to orc::MemoryAccess. (#145834)
This commit adds operations to orc::MemoryAccess for reading basic types
(uint8_t, uint16_t, uint32_t, uint64_t, pointers, buffers, and strings)
from executor memory.

The InProcessMemoryAccess and EPCGenericMemoryAccess implementations are
updated to support the new operations.
2025-06-27 09:49:17 +10:00

199 lines
8.0 KiB
C++

//===------------------------ OrcRTBootstrap.cpp --------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
#include "OrcRTBootstrap.h"
#include "llvm/ExecutionEngine/Orc/Shared/OrcRTBridge.h"
#include "llvm/ExecutionEngine/Orc/Shared/WrapperFunctionUtils.h"
#include "llvm/ExecutionEngine/Orc/TargetProcess/RegisterEHFrames.h"
#include "llvm/ExecutionEngine/Orc/TargetProcess/TargetExecutionUtils.h"
#define DEBUG_TYPE "orc"
using namespace llvm::orc::shared;
namespace llvm {
namespace orc {
namespace rt_bootstrap {
template <typename WriteT, typename SPSWriteT>
static llvm::orc::shared::CWrapperFunctionResult
writeUIntsWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<void(SPSSequence<SPSWriteT>)>::handle(
ArgData, ArgSize,
[](std::vector<WriteT> Ws) {
for (auto &W : Ws)
*W.Addr.template toPtr<decltype(W.Value) *>() = W.Value;
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
writePointersWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<void(SPSSequence<SPSMemoryAccessPointerWrite>)>::
handle(ArgData, ArgSize,
[](std::vector<tpctypes::PointerWrite> Ws) {
for (auto &W : Ws)
*W.Addr.template toPtr<void **>() =
W.Value.template toPtr<void *>();
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
writeBuffersWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<void(SPSSequence<SPSMemoryAccessBufferWrite>)>::handle(
ArgData, ArgSize,
[](std::vector<tpctypes::BufferWrite> Ws) {
for (auto &W : Ws)
memcpy(W.Addr.template toPtr<char *>(), W.Buffer.data(),
W.Buffer.size());
})
.release();
}
template <typename ReadT>
static llvm::orc::shared::CWrapperFunctionResult
readUIntsWrapper(const char *ArgData, size_t ArgSize) {
using SPSSig = SPSSequence<ReadT>(SPSSequence<SPSExecutorAddr>);
return WrapperFunction<SPSSig>::handle(ArgData, ArgSize,
[](std::vector<ExecutorAddr> Rs) {
std::vector<ReadT> Result;
Result.reserve(Rs.size());
for (auto &R : Rs)
Result.push_back(
*R.toPtr<ReadT *>());
return Result;
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
readPointersWrapper(const char *ArgData, size_t ArgSize) {
using SPSSig = SPSSequence<SPSExecutorAddr>(SPSSequence<SPSExecutorAddr>);
return WrapperFunction<SPSSig>::handle(
ArgData, ArgSize,
[](std::vector<ExecutorAddr> Rs) {
std::vector<ExecutorAddr> Result;
Result.reserve(Rs.size());
for (auto &R : Rs)
Result.push_back(ExecutorAddr::fromPtr(*R.toPtr<void **>()));
return Result;
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
readBuffersWrapper(const char *ArgData, size_t ArgSize) {
using SPSSig =
SPSSequence<SPSSequence<uint8_t>>(SPSSequence<SPSExecutorAddrRange>);
return WrapperFunction<SPSSig>::handle(
ArgData, ArgSize,
[](std::vector<ExecutorAddrRange> Rs) {
std::vector<std::vector<uint8_t>> Result;
Result.reserve(Rs.size());
for (auto &R : Rs) {
Result.push_back({});
Result.back().resize(R.size());
memcpy(reinterpret_cast<char *>(Result.back().data()),
R.Start.toPtr<char *>(), R.size());
}
return Result;
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
readStringsWrapper(const char *ArgData, size_t ArgSize) {
using SPSSig = SPSSequence<SPSString>(SPSSequence<SPSExecutorAddr>);
return WrapperFunction<SPSSig>::handle(ArgData, ArgSize,
[](std::vector<ExecutorAddr> Rs) {
std::vector<std::string> Result;
Result.reserve(Rs.size());
for (auto &R : Rs)
Result.push_back(
R.toPtr<char *>());
return Result;
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
runAsMainWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<rt::SPSRunAsMainSignature>::handle(
ArgData, ArgSize,
[](ExecutorAddr MainAddr,
std::vector<std::string> Args) -> int64_t {
return runAsMain(MainAddr.toPtr<int (*)(int, char *[])>(), Args);
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
runAsVoidFunctionWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<rt::SPSRunAsVoidFunctionSignature>::handle(
ArgData, ArgSize,
[](ExecutorAddr MainAddr) -> int32_t {
return runAsVoidFunction(MainAddr.toPtr<int32_t (*)(void)>());
})
.release();
}
static llvm::orc::shared::CWrapperFunctionResult
runAsIntFunctionWrapper(const char *ArgData, size_t ArgSize) {
return WrapperFunction<rt::SPSRunAsIntFunctionSignature>::handle(
ArgData, ArgSize,
[](ExecutorAddr MainAddr, int32_t Arg) -> int32_t {
return runAsIntFunction(MainAddr.toPtr<int32_t (*)(int32_t)>(),
Arg);
})
.release();
}
void addTo(StringMap<ExecutorAddr> &M) {
M[rt::MemoryWriteUInt8sWrapperName] = ExecutorAddr::fromPtr(
&writeUIntsWrapper<tpctypes::UInt8Write,
shared::SPSMemoryAccessUInt8Write>);
M[rt::MemoryWriteUInt16sWrapperName] = ExecutorAddr::fromPtr(
&writeUIntsWrapper<tpctypes::UInt16Write,
shared::SPSMemoryAccessUInt16Write>);
M[rt::MemoryWriteUInt32sWrapperName] = ExecutorAddr::fromPtr(
&writeUIntsWrapper<tpctypes::UInt32Write,
shared::SPSMemoryAccessUInt32Write>);
M[rt::MemoryWriteUInt64sWrapperName] = ExecutorAddr::fromPtr(
&writeUIntsWrapper<tpctypes::UInt64Write,
shared::SPSMemoryAccessUInt64Write>);
M[rt::MemoryWritePointersWrapperName] =
ExecutorAddr::fromPtr(&writePointersWrapper);
M[rt::MemoryWriteBuffersWrapperName] =
ExecutorAddr::fromPtr(&writeBuffersWrapper);
M[rt::MemoryReadUInt8sWrapperName] =
ExecutorAddr::fromPtr(&readUIntsWrapper<uint8_t>);
M[rt::MemoryReadUInt16sWrapperName] =
ExecutorAddr::fromPtr(&readUIntsWrapper<uint16_t>);
M[rt::MemoryReadUInt32sWrapperName] =
ExecutorAddr::fromPtr(&readUIntsWrapper<uint32_t>);
M[rt::MemoryReadUInt64sWrapperName] =
ExecutorAddr::fromPtr(&readUIntsWrapper<uint64_t>);
M[rt::MemoryReadPointersWrapperName] =
ExecutorAddr::fromPtr(&readPointersWrapper);
M[rt::MemoryReadBuffersWrapperName] =
ExecutorAddr::fromPtr(&readBuffersWrapper);
M[rt::MemoryReadStringsWrapperName] =
ExecutorAddr::fromPtr(&readStringsWrapper);
M[rt::RunAsMainWrapperName] = ExecutorAddr::fromPtr(&runAsMainWrapper);
M[rt::RunAsVoidFunctionWrapperName] =
ExecutorAddr::fromPtr(&runAsVoidFunctionWrapper);
M[rt::RunAsIntFunctionWrapperName] =
ExecutorAddr::fromPtr(&runAsIntFunctionWrapper);
}
} // end namespace rt_bootstrap
} // end namespace orc
} // end namespace llvm