| /* |
| * 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 |
| |