author | rehn |
Thu, 15 Aug 2019 09:06:42 +0200 | |
changeset 57758 | 91a758925be7 |
parent 55569 | 8e3a0ebf3497 |
child 58679 | 9c3209ff7550 |
child 59258 | 4c2557ab304e |
permissions | -rw-r--r-- |
1 | 1 |
/* |
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
52673
diff
changeset
|
2 |
* Copyright (c) 1997, 2019, Oracle and/or its affiliates. All rights reserved. |
1 | 3 |
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
4 |
* |
|
5 |
* This code is free software; you can redistribute it and/or modify it |
|
6 |
* under the terms of the GNU General Public License version 2 only, as |
|
7 |
* published by the Free Software Foundation. |
|
8 |
* |
|
9 |
* This code is distributed in the hope that it will be useful, but WITHOUT |
|
10 |
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
|
11 |
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
|
12 |
* version 2 for more details (a copy is included in the LICENSE file that |
|
13 |
* accompanied this code). |
|
14 |
* |
|
15 |
* You should have received a copy of the GNU General Public License version |
|
16 |
* 2 along with this work; if not, write to the Free Software Foundation, |
|
17 |
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
|
18 |
* |
|
5547
f4b087cbb361
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
5042
diff
changeset
|
19 |
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
f4b087cbb361
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
5042
diff
changeset
|
20 |
* or visit www.oracle.com if you need additional information or have any |
f4b087cbb361
6941466: Oracle rebranding changes for Hotspot repositories
trims
parents:
5042
diff
changeset
|
21 |
* questions. |
1 | 22 |
* |
23 |
*/ |
|
24 |
||
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
52673
diff
changeset
|
25 |
#ifndef SHARE_RUNTIME_SAFEPOINT_HPP |
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
52673
diff
changeset
|
26 |
#define SHARE_RUNTIME_SAFEPOINT_HPP |
7397 | 27 |
|
28 |
#include "memory/allocation.hpp" |
|
29 |
#include "runtime/os.hpp" |
|
53775 | 30 |
#include "runtime/thread.hpp" |
53895 | 31 |
#include "runtime/vmOperations.hpp" |
7397 | 32 |
#include "utilities/ostream.hpp" |
53775 | 33 |
#include "utilities/waitBarrier.hpp" |
7397 | 34 |
|
1 | 35 |
// |
36 |
// Safepoint synchronization |
|
37 |
//// |
|
53775 | 38 |
// The VMThread uses the SafepointSynchronize::begin/end |
1 | 39 |
// methods to enter/exit a safepoint region. The begin method will roll |
40 |
// all JavaThreads forward to a safepoint. |
|
41 |
// |
|
42 |
// JavaThreads must use the ThreadSafepointState abstraction (defined in |
|
43 |
// thread.hpp) to indicate that that they are at a safepoint. |
|
44 |
// |
|
45 |
// The Mutex/Condition variable and ObjectLocker classes calls the enter/ |
|
46 |
// exit safepoint methods, when a thread is blocked/restarted. Hence, all mutex exter/ |
|
47 |
// exit points *must* be at a safepoint. |
|
48 |
||
49 |
class ThreadSafepointState; |
|
50 |
||
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
51 |
class SafepointStateTracker { |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
52 |
uint64_t _safepoint_id; |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
53 |
bool _at_safepoint; |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
54 |
public: |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
55 |
SafepointStateTracker(uint64_t safepoint_id, bool at_safepoint); |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
56 |
bool safepoint_state_changed(); |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
57 |
}; |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
58 |
|
1 | 59 |
// |
60 |
// Implements roll-forward to safepoint (safepoint synchronization) |
|
61 |
// |
|
62 |
class SafepointSynchronize : AllStatic { |
|
63 |
public: |
|
64 |
enum SynchronizeState { |
|
53775 | 65 |
_not_synchronized = 0, // Threads not synchronized at a safepoint. Keep this value 0. |
1 | 66 |
_synchronizing = 1, // Synchronizing in progress |
53775 | 67 |
_synchronized = 2 // All Java threads are running in native, blocked in OS or stopped at safepoint. |
68 |
// VM thread and any NonJavaThread may be running. |
|
1 | 69 |
}; |
70 |
||
46702 | 71 |
// The enums are listed in the order of the tasks when done serially. |
72 |
enum SafepointCleanupTasks { |
|
73 |
SAFEPOINT_CLEANUP_DEFLATE_MONITORS, |
|
74 |
SAFEPOINT_CLEANUP_UPDATE_INLINE_CACHES, |
|
75 |
SAFEPOINT_CLEANUP_COMPILATION_POLICY, |
|
76 |
SAFEPOINT_CLEANUP_SYMBOL_TABLE_REHASH, |
|
77 |
SAFEPOINT_CLEANUP_STRING_TABLE_REHASH, |
|
78 |
SAFEPOINT_CLEANUP_CLD_PURGE, |
|
47774 | 79 |
SAFEPOINT_CLEANUP_SYSTEM_DICTIONARY_RESIZE, |
55569
8e3a0ebf3497
8226366: Excessive ServiceThread wakeups for OopStorage cleanup
kbarrett
parents:
55514
diff
changeset
|
80 |
SAFEPOINT_CLEANUP_REQUEST_OOPSTORAGE_CLEANUP, |
46702 | 81 |
// Leave this one last. |
82 |
SAFEPOINT_CLEANUP_NUM_TASKS |
|
83 |
}; |
|
84 |
||
1 | 85 |
private: |
53775 | 86 |
friend class SafepointMechanism; |
87 |
friend class ThreadSafepointState; |
|
88 |
friend class HandshakeState; |
|
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
89 |
friend class SafepointStateTracker; |
53775 | 90 |
|
91 |
// Threads might read this flag directly, without acquiring the Threads_lock: |
|
92 |
static volatile SynchronizeState _state; |
|
93 |
// Number of threads we are waiting for to block: |
|
94 |
static int _waiting_to_block; |
|
95 |
// Counts the number of active critical natives during the safepoint: |
|
96 |
static int _current_jni_active_count; |
|
1 | 97 |
|
98 |
// This counter is used for fast versions of jni_Get<Primitive>Field. |
|
53775 | 99 |
// An even value means there are no ongoing safepoint operations. |
1 | 100 |
// The counter is incremented ONLY at the beginning and end of each |
53775 | 101 |
// safepoint. |
52672
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
102 |
static volatile uint64_t _safepoint_counter; |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
103 |
|
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
104 |
// A change in this counter or a change in the result of |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
105 |
// is_at_safepoint() are used by SafepointStateTracker:: |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
106 |
// safepoint_state_changed() to determine its answer. |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
107 |
static uint64_t _safepoint_id; |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
108 |
|
53775 | 109 |
// JavaThreads that need to block for the safepoint will stop on the |
110 |
// _wait_barrier, where they can quickly be started again. |
|
111 |
static WaitBarrier* _wait_barrier; |
|
112 |
static long _end_of_last_safepoint; // Time of last safepoint in milliseconds |
|
113 |
static julong _coalesced_vmop_count; // coalesced vmop count |
|
1 | 114 |
|
115 |
// For debug long safepoint |
|
53895 | 116 |
static void print_safepoint_timeout(); |
1 | 117 |
|
53775 | 118 |
// Helper methods for safepoint procedure: |
119 |
static void arm_safepoint(); |
|
120 |
static int synchronize_threads(jlong safepoint_limit_time, int nof_threads, int* initial_running); |
|
121 |
static void disarm_safepoint(); |
|
122 |
static void increment_jni_active_count(); |
|
123 |
static void decrement_waiting_to_block(); |
|
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
124 |
static bool thread_not_running(ThreadSafepointState *cur_state); |
53775 | 125 |
|
126 |
// Used in safepoint_safe to do a stable load of the thread state. |
|
127 |
static bool try_stable_load_state(JavaThreadState *state, |
|
128 |
JavaThread *thread, |
|
129 |
uint64_t safepoint_count); |
|
130 |
||
131 |
// Called when a thread voluntarily blocks |
|
132 |
static void block(JavaThread *thread); |
|
133 |
||
134 |
// Called from VMThread during handshakes. |
|
135 |
// If true the VMThread may safely process the handshake operation for the JavaThread. |
|
136 |
static bool handshake_safe(JavaThread *thread); |
|
137 |
||
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
138 |
static uint64_t safepoint_counter() { return _safepoint_counter; } |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
139 |
|
1 | 140 |
public: |
141 |
||
53775 | 142 |
static void init(Thread* vmthread); |
1 | 143 |
|
53775 | 144 |
// Roll all threads forward to safepoint. Must be called by the VMThread. |
1 | 145 |
static void begin(); |
146 |
static void end(); // Start all suspended threads again... |
|
147 |
||
53775 | 148 |
// The value for a not set safepoint id. |
149 |
static const uint64_t InactiveSafepointCounter; |
|
11637
030466036615
7013347: allow crypto functions to be called inline to enhance performance
never
parents:
11631
diff
changeset
|
150 |
|
1 | 151 |
// Query |
53775 | 152 |
static bool is_at_safepoint() { return _state == _synchronized; } |
153 |
static bool is_synchronizing() { return _state == _synchronizing; } |
|
55514
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
154 |
|
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
155 |
static uint64_t safepoint_id() { |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
156 |
return _safepoint_id; |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
157 |
} |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
158 |
|
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
159 |
static SafepointStateTracker safepoint_state_tracker() { |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
160 |
return SafepointStateTracker(safepoint_id(), is_at_safepoint()); |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
161 |
} |
03468b206457
8225702: Safepoint counter can't be used for safepoint detection
rehn
parents:
54807
diff
changeset
|
162 |
|
1 | 163 |
// Exception handling for page polling |
164 |
static void handle_polling_page_exception(JavaThread *thread); |
|
165 |
||
166 |
static bool is_cleanup_needed(); |
|
167 |
static void do_cleanup_tasks(); |
|
168 |
||
53775 | 169 |
static void set_is_at_safepoint() { _state = _synchronized; } |
170 |
static void set_is_not_at_safepoint() { _state = _not_synchronized; } |
|
1 | 171 |
|
22551 | 172 |
// Assembly support |
53775 | 173 |
static address address_of_state() { return (address)&_state; } |
1 | 174 |
|
52672
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
175 |
// Only used for making sure that no safepoint has happened in |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
176 |
// JNI_FastGetField. Therefore only the low 32-bits are needed |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
177 |
// even if this is a 64-bit counter. |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
178 |
static address safepoint_counter_addr() { |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
179 |
#ifdef VM_LITTLE_ENDIAN |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
180 |
return (address)&_safepoint_counter; |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
181 |
#else /* BIG */ |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
182 |
// Return pointer to the 32 LSB: |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
183 |
return (address) (((uint32_t*)(&_safepoint_counter)) + 1); |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
184 |
#endif |
bbfa1b3aaf7e
8212108: SafepointSynchronizer never ending counter (big enough)
rehn
parents:
52518
diff
changeset
|
185 |
} |
1 | 186 |
}; |
187 |
||
49333
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
188 |
// Some helper assert macros for safepoint checks. |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
189 |
|
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
190 |
#define assert_at_safepoint() \ |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
191 |
assert(SafepointSynchronize::is_at_safepoint(), "should be at a safepoint") |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
192 |
|
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
193 |
#define assert_at_safepoint_msg(...) \ |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
194 |
assert(SafepointSynchronize::is_at_safepoint(), __VA_ARGS__) |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
195 |
|
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
196 |
#define assert_not_at_safepoint() \ |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
197 |
assert(!SafepointSynchronize::is_at_safepoint(), "should not be at a safepoint") |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
198 |
|
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
199 |
#define assert_not_at_safepoint_msg(...) \ |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
200 |
assert(!SafepointSynchronize::is_at_safepoint(), __VA_ARGS__) |
489f1dd40582
8196876: OopStorage::assert_at_safepoint clashes with assert_at_safepoint macros in g1CollectedHeap.hpp
kbarrett
parents:
49169
diff
changeset
|
201 |
|
1 | 202 |
// State class for a thread suspended at a safepoint |
53847
74b616640b00
8219244: NMT: Change ThreadSafepointState's allocation type from mtInternal to mtThread
zgu
parents:
53775
diff
changeset
|
203 |
class ThreadSafepointState: public CHeapObj<mtThread> { |
1 | 204 |
private: |
53775 | 205 |
// At polling page safepoint (NOT a poll return safepoint): |
206 |
volatile bool _at_poll_safepoint; |
|
207 |
JavaThread* _thread; |
|
208 |
bool _safepoint_safe; |
|
209 |
volatile uint64_t _safepoint_id; |
|
1 | 210 |
|
53775 | 211 |
ThreadSafepointState* _next; |
1 | 212 |
|
53775 | 213 |
void account_safe_thread(); |
1 | 214 |
|
215 |
public: |
|
216 |
ThreadSafepointState(JavaThread *thread); |
|
217 |
||
53775 | 218 |
// Linked list support: |
219 |
ThreadSafepointState* get_next() const { return _next; } |
|
220 |
void set_next(ThreadSafepointState* value) { _next = value; } |
|
221 |
ThreadSafepointState** next_ptr() { return &_next; } |
|
222 |
||
223 |
// examine/restart |
|
224 |
void examine_state_of_thread(uint64_t safepoint_count); |
|
1 | 225 |
void restart(); |
226 |
||
227 |
// Query |
|
228 |
JavaThread* thread() const { return _thread; } |
|
53775 | 229 |
bool is_running() const { return !_safepoint_safe; } |
230 |
||
231 |
uint64_t get_safepoint_id() const; |
|
232 |
void reset_safepoint_id(); |
|
233 |
void set_safepoint_id(uint64_t sid); |
|
234 |
||
1 | 235 |
// Support for safepoint timeout (debugging) |
53775 | 236 |
bool is_at_poll_safepoint() { return _at_poll_safepoint; } |
237 |
void set_at_poll_safepoint(bool val) { _at_poll_safepoint = val; } |
|
1 | 238 |
|
239 |
void handle_polling_page_exception(); |
|
240 |
||
241 |
// debugging |
|
242 |
void print_on(outputStream* st) const; |
|
243 |
||
244 |
// Initialize |
|
245 |
static void create(JavaThread *thread); |
|
246 |
static void destroy(JavaThread *thread); |
|
247 |
}; |
|
248 |
||
53895 | 249 |
class SafepointTracing : public AllStatic { |
250 |
private: |
|
251 |
// Absolute |
|
252 |
static jlong _last_safepoint_begin_time_ns; |
|
253 |
static jlong _last_safepoint_sync_time_ns; |
|
254 |
static jlong _last_safepoint_cleanup_time_ns; |
|
255 |
static jlong _last_safepoint_end_time_ns; |
|
54009
13acb4339895
8220151: SafepointTracing::end_of_last_safepoint_ms should return ms since epoch.
rehn
parents:
53895
diff
changeset
|
256 |
// amount of ms since epoch |
13acb4339895
8220151: SafepointTracing::end_of_last_safepoint_ms should return ms since epoch.
rehn
parents:
53895
diff
changeset
|
257 |
static jlong _last_safepoint_end_time_epoch_ms; |
53895 | 258 |
// Relative |
259 |
static jlong _last_app_time_ns; |
|
6453 | 260 |
|
53895 | 261 |
static int _nof_threads; |
262 |
static int _nof_running; |
|
263 |
static int _page_trap; |
|
264 |
||
265 |
static VM_Operation::VMOp_Type _current_type; |
|
266 |
static jlong _max_sync_time; |
|
267 |
static jlong _max_vmop_time; |
|
268 |
static uint64_t _op_count[VM_Operation::VMOp_Terminating]; |
|
269 |
||
270 |
static void statistics_log(); |
|
271 |
||
272 |
public: |
|
273 |
static void init(); |
|
274 |
||
275 |
static void begin(VM_Operation::VMOp_Type type); |
|
276 |
static void synchronized(int nof_threads, int nof_running, int traps); |
|
277 |
static void cleanup(); |
|
278 |
static void end(); |
|
279 |
||
280 |
static void statistics_exit_log(); |
|
281 |
||
282 |
static jlong time_since_last_safepoint_ms() { |
|
283 |
return (os::javaTimeNanos() - _last_safepoint_end_time_ns) / (NANOUNITS / MILLIUNITS); |
|
284 |
} |
|
285 |
||
54009
13acb4339895
8220151: SafepointTracing::end_of_last_safepoint_ms should return ms since epoch.
rehn
parents:
53895
diff
changeset
|
286 |
static jlong end_of_last_safepoint_epoch_ms() { |
13acb4339895
8220151: SafepointTracing::end_of_last_safepoint_ms should return ms since epoch.
rehn
parents:
53895
diff
changeset
|
287 |
return _last_safepoint_end_time_epoch_ms; |
53895 | 288 |
} |
289 |
||
290 |
static jlong start_of_safepoint() { |
|
291 |
return _last_safepoint_begin_time_ns; |
|
292 |
} |
|
293 |
}; |
|
7397 | 294 |
|
53244
9807daeb47c4
8216167: Update include guards to reflect correct directories
coleenp
parents:
52673
diff
changeset
|
295 |
#endif // SHARE_RUNTIME_SAFEPOINT_HPP |