| /* |
| * 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 <cstdio> |
| #include <memory> |
| #include <mutex> |
| #include <string> |
| #include <vector> |
| |
| #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 <typename T> static void Dealloc(T* t) { |
| jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(t)); |
| } |
| |
| template <typename T, typename... Rest> 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<jclass> 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<jclass> 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<ChangeArraySize>( |
| 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<jobject>(static_cast<intptr_t>(r))); |
| } |
| |
| extern "C" JNIEXPORT jlong JNICALL |
| Java_art_Test1974_GetWeakGlobalJniRef(JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject r) { |
| return static_cast<jlong>(reinterpret_cast<intptr_t>(env->NewWeakGlobalRef(r))); |
| } |
| |
| extern "C" JNIEXPORT jlong JNICALL Java_art_Test1974_GetGlobalJniRef(JNIEnv* env, |
| jclass klass ATTRIBUTE_UNUSED, |
| jobject r) { |
| return static_cast<jlong>(reinterpret_cast<intptr_t>(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<unsigned char*>(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<jlong> 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<void**>(&dat)), |
| JVMTI_ERROR_NONE); |
| std::lock_guard<std::mutex> 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<jvmtiExtensionEvent>(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<void**>(&info_p)))) { |
| return nullptr; |
| } |
| if (JvmtiErrorToException(env, jvmti_env, jvmti_env->SetEnvironmentLocalStorage(nullptr))) { |
| return nullptr; |
| } |
| std::unique_ptr<JvmtiInfo> info(info_p); |
| ScopedLocalRef<jlongArray> 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 |