2018-06-12 11:29:39 +08:00
|
|
|
//===-- xray_profiling.cc ---------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file is a part of XRay, a dynamic runtime instrumentation system.
|
|
|
|
//
|
|
|
|
// This is the implementation of a profiling handler.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include <memory>
|
2018-06-12 12:06:25 +08:00
|
|
|
#include <time.h>
|
2018-06-12 11:29:39 +08:00
|
|
|
|
|
|
|
#include "sanitizer_common/sanitizer_atomic.h"
|
|
|
|
#include "sanitizer_common/sanitizer_flags.h"
|
|
|
|
#include "xray/xray_interface.h"
|
|
|
|
#include "xray/xray_log_interface.h"
|
|
|
|
|
|
|
|
#include "xray_flags.h"
|
|
|
|
#include "xray_profile_collector.h"
|
|
|
|
#include "xray_profiling_flags.h"
|
|
|
|
#include "xray_recursion_guard.h"
|
|
|
|
#include "xray_tsc.h"
|
|
|
|
#include "xray_utils.h"
|
|
|
|
#include <pthread.h>
|
|
|
|
|
|
|
|
namespace __xray {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2018-06-12 12:06:25 +08:00
|
|
|
constexpr uptr XRayProfilingVersion = 0x20180424;
|
|
|
|
|
|
|
|
struct XRayProfilingFileHeader {
|
|
|
|
const u64 MagicBytes = 0x7872617970726f66; // Identifier for XRay profiling
|
|
|
|
// files 'xrayprof' in hex.
|
|
|
|
const uptr Version = XRayProfilingVersion;
|
|
|
|
uptr Timestamp = 0; // System time in nanoseconds.
|
|
|
|
uptr PID = 0; // Process ID.
|
|
|
|
};
|
|
|
|
|
2018-06-12 11:29:39 +08:00
|
|
|
atomic_sint32_t ProfilerLogFlushStatus = {
|
|
|
|
XRayLogFlushStatus::XRAY_LOG_NOT_FLUSHING};
|
|
|
|
|
|
|
|
atomic_sint32_t ProfilerLogStatus = {XRayLogInitStatus::XRAY_LOG_UNINITIALIZED};
|
|
|
|
|
|
|
|
SpinMutex ProfilerOptionsMutex;
|
|
|
|
|
|
|
|
struct alignas(64) ProfilingData {
|
|
|
|
FunctionCallTrie::Allocators *Allocators = nullptr;
|
|
|
|
FunctionCallTrie *FCT = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
static pthread_key_t ProfilingKey;
|
|
|
|
|
|
|
|
ProfilingData &getThreadLocalData() XRAY_NEVER_INSTRUMENT {
|
|
|
|
thread_local std::aligned_storage<sizeof(ProfilingData)>::type ThreadStorage;
|
|
|
|
if (pthread_getspecific(ProfilingKey) == NULL) {
|
|
|
|
new (&ThreadStorage) ProfilingData{};
|
|
|
|
pthread_setspecific(ProfilingKey, &ThreadStorage);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto &TLD = *reinterpret_cast<ProfilingData *>(&ThreadStorage);
|
|
|
|
|
|
|
|
// We need to check whether the global flag to finalizing/finalized has been
|
|
|
|
// switched. If it is, then we ought to not actually initialise the data.
|
|
|
|
auto Status = atomic_load(&ProfilerLogStatus, memory_order_acquire);
|
|
|
|
if (Status == XRayLogInitStatus::XRAY_LOG_FINALIZING ||
|
|
|
|
Status == XRayLogInitStatus::XRAY_LOG_FINALIZED)
|
|
|
|
return TLD;
|
|
|
|
|
|
|
|
// If we're live, then we re-initialize TLD if the pointers are not null.
|
|
|
|
if (UNLIKELY(TLD.Allocators == nullptr && TLD.FCT == nullptr)) {
|
|
|
|
TLD.Allocators = reinterpret_cast<FunctionCallTrie::Allocators *>(
|
|
|
|
InternalAlloc(sizeof(FunctionCallTrie::Allocators)));
|
|
|
|
new (TLD.Allocators) FunctionCallTrie::Allocators();
|
|
|
|
*TLD.Allocators = FunctionCallTrie::InitAllocators();
|
|
|
|
TLD.FCT = reinterpret_cast<FunctionCallTrie *>(
|
|
|
|
InternalAlloc(sizeof(FunctionCallTrie)));
|
|
|
|
new (TLD.FCT) FunctionCallTrie(*TLD.Allocators);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TLD;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
const char *profilingCompilerDefinedFlags() XRAY_NEVER_INSTRUMENT {
|
|
|
|
#ifdef XRAY_PROFILER_DEFAULT_OPTIONS
|
|
|
|
return SANITIZER_STRINGIFY(XRAY_PROFILER_DEFAULT_OPTIONS);
|
|
|
|
#else
|
|
|
|
return "";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
atomic_sint32_t ProfileFlushStatus = {
|
|
|
|
XRayLogFlushStatus::XRAY_LOG_NOT_FLUSHING};
|
|
|
|
|
|
|
|
XRayLogFlushStatus profilingFlush() XRAY_NEVER_INSTRUMENT {
|
|
|
|
if (atomic_load(&ProfilerLogStatus, memory_order_acquire) !=
|
|
|
|
XRayLogInitStatus::XRAY_LOG_FINALIZED) {
|
|
|
|
if (Verbosity())
|
|
|
|
Report("Not flushing profiles, profiling not been finalized.\n");
|
|
|
|
return XRayLogFlushStatus::XRAY_LOG_NOT_FLUSHING;
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 Result = XRayLogFlushStatus::XRAY_LOG_NOT_FLUSHING;
|
|
|
|
if (!atomic_compare_exchange_strong(&ProfilerLogFlushStatus, &Result,
|
|
|
|
XRayLogFlushStatus::XRAY_LOG_FLUSHING,
|
|
|
|
memory_order_acq_rel)) {
|
|
|
|
if (Verbosity())
|
|
|
|
Report("Not flushing profiles, implementation still finalizing.\n");
|
|
|
|
}
|
|
|
|
|
2018-06-12 12:06:25 +08:00
|
|
|
// At this point, we'll create the file that will contain the profile, but
|
|
|
|
// only if the options say so.
|
|
|
|
if (!profilingFlags()->no_flush) {
|
|
|
|
int Fd = -1;
|
|
|
|
Fd = getLogFD();
|
|
|
|
if (Fd == -1) {
|
|
|
|
if (__sanitizer::Verbosity())
|
|
|
|
Report(
|
|
|
|
"profiler: Failed to acquire a file descriptor, dropping data.\n");
|
|
|
|
} else {
|
|
|
|
XRayProfilingFileHeader Header;
|
|
|
|
Header.Timestamp = NanoTime();
|
|
|
|
Header.PID = internal_getpid();
|
|
|
|
retryingWriteAll(Fd, reinterpret_cast<const char *>(&Header),
|
|
|
|
reinterpret_cast<const char *>(&Header) +
|
|
|
|
sizeof(Header));
|
|
|
|
|
|
|
|
// Now for each of the threads, write out the profile data as we would see
|
|
|
|
// it in memory, verbatim.
|
|
|
|
XRayBuffer B = profileCollectorService::nextBuffer({nullptr, 0});
|
|
|
|
while (B.Data != nullptr && B.Size != 0) {
|
|
|
|
retryingWriteAll(Fd, reinterpret_cast<const char *>(B.Data),
|
|
|
|
reinterpret_cast<const char *>(B.Data) + B.Size);
|
|
|
|
B = profileCollectorService::nextBuffer(B);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Then we close out the file.
|
|
|
|
internal_close(Fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-12 11:29:39 +08:00
|
|
|
profileCollectorService::reset();
|
|
|
|
|
|
|
|
atomic_store(&ProfilerLogStatus, XRayLogFlushStatus::XRAY_LOG_FLUSHED,
|
|
|
|
memory_order_release);
|
|
|
|
|
|
|
|
return XRayLogFlushStatus::XRAY_LOG_FLUSHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
thread_local atomic_uint8_t ReentranceGuard{0};
|
|
|
|
|
|
|
|
void postCurrentThreadFCT(ProfilingData &TLD) {
|
|
|
|
if (TLD.Allocators == nullptr || TLD.FCT == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
profileCollectorService::post(*TLD.FCT, GetTid());
|
|
|
|
TLD.FCT->~FunctionCallTrie();
|
|
|
|
TLD.Allocators->~Allocators();
|
|
|
|
InternalFree(TLD.FCT);
|
|
|
|
InternalFree(TLD.Allocators);
|
|
|
|
TLD.FCT = nullptr;
|
|
|
|
TLD.Allocators = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void profilingHandleArg0(int32_t FuncId,
|
|
|
|
XRayEntryType Entry) XRAY_NEVER_INSTRUMENT {
|
|
|
|
unsigned char CPU;
|
|
|
|
auto TSC = readTSC(CPU);
|
|
|
|
RecursionGuard G(ReentranceGuard);
|
|
|
|
if (!G)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto Status = atomic_load(&ProfilerLogStatus, memory_order_acquire);
|
|
|
|
auto &TLD = getThreadLocalData();
|
|
|
|
if (UNLIKELY(Status == XRayLogInitStatus::XRAY_LOG_FINALIZED ||
|
|
|
|
Status == XRayLogInitStatus::XRAY_LOG_FINALIZING)) {
|
|
|
|
postCurrentThreadFCT(TLD);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (Entry) {
|
|
|
|
case XRayEntryType::ENTRY:
|
|
|
|
case XRayEntryType::LOG_ARGS_ENTRY:
|
|
|
|
TLD.FCT->enterFunction(FuncId, TSC);
|
|
|
|
break;
|
|
|
|
case XRayEntryType::EXIT:
|
|
|
|
case XRayEntryType::TAIL:
|
|
|
|
TLD.FCT->exitFunction(FuncId, TSC);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// FIXME: Handle bugs.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void profilingHandleArg1(int32_t FuncId, XRayEntryType Entry,
|
|
|
|
uint64_t) XRAY_NEVER_INSTRUMENT {
|
|
|
|
return profilingHandleArg0(FuncId, Entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
XRayLogInitStatus profilingFinalize() XRAY_NEVER_INSTRUMENT {
|
|
|
|
s32 CurrentStatus = XRayLogInitStatus::XRAY_LOG_INITIALIZED;
|
|
|
|
if (!atomic_compare_exchange_strong(&ProfilerLogStatus, &CurrentStatus,
|
|
|
|
XRayLogInitStatus::XRAY_LOG_FINALIZING,
|
|
|
|
memory_order_release)) {
|
|
|
|
if (Verbosity())
|
|
|
|
Report("Cannot finalize profile, the profiling is not initialized.\n");
|
|
|
|
return static_cast<XRayLogInitStatus>(CurrentStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait a grace period to allow threads to see that we're finalizing.
|
|
|
|
SleepForMillis(profilingFlags()->grace_period_ms);
|
|
|
|
|
|
|
|
// We also want to make sure that the current thread's data is cleaned up,
|
|
|
|
// if we have any.
|
|
|
|
auto &TLD = getThreadLocalData();
|
|
|
|
postCurrentThreadFCT(TLD);
|
|
|
|
|
|
|
|
// Then we force serialize the log data.
|
|
|
|
profileCollectorService::serialize();
|
|
|
|
|
|
|
|
atomic_store(&ProfilerLogStatus, XRayLogInitStatus::XRAY_LOG_FINALIZED,
|
|
|
|
memory_order_release);
|
|
|
|
return XRayLogInitStatus::XRAY_LOG_FINALIZED;
|
|
|
|
}
|
|
|
|
|
|
|
|
XRayLogInitStatus
|
|
|
|
profilingLoggingInit(size_t BufferSize, size_t BufferMax, void *Options,
|
|
|
|
size_t OptionsSize) XRAY_NEVER_INSTRUMENT {
|
|
|
|
if (BufferSize != 0 || BufferMax != 0) {
|
|
|
|
if (Verbosity())
|
|
|
|
Report("__xray_log_init() being used, and is unsupported. Use "
|
|
|
|
"__xray_log_init_mode(...) instead. Bailing out.");
|
|
|
|
return XRayLogInitStatus::XRAY_LOG_UNINITIALIZED;
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 CurrentStatus = XRayLogInitStatus::XRAY_LOG_UNINITIALIZED;
|
|
|
|
if (!atomic_compare_exchange_strong(&ProfilerLogStatus, &CurrentStatus,
|
|
|
|
XRayLogInitStatus::XRAY_LOG_INITIALIZING,
|
|
|
|
memory_order_release)) {
|
|
|
|
if (Verbosity())
|
|
|
|
Report("Cannot initialize already initialised profiling "
|
|
|
|
"implementation.\n");
|
|
|
|
return static_cast<XRayLogInitStatus>(CurrentStatus);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
SpinMutexLock Lock(&ProfilerOptionsMutex);
|
|
|
|
FlagParser ConfigParser;
|
|
|
|
auto *F = profilingFlags();
|
|
|
|
F->setDefaults();
|
|
|
|
registerProfilerFlags(&ConfigParser, F);
|
2018-06-12 12:06:25 +08:00
|
|
|
ConfigParser.ParseString(profilingCompilerDefinedFlags());
|
|
|
|
const char *Env = GetEnv("XRAY_PROFILING_OPTIONS");
|
|
|
|
if (Env == nullptr)
|
|
|
|
Env = "";
|
|
|
|
ConfigParser.ParseString(Env);
|
|
|
|
|
|
|
|
// Then parse the configuration string provided.
|
2018-06-12 11:29:39 +08:00
|
|
|
ConfigParser.ParseString(static_cast<const char *>(Options));
|
|
|
|
if (Verbosity())
|
|
|
|
ReportUnrecognizedFlags();
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to reset the profile data collection implementation now.
|
|
|
|
profileCollectorService::reset();
|
|
|
|
|
|
|
|
// We need to set up the at-thread-exit handler.
|
|
|
|
static pthread_once_t Once = PTHREAD_ONCE_INIT;
|
|
|
|
pthread_once(&Once, +[] {
|
|
|
|
pthread_key_create(&ProfilingKey, +[](void *P) {
|
|
|
|
// This is the thread-exit handler.
|
|
|
|
auto &TLD = *reinterpret_cast<ProfilingData *>(P);
|
|
|
|
if (TLD.Allocators == nullptr && TLD.FCT == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
postCurrentThreadFCT(TLD);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
__xray_log_set_buffer_iterator(profileCollectorService::nextBuffer);
|
|
|
|
__xray_set_handler(profilingHandleArg0);
|
|
|
|
__xray_set_handler_arg1(profilingHandleArg1);
|
|
|
|
|
|
|
|
atomic_store(&ProfilerLogStatus, XRayLogInitStatus::XRAY_LOG_INITIALIZED,
|
|
|
|
memory_order_release);
|
|
|
|
if (Verbosity())
|
|
|
|
Report("XRay Profiling init successful.\n");
|
|
|
|
|
|
|
|
return XRayLogInitStatus::XRAY_LOG_INITIALIZED;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool profilingDynamicInitializer() XRAY_NEVER_INSTRUMENT {
|
|
|
|
// Set up the flag defaults from the static defaults and the
|
|
|
|
// compiler-provided defaults.
|
|
|
|
{
|
|
|
|
SpinMutexLock Lock(&ProfilerOptionsMutex);
|
|
|
|
auto *F = profilingFlags();
|
|
|
|
F->setDefaults();
|
|
|
|
FlagParser ProfilingParser;
|
|
|
|
registerProfilerFlags(&ProfilingParser, F);
|
2018-06-12 12:06:25 +08:00
|
|
|
ProfilingParser.ParseString(profilingCompilerDefinedFlags());
|
2018-06-12 11:29:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
XRayLogImpl Impl{
|
|
|
|
profilingLoggingInit,
|
|
|
|
profilingFinalize,
|
|
|
|
profilingHandleArg0,
|
|
|
|
profilingFlush,
|
|
|
|
};
|
|
|
|
auto RegistrationResult = __xray_log_register_mode("xray-profiling", Impl);
|
|
|
|
if (RegistrationResult != XRayLogRegisterStatus::XRAY_REGISTRATION_OK &&
|
|
|
|
Verbosity())
|
|
|
|
Report("Cannot register XRay Profiling mode to 'xray-profiling'; error = "
|
|
|
|
"%d\n",
|
|
|
|
RegistrationResult);
|
|
|
|
if (!internal_strcmp(flags()->xray_mode, "xray-profiling"))
|
|
|
|
__xray_set_log_impl(Impl);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace __xray
|
|
|
|
|
|
|
|
static auto UNUSED Unused = __xray::profilingDynamicInitializer();
|