|  | /* | 
|  | * 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_breakpoint { | 
|  |  | 
|  | struct BreakpointData { | 
|  | jclass test_klass; | 
|  | jmethodID breakpoint_method; | 
|  | bool in_callback; | 
|  | bool allow_recursive; | 
|  | }; | 
|  |  | 
|  | extern "C" void breakpointCB(jvmtiEnv* jvmti, | 
|  | JNIEnv* jnienv, | 
|  | jthread thread, | 
|  | jmethodID method, | 
|  | jlocation location) { | 
|  | BreakpointData* data = nullptr; | 
|  | if (JvmtiErrorToException(jnienv, jvmti, | 
|  | jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)))) { | 
|  | return; | 
|  | } | 
|  | if (data->in_callback && !data->allow_recursive) { | 
|  | return; | 
|  | } | 
|  | data->in_callback = true; | 
|  | jobject method_arg = GetJavaMethod(jvmti, jnienv, method); | 
|  | jnienv->CallStaticVoidMethod(data->test_klass, | 
|  | data->breakpoint_method, | 
|  | thread, | 
|  | method_arg, | 
|  | static_cast<jlong>(location)); | 
|  | jnienv->DeleteLocalRef(method_arg); | 
|  | data->in_callback = false; | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT jobjectArray JNICALL Java_art_Breakpoint_getLineNumberTableNative( | 
|  | JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jobject target) { | 
|  | jmethodID method = env->FromReflectedMethod(target); | 
|  | if (env->ExceptionCheck()) { | 
|  | return nullptr; | 
|  | } | 
|  | jint nlines; | 
|  | jvmtiLineNumberEntry* lines = nullptr; | 
|  | if (JvmtiErrorToException(env, jvmti_env, | 
|  | jvmti_env->GetLineNumberTable(method, &nlines, &lines))) { | 
|  | return nullptr; | 
|  | } | 
|  | jintArray lines_array = env->NewIntArray(nlines); | 
|  | if (env->ExceptionCheck()) { | 
|  | jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(lines)); | 
|  | return nullptr; | 
|  | } | 
|  | jlongArray locs_array = env->NewLongArray(nlines); | 
|  | if (env->ExceptionCheck()) { | 
|  | jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(lines)); | 
|  | return nullptr; | 
|  | } | 
|  | ScopedLocalRef<jclass> object_class(env, env->FindClass("java/lang/Object")); | 
|  | if (env->ExceptionCheck()) { | 
|  | jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(lines)); | 
|  | return nullptr; | 
|  | } | 
|  | jobjectArray ret = env->NewObjectArray(2, object_class.get(), nullptr); | 
|  | if (env->ExceptionCheck()) { | 
|  | jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(lines)); | 
|  | return nullptr; | 
|  | } | 
|  | jint* temp_lines = env->GetIntArrayElements(lines_array, /*isCopy*/nullptr); | 
|  | jlong* temp_locs = env->GetLongArrayElements(locs_array, /*isCopy*/nullptr); | 
|  | for (jint i = 0; i < nlines; i++) { | 
|  | temp_lines[i] = lines[i].line_number; | 
|  | temp_locs[i] = lines[i].start_location; | 
|  | } | 
|  | env->ReleaseIntArrayElements(lines_array, temp_lines, 0); | 
|  | env->ReleaseLongArrayElements(locs_array, temp_locs, 0); | 
|  | env->SetObjectArrayElement(ret, 0, locs_array); | 
|  | env->SetObjectArrayElement(ret, 1, lines_array); | 
|  | jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(lines)); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT jlong JNICALL Java_art_Breakpoint_getStartLocation(JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jobject target) { | 
|  | jmethodID method = env->FromReflectedMethod(target); | 
|  | if (env->ExceptionCheck()) { | 
|  | return 0; | 
|  | } | 
|  | jlong start = 0; | 
|  | jlong end; | 
|  | JvmtiErrorToException(env, jvmti_env, jvmti_env->GetMethodLocation(method, &start, &end)); | 
|  | return start; | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT void JNICALL Java_art_Breakpoint_clearBreakpoint(JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jobject target, | 
|  | jlocation location) { | 
|  | jmethodID method = env->FromReflectedMethod(target); | 
|  | if (env->ExceptionCheck()) { | 
|  | return; | 
|  | } | 
|  | JvmtiErrorToException(env, jvmti_env, jvmti_env->ClearBreakpoint(method, location)); | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT void JNICALL Java_art_Breakpoint_setBreakpoint(JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jobject target, | 
|  | jlocation location) { | 
|  | jmethodID method = env->FromReflectedMethod(target); | 
|  | if (env->ExceptionCheck()) { | 
|  | return; | 
|  | } | 
|  | JvmtiErrorToException(env, jvmti_env, jvmti_env->SetBreakpoint(method, location)); | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT void JNICALL Java_art_Breakpoint_startBreakpointWatch( | 
|  | JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jclass method_klass, | 
|  | jobject method, | 
|  | jboolean allow_recursive, | 
|  | jthread thr) { | 
|  | BreakpointData* data = nullptr; | 
|  | if (JvmtiErrorToException(env, | 
|  | jvmti_env, | 
|  | jvmti_env->Allocate(sizeof(BreakpointData), | 
|  | reinterpret_cast<unsigned char**>(&data)))) { | 
|  | return; | 
|  | } | 
|  | memset(data, 0, sizeof(BreakpointData)); | 
|  | data->test_klass = reinterpret_cast<jclass>(env->NewGlobalRef(method_klass)); | 
|  | data->breakpoint_method = env->FromReflectedMethod(method); | 
|  | data->in_callback = false; | 
|  | data->allow_recursive = allow_recursive; | 
|  |  | 
|  | 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; | 
|  | } | 
|  | current_callbacks.Breakpoint = breakpointCB; | 
|  | 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_BREAKPOINT, | 
|  | thr))) { | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | extern "C" JNIEXPORT void JNICALL Java_art_Breakpoint_stopBreakpointWatch( | 
|  | JNIEnv* env, | 
|  | jclass k ATTRIBUTE_UNUSED, | 
|  | jthread thr) { | 
|  | if (JvmtiErrorToException(env, jvmti_env, | 
|  | jvmti_env->SetEventNotificationMode(JVMTI_DISABLE, | 
|  | JVMTI_EVENT_BREAKPOINT, | 
|  | thr))) { | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace common_breakpoint | 
|  |  | 
|  | }  // namespace art |