OLD | NEW |
1 // Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2015, 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 #ifndef VM_THREAD_H_ | 5 #ifndef VM_THREAD_H_ |
6 #define VM_THREAD_H_ | 6 #define VM_THREAD_H_ |
7 | 7 |
8 #include "vm/globals.h" | 8 #include "vm/globals.h" |
9 #include "vm/handles.h" | 9 #include "vm/handles.h" |
10 #include "vm/os_thread.h" | 10 #include "vm/os_thread.h" |
(...skipping 74 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
85 V(uword, native_call_wrapper_entry_point_, \ | 85 V(uword, native_call_wrapper_entry_point_, \ |
86 NativeEntry::NativeCallWrapperEntry(), 0) \ | 86 NativeEntry::NativeCallWrapperEntry(), 0) \ |
87 V(RawString**, predefined_symbols_address_, \ | 87 V(RawString**, predefined_symbols_address_, \ |
88 Symbols::PredefinedAddress(), NULL) \ | 88 Symbols::PredefinedAddress(), NULL) \ |
89 | 89 |
90 #define CACHED_CONSTANTS_LIST(V) \ | 90 #define CACHED_CONSTANTS_LIST(V) \ |
91 CACHED_VM_OBJECTS_LIST(V) \ | 91 CACHED_VM_OBJECTS_LIST(V) \ |
92 CACHED_ADDRESSES_LIST(V) \ | 92 CACHED_ADDRESSES_LIST(V) \ |
93 | 93 |
94 struct InterruptedThreadState { | 94 struct InterruptedThreadState { |
95 ThreadId tid; | |
96 uintptr_t pc; | 95 uintptr_t pc; |
97 uintptr_t csp; | 96 uintptr_t csp; |
98 uintptr_t dsp; | 97 uintptr_t dsp; |
99 uintptr_t fp; | 98 uintptr_t fp; |
100 uintptr_t lr; | 99 uintptr_t lr; |
101 }; | 100 }; |
102 | 101 |
103 // When a thread is interrupted the thread specific interrupt callback will be | 102 // When a thread is interrupted the thread specific interrupt callback will be |
104 // invoked. Each callback is given an InterruptedThreadState and the user data | 103 // invoked. Each callback is given an InterruptedThreadState and the user data |
105 // pointer. When inside a thread interrupt callback doing any of the following | 104 // pointer. When inside a thread interrupt callback doing any of the following |
106 // is forbidden: | 105 // is forbidden: |
107 // * Accessing TLS -- Because on Windows the callback will be running in a | 106 // * Accessing TLS -- Because on Windows the callback will be running in a |
108 // different thread. | 107 // different thread. |
109 // * Allocating memory -- Because this takes locks which may already be held, | 108 // * Allocating memory -- Because this takes locks which may already be held, |
110 // resulting in a dead lock. | 109 // resulting in a dead lock. |
111 // * Taking a lock -- See above. | 110 // * Taking a lock -- See above. |
112 typedef void (*ThreadInterruptCallback)(const InterruptedThreadState& state, | 111 typedef void (*ThreadInterruptCallback)(Thread* thread, |
113 void* data); | 112 const InterruptedThreadState& state); |
114 | 113 |
115 // A VM thread; may be executing Dart code or performing helper tasks like | 114 // A VM thread; may be executing Dart code or performing helper tasks like |
116 // garbage collection or compilation. The Thread structure associated with | 115 // garbage collection or compilation. The Thread structure associated with |
117 // a thread is allocated by EnsureInit before entering an isolate, and destroyed | 116 // a thread is allocated by EnsureInit before entering an isolate, and destroyed |
118 // automatically when the underlying OS thread exits. NOTE: On Windows, CleanUp | 117 // automatically when the underlying OS thread exits. NOTE: On Windows, CleanUp |
119 // must currently be called manually (issue 23474). | 118 // must currently be called manually (issue 23474). |
120 class Thread { | 119 class Thread { |
121 public: | 120 public: |
122 // The currently executing thread, or NULL if not yet initialized. | 121 // The currently executing thread, or NULL if not yet initialized. |
123 static Thread* Current() { | 122 static Thread* Current() { |
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
156 // The topmost zone used for allocation in this thread. | 155 // The topmost zone used for allocation in this thread. |
157 Zone* zone() const { return state_.zone; } | 156 Zone* zone() const { return state_.zone; } |
158 | 157 |
159 // The isolate that this thread is operating on, or NULL if none. | 158 // The isolate that this thread is operating on, or NULL if none. |
160 Isolate* isolate() const { return isolate_; } | 159 Isolate* isolate() const { return isolate_; } |
161 static intptr_t isolate_offset() { | 160 static intptr_t isolate_offset() { |
162 return OFFSET_OF(Thread, isolate_); | 161 return OFFSET_OF(Thread, isolate_); |
163 } | 162 } |
164 bool IsMutatorThread() const; | 163 bool IsMutatorThread() const; |
165 | 164 |
| 165 // Is |this| executing Dart code? |
| 166 bool IsExecutingDartCode() const; |
| 167 |
| 168 // Has |this| exited Dart code? |
| 169 bool HasExitedDartCode() const; |
| 170 |
166 // The (topmost) CHA for the compilation in this thread. | 171 // The (topmost) CHA for the compilation in this thread. |
167 CHA* cha() const; | 172 CHA* cha() const; |
168 void set_cha(CHA* value); | 173 void set_cha(CHA* value); |
169 | 174 |
170 int32_t no_callback_scope_depth() const { | 175 int32_t no_callback_scope_depth() const { |
171 return no_callback_scope_depth_; | 176 return no_callback_scope_depth_; |
172 } | 177 } |
173 | 178 |
174 void IncrementNoCallbackScopeDepth() { | 179 void IncrementNoCallbackScopeDepth() { |
175 ASSERT(no_callback_scope_depth_ < INT_MAX); | 180 ASSERT(no_callback_scope_depth_ < INT_MAX); |
(...skipping 194 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
370 ThreadId id() const { | 375 ThreadId id() const { |
371 ASSERT(id_ != OSThread::kInvalidThreadId); | 376 ASSERT(id_ != OSThread::kInvalidThreadId); |
372 return id_; | 377 return id_; |
373 } | 378 } |
374 | 379 |
375 ThreadId join_id() const { | 380 ThreadId join_id() const { |
376 ASSERT(join_id_ != OSThread::kInvalidThreadJoinId); | 381 ASSERT(join_id_ != OSThread::kInvalidThreadJoinId); |
377 return join_id_; | 382 return join_id_; |
378 } | 383 } |
379 | 384 |
380 void SetThreadInterrupter(ThreadInterruptCallback callback, void* data); | 385 // Used to temporarily disable or enable thread interrupts. |
381 | 386 void DisableThreadInterrupts(); |
382 bool IsThreadInterrupterEnabled(ThreadInterruptCallback* callback, | 387 void EnableThreadInterrupts(); |
383 void** data) const; | 388 bool ThreadInterruptsEnabled(); |
384 | 389 |
385 #if defined(DEBUG) | 390 #if defined(DEBUG) |
386 #define REUSABLE_HANDLE_SCOPE_ACCESSORS(object) \ | 391 #define REUSABLE_HANDLE_SCOPE_ACCESSORS(object) \ |
387 void set_reusable_##object##_handle_scope_active(bool value) { \ | 392 void set_reusable_##object##_handle_scope_active(bool value) { \ |
388 reusable_##object##_handle_scope_active_ = value; \ | 393 reusable_##object##_handle_scope_active_ = value; \ |
389 } \ | 394 } \ |
390 bool reusable_##object##_handle_scope_active() const { \ | 395 bool reusable_##object##_handle_scope_active() const { \ |
391 return reusable_##object##_handle_scope_active_; \ | 396 return reusable_##object##_handle_scope_active_; \ |
392 } | 397 } |
393 REUSABLE_HANDLE_LIST(REUSABLE_HANDLE_SCOPE_ACCESSORS) | 398 REUSABLE_HANDLE_LIST(REUSABLE_HANDLE_SCOPE_ACCESSORS) |
(...skipping 23 matching lines...) Expand all Loading... |
417 | 422 |
418 static bool IsThreadInList(ThreadId join_id); | 423 static bool IsThreadInList(ThreadId join_id); |
419 | 424 |
420 private: | 425 private: |
421 template<class T> T* AllocateReusableHandle(); | 426 template<class T> T* AllocateReusableHandle(); |
422 | 427 |
423 static ThreadLocalKey thread_key_; | 428 static ThreadLocalKey thread_key_; |
424 | 429 |
425 const ThreadId id_; | 430 const ThreadId id_; |
426 const ThreadId join_id_; | 431 const ThreadId join_id_; |
427 ThreadInterruptCallback thread_interrupt_callback_; | 432 uintptr_t thread_interrupt_disabled_; |
428 void* thread_interrupt_data_; | |
429 Isolate* isolate_; | 433 Isolate* isolate_; |
430 Heap* heap_; | 434 Heap* heap_; |
431 State state_; | 435 State state_; |
432 Mutex timeline_block_lock_; | 436 Mutex timeline_block_lock_; |
433 TimelineEventBlock* timeline_block_; | 437 TimelineEventBlock* timeline_block_; |
434 StoreBufferBlock* store_buffer_block_; | 438 StoreBufferBlock* store_buffer_block_; |
435 class Log* log_; | 439 class Log* log_; |
436 #define DECLARE_MEMBERS(type_name, member_name, expr, default_init_value) \ | 440 #define DECLARE_MEMBERS(type_name, member_name, expr, default_init_value) \ |
437 type_name member_name; | 441 type_name member_name; |
438 CACHED_CONSTANTS_LIST(DECLARE_MEMBERS) | 442 CACHED_CONSTANTS_LIST(DECLARE_MEMBERS) |
(...skipping 96 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
535 | 539 |
536 private: | 540 private: |
537 Thread* next_; | 541 Thread* next_; |
538 }; | 542 }; |
539 | 543 |
540 #if defined(TARGET_OS_WINDOWS) | 544 #if defined(TARGET_OS_WINDOWS) |
541 // Clears the state of the current thread and frees the allocation. | 545 // Clears the state of the current thread and frees the allocation. |
542 void WindowsThreadCleanUp(); | 546 void WindowsThreadCleanUp(); |
543 #endif | 547 #endif |
544 | 548 |
| 549 |
| 550 // Disable thread interrupts. |
| 551 class DisableThreadInterruptsScope : public StackResource { |
| 552 public: |
| 553 explicit DisableThreadInterruptsScope(Thread* thread); |
| 554 ~DisableThreadInterruptsScope(); |
| 555 }; |
| 556 |
545 } // namespace dart | 557 } // namespace dart |
546 | 558 |
547 #endif // VM_THREAD_H_ | 559 #endif // VM_THREAD_H_ |
OLD | NEW |