Fri, 12 Jun 2020 02:59:56 +0100
8233197: Invert JvmtiExport::post_vm_initialized() and Jfr:on_vm_start() start-up order for correct option parsing
8246703: [TESTBUG] Add test for JDK-8233197
Reviewed-by: aph, adinn, neugens
1 /*
2 * Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
25 #include "precompiled.hpp"
26 #include "jvm.h"
27 #include "jfr/instrumentation/jfrJvmtiAgent.hpp"
28 #include "jfr/jni/jfrJavaSupport.hpp"
29 #include "jfr/jni/jfrUpcalls.hpp"
30 #include "jfr/recorder/checkpoint/types/traceid/jfrTraceId.inline.hpp"
31 #include "jfr/recorder/service/jfrOptionSet.hpp"
32 #include "jfr/support/jfrEventClass.hpp"
33 #include "memory/resourceArea.hpp"
34 #include "prims/jvmtiEnvBase.hpp"
35 #include "prims/jvmtiExport.hpp"
36 #include "prims/jvmtiUtil.hpp"
37 #include "runtime/interfaceSupport.hpp"
38 #include "runtime/thread.inline.hpp"
39 #include "utilities/exceptions.hpp"
41 static const size_t ERROR_MSG_BUFFER_SIZE = 256;
42 static JfrJvmtiAgent* agent = NULL;
43 static jvmtiEnv* jfr_jvmti_env = NULL;
45 static void check_jvmti_error(jvmtiEnv* jvmti, jvmtiError errnum, const char* str) {
46 if (errnum != JVMTI_ERROR_NONE) {
47 char* errnum_str = NULL;
48 jvmti->GetErrorName(errnum, &errnum_str);
49 if (true) tty->print_cr("ERROR: JfrJvmtiAgent: " INT32_FORMAT " (%s): %s\n",
50 errnum,
51 NULL == errnum_str ? "Unknown" : errnum_str,
52 NULL == str ? "" : str);
53 }
54 }
56 static bool set_event_notification_mode(jvmtiEventMode mode,
57 jvmtiEvent event,
58 jthread event_thread,
59 ...) {
60 assert(jfr_jvmti_env != NULL, "invariant");
61 const jvmtiError jvmti_ret_code = jfr_jvmti_env->SetEventNotificationMode(mode, event, event_thread);
62 check_jvmti_error(jfr_jvmti_env, jvmti_ret_code, "SetEventNotificationMode");
63 return jvmti_ret_code == JVMTI_ERROR_NONE;
64 }
66 static bool update_class_file_load_hook_event(jvmtiEventMode mode) {
67 return set_event_notification_mode(mode, JVMTI_EVENT_CLASS_FILE_LOAD_HOOK, NULL);
68 }
70 static JavaThread* current_java_thread() {
71 Thread* this_thread = Thread::current();
72 assert(this_thread != NULL && this_thread->is_Java_thread(), "invariant");
73 return static_cast<JavaThread*>(this_thread);
74 }
76 // jvmti event callbacks require C linkage
77 extern "C" void JNICALL jfr_on_class_file_load_hook(jvmtiEnv *jvmti_env,
78 JNIEnv* jni_env,
79 jclass class_being_redefined,
80 jobject loader,
81 const char* name,
82 jobject protection_domain,
83 jint class_data_len,
84 const unsigned char* class_data,
85 jint* new_class_data_len,
86 unsigned char** new_class_data) {
87 if (class_being_redefined == NULL) {
88 return;
89 }
90 JavaThread* jt = JavaThread::thread_from_jni_environment(jni_env);
91 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(jt));;
92 ThreadInVMfromNative tvmfn(jt);
93 JfrUpcalls::on_retransform(JfrTraceId::get(class_being_redefined),
94 class_being_redefined,
95 class_data_len,
96 class_data,
97 new_class_data_len,
98 new_class_data,
99 jt);
100 }
102 // caller needs ResourceMark
103 static jclass* create_classes_array(jint classes_count, TRAPS) {
104 assert(classes_count > 0, "invariant");
105 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(THREAD));
106 ThreadInVMfromNative tvmfn((JavaThread*)THREAD);
107 jclass* const classes = NEW_RESOURCE_ARRAY_IN_THREAD_RETURN_NULL(THREAD, jclass, classes_count);
108 if (NULL == classes) {
109 char error_buffer[ERROR_MSG_BUFFER_SIZE];
110 jio_snprintf(error_buffer, ERROR_MSG_BUFFER_SIZE,
111 "Thread local allocation (native) of " SIZE_FORMAT " bytes failed "
112 "in retransform classes", sizeof(jclass) * classes_count);
113 if (true) tty->print_cr("%s", error_buffer);
114 JfrJavaSupport::throw_out_of_memory_error(error_buffer, CHECK_NULL);
115 }
116 return classes;
117 }
119 // caller needs ResourceMark
120 static void log_and_throw(jvmtiError error, TRAPS) {
121 if (!HAS_PENDING_EXCEPTION) {
122 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(THREAD));
123 ThreadInVMfromNative tvmfn((JavaThread*)THREAD);
124 const char base_error_msg[] = "JfrJvmtiAgent::retransformClasses failed: ";
125 size_t length = sizeof base_error_msg; // includes terminating null
126 const char* const jvmti_error_name = JvmtiUtil::error_name(error);
127 assert(jvmti_error_name != NULL, "invariant");
128 length += strlen(jvmti_error_name);
129 char* error_msg = NEW_RESOURCE_ARRAY(char, length);
130 jio_snprintf(error_msg, length, "%s%s", base_error_msg, jvmti_error_name);
131 if (JVMTI_ERROR_INVALID_CLASS_FORMAT == error) {
132 JfrJavaSupport::throw_class_format_error(error_msg, THREAD);
133 } else {
134 JfrJavaSupport::throw_runtime_exception(error_msg, THREAD);
135 }
136 }
137 }
139 static void check_exception_and_log(JNIEnv* env, TRAPS) {
140 assert(env != NULL, "invariant");
141 if (env->ExceptionOccurred()) {
142 // array index out of bound
143 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(THREAD));
144 ThreadInVMfromNative tvmfn((JavaThread*)THREAD);
145 if (true) tty->print_cr("GetObjectArrayElement threw an exception");
146 return;
147 }
148 }
150 static bool is_valid_jvmti_phase() {
151 return JvmtiEnvBase::get_phase() == JVMTI_PHASE_LIVE;
152 }
154 void JfrJvmtiAgent::retransform_classes(JNIEnv* env, jobjectArray classes_array, TRAPS) {
155 assert(env != NULL, "invariant");
156 assert(classes_array != NULL, "invariant");
157 assert(is_valid_jvmti_phase(), "invariant");
158 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(THREAD));
159 const jint classes_count = env->GetArrayLength(classes_array);
160 if (classes_count <= 0) {
161 return;
162 }
163 ResourceMark rm(THREAD);
164 jclass* const classes = create_classes_array(classes_count, CHECK);
165 assert(classes != NULL, "invariant");
166 for (jint i = 0; i < classes_count; i++) {
167 jclass clz = (jclass)env->GetObjectArrayElement(classes_array, i);
168 check_exception_and_log(env, THREAD);
169 classes[i] = clz;
170 }
171 {
172 // inspecting the oop/klass requires a thread transition
173 ThreadInVMfromNative transition((JavaThread*)THREAD);
174 for (jint i = 0; i < classes_count; ++i) {
175 jclass clz = classes[i];
176 if (!JdkJfrEvent::is_a(clz)) {
177 // outside the event hierarchy
178 JdkJfrEvent::tag_as_host(clz);
179 }
180 }
181 }
182 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(THREAD));
183 const jvmtiError result = jfr_jvmti_env->RetransformClasses(classes_count, classes);
184 if (result != JVMTI_ERROR_NONE) {
185 log_and_throw(result, THREAD);
186 }
187 }
189 static bool register_callbacks(JavaThread* jt) {
190 assert(jfr_jvmti_env != NULL, "invariant");
191 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(jt));
192 jvmtiEventCallbacks callbacks;
193 /* Set callbacks */
194 memset(&callbacks, 0, sizeof(callbacks));
195 callbacks.ClassFileLoadHook = jfr_on_class_file_load_hook;
196 const jvmtiError jvmti_ret_code = jfr_jvmti_env->SetEventCallbacks(&callbacks, sizeof(callbacks));
197 check_jvmti_error(jfr_jvmti_env, jvmti_ret_code, "SetEventCallbacks");
198 return jvmti_ret_code == JVMTI_ERROR_NONE;
199 }
201 static bool register_capabilities(JavaThread* jt) {
202 assert(jfr_jvmti_env != NULL, "invariant");
203 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(jt));
204 jvmtiCapabilities capabilities;
205 /* Add JVMTI capabilities */
206 (void)memset(&capabilities, 0, sizeof(capabilities));
207 capabilities.can_retransform_classes = 1;
208 capabilities.can_retransform_any_class = 1;
209 const jvmtiError jvmti_ret_code = jfr_jvmti_env->AddCapabilities(&capabilities);
210 check_jvmti_error(jfr_jvmti_env, jvmti_ret_code, "Add Capabilities");
211 return jvmti_ret_code == JVMTI_ERROR_NONE;
212 }
214 static jint create_jvmti_env(JavaThread* jt) {
215 assert(jfr_jvmti_env == NULL, "invariant");
216 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_native(jt));
217 extern struct JavaVM_ main_vm;
218 JavaVM* vm = &main_vm;
219 return vm->GetEnv((void **)&jfr_jvmti_env, JVMTI_VERSION);
220 }
222 static bool unregister_callbacks(JavaThread* jt) {
223 assert(jfr_jvmti_env != NULL, "invariant");
224 jvmtiEventCallbacks callbacks;
225 /* Set empty callbacks */
226 memset(&callbacks, 0, sizeof(callbacks));
227 const jvmtiError jvmti_ret_code = jfr_jvmti_env->SetEventCallbacks(&callbacks, sizeof(callbacks));
228 check_jvmti_error(jfr_jvmti_env, jvmti_ret_code, "SetEventCallbacks");
229 return jvmti_ret_code == JVMTI_ERROR_NONE;
230 }
232 JfrJvmtiAgent::JfrJvmtiAgent() {}
234 JfrJvmtiAgent::~JfrJvmtiAgent() {
235 JavaThread* jt = current_java_thread();
236 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(jt));
237 if (jfr_jvmti_env != NULL) {
238 ThreadToNativeFromVM transition(jt);
239 update_class_file_load_hook_event(JVMTI_DISABLE);
240 unregister_callbacks(jt);
241 jfr_jvmti_env->DisposeEnvironment();
242 jfr_jvmti_env = NULL;
243 }
244 }
246 static bool initialize(JavaThread* jt) {
247 assert(jt != NULL, "invariant");
248 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(jt));
249 ThreadToNativeFromVM transition(jt);
250 if (create_jvmti_env(jt) != JNI_OK) {
251 assert(jfr_jvmti_env == NULL, "invariant");
252 return false;
253 }
254 assert(jfr_jvmti_env != NULL, "invariant");
255 if (!register_capabilities(jt)) {
256 return false;
257 }
258 if (!register_callbacks(jt)) {
259 return false;
260 }
261 return update_class_file_load_hook_event(JVMTI_ENABLE);
262 }
264 static void log_and_throw_illegal_state_exception(TRAPS) {
265 DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(THREAD));
266 const char* const illegal_state_msg = "An attempt was made to start JFR too early in the VM initialization sequence.";
267 if (true) {
268 tty->print_cr("%s\n", illegal_state_msg);
269 tty->print_cr("JFR uses JVMTI RetransformClasses and requires the JVMTI state to have entered JVMTI_PHASE_LIVE.\n");
270 tty->print_cr("Please initialize JFR in response to event JVMTI_EVENT_VM_INIT instead of JVMTI_EVENT_VM_START.\n");
271 }
272 JfrJavaSupport::throw_illegal_state_exception(illegal_state_msg, THREAD);
273 }
275 bool JfrJvmtiAgent::create() {
276 assert(agent == NULL, "invariant");
277 JavaThread* const jt = current_java_thread();
278 if (!is_valid_jvmti_phase()) {
279 log_and_throw_illegal_state_exception(jt);
280 return false;
281 }
282 agent = new JfrJvmtiAgent();
283 if (agent == NULL) {
284 return false;
285 }
286 if (!initialize(jt)) {
287 delete agent;
288 agent = NULL;
289 return false;
290 }
291 return true;
292 }
294 void JfrJvmtiAgent::destroy() {
295 if (agent != NULL) {
296 delete agent;
297 agent = NULL;
298 }
299 }