blob: 4571032ce6fb5d78278b89ef91647045002691ce [file] [log] [blame]
/*
* 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;
}
jvmtiEventCallbacks cb;
memset(&cb, 0, sizeof(cb));
cb.FramePop = framePopCB;
if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEventCallbacks(&cb, sizeof(cb)))) {
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