/* * 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 "common_helper.h" #include "jni.h" #include "jvmti.h" #include "jvmti_helper.h" #include "scoped_local_ref.h" #include "test_env.h" namespace art { namespace common_frame_pop { struct FramePopData { jclass test_klass; jmethodID pop_method; }; static void framePopCB(jvmtiEnv* jvmti, JNIEnv* jnienv, jthread thr, jmethodID method ATTRIBUTE_UNUSED, jboolean was_popped_by_exception) { FramePopData* data = nullptr; if (JvmtiErrorToException(jnienv, jvmti, jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) { return; } jlong location; jmethodID frame_method; if (JvmtiErrorToException(jnienv, jvmti, jvmti->GetFrameLocation(thr, 0, &frame_method, &location))) { return; } CHECK(data->pop_method != nullptr); jobject method_arg = GetJavaMethod(jvmti, jnienv, frame_method); jnienv->CallStaticVoidMethod(data->test_klass, data->pop_method, method_arg, was_popped_by_exception, location); jnienv->DeleteLocalRef(method_arg); } extern "C" JNIEXPORT void JNICALL Java_art_FramePop_enableFramePopEvent( JNIEnv* env, jclass, jclass klass, jobject notify_method, jthread thr) { FramePopData* data = nullptr; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->Allocate(sizeof(FramePopData), reinterpret_cast<unsigned char**>(&data)))) { return; } memset(data, 0, sizeof(FramePopData)); data->test_klass = reinterpret_cast<jclass>(env->NewGlobalRef(klass)); data->pop_method = env->FromReflectedMethod(notify_method); if (env->ExceptionCheck()) { return; } void* old_data = nullptr; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->GetEnvironmentLocalStorage(&old_data))) { return; } else if (old_data != 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; } jvmtiCapabilities caps; memset(&caps, 0, sizeof(caps)); caps.can_generate_frame_pop_events = 1; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->AddCapabilities(&caps))) { return; } current_callbacks.FramePop = framePopCB; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEventCallbacks(¤t_callbacks, sizeof(current_callbacks)))) { return; } JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_FRAME_POP, thr)); } extern "C" JNIEXPORT jlong JNICALL Java_art_FramePop_makeJvmtiEnvForFramePop(JNIEnv* env, jclass) { JavaVM* vm; jvmtiEnv* out_jvmti_env = nullptr; if (env->GetJavaVM(&vm) != JNI_OK || vm->GetEnv(reinterpret_cast<void**>(&out_jvmti_env), JVMTI_VERSION_1_0) != JNI_OK) { ScopedLocalRef<jclass> rt_exception(env, env->FindClass("java/lang/RuntimeException")); if (rt_exception.get() == nullptr) { // CNFE should be pending. return 0L; } env->ThrowNew(rt_exception.get(), "Unable to create new jvmti_env"); return 0L; } SetAllCapabilities(out_jvmti_env); return static_cast<jlong>(reinterpret_cast<intptr_t>(out_jvmti_env)); } extern "C" JNIEXPORT void JNICALL Java_art_FramePop_notifyFramePop( JNIEnv* env, jclass, jthread thr, jint depth) { JvmtiErrorToException(env, jvmti_env, jvmti_env->NotifyFramePop(thr, depth)); } } // namespace common_frame_pop } // namespace art