OLD | NEW |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 #include "platform/globals.h" | 5 #include "platform/globals.h" |
6 #if defined(TARGET_OS_MACOS) | 6 #if defined(TARGET_OS_MACOS) |
7 | 7 |
8 #include "vm/signal_handler.h" | 8 #include "vm/signal_handler.h" |
9 #include "vm/thread_interrupter.h" | 9 #include "vm/thread_interrupter.h" |
10 | 10 |
11 namespace dart { | 11 namespace dart { |
12 | 12 |
13 DECLARE_FLAG(bool, thread_interrupter); | 13 DECLARE_FLAG(bool, thread_interrupter); |
14 DECLARE_FLAG(bool, trace_thread_interrupter); | 14 DECLARE_FLAG(bool, trace_thread_interrupter); |
15 | 15 |
16 class ThreadInterrupterMacOS : public AllStatic { | 16 class ThreadInterrupterMacOS : public AllStatic { |
17 public: | 17 public: |
18 static void ThreadInterruptSignalHandler(int signal, siginfo_t* info, | 18 static void ThreadInterruptSignalHandler(int signal, siginfo_t* info, |
19 void* context_) { | 19 void* context_) { |
20 if (signal != SIGPROF) { | 20 if (signal != SIGPROF) { |
21 return; | 21 return; |
22 } | 22 } |
23 ThreadInterrupter::ThreadState* state = | 23 InterruptableThreadState* state = ThreadInterrupter::CurrentThreadState(); |
24 ThreadInterrupter::CurrentThreadState(); | |
25 if ((state == NULL) || (state->callback == NULL)) { | 24 if ((state == NULL) || (state->callback == NULL)) { |
26 // No interrupter state or callback. | 25 // No interrupter state or callback. |
27 return; | 26 return; |
28 } | 27 } |
29 ASSERT(Thread::Compare(state->id, Thread::GetCurrentThreadId())); | 28 ASSERT(Thread::Compare(state->id, Thread::GetCurrentThreadId())); |
30 | |
31 // Extract thread state. | 29 // Extract thread state. |
32 ucontext_t* context = reinterpret_cast<ucontext_t*>(context_); | 30 ucontext_t* context = reinterpret_cast<ucontext_t*>(context_); |
33 mcontext_t mcontext = context->uc_mcontext; | 31 mcontext_t mcontext = context->uc_mcontext; |
34 InterruptedThreadState its; | 32 InterruptedThreadState its; |
35 its.tid = state->id; | 33 its.tid = state->id; |
36 its.pc = SignalHandler::GetProgramCounter(mcontext); | 34 its.pc = SignalHandler::GetProgramCounter(mcontext); |
37 its.fp = SignalHandler::GetFramePointer(mcontext); | 35 its.fp = SignalHandler::GetFramePointer(mcontext); |
38 its.sp = SignalHandler::GetStackPointer(mcontext); | 36 its.sp = SignalHandler::GetStackPointer(mcontext); |
39 state->callback(its, state->data); | 37 state->callback(its, state->data); |
40 } | 38 } |
41 }; | 39 }; |
42 | 40 |
43 | 41 |
44 void ThreadInterrupter::InterruptThreads(int64_t current_time) { | 42 void ThreadInterrupter::InterruptThread(InterruptableThreadState* state) { |
45 for (intptr_t i = 0; i < threads_size_; i++) { | 43 if (FLAG_trace_thread_interrupter) { |
46 ThreadState* state = threads_[i]; | 44 OS::Print("ThreadInterrupter interrupting %p\n", state->id); |
47 ASSERT(state->id != Thread::kInvalidThreadId); | |
48 if (FLAG_trace_thread_interrupter) { | |
49 OS::Print("ThreadInterrupter interrupting %p\n", state->id); | |
50 } | |
51 pthread_kill(state->id, SIGPROF); | |
52 } | 45 } |
| 46 pthread_kill(state->id, SIGPROF); |
53 } | 47 } |
54 | 48 |
55 | 49 |
56 void ThreadInterrupter::InstallSignalHandler() { | 50 void ThreadInterrupter::InstallSignalHandler() { |
57 SignalHandler::Install(ThreadInterrupterMacOS::ThreadInterruptSignalHandler); | 51 SignalHandler::Install(ThreadInterrupterMacOS::ThreadInterruptSignalHandler); |
58 } | 52 } |
59 | 53 |
60 | 54 |
61 } // namespace dart | 55 } // namespace dart |
62 | 56 |
63 #endif // defined(TARGET_OS_MACOS) | 57 #endif // defined(TARGET_OS_MACOS) |
64 | 58 |
OLD | NEW |