#include "llvm/ExecutionEngine/Orc/TargetProcess/JITLoaderVTune.h"
#include "llvm/ExecutionEngine/Orc/Shared/VTuneSharedStructs.h"
#include <map>
#if LLVM_USE_INTEL_JITEVENTS
#include "IntelJITEventsWrapper.h"
#include "ittnotify.h"
using namespace llvm;
using namespace llvm::orc;
namespace {
class JITEventWrapper {
public:
static std::unique_ptr<IntelJITEventsWrapper> Wrapper;
};
std::unique_ptr<IntelJITEventsWrapper> JITEventWrapper::Wrapper;
}
static Error registerJITLoaderVTuneRegisterImpl(const VTuneMethodBatch &MB) {
const size_t StringsSize = MB.Strings.size();
for (const auto &MethodInfo : MB.Methods) {
iJIT_Method_Load MethodMessage;
memset(&MethodMessage, 0, sizeof(iJIT_Method_Load));
MethodMessage.method_id = MethodInfo.MethodID;
if (MethodInfo.NameSI != 0 && MethodInfo.NameSI < StringsSize) {
MethodMessage.method_name =
const_cast<char *>(MB.Strings.at(MethodInfo.NameSI).data());
} else {
MethodMessage.method_name = NULL;
}
if (MethodInfo.ClassFileSI != 0 && MethodInfo.ClassFileSI < StringsSize) {
MethodMessage.class_file_name =
const_cast<char *>(MB.Strings.at(MethodInfo.ClassFileSI).data());
} else {
MethodMessage.class_file_name = NULL;
}
if (MethodInfo.SourceFileSI != 0 && MethodInfo.SourceFileSI < StringsSize) {
MethodMessage.source_file_name =
const_cast<char *>(MB.Strings.at(MethodInfo.SourceFileSI).data());
} else {
MethodMessage.source_file_name = NULL;
}
MethodMessage.method_load_address = MethodInfo.LoadAddr.toPtr<void *>();
MethodMessage.method_size = MethodInfo.LoadSize;
MethodMessage.class_id = 0;
MethodMessage.user_data = NULL;
MethodMessage.user_data_size = 0;
MethodMessage.env = iJDE_JittingAPI;
std::vector<LineNumberInfo> LineInfo;
for (const auto &LInfo : MethodInfo.LineTable) {
LineInfo.push_back(LineNumberInfo{LInfo.first, LInfo.second});
}
if (LineInfo.size() == 0) {
MethodMessage.line_number_size = 0;
MethodMessage.line_number_table = 0;
} else {
MethodMessage.line_number_size = LineInfo.size();
MethodMessage.line_number_table = &*LineInfo.begin();
}
JITEventWrapper::Wrapper->iJIT_NotifyEvent(
iJVM_EVENT_TYPE_METHOD_LOAD_FINISHED, &MethodMessage);
}
return Error::success();
}
static void registerJITLoaderVTuneUnregisterImpl(
const std::vector<std::pair<uint64_t, uint64_t>> &UM) {
for (auto &Method : UM) {
JITEventWrapper::Wrapper->iJIT_NotifyEvent(
iJVM_EVENT_TYPE_METHOD_UNLOAD_START,
const_cast<uint64_t *>(&Method.first));
}
}
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_registerVTuneImpl(const char *Data, uint64_t Size) {
using namespace orc::shared;
if (!JITEventWrapper::Wrapper)
JITEventWrapper::Wrapper.reset(new IntelJITEventsWrapper);
return WrapperFunction<SPSError(SPSVTuneMethodBatch)>::handle(
Data, Size, registerJITLoaderVTuneRegisterImpl)
.release();
}
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_unregisterVTuneImpl(const char *Data, uint64_t Size) {
using namespace orc::shared;
return WrapperFunction<void(SPSVTuneUnloadedMethodIDs)>::handle(
Data, Size, registerJITLoaderVTuneUnregisterImpl)
.release();
}
namespace {
using SourceLocations = std::vector<std::pair<std::string, unsigned int>>;
using NativeCodeMap = std::map<uint64_t, SourceLocations>;
NativeCodeMap ReportedDebugFuncs;
}
static int NotifyEvent(iJIT_JVM_EVENT EventType, void *EventSpecificData) {
switch (EventType) {
case iJVM_EVENT_TYPE_METHOD_LOAD_FINISHED: {
if (!EventSpecificData) {
errs() << "Error: The JIT event listener did not provide a event data.";
return -1;
}
iJIT_Method_Load *msg = static_cast<iJIT_Method_Load *>(EventSpecificData);
ReportedDebugFuncs[msg->method_id];
outs() << "Method load [" << msg->method_id << "]: " << msg->method_name
<< ", Size = " << msg->method_size << "\n";
for (unsigned int i = 0; i < msg->line_number_size; ++i) {
if (!msg->line_number_table) {
errs() << "A function with a non-zero line count had no line table.";
return -1;
}
std::pair<std::string, unsigned int> loc(
std::string(msg->source_file_name),
msg->line_number_table[i].LineNumber);
ReportedDebugFuncs[msg->method_id].push_back(loc);
outs() << " Line info @ " << msg->line_number_table[i].Offset << ": "
<< msg->source_file_name << ", line "
<< msg->line_number_table[i].LineNumber << "\n";
}
outs() << "\n";
} break;
case iJVM_EVENT_TYPE_METHOD_UNLOAD_START: {
if (!EventSpecificData) {
errs() << "Error: The JIT event listener did not provide a event data.";
return -1;
}
unsigned int UnloadId =
*reinterpret_cast<unsigned int *>(EventSpecificData);
assert(1 == ReportedDebugFuncs.erase(UnloadId));
outs() << "Method unload [" << UnloadId << "]\n";
} break;
default:
break;
}
return 0;
}
static iJIT_IsProfilingActiveFlags IsProfilingActive(void) {
return iJIT_SAMPLING_ON;
}
static unsigned int GetNewMethodID(void) {
static unsigned int id = 0;
return ++id;
}
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_test_registerVTuneImpl(const char *Data, uint64_t Size) {
using namespace orc::shared;
JITEventWrapper::Wrapper.reset(new IntelJITEventsWrapper(
NotifyEvent, NULL, NULL, IsProfilingActive, 0, 0, GetNewMethodID));
return WrapperFunction<SPSError(SPSVTuneMethodBatch)>::handle(
Data, Size, registerJITLoaderVTuneRegisterImpl)
.release();
}
#else
usingnamespacellvm;
usingnamespacellvm::orc;
static Error unsupportedBatch(const VTuneMethodBatch &MB) { … }
static void unsuppported(const std::vector<std::pair<uint64_t, uint64_t>> &UM) { … }
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_registerVTuneImpl(const char *Data, uint64_t Size) { … }
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_unregisterVTuneImpl(const char *Data, uint64_t Size) { … }
extern "C" llvm::orc::shared::CWrapperFunctionResult
llvm_orc_test_registerVTuneImpl(const char *Data, uint64_t Size) { … }
#endif