269 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			269 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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
 |