Index: runtime/vm/thread_interrupter_win.cc |
diff --git a/runtime/vm/thread_interrupter_win.cc b/runtime/vm/thread_interrupter_win.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..c933cbb73c98c866a59e8c0dc75fce17417811cb |
--- /dev/null |
+++ b/runtime/vm/thread_interrupter_win.cc |
@@ -0,0 +1,98 @@ |
+// 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(TARGET_OS_WINDOWS) |
+ |
+#include "vm/thread_interrupter.h" |
+ |
+namespace dart { |
+ |
+DECLARE_FLAG(bool, thread_interrupter); |
+DECLARE_FLAG(bool, trace_thread_interrupter); |
+ |
+#define kThreadError -1 |
+ |
+class ThreadInterrupterWin : public AllStatic { |
+ public: |
+ static bool GrabRegisters(ThreadId thread, InterruptedThreadState* state) { |
+ CONTEXT context; |
+ memset(&context, 0, sizeof(context)); |
+ context.ContextFlags = CONTEXT_FULL; |
+ if (GetThreadContext(thread, &context) != 0) { |
+#if defined(TARGET_ARCH_IA32) |
+ state->pc = static_cast<uintptr_t>(context.Eip); |
+ state->fp = static_cast<uintptr_t>(context.Ebp); |
+ state->sp = static_cast<uintptr_t>(context.Esp); |
+#elif defined(TARGET_ARCH_X64) |
+ state->pc = reinterpret_cast<uintptr_t>(context.Rip); |
+ state->fp = reinterpret_cast<uintptr_t>(context.Rbp); |
+ state->sp = reinterpret_cast<uintptr_t>(context.Rsp); |
+#else |
+ UNIMPLEMENTED(); |
+#endif |
+ return true; |
+ } |
+ return false; |
+ } |
+ |
+ |
+ static void Interrupt(ThreadInterrupter::ThreadState* state) { |
+ ASSERT(GetCurrentThread() != state->id); |
+ DWORD result = SuspendThread(state->id); |
+ if (result == kThreadError) { |
+ if (FLAG_trace_thread_interrupter) { |
+ OS::Print("ThreadInterrupted failed to suspend thread %p\n", |
+ reinterpret_cast<void*>(state->id)); |
+ } |
+ return; |
+ } |
+ InterruptedThreadState its; |
+ its.tid = state->id; |
+ if (!GrabRegisters(state->id, &its)) { |
+ // Failed to get thread registers. |
+ ResumeThread(state->id); |
+ if (FLAG_trace_thread_interrupter) { |
+ OS::Print("ThreadInterrupted failed to get registers for %p\n", |
+ reinterpret_cast<void*>(state->id)); |
+ } |
+ return; |
+ } |
+ if (state->callback == NULL) { |
+ // No callback registered. |
+ ResumeThread(state->id); |
+ return; |
+ } |
+ state->callback(its, state->data); |
+ ResumeThread(state->id); |
+ } |
+}; |
+ |
+ |
+void ThreadInterrupter::InterruptThreads(int64_t current_time) { |
+ for (intptr_t i = 0; i < threads_size_; i++) { |
+ ThreadState* state = threads_[i]; |
+ ASSERT(state->id != Thread::kInvalidThreadId); |
+ if (FLAG_trace_thread_interrupter) { |
+ OS::Print("ThreadInterrupter suspending %p\n", |
+ reinterpret_cast<void*>(state->id)); |
+ } |
+ ThreadInterrupterWin::Interrupt(state); |
+ if (FLAG_trace_thread_interrupter) { |
+ OS::Print("ThreadInterrupter resuming %p\n", |
+ reinterpret_cast<void*>(state->id)); |
+ } |
+ } |
+} |
+ |
+ |
+void ThreadInterrupter::InstallSignalHandler() { |
+ // Nothing to do on Windows. |
+} |
+ |
+ |
+} // namespace dart |
+ |
+#endif // defined(TARGET_OS_WINDOWS) |
+ |