224 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * Copyright (C) 2017 The Android Open Source Project
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  *      http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| #include "jni.h"
 | |
| #include "jvmti.h"
 | |
| 
 | |
| #include <vector>
 | |
| 
 | |
| #include "jvmti_helper.h"
 | |
| #include "jni_helper.h"
 | |
| #include "test_env.h"
 | |
| #include "scoped_local_ref.h"
 | |
| 
 | |
| namespace art {
 | |
| namespace common_monitors {
 | |
| 
 | |
| extern "C" JNIEXPORT jobject JNICALL Java_art_Monitors_getCurrentContendedMonitor(
 | |
|     JNIEnv* env, jclass, jthread thr) {
 | |
|   jobject out = nullptr;
 | |
|   JvmtiErrorToException(env, jvmti_env, jvmti_env->GetCurrentContendedMonitor(thr, &out));
 | |
|   return out;
 | |
| }
 | |
| 
 | |
| extern "C" JNIEXPORT jobject JNICALL Java_art_Monitors_getObjectMonitorUsage(
 | |
|     JNIEnv* env, jclass, jobject obj) {
 | |
|   ScopedLocalRef<jclass> klass(env, env->FindClass("art/Monitors$MonitorUsage"));
 | |
|   if (env->ExceptionCheck()) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   jmethodID constructor = env->GetMethodID(
 | |
|       klass.get(),
 | |
|       "<init>",
 | |
|       "(Ljava/lang/Object;Ljava/lang/Thread;I[Ljava/lang/Thread;[Ljava/lang/Thread;)V");
 | |
|   if (env->ExceptionCheck()) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   jvmtiMonitorUsage usage;
 | |
|   if (JvmtiErrorToException(env, jvmti_env, jvmti_env->GetObjectMonitorUsage(obj, &usage))) {
 | |
|     return nullptr;
 | |
|   }
 | |
|   jobjectArray wait = CreateObjectArray(env, usage.waiter_count, "java/lang/Thread",
 | |
|                                         [&](jint i) { return usage.waiters[i]; });
 | |
|   if (env->ExceptionCheck()) {
 | |
|     jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(usage.waiters));
 | |
|     jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(usage.notify_waiters));
 | |
|     return nullptr;
 | |
|   }
 | |
|   jobjectArray notify_wait = CreateObjectArray(env, usage.notify_waiter_count, "java/lang/Thread",
 | |
|                                                [&](jint i) { return usage.notify_waiters[i]; });
 | |
|   if (env->ExceptionCheck()) {
 | |
|     jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(usage.waiters));
 | |
|     jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(usage.notify_waiters));
 | |
|     return nullptr;
 | |
|   }
 | |
|   return env->NewObject(klass.get(), constructor,
 | |
|                         obj, usage.owner, usage.entry_count, wait, notify_wait);
 | |
| }
 | |
| 
 | |
| struct MonitorsData {
 | |
|   jclass test_klass;
 | |
|   jmethodID monitor_enter;
 | |
|   jmethodID monitor_entered;
 | |
|   jmethodID monitor_wait;
 | |
|   jmethodID monitor_waited;
 | |
|   jclass monitor_klass;
 | |
| };
 | |
| 
 | |
| static void monitorEnterCB(jvmtiEnv* jvmti,
 | |
|                            JNIEnv* jnienv,
 | |
|                            jthread thr,
 | |
|                            jobject obj) {
 | |
|   MonitorsData* data = nullptr;
 | |
|   if (JvmtiErrorToException(jnienv, jvmti,
 | |
|                             jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) {
 | |
|     return;
 | |
|   }
 | |
|   if (!jnienv->IsInstanceOf(obj, data->monitor_klass)) {
 | |
|     return;
 | |
|   }
 | |
|   jnienv->CallStaticVoidMethod(data->test_klass, data->monitor_enter, thr, obj);
 | |
| }
 | |
| static void monitorEnteredCB(jvmtiEnv* jvmti,
 | |
|                              JNIEnv* jnienv,
 | |
|                              jthread thr,
 | |
|                              jobject obj) {
 | |
|   MonitorsData* data = nullptr;
 | |
|   if (JvmtiErrorToException(jnienv, jvmti,
 | |
|                             jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) {
 | |
|     return;
 | |
|   }
 | |
|   if (!jnienv->IsInstanceOf(obj, data->monitor_klass)) {
 | |
|     return;
 | |
|   }
 | |
|   jnienv->CallStaticVoidMethod(data->test_klass, data->monitor_entered, thr, obj);
 | |
| }
 | |
| static void monitorWaitCB(jvmtiEnv* jvmti,
 | |
|                           JNIEnv* jnienv,
 | |
|                           jthread thr,
 | |
|                           jobject obj,
 | |
|                           jlong timeout) {
 | |
|   MonitorsData* data = nullptr;
 | |
|   if (JvmtiErrorToException(jnienv, jvmti,
 | |
|                             jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) {
 | |
|     return;
 | |
|   }
 | |
|   if (!jnienv->IsInstanceOf(obj, data->monitor_klass)) {
 | |
|     return;
 | |
|   }
 | |
|   jnienv->CallStaticVoidMethod(data->test_klass, data->monitor_wait, thr, obj, timeout);
 | |
| }
 | |
| static void monitorWaitedCB(jvmtiEnv* jvmti,
 | |
|                             JNIEnv* jnienv,
 | |
|                             jthread thr,
 | |
|                             jobject obj,
 | |
|                             jboolean timed_out) {
 | |
|   MonitorsData* data = nullptr;
 | |
|   if (JvmtiErrorToException(jnienv, jvmti,
 | |
|                             jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) {
 | |
|     return;
 | |
|   }
 | |
|   if (!jnienv->IsInstanceOf(obj, data->monitor_klass)) {
 | |
|     return;
 | |
|   }
 | |
|   jnienv->CallStaticVoidMethod(data->test_klass, data->monitor_waited, thr, obj, timed_out);
 | |
| }
 | |
| 
 | |
| extern "C" JNIEXPORT void JNICALL Java_art_Monitors_setupMonitorEvents(
 | |
|     JNIEnv* env,
 | |
|     jclass,
 | |
|     jclass test_klass,
 | |
|     jobject monitor_enter,
 | |
|     jobject monitor_entered,
 | |
|     jobject monitor_wait,
 | |
|     jobject monitor_waited,
 | |
|     jclass monitor_klass,
 | |
|     jthread thr) {
 | |
|   MonitorsData* data = nullptr;
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->Allocate(sizeof(MonitorsData),
 | |
|                                                 reinterpret_cast<unsigned char**>(&data)))) {
 | |
|     return;
 | |
|   }
 | |
|   jvmtiCapabilities caps;
 | |
|   memset(&caps, 0, sizeof(caps));
 | |
|   caps.can_generate_monitor_events = 1;
 | |
|   if (JvmtiErrorToException(env, jvmti_env, jvmti_env->AddCapabilities(&caps))) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   memset(data, 0, sizeof(MonitorsData));
 | |
|   data->test_klass = reinterpret_cast<jclass>(env->NewGlobalRef(test_klass));
 | |
|   data->monitor_enter = env->FromReflectedMethod(monitor_enter);
 | |
|   data->monitor_entered = env->FromReflectedMethod(monitor_entered);
 | |
|   data->monitor_wait = env->FromReflectedMethod(monitor_wait);
 | |
|   data->monitor_waited = env->FromReflectedMethod(monitor_waited);
 | |
|   data->monitor_klass = reinterpret_cast<jclass>(env->NewGlobalRef(monitor_klass));
 | |
|   MonitorsData* old_data = nullptr;
 | |
|   if (JvmtiErrorToException(env, jvmti_env,
 | |
|                             jvmti_env->GetEnvironmentLocalStorage(
 | |
|                                 reinterpret_cast<void**>(&old_data)))) {
 | |
|     return;
 | |
|   } else if (old_data != nullptr && old_data->test_klass != nullptr) {
 | |
|     ScopedLocalRef<jclass> rt_exception(env, env->FindClass("java/lang/RuntimeException"));
 | |
|     env->ThrowNew(rt_exception.get(), "Environment already has local storage set!");
 | |
|     return;
 | |
|   }
 | |
|   if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEnvironmentLocalStorage(data))) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   current_callbacks.MonitorContendedEnter = monitorEnterCB;
 | |
|   current_callbacks.MonitorContendedEntered = monitorEnteredCB;
 | |
|   current_callbacks.MonitorWait = monitorWaitCB;
 | |
|   current_callbacks.MonitorWaited = monitorWaitedCB;
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->SetEventCallbacks(¤t_callbacks,
 | |
|                                                          sizeof(current_callbacks)))) {
 | |
|     return;
 | |
|   }
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->SetEventNotificationMode(
 | |
|                                 JVMTI_ENABLE, JVMTI_EVENT_MONITOR_CONTENDED_ENTER, thr))) {
 | |
|     return;
 | |
|   }
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->SetEventNotificationMode(
 | |
|                                 JVMTI_ENABLE, JVMTI_EVENT_MONITOR_CONTENDED_ENTERED, thr))) {
 | |
|     return;
 | |
|   }
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->SetEventNotificationMode(
 | |
|                                 JVMTI_ENABLE, JVMTI_EVENT_MONITOR_WAIT, thr))) {
 | |
|     return;
 | |
|   }
 | |
|   if (JvmtiErrorToException(env,
 | |
|                             jvmti_env,
 | |
|                             jvmti_env->SetEventNotificationMode(
 | |
|                                 JVMTI_ENABLE, JVMTI_EVENT_MONITOR_WAITED, thr))) {
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| 
 | |
| }  // namespace common_monitors
 | |
| }  // namespace art
 | |
| 
 |