annotate src/share/vm/prims/jvmtiEnvBase.cpp @ 452:00b023ae2d78

6722113: CMS: Incorrect overflow handling during precleaning of Reference lists Summary: When we encounter marking stack overflow during precleaning of Reference lists, we were using the overflow list mechanism, which can cause problems on account of mutating the mark word of the header because of conflicts with mutator accesses and updates of that field. Instead we should use the usual mechanism for overflow handling in concurrent phases, namely dirtying of the card on which the overflowed object lies. Since precleaning effectively does a form of discovered list processing, albeit with discovery enabled, we needed to adjust some code to be correct in the face of interleaved processing and discovery. Reviewed-by: apetrusenko, jcoomes
author ysr
date Thu, 20 Nov 2008 12:27:41 -0800
parents dbec32712472
children ad8c8ca4ab0f
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
a61af66fc99e Initial load
duke
parents:
diff changeset
2 * Copyright 2003-2007 Sun Microsystems, Inc. All Rights Reserved.
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
a61af66fc99e Initial load
duke
parents:
diff changeset
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
a61af66fc99e Initial load
duke
parents:
diff changeset
20 * CA 95054 USA or visit www.sun.com if you need additional information or
a61af66fc99e Initial load
duke
parents:
diff changeset
21 * have any questions.
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24 # include "incls/_precompiled.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
25 # include "incls/_jvmtiEnvBase.cpp.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
26
a61af66fc99e Initial load
duke
parents:
diff changeset
27
a61af66fc99e Initial load
duke
parents:
diff changeset
28 ///////////////////////////////////////////////////////////////
a61af66fc99e Initial load
duke
parents:
diff changeset
29 //
a61af66fc99e Initial load
duke
parents:
diff changeset
30 // JvmtiEnvBase
a61af66fc99e Initial load
duke
parents:
diff changeset
31 //
a61af66fc99e Initial load
duke
parents:
diff changeset
32
a61af66fc99e Initial load
duke
parents:
diff changeset
33 JvmtiEnvBase* JvmtiEnvBase::_head_environment = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
34
a61af66fc99e Initial load
duke
parents:
diff changeset
35 bool JvmtiEnvBase::_globally_initialized = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
36 volatile bool JvmtiEnvBase::_needs_clean_up = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
37
a61af66fc99e Initial load
duke
parents:
diff changeset
38 jvmtiPhase JvmtiEnvBase::_phase = JVMTI_PHASE_PRIMORDIAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
39
a61af66fc99e Initial load
duke
parents:
diff changeset
40 volatile int JvmtiEnvBase::_dying_thread_env_iteration_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
41
a61af66fc99e Initial load
duke
parents:
diff changeset
42 extern jvmtiInterface_1_ jvmti_Interface;
a61af66fc99e Initial load
duke
parents:
diff changeset
43 extern jvmtiInterface_1_ jvmtiTrace_Interface;
a61af66fc99e Initial load
duke
parents:
diff changeset
44
a61af66fc99e Initial load
duke
parents:
diff changeset
45
a61af66fc99e Initial load
duke
parents:
diff changeset
46 // perform initializations that must occur before any JVMTI environments
a61af66fc99e Initial load
duke
parents:
diff changeset
47 // are released but which should only be initialized once (no matter
a61af66fc99e Initial load
duke
parents:
diff changeset
48 // how many environments are created).
a61af66fc99e Initial load
duke
parents:
diff changeset
49 void
a61af66fc99e Initial load
duke
parents:
diff changeset
50 JvmtiEnvBase::globally_initialize() {
a61af66fc99e Initial load
duke
parents:
diff changeset
51 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
52 assert(_globally_initialized == false, "bad call");
a61af66fc99e Initial load
duke
parents:
diff changeset
53
a61af66fc99e Initial load
duke
parents:
diff changeset
54 JvmtiManageCapabilities::initialize();
a61af66fc99e Initial load
duke
parents:
diff changeset
55
a61af66fc99e Initial load
duke
parents:
diff changeset
56 #ifndef JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
57 // register extension functions and events
a61af66fc99e Initial load
duke
parents:
diff changeset
58 JvmtiExtensions::register_extensions();
a61af66fc99e Initial load
duke
parents:
diff changeset
59 #endif // !JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
60
a61af66fc99e Initial load
duke
parents:
diff changeset
61 #ifdef JVMTI_TRACE
a61af66fc99e Initial load
duke
parents:
diff changeset
62 JvmtiTrace::initialize();
a61af66fc99e Initial load
duke
parents:
diff changeset
63 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
64
a61af66fc99e Initial load
duke
parents:
diff changeset
65 _globally_initialized = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
66 }
a61af66fc99e Initial load
duke
parents:
diff changeset
67
a61af66fc99e Initial load
duke
parents:
diff changeset
68
a61af66fc99e Initial load
duke
parents:
diff changeset
69 void
a61af66fc99e Initial load
duke
parents:
diff changeset
70 JvmtiEnvBase::initialize() {
a61af66fc99e Initial load
duke
parents:
diff changeset
71 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
72
a61af66fc99e Initial load
duke
parents:
diff changeset
73 // Add this environment to the end of the environment list (order is important)
a61af66fc99e Initial load
duke
parents:
diff changeset
74 {
a61af66fc99e Initial load
duke
parents:
diff changeset
75 // This block of code must not contain any safepoints, as list deallocation
a61af66fc99e Initial load
duke
parents:
diff changeset
76 // (which occurs at a safepoint) cannot occur simultaneously with this list
a61af66fc99e Initial load
duke
parents:
diff changeset
77 // addition. Note: No_Safepoint_Verifier cannot, currently, be used before
a61af66fc99e Initial load
duke
parents:
diff changeset
78 // threads exist.
a61af66fc99e Initial load
duke
parents:
diff changeset
79 JvmtiEnvIterator it;
a61af66fc99e Initial load
duke
parents:
diff changeset
80 JvmtiEnvBase *previous_env = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
81 for (JvmtiEnvBase* env = it.first(); env != NULL; env = it.next(env)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
82 previous_env = env;
a61af66fc99e Initial load
duke
parents:
diff changeset
83 }
a61af66fc99e Initial load
duke
parents:
diff changeset
84 if (previous_env == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
85 _head_environment = this;
a61af66fc99e Initial load
duke
parents:
diff changeset
86 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
87 previous_env->set_next_environment(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
88 }
a61af66fc99e Initial load
duke
parents:
diff changeset
89 }
a61af66fc99e Initial load
duke
parents:
diff changeset
90
a61af66fc99e Initial load
duke
parents:
diff changeset
91 if (_globally_initialized == false) {
a61af66fc99e Initial load
duke
parents:
diff changeset
92 globally_initialize();
a61af66fc99e Initial load
duke
parents:
diff changeset
93 }
a61af66fc99e Initial load
duke
parents:
diff changeset
94 }
a61af66fc99e Initial load
duke
parents:
diff changeset
95
a61af66fc99e Initial load
duke
parents:
diff changeset
96
a61af66fc99e Initial load
duke
parents:
diff changeset
97 JvmtiEnvBase::JvmtiEnvBase() : _env_event_enable() {
a61af66fc99e Initial load
duke
parents:
diff changeset
98 _env_local_storage = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
99 _tag_map = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
100 _native_method_prefix_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
101 _native_method_prefixes = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
102 _next = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
103 _class_file_load_hook_ever_enabled = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
104
a61af66fc99e Initial load
duke
parents:
diff changeset
105 // Moot since ClassFileLoadHook not yet enabled.
a61af66fc99e Initial load
duke
parents:
diff changeset
106 // But "true" will give a more predictable ClassFileLoadHook behavior
a61af66fc99e Initial load
duke
parents:
diff changeset
107 // for environment creation during ClassFileLoadHook.
a61af66fc99e Initial load
duke
parents:
diff changeset
108 _is_retransformable = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
109
a61af66fc99e Initial load
duke
parents:
diff changeset
110 // all callbacks initially NULL
a61af66fc99e Initial load
duke
parents:
diff changeset
111 memset(&_event_callbacks,0,sizeof(jvmtiEventCallbacks));
a61af66fc99e Initial load
duke
parents:
diff changeset
112
a61af66fc99e Initial load
duke
parents:
diff changeset
113 // all capabilities initially off
a61af66fc99e Initial load
duke
parents:
diff changeset
114 memset(&_current_capabilities, 0, sizeof(_current_capabilities));
a61af66fc99e Initial load
duke
parents:
diff changeset
115
a61af66fc99e Initial load
duke
parents:
diff changeset
116 // all prohibited capabilities initially off
a61af66fc99e Initial load
duke
parents:
diff changeset
117 memset(&_prohibited_capabilities, 0, sizeof(_prohibited_capabilities));
a61af66fc99e Initial load
duke
parents:
diff changeset
118
a61af66fc99e Initial load
duke
parents:
diff changeset
119 _magic = JVMTI_MAGIC;
a61af66fc99e Initial load
duke
parents:
diff changeset
120
a61af66fc99e Initial load
duke
parents:
diff changeset
121 JvmtiEventController::env_initialize((JvmtiEnv*)this);
a61af66fc99e Initial load
duke
parents:
diff changeset
122
a61af66fc99e Initial load
duke
parents:
diff changeset
123 #ifdef JVMTI_TRACE
371
dbec32712472 6753795: HotSpot crash in strlen() when JVMTI is used
never
parents: 0
diff changeset
124 _jvmti_external.functions = TraceJVMTI != NULL ? &jvmtiTrace_Interface : &jvmti_Interface;
0
a61af66fc99e Initial load
duke
parents:
diff changeset
125 #else
a61af66fc99e Initial load
duke
parents:
diff changeset
126 _jvmti_external.functions = &jvmti_Interface;
a61af66fc99e Initial load
duke
parents:
diff changeset
127 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
128 }
a61af66fc99e Initial load
duke
parents:
diff changeset
129
a61af66fc99e Initial load
duke
parents:
diff changeset
130
a61af66fc99e Initial load
duke
parents:
diff changeset
131 void
a61af66fc99e Initial load
duke
parents:
diff changeset
132 JvmtiEnvBase::dispose() {
a61af66fc99e Initial load
duke
parents:
diff changeset
133
a61af66fc99e Initial load
duke
parents:
diff changeset
134 #ifdef JVMTI_TRACE
a61af66fc99e Initial load
duke
parents:
diff changeset
135 JvmtiTrace::shutdown();
a61af66fc99e Initial load
duke
parents:
diff changeset
136 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
137
a61af66fc99e Initial load
duke
parents:
diff changeset
138 // Dispose of event info and let the event controller call us back
a61af66fc99e Initial load
duke
parents:
diff changeset
139 // in a locked state (env_dispose, below)
a61af66fc99e Initial load
duke
parents:
diff changeset
140 JvmtiEventController::env_dispose(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
141 }
a61af66fc99e Initial load
duke
parents:
diff changeset
142
a61af66fc99e Initial load
duke
parents:
diff changeset
143 void
a61af66fc99e Initial load
duke
parents:
diff changeset
144 JvmtiEnvBase::env_dispose() {
a61af66fc99e Initial load
duke
parents:
diff changeset
145 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
146
a61af66fc99e Initial load
duke
parents:
diff changeset
147 // We have been entered with all events disabled on this environment.
a61af66fc99e Initial load
duke
parents:
diff changeset
148 // A race to re-enable events (by setting callbacks) is prevented by
a61af66fc99e Initial load
duke
parents:
diff changeset
149 // checking for a valid environment when setting callbacks (while
a61af66fc99e Initial load
duke
parents:
diff changeset
150 // holding the JvmtiThreadState_lock).
a61af66fc99e Initial load
duke
parents:
diff changeset
151
a61af66fc99e Initial load
duke
parents:
diff changeset
152 // Mark as invalid.
a61af66fc99e Initial load
duke
parents:
diff changeset
153 _magic = DISPOSED_MAGIC;
a61af66fc99e Initial load
duke
parents:
diff changeset
154
a61af66fc99e Initial load
duke
parents:
diff changeset
155 // Relinquish all capabilities.
a61af66fc99e Initial load
duke
parents:
diff changeset
156 jvmtiCapabilities *caps = get_capabilities();
a61af66fc99e Initial load
duke
parents:
diff changeset
157 JvmtiManageCapabilities::relinquish_capabilities(caps, caps, caps);
a61af66fc99e Initial load
duke
parents:
diff changeset
158
a61af66fc99e Initial load
duke
parents:
diff changeset
159 // Same situation as with events (see above)
a61af66fc99e Initial load
duke
parents:
diff changeset
160 set_native_method_prefixes(0, NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
161
a61af66fc99e Initial load
duke
parents:
diff changeset
162 #ifndef JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
163 JvmtiTagMap* tag_map_to_deallocate = _tag_map;
a61af66fc99e Initial load
duke
parents:
diff changeset
164 set_tag_map(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
165 // A tag map can be big, deallocate it now
a61af66fc99e Initial load
duke
parents:
diff changeset
166 if (tag_map_to_deallocate != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
167 delete tag_map_to_deallocate;
a61af66fc99e Initial load
duke
parents:
diff changeset
168 }
a61af66fc99e Initial load
duke
parents:
diff changeset
169 #endif // !JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
170
a61af66fc99e Initial load
duke
parents:
diff changeset
171 _needs_clean_up = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
172 }
a61af66fc99e Initial load
duke
parents:
diff changeset
173
a61af66fc99e Initial load
duke
parents:
diff changeset
174
a61af66fc99e Initial load
duke
parents:
diff changeset
175 JvmtiEnvBase::~JvmtiEnvBase() {
a61af66fc99e Initial load
duke
parents:
diff changeset
176 assert(SafepointSynchronize::is_at_safepoint(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
177
a61af66fc99e Initial load
duke
parents:
diff changeset
178 // There is a small window of time during which the tag map of a
a61af66fc99e Initial load
duke
parents:
diff changeset
179 // disposed environment could have been reallocated.
a61af66fc99e Initial load
duke
parents:
diff changeset
180 // Make sure it is gone.
a61af66fc99e Initial load
duke
parents:
diff changeset
181 #ifndef JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
182 JvmtiTagMap* tag_map_to_deallocate = _tag_map;
a61af66fc99e Initial load
duke
parents:
diff changeset
183 set_tag_map(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
184 // A tag map can be big, deallocate it now
a61af66fc99e Initial load
duke
parents:
diff changeset
185 if (tag_map_to_deallocate != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
186 delete tag_map_to_deallocate;
a61af66fc99e Initial load
duke
parents:
diff changeset
187 }
a61af66fc99e Initial load
duke
parents:
diff changeset
188 #endif // !JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
189
a61af66fc99e Initial load
duke
parents:
diff changeset
190 _magic = BAD_MAGIC;
a61af66fc99e Initial load
duke
parents:
diff changeset
191 }
a61af66fc99e Initial load
duke
parents:
diff changeset
192
a61af66fc99e Initial load
duke
parents:
diff changeset
193
a61af66fc99e Initial load
duke
parents:
diff changeset
194 void
a61af66fc99e Initial load
duke
parents:
diff changeset
195 JvmtiEnvBase::periodic_clean_up() {
a61af66fc99e Initial load
duke
parents:
diff changeset
196 assert(SafepointSynchronize::is_at_safepoint(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
197
a61af66fc99e Initial load
duke
parents:
diff changeset
198 // JvmtiEnvBase reference is saved in JvmtiEnvThreadState. So
a61af66fc99e Initial load
duke
parents:
diff changeset
199 // clean up JvmtiThreadState before deleting JvmtiEnv pointer.
a61af66fc99e Initial load
duke
parents:
diff changeset
200 JvmtiThreadState::periodic_clean_up();
a61af66fc99e Initial load
duke
parents:
diff changeset
201
a61af66fc99e Initial load
duke
parents:
diff changeset
202 // Unlink all invalid environments from the list of environments
a61af66fc99e Initial load
duke
parents:
diff changeset
203 // and deallocate them
a61af66fc99e Initial load
duke
parents:
diff changeset
204 JvmtiEnvIterator it;
a61af66fc99e Initial load
duke
parents:
diff changeset
205 JvmtiEnvBase* previous_env = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
206 JvmtiEnvBase* env = it.first();
a61af66fc99e Initial load
duke
parents:
diff changeset
207 while (env != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
208 if (env->is_valid()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
209 previous_env = env;
a61af66fc99e Initial load
duke
parents:
diff changeset
210 env = it.next(env);
a61af66fc99e Initial load
duke
parents:
diff changeset
211 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
212 // This one isn't valid, remove it from the list and deallocate it
a61af66fc99e Initial load
duke
parents:
diff changeset
213 JvmtiEnvBase* defunct_env = env;
a61af66fc99e Initial load
duke
parents:
diff changeset
214 env = it.next(env);
a61af66fc99e Initial load
duke
parents:
diff changeset
215 if (previous_env == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
216 _head_environment = env;
a61af66fc99e Initial load
duke
parents:
diff changeset
217 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
218 previous_env->set_next_environment(env);
a61af66fc99e Initial load
duke
parents:
diff changeset
219 }
a61af66fc99e Initial load
duke
parents:
diff changeset
220 delete defunct_env;
a61af66fc99e Initial load
duke
parents:
diff changeset
221 }
a61af66fc99e Initial load
duke
parents:
diff changeset
222 }
a61af66fc99e Initial load
duke
parents:
diff changeset
223
a61af66fc99e Initial load
duke
parents:
diff changeset
224 }
a61af66fc99e Initial load
duke
parents:
diff changeset
225
a61af66fc99e Initial load
duke
parents:
diff changeset
226
a61af66fc99e Initial load
duke
parents:
diff changeset
227 void
a61af66fc99e Initial load
duke
parents:
diff changeset
228 JvmtiEnvBase::check_for_periodic_clean_up() {
a61af66fc99e Initial load
duke
parents:
diff changeset
229 assert(SafepointSynchronize::is_at_safepoint(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
230
a61af66fc99e Initial load
duke
parents:
diff changeset
231 class ThreadInsideIterationClosure: public ThreadClosure {
a61af66fc99e Initial load
duke
parents:
diff changeset
232 private:
a61af66fc99e Initial load
duke
parents:
diff changeset
233 bool _inside;
a61af66fc99e Initial load
duke
parents:
diff changeset
234 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
235 ThreadInsideIterationClosure() : _inside(false) {};
a61af66fc99e Initial load
duke
parents:
diff changeset
236
a61af66fc99e Initial load
duke
parents:
diff changeset
237 void do_thread(Thread* thread) {
a61af66fc99e Initial load
duke
parents:
diff changeset
238 _inside |= thread->is_inside_jvmti_env_iteration();
a61af66fc99e Initial load
duke
parents:
diff changeset
239 }
a61af66fc99e Initial load
duke
parents:
diff changeset
240
a61af66fc99e Initial load
duke
parents:
diff changeset
241 bool is_inside_jvmti_env_iteration() {
a61af66fc99e Initial load
duke
parents:
diff changeset
242 return _inside;
a61af66fc99e Initial load
duke
parents:
diff changeset
243 }
a61af66fc99e Initial load
duke
parents:
diff changeset
244 };
a61af66fc99e Initial load
duke
parents:
diff changeset
245
a61af66fc99e Initial load
duke
parents:
diff changeset
246 if (_needs_clean_up) {
a61af66fc99e Initial load
duke
parents:
diff changeset
247 // Check if we are currently iterating environment,
a61af66fc99e Initial load
duke
parents:
diff changeset
248 // deallocation should not occur if we are
a61af66fc99e Initial load
duke
parents:
diff changeset
249 ThreadInsideIterationClosure tiic;
a61af66fc99e Initial load
duke
parents:
diff changeset
250 Threads::threads_do(&tiic);
a61af66fc99e Initial load
duke
parents:
diff changeset
251 if (!tiic.is_inside_jvmti_env_iteration() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
252 !is_inside_dying_thread_env_iteration()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
253 _needs_clean_up = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
254 JvmtiEnvBase::periodic_clean_up();
a61af66fc99e Initial load
duke
parents:
diff changeset
255 }
a61af66fc99e Initial load
duke
parents:
diff changeset
256 }
a61af66fc99e Initial load
duke
parents:
diff changeset
257 }
a61af66fc99e Initial load
duke
parents:
diff changeset
258
a61af66fc99e Initial load
duke
parents:
diff changeset
259
a61af66fc99e Initial load
duke
parents:
diff changeset
260 void
a61af66fc99e Initial load
duke
parents:
diff changeset
261 JvmtiEnvBase::record_first_time_class_file_load_hook_enabled() {
a61af66fc99e Initial load
duke
parents:
diff changeset
262 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(),
a61af66fc99e Initial load
duke
parents:
diff changeset
263 "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
264
a61af66fc99e Initial load
duke
parents:
diff changeset
265 if (!_class_file_load_hook_ever_enabled) {
a61af66fc99e Initial load
duke
parents:
diff changeset
266 _class_file_load_hook_ever_enabled = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
267
a61af66fc99e Initial load
duke
parents:
diff changeset
268 if (get_capabilities()->can_retransform_classes) {
a61af66fc99e Initial load
duke
parents:
diff changeset
269 _is_retransformable = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
270 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
271 _is_retransformable = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
272
a61af66fc99e Initial load
duke
parents:
diff changeset
273 // cannot add retransform capability after ClassFileLoadHook has been enabled
a61af66fc99e Initial load
duke
parents:
diff changeset
274 get_prohibited_capabilities()->can_retransform_classes = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
275 }
a61af66fc99e Initial load
duke
parents:
diff changeset
276 }
a61af66fc99e Initial load
duke
parents:
diff changeset
277 }
a61af66fc99e Initial load
duke
parents:
diff changeset
278
a61af66fc99e Initial load
duke
parents:
diff changeset
279
a61af66fc99e Initial load
duke
parents:
diff changeset
280 void
a61af66fc99e Initial load
duke
parents:
diff changeset
281 JvmtiEnvBase::record_class_file_load_hook_enabled() {
a61af66fc99e Initial load
duke
parents:
diff changeset
282 if (!_class_file_load_hook_ever_enabled) {
a61af66fc99e Initial load
duke
parents:
diff changeset
283 if (Threads::number_of_threads() == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
284 record_first_time_class_file_load_hook_enabled();
a61af66fc99e Initial load
duke
parents:
diff changeset
285 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
286 MutexLocker mu(JvmtiThreadState_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
287 record_first_time_class_file_load_hook_enabled();
a61af66fc99e Initial load
duke
parents:
diff changeset
288 }
a61af66fc99e Initial load
duke
parents:
diff changeset
289 }
a61af66fc99e Initial load
duke
parents:
diff changeset
290 }
a61af66fc99e Initial load
duke
parents:
diff changeset
291
a61af66fc99e Initial load
duke
parents:
diff changeset
292
a61af66fc99e Initial load
duke
parents:
diff changeset
293 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
294 JvmtiEnvBase::set_native_method_prefixes(jint prefix_count, char** prefixes) {
a61af66fc99e Initial load
duke
parents:
diff changeset
295 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(),
a61af66fc99e Initial load
duke
parents:
diff changeset
296 "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
297
a61af66fc99e Initial load
duke
parents:
diff changeset
298 int old_prefix_count = get_native_method_prefix_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
299 char **old_prefixes = get_native_method_prefixes();
a61af66fc99e Initial load
duke
parents:
diff changeset
300
a61af66fc99e Initial load
duke
parents:
diff changeset
301 // allocate and install the new prefixex
a61af66fc99e Initial load
duke
parents:
diff changeset
302 if (prefix_count == 0 || !is_valid()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
303 _native_method_prefix_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
304 _native_method_prefixes = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
305 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
306 // there are prefixes, allocate an array to hold them, and fill it
a61af66fc99e Initial load
duke
parents:
diff changeset
307 char** new_prefixes = (char**)os::malloc((prefix_count) * sizeof(char*));
a61af66fc99e Initial load
duke
parents:
diff changeset
308 if (new_prefixes == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
309 return JVMTI_ERROR_OUT_OF_MEMORY;
a61af66fc99e Initial load
duke
parents:
diff changeset
310 }
a61af66fc99e Initial load
duke
parents:
diff changeset
311 for (int i = 0; i < prefix_count; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
312 char* prefix = prefixes[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
313 if (prefix == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
314 for (int j = 0; j < (i-1); j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
315 os::free(new_prefixes[j]);
a61af66fc99e Initial load
duke
parents:
diff changeset
316 }
a61af66fc99e Initial load
duke
parents:
diff changeset
317 os::free(new_prefixes);
a61af66fc99e Initial load
duke
parents:
diff changeset
318 return JVMTI_ERROR_NULL_POINTER;
a61af66fc99e Initial load
duke
parents:
diff changeset
319 }
a61af66fc99e Initial load
duke
parents:
diff changeset
320 prefix = os::strdup(prefixes[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
321 if (prefix == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
322 for (int j = 0; j < (i-1); j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
323 os::free(new_prefixes[j]);
a61af66fc99e Initial load
duke
parents:
diff changeset
324 }
a61af66fc99e Initial load
duke
parents:
diff changeset
325 os::free(new_prefixes);
a61af66fc99e Initial load
duke
parents:
diff changeset
326 return JVMTI_ERROR_OUT_OF_MEMORY;
a61af66fc99e Initial load
duke
parents:
diff changeset
327 }
a61af66fc99e Initial load
duke
parents:
diff changeset
328 new_prefixes[i] = prefix;
a61af66fc99e Initial load
duke
parents:
diff changeset
329 }
a61af66fc99e Initial load
duke
parents:
diff changeset
330 _native_method_prefix_count = prefix_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
331 _native_method_prefixes = new_prefixes;
a61af66fc99e Initial load
duke
parents:
diff changeset
332 }
a61af66fc99e Initial load
duke
parents:
diff changeset
333
a61af66fc99e Initial load
duke
parents:
diff changeset
334 // now that we know the new prefixes have been successfully installed we can
a61af66fc99e Initial load
duke
parents:
diff changeset
335 // safely remove the old ones
a61af66fc99e Initial load
duke
parents:
diff changeset
336 if (old_prefix_count != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
337 for (int i = 0; i < old_prefix_count; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
338 os::free(old_prefixes[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
339 }
a61af66fc99e Initial load
duke
parents:
diff changeset
340 os::free(old_prefixes);
a61af66fc99e Initial load
duke
parents:
diff changeset
341 }
a61af66fc99e Initial load
duke
parents:
diff changeset
342
a61af66fc99e Initial load
duke
parents:
diff changeset
343 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
344 }
a61af66fc99e Initial load
duke
parents:
diff changeset
345
a61af66fc99e Initial load
duke
parents:
diff changeset
346
a61af66fc99e Initial load
duke
parents:
diff changeset
347 // Collect all the prefixes which have been set in any JVM TI environments
a61af66fc99e Initial load
duke
parents:
diff changeset
348 // by the SetNativeMethodPrefix(es) functions. Be sure to maintain the
a61af66fc99e Initial load
duke
parents:
diff changeset
349 // order of environments and the order of prefixes within each environment.
a61af66fc99e Initial load
duke
parents:
diff changeset
350 // Return in a resource allocated array.
a61af66fc99e Initial load
duke
parents:
diff changeset
351 char**
a61af66fc99e Initial load
duke
parents:
diff changeset
352 JvmtiEnvBase::get_all_native_method_prefixes(int* count_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
353 assert(Threads::number_of_threads() == 0 ||
a61af66fc99e Initial load
duke
parents:
diff changeset
354 SafepointSynchronize::is_at_safepoint() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
355 JvmtiThreadState_lock->is_locked(),
a61af66fc99e Initial load
duke
parents:
diff changeset
356 "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
357
a61af66fc99e Initial load
duke
parents:
diff changeset
358 int total_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
359 GrowableArray<char*>* prefix_array =new GrowableArray<char*>(5);
a61af66fc99e Initial load
duke
parents:
diff changeset
360
a61af66fc99e Initial load
duke
parents:
diff changeset
361 JvmtiEnvIterator it;
a61af66fc99e Initial load
duke
parents:
diff changeset
362 for (JvmtiEnvBase* env = it.first(); env != NULL; env = it.next(env)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
363 int prefix_count = env->get_native_method_prefix_count();
a61af66fc99e Initial load
duke
parents:
diff changeset
364 char** prefixes = env->get_native_method_prefixes();
a61af66fc99e Initial load
duke
parents:
diff changeset
365 for (int j = 0; j < prefix_count; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
366 // retrieve a prefix and so that it is safe against asynchronous changes
a61af66fc99e Initial load
duke
parents:
diff changeset
367 // copy it into the resource area
a61af66fc99e Initial load
duke
parents:
diff changeset
368 char* prefix = prefixes[j];
a61af66fc99e Initial load
duke
parents:
diff changeset
369 char* prefix_copy = NEW_RESOURCE_ARRAY(char, strlen(prefix)+1);
a61af66fc99e Initial load
duke
parents:
diff changeset
370 strcpy(prefix_copy, prefix);
a61af66fc99e Initial load
duke
parents:
diff changeset
371 prefix_array->at_put_grow(total_count++, prefix_copy);
a61af66fc99e Initial load
duke
parents:
diff changeset
372 }
a61af66fc99e Initial load
duke
parents:
diff changeset
373 }
a61af66fc99e Initial load
duke
parents:
diff changeset
374
a61af66fc99e Initial load
duke
parents:
diff changeset
375 char** all_prefixes = NEW_RESOURCE_ARRAY(char*, total_count);
a61af66fc99e Initial load
duke
parents:
diff changeset
376 char** p = all_prefixes;
a61af66fc99e Initial load
duke
parents:
diff changeset
377 for (int i = 0; i < total_count; ++i) {
a61af66fc99e Initial load
duke
parents:
diff changeset
378 *p++ = prefix_array->at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
379 }
a61af66fc99e Initial load
duke
parents:
diff changeset
380 *count_ptr = total_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
381 return all_prefixes;
a61af66fc99e Initial load
duke
parents:
diff changeset
382 }
a61af66fc99e Initial load
duke
parents:
diff changeset
383
a61af66fc99e Initial load
duke
parents:
diff changeset
384 void
a61af66fc99e Initial load
duke
parents:
diff changeset
385 JvmtiEnvBase::set_event_callbacks(const jvmtiEventCallbacks* callbacks,
a61af66fc99e Initial load
duke
parents:
diff changeset
386 jint size_of_callbacks) {
a61af66fc99e Initial load
duke
parents:
diff changeset
387 assert(Threads::number_of_threads() == 0 || JvmtiThreadState_lock->is_locked(), "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
388
a61af66fc99e Initial load
duke
parents:
diff changeset
389 size_t byte_cnt = sizeof(jvmtiEventCallbacks);
a61af66fc99e Initial load
duke
parents:
diff changeset
390
a61af66fc99e Initial load
duke
parents:
diff changeset
391 // clear in either case to be sure we got any gap between sizes
a61af66fc99e Initial load
duke
parents:
diff changeset
392 memset(&_event_callbacks, 0, byte_cnt);
a61af66fc99e Initial load
duke
parents:
diff changeset
393
a61af66fc99e Initial load
duke
parents:
diff changeset
394 // Now that JvmtiThreadState_lock is held, prevent a possible race condition where events
a61af66fc99e Initial load
duke
parents:
diff changeset
395 // are re-enabled by a call to set event callbacks where the DisposeEnvironment
a61af66fc99e Initial load
duke
parents:
diff changeset
396 // occurs after the boiler-plate environment check and before the lock is acquired.
a61af66fc99e Initial load
duke
parents:
diff changeset
397 if (callbacks != NULL && is_valid()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
398 if (size_of_callbacks < (jint)byte_cnt) {
a61af66fc99e Initial load
duke
parents:
diff changeset
399 byte_cnt = size_of_callbacks;
a61af66fc99e Initial load
duke
parents:
diff changeset
400 }
a61af66fc99e Initial load
duke
parents:
diff changeset
401 memcpy(&_event_callbacks, callbacks, byte_cnt);
a61af66fc99e Initial load
duke
parents:
diff changeset
402 }
a61af66fc99e Initial load
duke
parents:
diff changeset
403 }
a61af66fc99e Initial load
duke
parents:
diff changeset
404
a61af66fc99e Initial load
duke
parents:
diff changeset
405 // Called from JVMTI entry points which perform stack walking. If the
a61af66fc99e Initial load
duke
parents:
diff changeset
406 // associated JavaThread is the current thread, then wait_for_suspend
a61af66fc99e Initial load
duke
parents:
diff changeset
407 // is not used. Otherwise, it determines if we should wait for the
a61af66fc99e Initial load
duke
parents:
diff changeset
408 // "other" thread to complete external suspension. (NOTE: in future
a61af66fc99e Initial load
duke
parents:
diff changeset
409 // releases the suspension mechanism should be reimplemented so this
a61af66fc99e Initial load
duke
parents:
diff changeset
410 // is not necessary.)
a61af66fc99e Initial load
duke
parents:
diff changeset
411 //
a61af66fc99e Initial load
duke
parents:
diff changeset
412 bool
a61af66fc99e Initial load
duke
parents:
diff changeset
413 JvmtiEnvBase::is_thread_fully_suspended(JavaThread* thr, bool wait_for_suspend, uint32_t *bits) {
a61af66fc99e Initial load
duke
parents:
diff changeset
414 // "other" threads require special handling
a61af66fc99e Initial load
duke
parents:
diff changeset
415 if (thr != JavaThread::current()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
416 if (wait_for_suspend) {
a61af66fc99e Initial load
duke
parents:
diff changeset
417 // We are allowed to wait for the external suspend to complete
a61af66fc99e Initial load
duke
parents:
diff changeset
418 // so give the other thread a chance to get suspended.
a61af66fc99e Initial load
duke
parents:
diff changeset
419 if (!thr->wait_for_ext_suspend_completion(SuspendRetryCount,
a61af66fc99e Initial load
duke
parents:
diff changeset
420 SuspendRetryDelay, bits)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
421 // didn't make it so let the caller know
a61af66fc99e Initial load
duke
parents:
diff changeset
422 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
423 }
a61af66fc99e Initial load
duke
parents:
diff changeset
424 }
a61af66fc99e Initial load
duke
parents:
diff changeset
425 // We aren't allowed to wait for the external suspend to complete
a61af66fc99e Initial load
duke
parents:
diff changeset
426 // so if the other thread isn't externally suspended we need to
a61af66fc99e Initial load
duke
parents:
diff changeset
427 // let the caller know.
a61af66fc99e Initial load
duke
parents:
diff changeset
428 else if (!thr->is_ext_suspend_completed_with_lock(bits)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
429 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
430 }
a61af66fc99e Initial load
duke
parents:
diff changeset
431 }
a61af66fc99e Initial load
duke
parents:
diff changeset
432
a61af66fc99e Initial load
duke
parents:
diff changeset
433 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
434 }
a61af66fc99e Initial load
duke
parents:
diff changeset
435
a61af66fc99e Initial load
duke
parents:
diff changeset
436
a61af66fc99e Initial load
duke
parents:
diff changeset
437 // In the fullness of time, all users of the method should instead
a61af66fc99e Initial load
duke
parents:
diff changeset
438 // directly use allocate, besides being cleaner and faster, this will
a61af66fc99e Initial load
duke
parents:
diff changeset
439 // mean much better out of memory handling
a61af66fc99e Initial load
duke
parents:
diff changeset
440 unsigned char *
a61af66fc99e Initial load
duke
parents:
diff changeset
441 JvmtiEnvBase::jvmtiMalloc(jlong size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
442 unsigned char* mem;
a61af66fc99e Initial load
duke
parents:
diff changeset
443 jvmtiError result = allocate(size, &mem);
a61af66fc99e Initial load
duke
parents:
diff changeset
444 assert(result == JVMTI_ERROR_NONE, "Allocate failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
445 return mem;
a61af66fc99e Initial load
duke
parents:
diff changeset
446 }
a61af66fc99e Initial load
duke
parents:
diff changeset
447
a61af66fc99e Initial load
duke
parents:
diff changeset
448
a61af66fc99e Initial load
duke
parents:
diff changeset
449 //
a61af66fc99e Initial load
duke
parents:
diff changeset
450 // Threads
a61af66fc99e Initial load
duke
parents:
diff changeset
451 //
a61af66fc99e Initial load
duke
parents:
diff changeset
452
a61af66fc99e Initial load
duke
parents:
diff changeset
453 jobject *
a61af66fc99e Initial load
duke
parents:
diff changeset
454 JvmtiEnvBase::new_jobjectArray(int length, Handle *handles) {
a61af66fc99e Initial load
duke
parents:
diff changeset
455 if (length == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
456 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
457 }
a61af66fc99e Initial load
duke
parents:
diff changeset
458
a61af66fc99e Initial load
duke
parents:
diff changeset
459 jobject *objArray = (jobject *) jvmtiMalloc(sizeof(jobject) * length);
a61af66fc99e Initial load
duke
parents:
diff changeset
460 NULL_CHECK(objArray, NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
461
a61af66fc99e Initial load
duke
parents:
diff changeset
462 for (int i=0; i<length; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
463 objArray[i] = jni_reference(handles[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
464 }
a61af66fc99e Initial load
duke
parents:
diff changeset
465 return objArray;
a61af66fc99e Initial load
duke
parents:
diff changeset
466 }
a61af66fc99e Initial load
duke
parents:
diff changeset
467
a61af66fc99e Initial load
duke
parents:
diff changeset
468 jthread *
a61af66fc99e Initial load
duke
parents:
diff changeset
469 JvmtiEnvBase::new_jthreadArray(int length, Handle *handles) {
a61af66fc99e Initial load
duke
parents:
diff changeset
470 return (jthread *) new_jobjectArray(length,handles);
a61af66fc99e Initial load
duke
parents:
diff changeset
471 }
a61af66fc99e Initial load
duke
parents:
diff changeset
472
a61af66fc99e Initial load
duke
parents:
diff changeset
473 jthreadGroup *
a61af66fc99e Initial load
duke
parents:
diff changeset
474 JvmtiEnvBase::new_jthreadGroupArray(int length, Handle *handles) {
a61af66fc99e Initial load
duke
parents:
diff changeset
475 return (jthreadGroup *) new_jobjectArray(length,handles);
a61af66fc99e Initial load
duke
parents:
diff changeset
476 }
a61af66fc99e Initial load
duke
parents:
diff changeset
477
a61af66fc99e Initial load
duke
parents:
diff changeset
478
a61af66fc99e Initial load
duke
parents:
diff changeset
479 JavaThread *
a61af66fc99e Initial load
duke
parents:
diff changeset
480 JvmtiEnvBase::get_JavaThread(jthread jni_thread) {
a61af66fc99e Initial load
duke
parents:
diff changeset
481 oop t = JNIHandles::resolve_external_guard(jni_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
482 if (t == NULL || !t->is_a(SystemDictionary::thread_klass())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
483 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
484 }
a61af66fc99e Initial load
duke
parents:
diff changeset
485 // The following returns NULL if the thread has not yet run or is in
a61af66fc99e Initial load
duke
parents:
diff changeset
486 // process of exiting
a61af66fc99e Initial load
duke
parents:
diff changeset
487 return java_lang_Thread::thread(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
488 }
a61af66fc99e Initial load
duke
parents:
diff changeset
489
a61af66fc99e Initial load
duke
parents:
diff changeset
490
a61af66fc99e Initial load
duke
parents:
diff changeset
491 // update the access_flags for the field in the klass
a61af66fc99e Initial load
duke
parents:
diff changeset
492 void
a61af66fc99e Initial load
duke
parents:
diff changeset
493 JvmtiEnvBase::update_klass_field_access_flag(fieldDescriptor *fd) {
a61af66fc99e Initial load
duke
parents:
diff changeset
494 instanceKlass* ik = instanceKlass::cast(fd->field_holder());
a61af66fc99e Initial load
duke
parents:
diff changeset
495 typeArrayOop fields = ik->fields();
a61af66fc99e Initial load
duke
parents:
diff changeset
496 fields->ushort_at_put(fd->index(), (jushort)fd->access_flags().as_short());
a61af66fc99e Initial load
duke
parents:
diff changeset
497 }
a61af66fc99e Initial load
duke
parents:
diff changeset
498
a61af66fc99e Initial load
duke
parents:
diff changeset
499
a61af66fc99e Initial load
duke
parents:
diff changeset
500 // return the vframe on the specified thread and depth, NULL if no such frame
a61af66fc99e Initial load
duke
parents:
diff changeset
501 vframe*
a61af66fc99e Initial load
duke
parents:
diff changeset
502 JvmtiEnvBase::vframeFor(JavaThread* java_thread, jint depth) {
a61af66fc99e Initial load
duke
parents:
diff changeset
503 if (!java_thread->has_last_Java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
504 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
505 }
a61af66fc99e Initial load
duke
parents:
diff changeset
506 RegisterMap reg_map(java_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
507 vframe *vf = java_thread->last_java_vframe(&reg_map);
a61af66fc99e Initial load
duke
parents:
diff changeset
508 int d = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
509 while ((vf != NULL) && (d < depth)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
510 vf = vf->java_sender();
a61af66fc99e Initial load
duke
parents:
diff changeset
511 d++;
a61af66fc99e Initial load
duke
parents:
diff changeset
512 }
a61af66fc99e Initial load
duke
parents:
diff changeset
513 return vf;
a61af66fc99e Initial load
duke
parents:
diff changeset
514 }
a61af66fc99e Initial load
duke
parents:
diff changeset
515
a61af66fc99e Initial load
duke
parents:
diff changeset
516
a61af66fc99e Initial load
duke
parents:
diff changeset
517 //
a61af66fc99e Initial load
duke
parents:
diff changeset
518 // utilities: JNI objects
a61af66fc99e Initial load
duke
parents:
diff changeset
519 //
a61af66fc99e Initial load
duke
parents:
diff changeset
520
a61af66fc99e Initial load
duke
parents:
diff changeset
521
a61af66fc99e Initial load
duke
parents:
diff changeset
522 jclass
a61af66fc99e Initial load
duke
parents:
diff changeset
523 JvmtiEnvBase::get_jni_class_non_null(klassOop k) {
a61af66fc99e Initial load
duke
parents:
diff changeset
524 assert(k != NULL, "k != NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
525 return (jclass)jni_reference(Klass::cast(k)->java_mirror());
a61af66fc99e Initial load
duke
parents:
diff changeset
526 }
a61af66fc99e Initial load
duke
parents:
diff changeset
527
a61af66fc99e Initial load
duke
parents:
diff changeset
528 #ifndef JVMTI_KERNEL
a61af66fc99e Initial load
duke
parents:
diff changeset
529
a61af66fc99e Initial load
duke
parents:
diff changeset
530 //
a61af66fc99e Initial load
duke
parents:
diff changeset
531 // Field Information
a61af66fc99e Initial load
duke
parents:
diff changeset
532 //
a61af66fc99e Initial load
duke
parents:
diff changeset
533
a61af66fc99e Initial load
duke
parents:
diff changeset
534 bool
a61af66fc99e Initial load
duke
parents:
diff changeset
535 JvmtiEnvBase::get_field_descriptor(klassOop k, jfieldID field, fieldDescriptor* fd) {
a61af66fc99e Initial load
duke
parents:
diff changeset
536 if (!jfieldIDWorkaround::is_valid_jfieldID(k, field)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
537 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
538 }
a61af66fc99e Initial load
duke
parents:
diff changeset
539 bool found = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
540 if (jfieldIDWorkaround::is_static_jfieldID(field)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
541 JNIid* id = jfieldIDWorkaround::from_static_jfieldID(field);
a61af66fc99e Initial load
duke
parents:
diff changeset
542 int offset = id->offset();
a61af66fc99e Initial load
duke
parents:
diff changeset
543 klassOop holder = id->holder();
a61af66fc99e Initial load
duke
parents:
diff changeset
544 found = instanceKlass::cast(holder)->find_local_field_from_offset(offset, true, fd);
a61af66fc99e Initial load
duke
parents:
diff changeset
545 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
546 // Non-static field. The fieldID is really the offset of the field within the object.
a61af66fc99e Initial load
duke
parents:
diff changeset
547 int offset = jfieldIDWorkaround::from_instance_jfieldID(k, field);
a61af66fc99e Initial load
duke
parents:
diff changeset
548 found = instanceKlass::cast(k)->find_field_from_offset(offset, false, fd);
a61af66fc99e Initial load
duke
parents:
diff changeset
549 }
a61af66fc99e Initial load
duke
parents:
diff changeset
550 return found;
a61af66fc99e Initial load
duke
parents:
diff changeset
551 }
a61af66fc99e Initial load
duke
parents:
diff changeset
552
a61af66fc99e Initial load
duke
parents:
diff changeset
553 //
a61af66fc99e Initial load
duke
parents:
diff changeset
554 // Object Monitor Information
a61af66fc99e Initial load
duke
parents:
diff changeset
555 //
a61af66fc99e Initial load
duke
parents:
diff changeset
556
a61af66fc99e Initial load
duke
parents:
diff changeset
557 //
a61af66fc99e Initial load
duke
parents:
diff changeset
558 // Count the number of objects for a lightweight monitor. The hobj
a61af66fc99e Initial load
duke
parents:
diff changeset
559 // parameter is object that owns the monitor so this routine will
a61af66fc99e Initial load
duke
parents:
diff changeset
560 // count the number of times the same object was locked by frames
a61af66fc99e Initial load
duke
parents:
diff changeset
561 // in java_thread.
a61af66fc99e Initial load
duke
parents:
diff changeset
562 //
a61af66fc99e Initial load
duke
parents:
diff changeset
563 jint
a61af66fc99e Initial load
duke
parents:
diff changeset
564 JvmtiEnvBase::count_locked_objects(JavaThread *java_thread, Handle hobj) {
a61af66fc99e Initial load
duke
parents:
diff changeset
565 jint ret = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
566 if (!java_thread->has_last_Java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
567 return ret; // no Java frames so no monitors
a61af66fc99e Initial load
duke
parents:
diff changeset
568 }
a61af66fc99e Initial load
duke
parents:
diff changeset
569
a61af66fc99e Initial load
duke
parents:
diff changeset
570 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
571 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
572 RegisterMap reg_map(java_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
573
a61af66fc99e Initial load
duke
parents:
diff changeset
574 for(javaVFrame *jvf=java_thread->last_java_vframe(&reg_map); jvf != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
575 jvf = jvf->java_sender()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
576 GrowableArray<MonitorInfo*>* mons = jvf->monitors();
a61af66fc99e Initial load
duke
parents:
diff changeset
577 if (!mons->is_empty()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
578 for (int i = 0; i < mons->length(); i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
579 MonitorInfo *mi = mons->at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
580
a61af66fc99e Initial load
duke
parents:
diff changeset
581 // see if owner of the monitor is our object
a61af66fc99e Initial load
duke
parents:
diff changeset
582 if (mi->owner() != NULL && mi->owner() == hobj()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
583 ret++;
a61af66fc99e Initial load
duke
parents:
diff changeset
584 }
a61af66fc99e Initial load
duke
parents:
diff changeset
585 }
a61af66fc99e Initial load
duke
parents:
diff changeset
586 }
a61af66fc99e Initial load
duke
parents:
diff changeset
587 }
a61af66fc99e Initial load
duke
parents:
diff changeset
588 return ret;
a61af66fc99e Initial load
duke
parents:
diff changeset
589 }
a61af66fc99e Initial load
duke
parents:
diff changeset
590
a61af66fc99e Initial load
duke
parents:
diff changeset
591
a61af66fc99e Initial load
duke
parents:
diff changeset
592
a61af66fc99e Initial load
duke
parents:
diff changeset
593 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
594 JvmtiEnvBase::get_current_contended_monitor(JavaThread *calling_thread, JavaThread *java_thread, jobject *monitor_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
595 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
596 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
597 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
598 assert((SafepointSynchronize::is_at_safepoint() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
599 is_thread_fully_suspended(java_thread, false, &debug_bits)),
a61af66fc99e Initial load
duke
parents:
diff changeset
600 "at safepoint or target thread is suspended");
a61af66fc99e Initial load
duke
parents:
diff changeset
601 oop obj = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
602 ObjectMonitor *mon = java_thread->current_waiting_monitor();
a61af66fc99e Initial load
duke
parents:
diff changeset
603 if (mon == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
604 // thread is not doing an Object.wait() call
a61af66fc99e Initial load
duke
parents:
diff changeset
605 mon = java_thread->current_pending_monitor();
a61af66fc99e Initial load
duke
parents:
diff changeset
606 if (mon != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
607 // The thread is trying to enter() or raw_enter() an ObjectMonitor.
a61af66fc99e Initial load
duke
parents:
diff changeset
608 obj = (oop)mon->object();
a61af66fc99e Initial load
duke
parents:
diff changeset
609 // If obj == NULL, then ObjectMonitor is raw which doesn't count
a61af66fc99e Initial load
duke
parents:
diff changeset
610 // as contended for this API
a61af66fc99e Initial load
duke
parents:
diff changeset
611 }
a61af66fc99e Initial load
duke
parents:
diff changeset
612 // implied else: no contended ObjectMonitor
a61af66fc99e Initial load
duke
parents:
diff changeset
613 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
614 // thread is doing an Object.wait() call
a61af66fc99e Initial load
duke
parents:
diff changeset
615 obj = (oop)mon->object();
a61af66fc99e Initial load
duke
parents:
diff changeset
616 assert(obj != NULL, "Object.wait() should have an object");
a61af66fc99e Initial load
duke
parents:
diff changeset
617 }
a61af66fc99e Initial load
duke
parents:
diff changeset
618
a61af66fc99e Initial load
duke
parents:
diff changeset
619 if (obj == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
620 *monitor_ptr = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
621 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
622 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
623 Handle hobj(obj);
a61af66fc99e Initial load
duke
parents:
diff changeset
624 *monitor_ptr = jni_reference(calling_thread, hobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
625 }
a61af66fc99e Initial load
duke
parents:
diff changeset
626 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
627 }
a61af66fc99e Initial load
duke
parents:
diff changeset
628
a61af66fc99e Initial load
duke
parents:
diff changeset
629
a61af66fc99e Initial load
duke
parents:
diff changeset
630 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
631 JvmtiEnvBase::get_owned_monitors(JavaThread *calling_thread, JavaThread* java_thread,
a61af66fc99e Initial load
duke
parents:
diff changeset
632 GrowableArray<jvmtiMonitorStackDepthInfo*> *owned_monitors_list) {
a61af66fc99e Initial load
duke
parents:
diff changeset
633 jvmtiError err = JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
634 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
635 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
636 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
637 assert((SafepointSynchronize::is_at_safepoint() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
638 is_thread_fully_suspended(java_thread, false, &debug_bits)),
a61af66fc99e Initial load
duke
parents:
diff changeset
639 "at safepoint or target thread is suspended");
a61af66fc99e Initial load
duke
parents:
diff changeset
640
a61af66fc99e Initial load
duke
parents:
diff changeset
641 if (java_thread->has_last_Java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
642 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
643 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
644 RegisterMap reg_map(java_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
645
a61af66fc99e Initial load
duke
parents:
diff changeset
646 int depth = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
647 for (javaVFrame *jvf = java_thread->last_java_vframe(&reg_map); jvf != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
648 jvf = jvf->java_sender()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
649 if (depth++ < MaxJavaStackTraceDepth) { // check for stack too deep
a61af66fc99e Initial load
duke
parents:
diff changeset
650 // add locked objects for this frame into list
a61af66fc99e Initial load
duke
parents:
diff changeset
651 err = get_locked_objects_in_frame(calling_thread, java_thread, jvf, owned_monitors_list, depth-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
652 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
653 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
654 }
a61af66fc99e Initial load
duke
parents:
diff changeset
655 }
a61af66fc99e Initial load
duke
parents:
diff changeset
656 }
a61af66fc99e Initial load
duke
parents:
diff changeset
657 }
a61af66fc99e Initial load
duke
parents:
diff changeset
658
a61af66fc99e Initial load
duke
parents:
diff changeset
659 // Get off stack monitors. (e.g. acquired via jni MonitorEnter).
a61af66fc99e Initial load
duke
parents:
diff changeset
660 JvmtiMonitorClosure jmc(java_thread, calling_thread, owned_monitors_list, this);
a61af66fc99e Initial load
duke
parents:
diff changeset
661 ObjectSynchronizer::monitors_iterate(&jmc);
a61af66fc99e Initial load
duke
parents:
diff changeset
662 err = jmc.error();
a61af66fc99e Initial load
duke
parents:
diff changeset
663
a61af66fc99e Initial load
duke
parents:
diff changeset
664 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
665 }
a61af66fc99e Initial load
duke
parents:
diff changeset
666
a61af66fc99e Initial load
duke
parents:
diff changeset
667 // Save JNI local handles for any objects that this frame owns.
a61af66fc99e Initial load
duke
parents:
diff changeset
668 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
669 JvmtiEnvBase::get_locked_objects_in_frame(JavaThread* calling_thread, JavaThread* java_thread,
a61af66fc99e Initial load
duke
parents:
diff changeset
670 javaVFrame *jvf, GrowableArray<jvmtiMonitorStackDepthInfo*>* owned_monitors_list, int stack_depth) {
a61af66fc99e Initial load
duke
parents:
diff changeset
671 jvmtiError err = JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
672 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
673
a61af66fc99e Initial load
duke
parents:
diff changeset
674 GrowableArray<MonitorInfo*>* mons = jvf->monitors();
a61af66fc99e Initial load
duke
parents:
diff changeset
675 if (mons->is_empty()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
676 return err; // this javaVFrame holds no monitors
a61af66fc99e Initial load
duke
parents:
diff changeset
677 }
a61af66fc99e Initial load
duke
parents:
diff changeset
678
a61af66fc99e Initial load
duke
parents:
diff changeset
679 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
680 oop wait_obj = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
681 {
a61af66fc99e Initial load
duke
parents:
diff changeset
682 // save object of current wait() call (if any) for later comparison
a61af66fc99e Initial load
duke
parents:
diff changeset
683 ObjectMonitor *mon = java_thread->current_waiting_monitor();
a61af66fc99e Initial load
duke
parents:
diff changeset
684 if (mon != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
685 wait_obj = (oop)mon->object();
a61af66fc99e Initial load
duke
parents:
diff changeset
686 }
a61af66fc99e Initial load
duke
parents:
diff changeset
687 }
a61af66fc99e Initial load
duke
parents:
diff changeset
688 oop pending_obj = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
689 {
a61af66fc99e Initial load
duke
parents:
diff changeset
690 // save object of current enter() call (if any) for later comparison
a61af66fc99e Initial load
duke
parents:
diff changeset
691 ObjectMonitor *mon = java_thread->current_pending_monitor();
a61af66fc99e Initial load
duke
parents:
diff changeset
692 if (mon != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
693 pending_obj = (oop)mon->object();
a61af66fc99e Initial load
duke
parents:
diff changeset
694 }
a61af66fc99e Initial load
duke
parents:
diff changeset
695 }
a61af66fc99e Initial load
duke
parents:
diff changeset
696
a61af66fc99e Initial load
duke
parents:
diff changeset
697 for (int i = 0; i < mons->length(); i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
698 MonitorInfo *mi = mons->at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
699
a61af66fc99e Initial load
duke
parents:
diff changeset
700 oop obj = mi->owner();
a61af66fc99e Initial load
duke
parents:
diff changeset
701 if (obj == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
702 // this monitor doesn't have an owning object so skip it
a61af66fc99e Initial load
duke
parents:
diff changeset
703 continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
704 }
a61af66fc99e Initial load
duke
parents:
diff changeset
705
a61af66fc99e Initial load
duke
parents:
diff changeset
706 if (wait_obj == obj) {
a61af66fc99e Initial load
duke
parents:
diff changeset
707 // the thread is waiting on this monitor so it isn't really owned
a61af66fc99e Initial load
duke
parents:
diff changeset
708 continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
709 }
a61af66fc99e Initial load
duke
parents:
diff changeset
710
a61af66fc99e Initial load
duke
parents:
diff changeset
711 if (pending_obj == obj) {
a61af66fc99e Initial load
duke
parents:
diff changeset
712 // the thread is pending on this monitor so it isn't really owned
a61af66fc99e Initial load
duke
parents:
diff changeset
713 continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
714 }
a61af66fc99e Initial load
duke
parents:
diff changeset
715
a61af66fc99e Initial load
duke
parents:
diff changeset
716 if (owned_monitors_list->length() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
717 // Our list has at least one object on it so we have to check
a61af66fc99e Initial load
duke
parents:
diff changeset
718 // for recursive object locking
a61af66fc99e Initial load
duke
parents:
diff changeset
719 bool found = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
720 for (int j = 0; j < owned_monitors_list->length(); j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
721 jobject jobj = ((jvmtiMonitorStackDepthInfo*)owned_monitors_list->at(j))->monitor;
a61af66fc99e Initial load
duke
parents:
diff changeset
722 oop check = JNIHandles::resolve(jobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
723 if (check == obj) {
a61af66fc99e Initial load
duke
parents:
diff changeset
724 found = true; // we found the object
a61af66fc99e Initial load
duke
parents:
diff changeset
725 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
726 }
a61af66fc99e Initial load
duke
parents:
diff changeset
727 }
a61af66fc99e Initial load
duke
parents:
diff changeset
728
a61af66fc99e Initial load
duke
parents:
diff changeset
729 if (found) {
a61af66fc99e Initial load
duke
parents:
diff changeset
730 // already have this object so don't include it
a61af66fc99e Initial load
duke
parents:
diff changeset
731 continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
732 }
a61af66fc99e Initial load
duke
parents:
diff changeset
733 }
a61af66fc99e Initial load
duke
parents:
diff changeset
734
a61af66fc99e Initial load
duke
parents:
diff changeset
735 // add the owning object to our list
a61af66fc99e Initial load
duke
parents:
diff changeset
736 jvmtiMonitorStackDepthInfo *jmsdi;
a61af66fc99e Initial load
duke
parents:
diff changeset
737 err = allocate(sizeof(jvmtiMonitorStackDepthInfo), (unsigned char **)&jmsdi);
a61af66fc99e Initial load
duke
parents:
diff changeset
738 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
739 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
740 }
a61af66fc99e Initial load
duke
parents:
diff changeset
741 Handle hobj(obj);
a61af66fc99e Initial load
duke
parents:
diff changeset
742 jmsdi->monitor = jni_reference(calling_thread, hobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
743 jmsdi->stack_depth = stack_depth;
a61af66fc99e Initial load
duke
parents:
diff changeset
744 owned_monitors_list->append(jmsdi);
a61af66fc99e Initial load
duke
parents:
diff changeset
745 }
a61af66fc99e Initial load
duke
parents:
diff changeset
746
a61af66fc99e Initial load
duke
parents:
diff changeset
747 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
748 }
a61af66fc99e Initial load
duke
parents:
diff changeset
749
a61af66fc99e Initial load
duke
parents:
diff changeset
750 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
751 JvmtiEnvBase::get_stack_trace(JavaThread *java_thread,
a61af66fc99e Initial load
duke
parents:
diff changeset
752 jint start_depth, jint max_count,
a61af66fc99e Initial load
duke
parents:
diff changeset
753 jvmtiFrameInfo* frame_buffer, jint* count_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
754 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
755 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
756 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
757 assert((SafepointSynchronize::is_at_safepoint() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
758 is_thread_fully_suspended(java_thread, false, &debug_bits)),
a61af66fc99e Initial load
duke
parents:
diff changeset
759 "at safepoint or target thread is suspended");
a61af66fc99e Initial load
duke
parents:
diff changeset
760 int count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
761 if (java_thread->has_last_Java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
762 RegisterMap reg_map(java_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
763 Thread* current_thread = Thread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
764 ResourceMark rm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
765 javaVFrame *jvf = java_thread->last_java_vframe(&reg_map);
a61af66fc99e Initial load
duke
parents:
diff changeset
766 HandleMark hm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
767 if (start_depth != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
768 if (start_depth > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
769 for (int j = 0; j < start_depth && jvf != NULL; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
770 jvf = jvf->java_sender();
a61af66fc99e Initial load
duke
parents:
diff changeset
771 }
a61af66fc99e Initial load
duke
parents:
diff changeset
772 if (jvf == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
773 // start_depth is deeper than the stack depth
a61af66fc99e Initial load
duke
parents:
diff changeset
774 return JVMTI_ERROR_ILLEGAL_ARGUMENT;
a61af66fc99e Initial load
duke
parents:
diff changeset
775 }
a61af66fc99e Initial load
duke
parents:
diff changeset
776 } else { // start_depth < 0
a61af66fc99e Initial load
duke
parents:
diff changeset
777 // we are referencing the starting depth based on the oldest
a61af66fc99e Initial load
duke
parents:
diff changeset
778 // part of the stack.
a61af66fc99e Initial load
duke
parents:
diff changeset
779 // optimize to limit the number of times that java_sender() is called
a61af66fc99e Initial load
duke
parents:
diff changeset
780 javaVFrame *jvf_cursor = jvf;
a61af66fc99e Initial load
duke
parents:
diff changeset
781 javaVFrame *jvf_prev = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
782 javaVFrame *jvf_prev_prev;
a61af66fc99e Initial load
duke
parents:
diff changeset
783 int j = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
784 while (jvf_cursor != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
785 jvf_prev_prev = jvf_prev;
a61af66fc99e Initial load
duke
parents:
diff changeset
786 jvf_prev = jvf_cursor;
a61af66fc99e Initial load
duke
parents:
diff changeset
787 for (j = 0; j > start_depth && jvf_cursor != NULL; j--) {
a61af66fc99e Initial load
duke
parents:
diff changeset
788 jvf_cursor = jvf_cursor->java_sender();
a61af66fc99e Initial load
duke
parents:
diff changeset
789 }
a61af66fc99e Initial load
duke
parents:
diff changeset
790 }
a61af66fc99e Initial load
duke
parents:
diff changeset
791 if (j == start_depth) {
a61af66fc99e Initial load
duke
parents:
diff changeset
792 // previous pointer is exactly where we want to start
a61af66fc99e Initial load
duke
parents:
diff changeset
793 jvf = jvf_prev;
a61af66fc99e Initial load
duke
parents:
diff changeset
794 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
795 // we need to back up further to get to the right place
a61af66fc99e Initial load
duke
parents:
diff changeset
796 if (jvf_prev_prev == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
797 // the -start_depth is greater than the stack depth
a61af66fc99e Initial load
duke
parents:
diff changeset
798 return JVMTI_ERROR_ILLEGAL_ARGUMENT;
a61af66fc99e Initial load
duke
parents:
diff changeset
799 }
a61af66fc99e Initial load
duke
parents:
diff changeset
800 // j now is the number of frames on the stack starting with
a61af66fc99e Initial load
duke
parents:
diff changeset
801 // jvf_prev, we start from jvf_prev_prev and move older on
a61af66fc99e Initial load
duke
parents:
diff changeset
802 // the stack that many, the result is -start_depth frames
a61af66fc99e Initial load
duke
parents:
diff changeset
803 // remaining.
a61af66fc99e Initial load
duke
parents:
diff changeset
804 jvf = jvf_prev_prev;
a61af66fc99e Initial load
duke
parents:
diff changeset
805 for (; j < 0; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
806 jvf = jvf->java_sender();
a61af66fc99e Initial load
duke
parents:
diff changeset
807 }
a61af66fc99e Initial load
duke
parents:
diff changeset
808 }
a61af66fc99e Initial load
duke
parents:
diff changeset
809 }
a61af66fc99e Initial load
duke
parents:
diff changeset
810 }
a61af66fc99e Initial load
duke
parents:
diff changeset
811 for (; count < max_count && jvf != NULL; count++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
812 frame_buffer[count].method = jvf->method()->jmethod_id();
a61af66fc99e Initial load
duke
parents:
diff changeset
813 frame_buffer[count].location = (jvf->method()->is_native() ? -1 : jvf->bci());
a61af66fc99e Initial load
duke
parents:
diff changeset
814 jvf = jvf->java_sender();
a61af66fc99e Initial load
duke
parents:
diff changeset
815 }
a61af66fc99e Initial load
duke
parents:
diff changeset
816 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
817 if (start_depth != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
818 // no frames and there is a starting depth
a61af66fc99e Initial load
duke
parents:
diff changeset
819 return JVMTI_ERROR_ILLEGAL_ARGUMENT;
a61af66fc99e Initial load
duke
parents:
diff changeset
820 }
a61af66fc99e Initial load
duke
parents:
diff changeset
821 }
a61af66fc99e Initial load
duke
parents:
diff changeset
822 *count_ptr = count;
a61af66fc99e Initial load
duke
parents:
diff changeset
823 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
824 }
a61af66fc99e Initial load
duke
parents:
diff changeset
825
a61af66fc99e Initial load
duke
parents:
diff changeset
826 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
827 JvmtiEnvBase::get_frame_count(JvmtiThreadState *state, jint *count_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
828 assert((state != NULL),
a61af66fc99e Initial load
duke
parents:
diff changeset
829 "JavaThread should create JvmtiThreadState before calling this method");
a61af66fc99e Initial load
duke
parents:
diff changeset
830 *count_ptr = state->count_frames();
a61af66fc99e Initial load
duke
parents:
diff changeset
831 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
832 }
a61af66fc99e Initial load
duke
parents:
diff changeset
833
a61af66fc99e Initial load
duke
parents:
diff changeset
834 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
835 JvmtiEnvBase::get_frame_location(JavaThread *java_thread, jint depth,
a61af66fc99e Initial load
duke
parents:
diff changeset
836 jmethodID* method_ptr, jlocation* location_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
837 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
838 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
839 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
840 assert((SafepointSynchronize::is_at_safepoint() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
841 is_thread_fully_suspended(java_thread, false, &debug_bits)),
a61af66fc99e Initial load
duke
parents:
diff changeset
842 "at safepoint or target thread is suspended");
a61af66fc99e Initial load
duke
parents:
diff changeset
843 Thread* current_thread = Thread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
844 ResourceMark rm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
845
a61af66fc99e Initial load
duke
parents:
diff changeset
846 vframe *vf = vframeFor(java_thread, depth);
a61af66fc99e Initial load
duke
parents:
diff changeset
847 if (vf == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
848 return JVMTI_ERROR_NO_MORE_FRAMES;
a61af66fc99e Initial load
duke
parents:
diff changeset
849 }
a61af66fc99e Initial load
duke
parents:
diff changeset
850
a61af66fc99e Initial load
duke
parents:
diff changeset
851 // vframeFor should return a java frame. If it doesn't
a61af66fc99e Initial load
duke
parents:
diff changeset
852 // it means we've got an internal error and we return the
a61af66fc99e Initial load
duke
parents:
diff changeset
853 // error in product mode. In debug mode we will instead
a61af66fc99e Initial load
duke
parents:
diff changeset
854 // attempt to cast the vframe to a javaVFrame and will
a61af66fc99e Initial load
duke
parents:
diff changeset
855 // cause an assertion/crash to allow further diagnosis.
a61af66fc99e Initial load
duke
parents:
diff changeset
856 #ifdef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
857 if (!vf->is_java_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
858 return JVMTI_ERROR_INTERNAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
859 }
a61af66fc99e Initial load
duke
parents:
diff changeset
860 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
861
a61af66fc99e Initial load
duke
parents:
diff changeset
862 HandleMark hm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
863 javaVFrame *jvf = javaVFrame::cast(vf);
a61af66fc99e Initial load
duke
parents:
diff changeset
864 methodOop method = jvf->method();
a61af66fc99e Initial load
duke
parents:
diff changeset
865 if (method->is_native()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
866 *location_ptr = -1;
a61af66fc99e Initial load
duke
parents:
diff changeset
867 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
868 *location_ptr = jvf->bci();
a61af66fc99e Initial load
duke
parents:
diff changeset
869 }
a61af66fc99e Initial load
duke
parents:
diff changeset
870 *method_ptr = method->jmethod_id();
a61af66fc99e Initial load
duke
parents:
diff changeset
871
a61af66fc99e Initial load
duke
parents:
diff changeset
872 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
873 }
a61af66fc99e Initial load
duke
parents:
diff changeset
874
a61af66fc99e Initial load
duke
parents:
diff changeset
875
a61af66fc99e Initial load
duke
parents:
diff changeset
876 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
877 JvmtiEnvBase::get_object_monitor_usage(JavaThread* calling_thread, jobject object, jvmtiMonitorUsage* info_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
878 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
879 Handle hobj;
a61af66fc99e Initial load
duke
parents:
diff changeset
880
a61af66fc99e Initial load
duke
parents:
diff changeset
881 bool at_safepoint = SafepointSynchronize::is_at_safepoint();
a61af66fc99e Initial load
duke
parents:
diff changeset
882
a61af66fc99e Initial load
duke
parents:
diff changeset
883 // Check arguments
a61af66fc99e Initial load
duke
parents:
diff changeset
884 {
a61af66fc99e Initial load
duke
parents:
diff changeset
885 oop mirror = JNIHandles::resolve_external_guard(object);
a61af66fc99e Initial load
duke
parents:
diff changeset
886 NULL_CHECK(mirror, JVMTI_ERROR_INVALID_OBJECT);
a61af66fc99e Initial load
duke
parents:
diff changeset
887 NULL_CHECK(info_ptr, JVMTI_ERROR_NULL_POINTER);
a61af66fc99e Initial load
duke
parents:
diff changeset
888
a61af66fc99e Initial load
duke
parents:
diff changeset
889 hobj = Handle(mirror);
a61af66fc99e Initial load
duke
parents:
diff changeset
890 }
a61af66fc99e Initial load
duke
parents:
diff changeset
891
a61af66fc99e Initial load
duke
parents:
diff changeset
892 JavaThread *owning_thread = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
893 ObjectMonitor *mon = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
894 jvmtiMonitorUsage ret = {
a61af66fc99e Initial load
duke
parents:
diff changeset
895 NULL, 0, 0, NULL, 0, NULL
a61af66fc99e Initial load
duke
parents:
diff changeset
896 };
a61af66fc99e Initial load
duke
parents:
diff changeset
897
a61af66fc99e Initial load
duke
parents:
diff changeset
898 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
899 // first derive the object's owner and entry_count (if any)
a61af66fc99e Initial load
duke
parents:
diff changeset
900 {
a61af66fc99e Initial load
duke
parents:
diff changeset
901 // Revoke any biases before querying the mark word
a61af66fc99e Initial load
duke
parents:
diff changeset
902 if (SafepointSynchronize::is_at_safepoint()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
903 BiasedLocking::revoke_at_safepoint(hobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
904 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
905 BiasedLocking::revoke_and_rebias(hobj, false, calling_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
906 }
a61af66fc99e Initial load
duke
parents:
diff changeset
907
a61af66fc99e Initial load
duke
parents:
diff changeset
908 address owner = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
909 {
a61af66fc99e Initial load
duke
parents:
diff changeset
910 markOop mark = hobj()->mark();
a61af66fc99e Initial load
duke
parents:
diff changeset
911
a61af66fc99e Initial load
duke
parents:
diff changeset
912 if (!mark->has_monitor()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
913 // this object has a lightweight monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
914
a61af66fc99e Initial load
duke
parents:
diff changeset
915 if (mark->has_locker()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
916 owner = (address)mark->locker(); // save the address of the Lock word
a61af66fc99e Initial load
duke
parents:
diff changeset
917 }
a61af66fc99e Initial load
duke
parents:
diff changeset
918 // implied else: no owner
a61af66fc99e Initial load
duke
parents:
diff changeset
919 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
920 // this object has a heavyweight monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
921 mon = mark->monitor();
a61af66fc99e Initial load
duke
parents:
diff changeset
922
a61af66fc99e Initial load
duke
parents:
diff changeset
923 // The owner field of a heavyweight monitor may be NULL for no
a61af66fc99e Initial load
duke
parents:
diff changeset
924 // owner, a JavaThread * or it may still be the address of the
a61af66fc99e Initial load
duke
parents:
diff changeset
925 // Lock word in a JavaThread's stack. A monitor can be inflated
a61af66fc99e Initial load
duke
parents:
diff changeset
926 // by a non-owning JavaThread, but only the owning JavaThread
a61af66fc99e Initial load
duke
parents:
diff changeset
927 // can change the owner field from the Lock word to the
a61af66fc99e Initial load
duke
parents:
diff changeset
928 // JavaThread * and it may not have done that yet.
a61af66fc99e Initial load
duke
parents:
diff changeset
929 owner = (address)mon->owner();
a61af66fc99e Initial load
duke
parents:
diff changeset
930 }
a61af66fc99e Initial load
duke
parents:
diff changeset
931 }
a61af66fc99e Initial load
duke
parents:
diff changeset
932
a61af66fc99e Initial load
duke
parents:
diff changeset
933 if (owner != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
934 // This monitor is owned so we have to find the owning JavaThread.
a61af66fc99e Initial load
duke
parents:
diff changeset
935 // Since owning_thread_from_monitor_owner() grabs a lock, GC can
a61af66fc99e Initial load
duke
parents:
diff changeset
936 // move our object at this point. However, our owner value is safe
a61af66fc99e Initial load
duke
parents:
diff changeset
937 // since it is either the Lock word on a stack or a JavaThread *.
a61af66fc99e Initial load
duke
parents:
diff changeset
938 owning_thread = Threads::owning_thread_from_monitor_owner(owner, !at_safepoint);
a61af66fc99e Initial load
duke
parents:
diff changeset
939 assert(owning_thread != NULL, "sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
940 if (owning_thread != NULL) { // robustness
a61af66fc99e Initial load
duke
parents:
diff changeset
941 // The monitor's owner either has to be the current thread, at safepoint
a61af66fc99e Initial load
duke
parents:
diff changeset
942 // or it has to be suspended. Any of these conditions will prevent both
a61af66fc99e Initial load
duke
parents:
diff changeset
943 // contending and waiting threads from modifying the state of
a61af66fc99e Initial load
duke
parents:
diff changeset
944 // the monitor.
a61af66fc99e Initial load
duke
parents:
diff changeset
945 if (!at_safepoint && !JvmtiEnv::is_thread_fully_suspended(owning_thread, true, &debug_bits)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
946 return JVMTI_ERROR_THREAD_NOT_SUSPENDED;
a61af66fc99e Initial load
duke
parents:
diff changeset
947 }
a61af66fc99e Initial load
duke
parents:
diff changeset
948 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
949 Handle th(owning_thread->threadObj());
a61af66fc99e Initial load
duke
parents:
diff changeset
950 ret.owner = (jthread)jni_reference(calling_thread, th);
a61af66fc99e Initial load
duke
parents:
diff changeset
951 }
a61af66fc99e Initial load
duke
parents:
diff changeset
952 // implied else: no owner
a61af66fc99e Initial load
duke
parents:
diff changeset
953 }
a61af66fc99e Initial load
duke
parents:
diff changeset
954
a61af66fc99e Initial load
duke
parents:
diff changeset
955 if (owning_thread != NULL) { // monitor is owned
a61af66fc99e Initial load
duke
parents:
diff changeset
956 if ((address)owning_thread == owner) {
a61af66fc99e Initial load
duke
parents:
diff changeset
957 // the owner field is the JavaThread *
a61af66fc99e Initial load
duke
parents:
diff changeset
958 assert(mon != NULL,
a61af66fc99e Initial load
duke
parents:
diff changeset
959 "must have heavyweight monitor with JavaThread * owner");
a61af66fc99e Initial load
duke
parents:
diff changeset
960 ret.entry_count = mon->recursions() + 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
961 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
962 // The owner field is the Lock word on the JavaThread's stack
a61af66fc99e Initial load
duke
parents:
diff changeset
963 // so the recursions field is not valid. We have to count the
a61af66fc99e Initial load
duke
parents:
diff changeset
964 // number of recursive monitor entries the hard way. We pass
a61af66fc99e Initial load
duke
parents:
diff changeset
965 // a handle to survive any GCs along the way.
a61af66fc99e Initial load
duke
parents:
diff changeset
966 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
967 ret.entry_count = count_locked_objects(owning_thread, hobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
968 }
a61af66fc99e Initial load
duke
parents:
diff changeset
969 }
a61af66fc99e Initial load
duke
parents:
diff changeset
970 // implied else: entry_count == 0
a61af66fc99e Initial load
duke
parents:
diff changeset
971 }
a61af66fc99e Initial load
duke
parents:
diff changeset
972
a61af66fc99e Initial load
duke
parents:
diff changeset
973 int nWant,nWait;
a61af66fc99e Initial load
duke
parents:
diff changeset
974 if (mon != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
975 // this object has a heavyweight monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
976 nWant = mon->contentions(); // # of threads contending for monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
977 nWait = mon->waiters(); // # of threads in Object.wait()
a61af66fc99e Initial load
duke
parents:
diff changeset
978 ret.waiter_count = nWant + nWait;
a61af66fc99e Initial load
duke
parents:
diff changeset
979 ret.notify_waiter_count = nWait;
a61af66fc99e Initial load
duke
parents:
diff changeset
980 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
981 // this object has a lightweight monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
982 ret.waiter_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
983 ret.notify_waiter_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
984 }
a61af66fc99e Initial load
duke
parents:
diff changeset
985
a61af66fc99e Initial load
duke
parents:
diff changeset
986 // Allocate memory for heavyweight and lightweight monitor.
a61af66fc99e Initial load
duke
parents:
diff changeset
987 jvmtiError err;
a61af66fc99e Initial load
duke
parents:
diff changeset
988 err = allocate(ret.waiter_count * sizeof(jthread *), (unsigned char**)&ret.waiters);
a61af66fc99e Initial load
duke
parents:
diff changeset
989 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
990 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
991 }
a61af66fc99e Initial load
duke
parents:
diff changeset
992 err = allocate(ret.notify_waiter_count * sizeof(jthread *),
a61af66fc99e Initial load
duke
parents:
diff changeset
993 (unsigned char**)&ret.notify_waiters);
a61af66fc99e Initial load
duke
parents:
diff changeset
994 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
995 deallocate((unsigned char*)ret.waiters);
a61af66fc99e Initial load
duke
parents:
diff changeset
996 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
997 }
a61af66fc99e Initial load
duke
parents:
diff changeset
998
a61af66fc99e Initial load
duke
parents:
diff changeset
999 // now derive the rest of the fields
a61af66fc99e Initial load
duke
parents:
diff changeset
1000 if (mon != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1001 // this object has a heavyweight monitor
a61af66fc99e Initial load
duke
parents:
diff changeset
1002
a61af66fc99e Initial load
duke
parents:
diff changeset
1003 // Number of waiters may actually be less than the waiter count.
a61af66fc99e Initial load
duke
parents:
diff changeset
1004 // So NULL out memory so that unused memory will be NULL.
a61af66fc99e Initial load
duke
parents:
diff changeset
1005 memset(ret.waiters, 0, ret.waiter_count * sizeof(jthread *));
a61af66fc99e Initial load
duke
parents:
diff changeset
1006 memset(ret.notify_waiters, 0, ret.notify_waiter_count * sizeof(jthread *));
a61af66fc99e Initial load
duke
parents:
diff changeset
1007
a61af66fc99e Initial load
duke
parents:
diff changeset
1008 if (ret.waiter_count > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1009 // we have contending and/or waiting threads
a61af66fc99e Initial load
duke
parents:
diff changeset
1010 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
1011 if (nWant > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1012 // we have contending threads
a61af66fc99e Initial load
duke
parents:
diff changeset
1013 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
1014 // get_pending_threads returns only java thread so we do not need to
a61af66fc99e Initial load
duke
parents:
diff changeset
1015 // check for non java threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
1016 GrowableArray<JavaThread*>* wantList = Threads::get_pending_threads(
a61af66fc99e Initial load
duke
parents:
diff changeset
1017 nWant, (address)mon, !at_safepoint);
a61af66fc99e Initial load
duke
parents:
diff changeset
1018 if (wantList->length() < nWant) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1019 // robustness: the pending list has gotten smaller
a61af66fc99e Initial load
duke
parents:
diff changeset
1020 nWant = wantList->length();
a61af66fc99e Initial load
duke
parents:
diff changeset
1021 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1022 for (int i = 0; i < nWant; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1023 JavaThread *pending_thread = wantList->at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1024 // If the monitor has no owner, then a non-suspended contending
a61af66fc99e Initial load
duke
parents:
diff changeset
1025 // thread could potentially change the state of the monitor by
a61af66fc99e Initial load
duke
parents:
diff changeset
1026 // entering it. The JVM/TI spec doesn't allow this.
a61af66fc99e Initial load
duke
parents:
diff changeset
1027 if (owning_thread == NULL && !at_safepoint &
a61af66fc99e Initial load
duke
parents:
diff changeset
1028 !JvmtiEnv::is_thread_fully_suspended(pending_thread, true, &debug_bits)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1029 if (ret.owner != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1030 destroy_jni_reference(calling_thread, ret.owner);
a61af66fc99e Initial load
duke
parents:
diff changeset
1031 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1032 for (int j = 0; j < i; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1033 destroy_jni_reference(calling_thread, ret.waiters[j]);
a61af66fc99e Initial load
duke
parents:
diff changeset
1034 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1035 deallocate((unsigned char*)ret.waiters);
a61af66fc99e Initial load
duke
parents:
diff changeset
1036 deallocate((unsigned char*)ret.notify_waiters);
a61af66fc99e Initial load
duke
parents:
diff changeset
1037 return JVMTI_ERROR_THREAD_NOT_SUSPENDED;
a61af66fc99e Initial load
duke
parents:
diff changeset
1038 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1039 Handle th(pending_thread->threadObj());
a61af66fc99e Initial load
duke
parents:
diff changeset
1040 ret.waiters[i] = (jthread)jni_reference(calling_thread, th);
a61af66fc99e Initial load
duke
parents:
diff changeset
1041 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1042 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1043 if (nWait > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1044 // we have threads in Object.wait()
a61af66fc99e Initial load
duke
parents:
diff changeset
1045 int offset = nWant; // add after any contending threads
a61af66fc99e Initial load
duke
parents:
diff changeset
1046 ObjectWaiter *waiter = mon->first_waiter();
a61af66fc99e Initial load
duke
parents:
diff changeset
1047 for (int i = 0, j = 0; i < nWait; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1048 if (waiter == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1049 // robustness: the waiting list has gotten smaller
a61af66fc99e Initial load
duke
parents:
diff changeset
1050 nWait = j;
a61af66fc99e Initial load
duke
parents:
diff changeset
1051 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1052 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1053 Thread *t = mon->thread_of_waiter(waiter);
a61af66fc99e Initial load
duke
parents:
diff changeset
1054 if (t != NULL && t->is_Java_thread()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1055 JavaThread *wjava_thread = (JavaThread *)t;
a61af66fc99e Initial load
duke
parents:
diff changeset
1056 // If the thread was found on the ObjectWaiter list, then
a61af66fc99e Initial load
duke
parents:
diff changeset
1057 // it has not been notified. This thread can't change the
a61af66fc99e Initial load
duke
parents:
diff changeset
1058 // state of the monitor so it doesn't need to be suspended.
a61af66fc99e Initial load
duke
parents:
diff changeset
1059 Handle th(wjava_thread->threadObj());
a61af66fc99e Initial load
duke
parents:
diff changeset
1060 ret.waiters[offset + j] = (jthread)jni_reference(calling_thread, th);
a61af66fc99e Initial load
duke
parents:
diff changeset
1061 ret.notify_waiters[j++] = (jthread)jni_reference(calling_thread, th);
a61af66fc99e Initial load
duke
parents:
diff changeset
1062 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1063 waiter = mon->next_waiter(waiter);
a61af66fc99e Initial load
duke
parents:
diff changeset
1064 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1065 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1066 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1067
a61af66fc99e Initial load
duke
parents:
diff changeset
1068 // Adjust count. nWant and nWait count values may be less than original.
a61af66fc99e Initial load
duke
parents:
diff changeset
1069 ret.waiter_count = nWant + nWait;
a61af66fc99e Initial load
duke
parents:
diff changeset
1070 ret.notify_waiter_count = nWait;
a61af66fc99e Initial load
duke
parents:
diff changeset
1071 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1072 // this object has a lightweight monitor and we have nothing more
a61af66fc99e Initial load
duke
parents:
diff changeset
1073 // to do here because the defaults are just fine.
a61af66fc99e Initial load
duke
parents:
diff changeset
1074 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1075
a61af66fc99e Initial load
duke
parents:
diff changeset
1076 // we don't update return parameter unless everything worked
a61af66fc99e Initial load
duke
parents:
diff changeset
1077 *info_ptr = ret;
a61af66fc99e Initial load
duke
parents:
diff changeset
1078
a61af66fc99e Initial load
duke
parents:
diff changeset
1079 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
1080 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1081
a61af66fc99e Initial load
duke
parents:
diff changeset
1082 ResourceTracker::ResourceTracker(JvmtiEnv* env) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1083 _env = env;
a61af66fc99e Initial load
duke
parents:
diff changeset
1084 _allocations = new (ResourceObj::C_HEAP) GrowableArray<unsigned char*>(20, true);
a61af66fc99e Initial load
duke
parents:
diff changeset
1085 _failed = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1086 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1087 ResourceTracker::~ResourceTracker() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1088 if (_failed) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1089 for (int i=0; i<_allocations->length(); i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1090 _env->deallocate(_allocations->at(i));
a61af66fc99e Initial load
duke
parents:
diff changeset
1091 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1092 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1093 delete _allocations;
a61af66fc99e Initial load
duke
parents:
diff changeset
1094 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1095
a61af66fc99e Initial load
duke
parents:
diff changeset
1096 jvmtiError ResourceTracker::allocate(jlong size, unsigned char** mem_ptr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1097 unsigned char *ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1098 jvmtiError err = _env->allocate(size, &ptr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1099 if (err == JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1100 _allocations->append(ptr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1101 *mem_ptr = ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1102 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1103 *mem_ptr = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1104 _failed = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1105 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1106 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
1107 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1108
a61af66fc99e Initial load
duke
parents:
diff changeset
1109 unsigned char* ResourceTracker::allocate(jlong size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1110 unsigned char* ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1111 allocate(size, &ptr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1112 return ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1113 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1114
a61af66fc99e Initial load
duke
parents:
diff changeset
1115 char* ResourceTracker::strdup(const char* str) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1116 char *dup_str = (char*)allocate(strlen(str)+1);
a61af66fc99e Initial load
duke
parents:
diff changeset
1117 if (dup_str != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1118 strcpy(dup_str, str);
a61af66fc99e Initial load
duke
parents:
diff changeset
1119 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1120 return dup_str;
a61af66fc99e Initial load
duke
parents:
diff changeset
1121 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1122
a61af66fc99e Initial load
duke
parents:
diff changeset
1123 struct StackInfoNode {
a61af66fc99e Initial load
duke
parents:
diff changeset
1124 struct StackInfoNode *next;
a61af66fc99e Initial load
duke
parents:
diff changeset
1125 jvmtiStackInfo info;
a61af66fc99e Initial load
duke
parents:
diff changeset
1126 };
a61af66fc99e Initial load
duke
parents:
diff changeset
1127
a61af66fc99e Initial load
duke
parents:
diff changeset
1128 // Create a jvmtiStackInfo inside a linked list node and create a
a61af66fc99e Initial load
duke
parents:
diff changeset
1129 // buffer for the frame information, both allocated as resource objects.
a61af66fc99e Initial load
duke
parents:
diff changeset
1130 // Fill in both the jvmtiStackInfo and the jvmtiFrameInfo.
a61af66fc99e Initial load
duke
parents:
diff changeset
1131 // Note that either or both of thr and thread_oop
a61af66fc99e Initial load
duke
parents:
diff changeset
1132 // may be null if the thread is new or has exited.
a61af66fc99e Initial load
duke
parents:
diff changeset
1133 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1134 VM_GetMultipleStackTraces::fill_frames(jthread jt, JavaThread *thr, oop thread_oop) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1135 assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint");
a61af66fc99e Initial load
duke
parents:
diff changeset
1136
a61af66fc99e Initial load
duke
parents:
diff changeset
1137 jint state = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1138 struct StackInfoNode *node = NEW_RESOURCE_OBJ(struct StackInfoNode);
a61af66fc99e Initial load
duke
parents:
diff changeset
1139 jvmtiStackInfo *infop = &(node->info);
a61af66fc99e Initial load
duke
parents:
diff changeset
1140 node->next = head();
a61af66fc99e Initial load
duke
parents:
diff changeset
1141 set_head(node);
a61af66fc99e Initial load
duke
parents:
diff changeset
1142 infop->frame_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1143 infop->thread = jt;
a61af66fc99e Initial load
duke
parents:
diff changeset
1144
a61af66fc99e Initial load
duke
parents:
diff changeset
1145 if (thread_oop != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1146 // get most state bits
a61af66fc99e Initial load
duke
parents:
diff changeset
1147 state = (jint)java_lang_Thread::get_thread_status(thread_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
1148 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1149
a61af66fc99e Initial load
duke
parents:
diff changeset
1150 if (thr != NULL) { // add more state bits if there is a JavaThead to query
a61af66fc99e Initial load
duke
parents:
diff changeset
1151 // same as is_being_ext_suspended() but without locking
a61af66fc99e Initial load
duke
parents:
diff changeset
1152 if (thr->is_ext_suspended() || thr->is_external_suspend()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1153 state |= JVMTI_THREAD_STATE_SUSPENDED;
a61af66fc99e Initial load
duke
parents:
diff changeset
1154 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1155 JavaThreadState jts = thr->thread_state();
a61af66fc99e Initial load
duke
parents:
diff changeset
1156 if (jts == _thread_in_native) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1157 state |= JVMTI_THREAD_STATE_IN_NATIVE;
a61af66fc99e Initial load
duke
parents:
diff changeset
1158 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1159 OSThread* osThread = thr->osthread();
a61af66fc99e Initial load
duke
parents:
diff changeset
1160 if (osThread != NULL && osThread->interrupted()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1161 state |= JVMTI_THREAD_STATE_INTERRUPTED;
a61af66fc99e Initial load
duke
parents:
diff changeset
1162 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1163 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1164 infop->state = state;
a61af66fc99e Initial load
duke
parents:
diff changeset
1165
a61af66fc99e Initial load
duke
parents:
diff changeset
1166 if (thr != NULL || (state & JVMTI_THREAD_STATE_ALIVE) != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1167 infop->frame_buffer = NEW_RESOURCE_ARRAY(jvmtiFrameInfo, max_frame_count());
a61af66fc99e Initial load
duke
parents:
diff changeset
1168 env()->get_stack_trace(thr, 0, max_frame_count(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1169 infop->frame_buffer, &(infop->frame_count));
a61af66fc99e Initial load
duke
parents:
diff changeset
1170 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1171 infop->frame_buffer = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1172 infop->frame_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1173 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1174 _frame_count_total += infop->frame_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
1175 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1176
a61af66fc99e Initial load
duke
parents:
diff changeset
1177 // Based on the stack information in the linked list, allocate memory
a61af66fc99e Initial load
duke
parents:
diff changeset
1178 // block to return and fill it from the info in the linked list.
a61af66fc99e Initial load
duke
parents:
diff changeset
1179 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1180 VM_GetMultipleStackTraces::allocate_and_fill_stacks(jint thread_count) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1181 // do I need to worry about alignment issues?
a61af66fc99e Initial load
duke
parents:
diff changeset
1182 jlong alloc_size = thread_count * sizeof(jvmtiStackInfo)
a61af66fc99e Initial load
duke
parents:
diff changeset
1183 + _frame_count_total * sizeof(jvmtiFrameInfo);
a61af66fc99e Initial load
duke
parents:
diff changeset
1184 env()->allocate(alloc_size, (unsigned char **)&_stack_info);
a61af66fc99e Initial load
duke
parents:
diff changeset
1185
a61af66fc99e Initial load
duke
parents:
diff changeset
1186 // pointers to move through the newly allocated space as it is filled in
a61af66fc99e Initial load
duke
parents:
diff changeset
1187 jvmtiStackInfo *si = _stack_info + thread_count; // bottom of stack info
a61af66fc99e Initial load
duke
parents:
diff changeset
1188 jvmtiFrameInfo *fi = (jvmtiFrameInfo *)si; // is the top of frame info
a61af66fc99e Initial load
duke
parents:
diff changeset
1189
a61af66fc99e Initial load
duke
parents:
diff changeset
1190 // copy information in resource area into allocated buffer
a61af66fc99e Initial load
duke
parents:
diff changeset
1191 // insert stack info backwards since linked list is backwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1192 // insert frame info forwards
a61af66fc99e Initial load
duke
parents:
diff changeset
1193 // walk the StackInfoNodes
a61af66fc99e Initial load
duke
parents:
diff changeset
1194 for (struct StackInfoNode *sin = head(); sin != NULL; sin = sin->next) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1195 jint frame_count = sin->info.frame_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
1196 size_t frames_size = frame_count * sizeof(jvmtiFrameInfo);
a61af66fc99e Initial load
duke
parents:
diff changeset
1197 --si;
a61af66fc99e Initial load
duke
parents:
diff changeset
1198 memcpy(si, &(sin->info), sizeof(jvmtiStackInfo));
a61af66fc99e Initial load
duke
parents:
diff changeset
1199 if (frames_size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1200 si->frame_buffer = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1201 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1202 memcpy(fi, sin->info.frame_buffer, frames_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1203 si->frame_buffer = fi; // point to the new allocated copy of the frames
a61af66fc99e Initial load
duke
parents:
diff changeset
1204 fi += frame_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
1205 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1206 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1207 assert(si == _stack_info, "the last copied stack info must be the first record");
a61af66fc99e Initial load
duke
parents:
diff changeset
1208 assert((unsigned char *)fi == ((unsigned char *)_stack_info) + alloc_size,
a61af66fc99e Initial load
duke
parents:
diff changeset
1209 "the last copied frame info must be the last record");
a61af66fc99e Initial load
duke
parents:
diff changeset
1210 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1211
a61af66fc99e Initial load
duke
parents:
diff changeset
1212
a61af66fc99e Initial load
duke
parents:
diff changeset
1213 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1214 VM_GetThreadListStackTraces::doit() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1215 assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint");
a61af66fc99e Initial load
duke
parents:
diff changeset
1216
a61af66fc99e Initial load
duke
parents:
diff changeset
1217 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
1218 for (int i = 0; i < _thread_count; ++i) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1219 jthread jt = _thread_list[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
1220 oop thread_oop = JNIHandles::resolve_external_guard(jt);
a61af66fc99e Initial load
duke
parents:
diff changeset
1221 if (thread_oop == NULL || !thread_oop->is_a(SystemDictionary::thread_klass())) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1222 set_result(JVMTI_ERROR_INVALID_THREAD);
a61af66fc99e Initial load
duke
parents:
diff changeset
1223 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1224 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1225 fill_frames(jt, java_lang_Thread::thread(thread_oop), thread_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
1226 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1227 allocate_and_fill_stacks(_thread_count);
a61af66fc99e Initial load
duke
parents:
diff changeset
1228 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1229
a61af66fc99e Initial load
duke
parents:
diff changeset
1230 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1231 VM_GetAllStackTraces::doit() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1232 assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint");
a61af66fc99e Initial load
duke
parents:
diff changeset
1233
a61af66fc99e Initial load
duke
parents:
diff changeset
1234 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
1235 _final_thread_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1236 for (JavaThread *jt = Threads::first(); jt != NULL; jt = jt->next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1237 oop thread_oop = jt->threadObj();
a61af66fc99e Initial load
duke
parents:
diff changeset
1238 if (thread_oop != NULL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1239 !jt->is_exiting() &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1240 java_lang_Thread::is_alive(thread_oop) &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1241 !jt->is_hidden_from_external_view()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1242 ++_final_thread_count;
a61af66fc99e Initial load
duke
parents:
diff changeset
1243 // Handle block of the calling thread is used to create local refs.
a61af66fc99e Initial load
duke
parents:
diff changeset
1244 fill_frames((jthread)JNIHandles::make_local(_calling_thread, thread_oop),
a61af66fc99e Initial load
duke
parents:
diff changeset
1245 jt, thread_oop);
a61af66fc99e Initial load
duke
parents:
diff changeset
1246 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1247 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1248 allocate_and_fill_stacks(_final_thread_count);
a61af66fc99e Initial load
duke
parents:
diff changeset
1249 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1250
a61af66fc99e Initial load
duke
parents:
diff changeset
1251 // Verifies that the top frame is a java frame in an expected state.
a61af66fc99e Initial load
duke
parents:
diff changeset
1252 // Deoptimizes frame if needed.
a61af66fc99e Initial load
duke
parents:
diff changeset
1253 // Checks that the frame method signature matches the return type (tos).
a61af66fc99e Initial load
duke
parents:
diff changeset
1254 // HandleMark must be defined in the caller only.
a61af66fc99e Initial load
duke
parents:
diff changeset
1255 // It is to keep a ret_ob_h handle alive after return to the caller.
a61af66fc99e Initial load
duke
parents:
diff changeset
1256 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
1257 JvmtiEnvBase::check_top_frame(JavaThread* current_thread, JavaThread* java_thread,
a61af66fc99e Initial load
duke
parents:
diff changeset
1258 jvalue value, TosState tos, Handle* ret_ob_h) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1259 ResourceMark rm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
1260
a61af66fc99e Initial load
duke
parents:
diff changeset
1261 vframe *vf = vframeFor(java_thread, 0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1262 NULL_CHECK(vf, JVMTI_ERROR_NO_MORE_FRAMES);
a61af66fc99e Initial load
duke
parents:
diff changeset
1263
a61af66fc99e Initial load
duke
parents:
diff changeset
1264 javaVFrame *jvf = (javaVFrame*) vf;
a61af66fc99e Initial load
duke
parents:
diff changeset
1265 if (!vf->is_java_frame() || jvf->method()->is_native()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1266 return JVMTI_ERROR_OPAQUE_FRAME;
a61af66fc99e Initial load
duke
parents:
diff changeset
1267 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1268
a61af66fc99e Initial load
duke
parents:
diff changeset
1269 // If the frame is a compiled one, need to deoptimize it.
a61af66fc99e Initial load
duke
parents:
diff changeset
1270 if (vf->is_compiled_frame()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1271 if (!vf->fr().can_be_deoptimized()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1272 return JVMTI_ERROR_OPAQUE_FRAME;
a61af66fc99e Initial load
duke
parents:
diff changeset
1273 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1274 VM_DeoptimizeFrame deopt(java_thread, jvf->fr().id());
a61af66fc99e Initial load
duke
parents:
diff changeset
1275 VMThread::execute(&deopt);
a61af66fc99e Initial load
duke
parents:
diff changeset
1276 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1277
a61af66fc99e Initial load
duke
parents:
diff changeset
1278 // Get information about method return type
a61af66fc99e Initial load
duke
parents:
diff changeset
1279 symbolHandle signature(current_thread, jvf->method()->signature());
a61af66fc99e Initial load
duke
parents:
diff changeset
1280
a61af66fc99e Initial load
duke
parents:
diff changeset
1281 ResultTypeFinder rtf(signature);
a61af66fc99e Initial load
duke
parents:
diff changeset
1282 TosState fr_tos = as_TosState(rtf.type());
a61af66fc99e Initial load
duke
parents:
diff changeset
1283 if (fr_tos != tos) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1284 if (tos != itos || (fr_tos != btos && fr_tos != ctos && fr_tos != stos)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1285 return JVMTI_ERROR_TYPE_MISMATCH;
a61af66fc99e Initial load
duke
parents:
diff changeset
1286 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1287 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1288
a61af66fc99e Initial load
duke
parents:
diff changeset
1289 // Check that the jobject class matches the return type signature.
a61af66fc99e Initial load
duke
parents:
diff changeset
1290 jobject jobj = value.l;
a61af66fc99e Initial load
duke
parents:
diff changeset
1291 if (tos == atos && jobj != NULL) { // NULL reference is allowed
a61af66fc99e Initial load
duke
parents:
diff changeset
1292 Handle ob_h = Handle(current_thread, JNIHandles::resolve_external_guard(jobj));
a61af66fc99e Initial load
duke
parents:
diff changeset
1293 NULL_CHECK(ob_h, JVMTI_ERROR_INVALID_OBJECT);
a61af66fc99e Initial load
duke
parents:
diff changeset
1294 KlassHandle ob_kh = KlassHandle(current_thread, ob_h()->klass());
a61af66fc99e Initial load
duke
parents:
diff changeset
1295 NULL_CHECK(ob_kh, JVMTI_ERROR_INVALID_OBJECT);
a61af66fc99e Initial load
duke
parents:
diff changeset
1296
a61af66fc99e Initial load
duke
parents:
diff changeset
1297 // Method return type signature.
a61af66fc99e Initial load
duke
parents:
diff changeset
1298 char* ty_sign = 1 + strchr(signature->as_C_string(), ')');
a61af66fc99e Initial load
duke
parents:
diff changeset
1299
a61af66fc99e Initial load
duke
parents:
diff changeset
1300 if (!VM_GetOrSetLocal::is_assignable(ty_sign, Klass::cast(ob_kh()), current_thread)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1301 return JVMTI_ERROR_TYPE_MISMATCH;
a61af66fc99e Initial load
duke
parents:
diff changeset
1302 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1303 *ret_ob_h = ob_h;
a61af66fc99e Initial load
duke
parents:
diff changeset
1304 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1305 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
1306 } /* end check_top_frame */
a61af66fc99e Initial load
duke
parents:
diff changeset
1307
a61af66fc99e Initial load
duke
parents:
diff changeset
1308
a61af66fc99e Initial load
duke
parents:
diff changeset
1309 // ForceEarlyReturn<type> follows the PopFrame approach in many aspects.
a61af66fc99e Initial load
duke
parents:
diff changeset
1310 // Main difference is on the last stage in the interpreter.
a61af66fc99e Initial load
duke
parents:
diff changeset
1311 // The PopFrame stops method execution to continue execution
a61af66fc99e Initial load
duke
parents:
diff changeset
1312 // from the same method call instruction.
a61af66fc99e Initial load
duke
parents:
diff changeset
1313 // The ForceEarlyReturn forces return from method so the execution
a61af66fc99e Initial load
duke
parents:
diff changeset
1314 // continues at the bytecode following the method call.
a61af66fc99e Initial load
duke
parents:
diff changeset
1315
a61af66fc99e Initial load
duke
parents:
diff changeset
1316 // Threads_lock NOT held, java_thread not protected by lock
a61af66fc99e Initial load
duke
parents:
diff changeset
1317 // java_thread - pre-checked
a61af66fc99e Initial load
duke
parents:
diff changeset
1318
a61af66fc99e Initial load
duke
parents:
diff changeset
1319 jvmtiError
a61af66fc99e Initial load
duke
parents:
diff changeset
1320 JvmtiEnvBase::force_early_return(JavaThread* java_thread, jvalue value, TosState tos) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1321 JavaThread* current_thread = JavaThread::current();
a61af66fc99e Initial load
duke
parents:
diff changeset
1322 HandleMark hm(current_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
1323 uint32_t debug_bits = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1324
a61af66fc99e Initial load
duke
parents:
diff changeset
1325 // Check if java_thread is fully suspended
a61af66fc99e Initial load
duke
parents:
diff changeset
1326 if (!is_thread_fully_suspended(java_thread,
a61af66fc99e Initial load
duke
parents:
diff changeset
1327 true /* wait for suspend completion */,
a61af66fc99e Initial load
duke
parents:
diff changeset
1328 &debug_bits)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1329 return JVMTI_ERROR_THREAD_NOT_SUSPENDED;
a61af66fc99e Initial load
duke
parents:
diff changeset
1330 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1331
a61af66fc99e Initial load
duke
parents:
diff changeset
1332 // retreive or create the state
a61af66fc99e Initial load
duke
parents:
diff changeset
1333 JvmtiThreadState* state = JvmtiThreadState::state_for(java_thread);
a61af66fc99e Initial load
duke
parents:
diff changeset
1334
a61af66fc99e Initial load
duke
parents:
diff changeset
1335 // Check to see if a ForceEarlyReturn was already in progress
a61af66fc99e Initial load
duke
parents:
diff changeset
1336 if (state->is_earlyret_pending()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1337 // Probably possible for JVMTI clients to trigger this, but the
a61af66fc99e Initial load
duke
parents:
diff changeset
1338 // JPDA backend shouldn't allow this to happen
a61af66fc99e Initial load
duke
parents:
diff changeset
1339 return JVMTI_ERROR_INTERNAL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1340 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1341 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1342 // The same as for PopFrame. Workaround bug:
a61af66fc99e Initial load
duke
parents:
diff changeset
1343 // 4812902: popFrame hangs if the method is waiting at a synchronize
a61af66fc99e Initial load
duke
parents:
diff changeset
1344 // Catch this condition and return an error to avoid hanging.
a61af66fc99e Initial load
duke
parents:
diff changeset
1345 // Now JVMTI spec allows an implementation to bail out with an opaque
a61af66fc99e Initial load
duke
parents:
diff changeset
1346 // frame error.
a61af66fc99e Initial load
duke
parents:
diff changeset
1347 OSThread* osThread = java_thread->osthread();
a61af66fc99e Initial load
duke
parents:
diff changeset
1348 if (osThread->get_state() == MONITOR_WAIT) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1349 return JVMTI_ERROR_OPAQUE_FRAME;
a61af66fc99e Initial load
duke
parents:
diff changeset
1350 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1351 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1352 Handle ret_ob_h = Handle();
a61af66fc99e Initial load
duke
parents:
diff changeset
1353 jvmtiError err = check_top_frame(current_thread, java_thread, value, tos, &ret_ob_h);
a61af66fc99e Initial load
duke
parents:
diff changeset
1354 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1355 return err;
a61af66fc99e Initial load
duke
parents:
diff changeset
1356 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1357 assert(tos != atos || value.l == NULL || ret_ob_h() != NULL,
a61af66fc99e Initial load
duke
parents:
diff changeset
1358 "return object oop must not be NULL if jobject is not NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
1359
a61af66fc99e Initial load
duke
parents:
diff changeset
1360 // Update the thread state to reflect that the top frame must be
a61af66fc99e Initial load
duke
parents:
diff changeset
1361 // forced to return.
a61af66fc99e Initial load
duke
parents:
diff changeset
1362 // The current frame will be returned later when the suspended
a61af66fc99e Initial load
duke
parents:
diff changeset
1363 // thread is resumed and right before returning from VM to Java.
a61af66fc99e Initial load
duke
parents:
diff changeset
1364 // (see call_VM_base() in assembler_<cpu>.cpp).
a61af66fc99e Initial load
duke
parents:
diff changeset
1365
a61af66fc99e Initial load
duke
parents:
diff changeset
1366 state->set_earlyret_pending();
a61af66fc99e Initial load
duke
parents:
diff changeset
1367 state->set_earlyret_oop(ret_ob_h());
a61af66fc99e Initial load
duke
parents:
diff changeset
1368 state->set_earlyret_value(value, tos);
a61af66fc99e Initial load
duke
parents:
diff changeset
1369
a61af66fc99e Initial load
duke
parents:
diff changeset
1370 // Set pending step flag for this early return.
a61af66fc99e Initial load
duke
parents:
diff changeset
1371 // It is cleared when next step event is posted.
a61af66fc99e Initial load
duke
parents:
diff changeset
1372 state->set_pending_step_for_earlyret();
a61af66fc99e Initial load
duke
parents:
diff changeset
1373
a61af66fc99e Initial load
duke
parents:
diff changeset
1374 return JVMTI_ERROR_NONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
1375 } /* end force_early_return */
a61af66fc99e Initial load
duke
parents:
diff changeset
1376
a61af66fc99e Initial load
duke
parents:
diff changeset
1377 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1378 JvmtiMonitorClosure::do_monitor(ObjectMonitor* mon) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1379 if ( _error != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1380 // Error occurred in previous iteration so no need to add
a61af66fc99e Initial load
duke
parents:
diff changeset
1381 // to the list.
a61af66fc99e Initial load
duke
parents:
diff changeset
1382 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1383 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1384 if (mon->owner() == _java_thread ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1385 // Filter out on stack monitors collected during stack walk.
a61af66fc99e Initial load
duke
parents:
diff changeset
1386 oop obj = (oop)mon->object();
a61af66fc99e Initial load
duke
parents:
diff changeset
1387 bool found = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1388 for (int j = 0; j < _owned_monitors_list->length(); j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1389 jobject jobj = ((jvmtiMonitorStackDepthInfo*)_owned_monitors_list->at(j))->monitor;
a61af66fc99e Initial load
duke
parents:
diff changeset
1390 oop check = JNIHandles::resolve(jobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
1391 if (check == obj) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1392 // On stack monitor already collected during the stack walk.
a61af66fc99e Initial load
duke
parents:
diff changeset
1393 found = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1394 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1395 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1396 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1397 if (found == false) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1398 // This is off stack monitor (e.g. acquired via jni MonitorEnter).
a61af66fc99e Initial load
duke
parents:
diff changeset
1399 jvmtiError err;
a61af66fc99e Initial load
duke
parents:
diff changeset
1400 jvmtiMonitorStackDepthInfo *jmsdi;
a61af66fc99e Initial load
duke
parents:
diff changeset
1401 err = _env->allocate(sizeof(jvmtiMonitorStackDepthInfo), (unsigned char **)&jmsdi);
a61af66fc99e Initial load
duke
parents:
diff changeset
1402 if (err != JVMTI_ERROR_NONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1403 _error = err;
a61af66fc99e Initial load
duke
parents:
diff changeset
1404 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1405 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1406 Handle hobj(obj);
a61af66fc99e Initial load
duke
parents:
diff changeset
1407 jmsdi->monitor = _env->jni_reference(_calling_thread, hobj);
a61af66fc99e Initial load
duke
parents:
diff changeset
1408 // stack depth is unknown for this monitor.
a61af66fc99e Initial load
duke
parents:
diff changeset
1409 jmsdi->stack_depth = -1;
a61af66fc99e Initial load
duke
parents:
diff changeset
1410 _owned_monitors_list->append(jmsdi);
a61af66fc99e Initial load
duke
parents:
diff changeset
1411 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1412 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1413 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1414
a61af66fc99e Initial load
duke
parents:
diff changeset
1415 #endif // !JVMTI_KERNEL