src/share/vm/jfr/instrumentation/jfrJvmtiAgent.cpp

Fri, 12 Jun 2020 02:59:56 +0100

author
jbachorik
date
Fri, 12 Jun 2020 02:59:56 +0100
changeset 9925
30fb8c8cceb9
parent 9858
b985cbb00e68
child 9986
85e682d8ab91
permissions
-rw-r--r--

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 }

mercurial