/* * 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 <stdio.h> #include "android-base/macros.h" #include "jni.h" #include "jvmti.h" #include "scoped_local_ref.h" // Test infrastructure #include "jni_helper.h" #include "test_env.h" namespace art { namespace Test918Fields { extern "C" JNIEXPORT jobjectArray JNICALL Java_art_Test918_getFieldName( JNIEnv* env, jclass klass, jobject field) { jfieldID id = env->FromReflectedField(field); char* name; char* sig; char* gen; // Note: technically putting the caller class here is wrong, but we don't need it, anyways. jvmtiError result = jvmti_env->GetFieldName(klass, id, &name, &sig, &gen); if (result != JVMTI_ERROR_NONE) { char* err; jvmti_env->GetErrorName(result, &err); printf("Failure running GetFieldName: %s\n", err); jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(err)); return nullptr; } auto callback = [&](jint i) { if (i == 0) { return name == nullptr ? nullptr : env->NewStringUTF(name); } else if (i == 1) { return sig == nullptr ? nullptr : env->NewStringUTF(sig); } else { return gen == nullptr ? nullptr : env->NewStringUTF(gen); } }; jobjectArray ret = CreateObjectArray(env, 3, "java/lang/String", callback); // Need to deallocate the strings. if (name != nullptr) { jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(name)); } if (sig != nullptr) { jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(sig)); } if (gen != nullptr) { jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(gen)); } // Also run GetMethodName with all parameter pointers null to check for segfaults. jvmtiError result2 = jvmti_env->GetFieldName(klass, id, nullptr, nullptr, nullptr); if (result2 != JVMTI_ERROR_NONE) { char* err; jvmti_env->GetErrorName(result2, &err); printf("Failure running GetFieldName(null, null, null): %s\n", err); jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(err)); return nullptr; } return ret; } extern "C" JNIEXPORT jclass JNICALL Java_art_Test918_getFieldDeclaringClass( JNIEnv* env, jclass klass, jobject field) { jfieldID id = env->FromReflectedField(field); jclass declaring_class; jvmtiError result = jvmti_env->GetFieldDeclaringClass(klass, id, &declaring_class); if (result != JVMTI_ERROR_NONE) { char* err; jvmti_env->GetErrorName(result, &err); printf("Failure running GetFieldDeclaringClass: %s\n", err); jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(err)); return nullptr; } return declaring_class; } extern "C" JNIEXPORT jint JNICALL Java_art_Test918_getFieldModifiers( JNIEnv* env, jclass klass, jobject field) { jfieldID id = env->FromReflectedField(field); jint modifiers; jvmtiError result = jvmti_env->GetFieldModifiers(klass, id, &modifiers); if (result != JVMTI_ERROR_NONE) { char* err; jvmti_env->GetErrorName(result, &err); printf("Failure running GetFieldModifiers: %s\n", err); jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(err)); return 0; } return modifiers; } extern "C" JNIEXPORT jboolean JNICALL Java_art_Test918_isFieldSynthetic( JNIEnv* env, jclass klass, jobject field) { jfieldID id = env->FromReflectedField(field); jboolean synth; jvmtiError result = jvmti_env->IsFieldSynthetic(klass, id, &synth); if (result != JVMTI_ERROR_NONE) { char* err; jvmti_env->GetErrorName(result, &err); printf("Failure running IsFieldSynthetic: %s\n", err); jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(err)); return 0; } return synth; } } // namespace Test918Fields } // namespace art