/* * Copyright (C) 2013 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 "jni.h" #include "jvmti.h" #include "android-base/logging.h" #include "android-base/macros.h" // Test infrastructure #include "jvmti_helper.h" #include "test_env.h" namespace art { namespace Test927JNITable { // This test is equivalent to the jni_internal_test JNIEnvExtTableOverride. static size_t gGlobalRefCount = 0; static JNINativeInterface* gOriginalEnv = nullptr; static jobject CountNewGlobalRef(JNIEnv* env, jobject o) { ++gGlobalRefCount; return gOriginalEnv->NewGlobalRef(env, o); } static void DoDeleteGlobalRef(JNIEnv* env, jobject o) { jclass thr = env->FindClass("java/lang/Thread"); CHECK(thr != nullptr); if (env->IsInstanceOf(o, thr)) { jvmtiThreadInfo jti; // b/146170834: This could cause DCHECK failures. CHECK_EQ(jvmti_env->GetThreadInfo(reinterpret_cast(o), &jti), JVMTI_ERROR_NONE); } gOriginalEnv->DeleteGlobalRef(env, o); } extern "C" JNIEXPORT void JNICALL Java_art_Test928_doOtherThreadTest(JNIEnv* env, jclass klass) { size_t start_other_thread_count = gGlobalRefCount; // Make sure it still works even on another thread. jobject global = env->NewGlobalRef(klass); CHECK_EQ(start_other_thread_count + 1, gGlobalRefCount); env->DeleteGlobalRef(global); } extern "C" JNIEXPORT void JNICALL Java_art_Test928_doJNITableTest( JNIEnv* env, jclass klass) { // Get the current table, as the delegate. jvmtiError getorig_result = jvmti_env->GetJNIFunctionTable(&gOriginalEnv); if (JvmtiErrorToException(env, jvmti_env, getorig_result)) { return; } // Get the current table, as the override we'll install. JNINativeInterface* env_override; jvmtiError getoverride_result = jvmti_env->GetJNIFunctionTable(&env_override); if (JvmtiErrorToException(env, jvmti_env, getoverride_result)) { return; } env_override->NewGlobalRef = CountNewGlobalRef; env_override->DeleteGlobalRef = DoDeleteGlobalRef; gGlobalRefCount = 0; // Install the override. jvmtiError setoverride_result = jvmti_env->SetJNIFunctionTable(env_override); if (JvmtiErrorToException(env, jvmti_env, setoverride_result)) { return; } jobject global = env->NewGlobalRef(klass); CHECK_EQ(1u, gGlobalRefCount); env->DeleteGlobalRef(global); // Try and create and destroy a thread. env->CallStaticVoidMethod(klass, env->GetStaticMethodID(klass, "runThreadTest", "()V")); // Make sure something got ref'd, in the other thread we make and then clear a global ref so that // should at least be present. CHECK_LT(1u, gGlobalRefCount); // Install the "original." There is no real reset. size_t final_global_ref_count = gGlobalRefCount; jvmtiError setoverride2_result = jvmti_env->SetJNIFunctionTable(gOriginalEnv); if (JvmtiErrorToException(env, jvmti_env, setoverride2_result)) { return; } jobject global2 = env->NewGlobalRef(klass); CHECK_EQ(final_global_ref_count, gGlobalRefCount); env->DeleteGlobalRef(global2); // Try to install null. Should return NULL_POINTER error. jvmtiError setoverride3_result = jvmti_env->SetJNIFunctionTable(nullptr); if (setoverride3_result != JVMTI_ERROR_NULL_POINTER) { LOG(FATAL) << "Didn't receive NULL_POINTER"; } jvmti_env->Deallocate(reinterpret_cast(env_override)); } } // namespace Test927JNITable } // namespace art