/* * Copyright (C) 2019 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 #include #include #include #include #include "android-base/logging.h" #include "android-base/macros.h" #include "android-base/stringprintf.h" #include "jni.h" #include "jvmti.h" #include "scoped_local_ref.h" #include "scoped_utf_chars.h" // Test infrastructure #include "jni_helper.h" #include "jvmti_helper.h" #include "test_env.h" #include "ti_macros.h" namespace art { namespace Test1974ResizeArray { using ChangeArraySize = jvmtiError (*)(jvmtiEnv* env, jobject arr, jint size); template static void Dealloc(T* t) { jvmti_env->Deallocate(reinterpret_cast(t)); } template static void Dealloc(T* t, Rest... rs) { Dealloc(t); Dealloc(rs...); } static void DeallocParams(jvmtiParamInfo* params, jint n_params) { for (jint i = 0; i < n_params; i++) { Dealloc(params[i].name); } } static jint FindExtensionEvent(JNIEnv* env, const std::string& name) { jint n_ext; jvmtiExtensionEventInfo* infos; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->GetExtensionEvents(&n_ext, &infos))) { return -1; } jint res = -1; bool found = false; for (jint i = 0; i < n_ext; i++) { jvmtiExtensionEventInfo* cur_info = &infos[i]; if (strcmp(name.c_str(), cur_info->id) == 0) { res = cur_info->extension_event_index; found = true; } // Cleanup the cur_info DeallocParams(cur_info->params, cur_info->param_count); Dealloc(cur_info->id, cur_info->short_description, cur_info->params); } // Cleanup the array. Dealloc(infos); if (!found) { ScopedLocalRef rt_exception(env, env->FindClass("java/lang/RuntimeException")); env->ThrowNew(rt_exception.get(), (name + " extensions not found").c_str()); return -1; } return res; } static jvmtiExtensionFunction FindExtensionMethod(JNIEnv* env, const std::string& name) { jint n_ext; jvmtiExtensionFunctionInfo* infos; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->GetExtensionFunctions(&n_ext, &infos))) { return nullptr; } jvmtiExtensionFunction res = nullptr; for (jint i = 0; i < n_ext; i++) { jvmtiExtensionFunctionInfo* cur_info = &infos[i]; if (strcmp(name.c_str(), cur_info->id) == 0) { res = cur_info->func; } // Cleanup the cur_info DeallocParams(cur_info->params, cur_info->param_count); Dealloc(cur_info->id, cur_info->short_description, cur_info->params, cur_info->errors); } // Cleanup the array. Dealloc(infos); if (res == nullptr) { ScopedLocalRef rt_exception(env, env->FindClass("java/lang/RuntimeException")); env->ThrowNew(rt_exception.get(), (name + " extensions not found").c_str()); return nullptr; } return res; } extern "C" JNIEXPORT void JNICALL Java_art_Test1974_ResizeArray(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject ref_gen, jint new_size) { ChangeArraySize change_array_size = reinterpret_cast( FindExtensionMethod(env, "com.android.art.heap.change_array_size")); if (change_array_size == nullptr) { return; } jmethodID getArr = env->GetMethodID( env->FindClass("java/util/function/Supplier"), "get", "()Ljava/lang/Object;"); jobject arr = env->CallObjectMethod(ref_gen, getArr); JvmtiErrorToException(env, jvmti_env, change_array_size(jvmti_env, arr, new_size)); } extern "C" JNIEXPORT jobject JNICALL Java_art_Test1974_ReadJniRef(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jlong r) { return env->NewLocalRef(reinterpret_cast(static_cast(r))); } extern "C" JNIEXPORT jlong JNICALL Java_art_Test1974_GetWeakGlobalJniRef(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject r) { return static_cast(reinterpret_cast(env->NewWeakGlobalRef(r))); } extern "C" JNIEXPORT jlong JNICALL Java_art_Test1974_GetGlobalJniRef(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject r) { return static_cast(reinterpret_cast(env->NewGlobalRef(r))); } extern "C" JNIEXPORT jobjectArray JNICALL Java_art_Test1974_GetObjectsWithTag(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jlong tag) { jsize cnt = 0; jobject* res = nullptr; if (JvmtiErrorToException( env, jvmti_env, jvmti_env->GetObjectsWithTags(1, &tag, &cnt, &res, nullptr))) { return nullptr; } jobjectArray ret = env->NewObjectArray(cnt, env->FindClass("java/lang/Object"), nullptr); if (ret == nullptr) { return nullptr; } for (jsize i = 0; i < cnt; i++) { env->SetObjectArrayElement(ret, i, res[i]); } jvmti_env->Deallocate(reinterpret_cast(res)); return ret; } extern "C" JNIEXPORT void JNICALL Java_art_Test1974_runNativeTest(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobjectArray arr, jobject resize, jobject print, jobject check) { jmethodID run = env->GetMethodID(env->FindClass("java/lang/Runnable"), "run", "()V"); jmethodID accept = env->GetMethodID( env->FindClass("java/util/function/Consumer"), "accept", "(Ljava/lang/Object;)V"); env->CallVoidMethod(print, accept, arr); env->CallVoidMethod(resize, run); env->CallVoidMethod(print, accept, arr); env->CallVoidMethod(check, accept, arr); } struct JvmtiInfo { std::mutex mu_; std::vector freed_tags_; }; extern "C" JNIEXPORT void JNICALL Java_art_Test1974_StartCollectFrees(JNIEnv* env, jclass k ATTRIBUTE_UNUSED) { jvmtiEventCallbacks cb{ .ObjectFree = [](jvmtiEnv* jvmti, jlong tag) { JvmtiInfo* dat = nullptr; CHECK_EQ(jvmti->GetEnvironmentLocalStorage(reinterpret_cast(&dat)), JVMTI_ERROR_NONE); std::lock_guard mu(dat->mu_); dat->freed_tags_.push_back(tag); }, }; JvmtiInfo* info = new JvmtiInfo; if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEnvironmentLocalStorage(info))) { LOG(INFO) << "couldn't set env-local storage"; return; } if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEventCallbacks(&cb, sizeof(cb)))) { LOG(INFO) << "couldn't set event callback"; return; } JvmtiErrorToException( env, jvmti_env, jvmti_env->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_OBJECT_FREE, nullptr)); } extern "C" JNIEXPORT void JNICALL Java_art_Test1974_StartAssignObsoleteIncrementedId(JNIEnv* env, jclass k ATTRIBUTE_UNUSED) { jint id = FindExtensionEvent(env, "com.android.art.heap.obsolete_object_created"); if (env->ExceptionCheck()) { LOG(INFO) << "Could not find extension event!"; return; } using ObsoleteEvent = void (*)(jvmtiEnv * env, jlong * obsolete, jlong * non_obsolete); ObsoleteEvent oe = [](jvmtiEnv* env ATTRIBUTE_UNUSED, jlong* obsolete, jlong* non_obsolete) { *non_obsolete = *obsolete; *obsolete = *obsolete + 1; }; JvmtiErrorToException( env, jvmti_env, jvmti_env->SetExtensionEventCallback(id, reinterpret_cast(oe))); } extern "C" JNIEXPORT void JNICALL Java_art_Test1974_EndAssignObsoleteIncrementedId(JNIEnv* env, jclass k ATTRIBUTE_UNUSED) { jint id = FindExtensionEvent(env, "com.android.art.heap.obsolete_object_created"); if (env->ExceptionCheck()) { LOG(INFO) << "Could not find extension event!"; return; } JvmtiErrorToException(env, jvmti_env, jvmti_env->SetExtensionEventCallback(id, nullptr)); } extern "C" JNIEXPORT jlongArray JNICALL Java_art_Test1974_CollectFreedTags(JNIEnv* env, jclass k ATTRIBUTE_UNUSED) { if (JvmtiErrorToException( env, jvmti_env, jvmti_env->SetEventNotificationMode(JVMTI_DISABLE, JVMTI_EVENT_OBJECT_FREE, nullptr))) { return nullptr; } JvmtiInfo* info_p = nullptr; if (JvmtiErrorToException( env, jvmti_env, jvmti_env->GetEnvironmentLocalStorage(reinterpret_cast(&info_p)))) { return nullptr; } if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEnvironmentLocalStorage(nullptr))) { return nullptr; } std::unique_ptr info(info_p); ScopedLocalRef arr(env, env->NewLongArray(info->freed_tags_.size())); if (env->ExceptionCheck()) { return nullptr; } env->SetLongArrayRegion(arr.get(), 0, info->freed_tags_.size(), info->freed_tags_.data()); if (env->ExceptionCheck()) { return nullptr; } return arr.release(); } } // namespace Test1974ResizeArray } // namespace art