blob: edbcce60b05e6be657347a8bf99976e6eba7e951 [file] [log] [blame] [edit]
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#include "platform/globals.h"
#if defined(DART_HOST_OS_ANDROID)
#include <errno.h> // NOLINT
#include <sys/syscall.h> // NOLINT
#include "vm/flags.h"
#include "vm/os.h"
#include "vm/profiler.h"
#include "vm/signal_handler.h"
#include "vm/thread_interrupter.h"
namespace dart {
#if defined(DART_INCLUDE_PROFILER)
// Old linux kernels on ARM might require a trampoline to
// work around incorrect Thumb -> ARM transitions.
// See thread_interrupted_android_arm.S for more details.
#if defined(HOST_ARCH_ARM) && \
(defined(DART_HOST_OS_LINUX) || defined(DART_HOST_OS_ANDROID)) && \
!defined(__thumb__)
#define USE_SIGNAL_HANDLER_TRAMPOLINE
#endif
DECLARE_FLAG(bool, trace_thread_interrupter);
class SignalState : public AllStatic {
public:
static bool Start() {
uword expected = state_.load(std::memory_order_relaxed);
uword desired;
do {
if (!Enabled::decode(expected)) return false;
desired = Pending::update(Pending::decode(expected) + 1, expected);
} while (!state_.compare_exchange_weak(expected, desired,
std::memory_order_relaxed));
return true;
}
static void End() {
uword expected = state_.load(std::memory_order_relaxed);
uword desired;
do {
intptr_t pending = Pending::decode(expected);
ASSERT(pending > 0);
desired = Pending::update(pending - 1, expected);
} while (!state_.compare_exchange_weak(expected, desired,
std::memory_order_relaxed));
}
static void Enable() {
uword expected = state_.load(std::memory_order_relaxed);
ASSERT(expected == (Enabled::encode(false) | Pending::encode(0)));
uword desired = Enabled::encode(true) | Pending::encode(0);
bool success = state_.compare_exchange_strong(expected, desired,
std::memory_order_relaxed);
ASSERT(success);
}
static void Disable() {
ASSERT(Enabled::decode(state_.load(std::memory_order_relaxed)));
uword expected;
uword desired = Enabled::encode(false) | Pending::encode(0);
do {
// Failed CAS updates [expected], recompute.
expected = Enabled::encode(true) | Pending::encode(0);
} while (!state_.compare_exchange_weak(expected, desired,
std::memory_order_relaxed));
}
private:
using Enabled = BitField<uword, bool, 0, 1>;
using Pending =
BitField<uword, intptr_t, Enabled::kNextBit, kBitsPerWord - 1>;
static std::atomic<uword> state_;
};
std::atomic<uword> SignalState::state_ = {Enabled::encode(false) |
Pending::encode(0)};
namespace {
#if defined(USE_SIGNAL_HANDLER_TRAMPOLINE)
extern "C" {
#endif
void ThreadInterruptSignalHandler(int signal, siginfo_t* info, void* context_) {
if (signal != SIGPROF) {
return;
}
Thread* thread = Thread::Current();
if (thread == nullptr) {
return;
}
if (!SignalState::Start()) {
return;
}
ThreadInterruptScope signal_handler_scope;
// Extract thread state.
ucontext_t* context = reinterpret_cast<ucontext_t*>(context_);
mcontext_t mcontext = context->uc_mcontext;
InterruptedThreadState its;
its.pc = SignalHandler::GetProgramCounter(mcontext);
its.fp = SignalHandler::GetFramePointer(mcontext);
its.csp = SignalHandler::GetCStackPointer(mcontext);
its.dsp = SignalHandler::GetDartStackPointer(mcontext);
its.lr = SignalHandler::GetLinkRegister(mcontext);
Profiler::SampleThread(thread, its);
SignalState::End();
}
#if defined(USE_SIGNAL_HANDLER_TRAMPOLINE)
} // extern "C"
#endif
} // namespace
void ThreadInterrupter::InterruptThread(OSThread* thread) {
if (FLAG_trace_thread_interrupter) {
OS::PrintErr("ThreadInterrupter interrupting %p\n",
reinterpret_cast<void*>(thread->id()));
}
int result = pthread_kill(thread->id(), SIGPROF);
ASSERT((result == 0) || (result == ESRCH));
}
#if defined(USE_SIGNAL_HANDLER_TRAMPOLINE)
// Defined in thread_interrupted_android_arm.S
extern "C" void ThreadInterruptSignalHandlerTrampoline(int signal,
siginfo_t* info,
void* context_);
#endif
void ThreadInterrupter::InstallSignalHandler() {
#if defined(USE_SIGNAL_HANDLER_TRAMPOLINE)
SignalHandler::Install(&ThreadInterruptSignalHandlerTrampoline);
#else
SignalHandler::Install(&ThreadInterruptSignalHandler);
#endif
SignalState::Enable();
}
void ThreadInterrupter::RemoveSignalHandler() {
SignalState::Disable();
SignalHandler::Remove();
}
void* ThreadInterrupter::PrepareCurrentThread() {
return SignalHandler::PrepareCurrentThread();
}
void ThreadInterrupter::CleanupCurrentThreadState(void* state) {
SignalHandler::CleanupCurrentThreadState(state);
}
#endif // defined(DART_INCLUDE_PROFILER)
} // namespace dart
#endif // defined(DART_HOST_OS_ANDROID)