| /* |
| * 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 <pthread.h> |
| |
| #include <cstdio> |
| #include <iostream> |
| #include <vector> |
| |
| #include "android-base/logging.h" |
| #include "jni.h" |
| #include "jvmti.h" |
| |
| #include "scoped_local_ref.h" |
| #include "scoped_primitive_array.h" |
| |
| // Test infrastructure |
| #include "jvmti_helper.h" |
| #include "test_env.h" |
| |
| namespace art { |
| namespace Test1934SignalThreads { |
| |
| struct NativeMonitor { |
| jrawMonitorID continue_monitor; |
| bool should_continue; |
| jrawMonitorID start_monitor; |
| bool should_start; |
| }; |
| |
| extern "C" JNIEXPORT jlong JNICALL Java_art_Test1934_allocNativeMonitor(JNIEnv* env, jclass) { |
| NativeMonitor* mon; |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->Allocate(sizeof(NativeMonitor), |
| reinterpret_cast<unsigned char**>(&mon)))) { |
| return -1L; |
| } |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->CreateRawMonitor("test-1934 start", |
| &mon->start_monitor))) { |
| return -1L; |
| } |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->CreateRawMonitor("test-1934 continue", |
| &mon->continue_monitor))) { |
| return -1L; |
| } |
| mon->should_continue = false; |
| mon->should_start = false; |
| return static_cast<jlong>(reinterpret_cast<intptr_t>(mon)); |
| } |
| |
| extern "C" JNIEXPORT void Java_art_Test1934_nativeWaitForOtherThread(JNIEnv* env, |
| jclass, |
| jlong id) { |
| NativeMonitor* mon = reinterpret_cast<NativeMonitor*>(static_cast<intptr_t>(id)); |
| // Start |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorEnter(mon->start_monitor))) { |
| return; |
| } |
| mon->should_start = true; |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->RawMonitorNotifyAll(mon->start_monitor))) { |
| JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->start_monitor)); |
| return; |
| } |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->start_monitor))) { |
| return; |
| } |
| |
| // Finish |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorEnter(mon->continue_monitor))) { |
| return; |
| } |
| while (!mon->should_continue) { |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->RawMonitorWait(mon->continue_monitor, -1L))) { |
| JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->continue_monitor)); |
| return; |
| } |
| } |
| JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->continue_monitor)); |
| } |
| |
| extern "C" JNIEXPORT void Java_art_Test1934_nativeDoInterleaved(JNIEnv* env, |
| jclass, |
| jlong id, |
| jobject closure) { |
| NativeMonitor* mon = reinterpret_cast<NativeMonitor*>(static_cast<intptr_t>(id)); |
| // Wait for start. |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorEnter(mon->start_monitor))) { |
| return; |
| } |
| while (!mon->should_start) { |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->RawMonitorWait(mon->start_monitor, -1L))) { |
| return; |
| } |
| } |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->start_monitor))) { |
| return; |
| } |
| |
| // Call closure. |
| ScopedLocalRef<jclass> runnable_klass(env, env->FindClass("java/lang/Runnable")); |
| if (env->ExceptionCheck()) { |
| return; |
| } |
| jmethodID doRun = env->GetMethodID(runnable_klass.get(), "run", "()V"); |
| if (env->ExceptionCheck()) { |
| return; |
| } |
| env->CallVoidMethod(closure, doRun); |
| |
| // Tell other thread to finish. |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorEnter(mon->continue_monitor))) { |
| return; |
| } |
| mon->should_continue = true; |
| if (JvmtiErrorToException(env, |
| jvmti_env, |
| jvmti_env->RawMonitorNotifyAll(mon->continue_monitor))) { |
| JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->continue_monitor)); |
| return; |
| } |
| JvmtiErrorToException(env, jvmti_env, jvmti_env->RawMonitorExit(mon->continue_monitor)); |
| } |
| |
| extern "C" JNIEXPORT void Java_art_Test1934_destroyNativeMonitor(JNIEnv*, jclass, jlong id) { |
| NativeMonitor* mon = reinterpret_cast<NativeMonitor*>(static_cast<intptr_t>(id)); |
| jvmti_env->DestroyRawMonitor(mon->start_monitor); |
| jvmti_env->DestroyRawMonitor(mon->continue_monitor); |
| jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(mon)); |
| } |
| |
| } // namespace Test1934SignalThreads |
| } // namespace art |
| |