/*
* 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 <pthread.h>
#include <cstdio>
#include <iostream>
#include <vector>
#include "android-base/logging.h"
#include "jni.h"
#include "jvmti.h"
#include "scoped_local_ref.h"
#include "scoped_primitive_array.h"
// Test infrastructure
#include "jvmti_helper.h"
#include "test_env.h"
namespace art {
namespace Test903HelloTagging {
extern "C" JNIEXPORT jobjectArray JNICALL Java_art_Test903_getTaggedObjects(
JNIEnv* env, jclass, jlongArray searchTags, jboolean returnObjects, jboolean returnTags) {
ScopedLongArrayRO scoped_array(env);
if (searchTags != nullptr) {
scoped_array.reset(searchTags);
}
const jlong* tag_ptr = scoped_array.get();
if (tag_ptr == nullptr) {
// Can never pass null.
tag_ptr = reinterpret_cast<const jlong*>(1);
}
jint result_count;
jobject* result_object_array;
jobject** result_object_array_ptr = returnObjects == JNI_TRUE ? &result_object_array : nullptr;
jlong* result_tag_array;
jlong** result_tag_array_ptr = returnTags == JNI_TRUE ? &result_tag_array : nullptr;
jvmtiError ret = jvmti_env->GetObjectsWithTags(scoped_array.size(),
tag_ptr,
&result_count,
result_object_array_ptr,
result_tag_array_ptr);
if (JvmtiErrorToException(env, jvmti_env, ret)) {
return nullptr;
}
CHECK_GE(result_count, 0);
ScopedLocalRef<jclass> obj_class(env, env->FindClass("java/lang/Object"));
if (obj_class.get() == nullptr) {
return nullptr;
}
jobjectArray resultObjectArray = nullptr;
if (returnObjects == JNI_TRUE) {
resultObjectArray = env->NewObjectArray(result_count, obj_class.get(), nullptr);
if (resultObjectArray == nullptr) {
return nullptr;
}
for (jint i = 0; i < result_count; ++i) {
env->SetObjectArrayElement(resultObjectArray, i, result_object_array[i]);
}
}
jlongArray resultTagArray = nullptr;
if (returnTags == JNI_TRUE) {
resultTagArray = env->NewLongArray(result_count);
env->SetLongArrayRegion(resultTagArray, 0, result_count, result_tag_array);
}
jobject count_integer;
{
ScopedLocalRef<jclass> integer_class(env, env->FindClass("java/lang/Integer"));
jmethodID methodID = env->GetMethodID(integer_class.get(), "<init>", "(I)V");
count_integer = env->NewObject(integer_class.get(), methodID, result_count);
if (count_integer == nullptr) {
return nullptr;
}
}
jobjectArray resultArray = env->NewObjectArray(3, obj_class.get(), nullptr);
if (resultArray == nullptr) {
return nullptr;
}
env->SetObjectArrayElement(resultArray, 0, resultObjectArray);
env->SetObjectArrayElement(resultArray, 1, resultTagArray);
env->SetObjectArrayElement(resultArray, 2, count_integer);
return resultArray;
}
static jvmtiEnv* CreateJvmtiEnv(JNIEnv* env) {
JavaVM* jvm;
CHECK_EQ(0, env->GetJavaVM(&jvm));
jvmtiEnv* new_jvmti_env;
CHECK_EQ(0, jvm->GetEnv(reinterpret_cast<void**>(&new_jvmti_env), JVMTI_VERSION_1_0));
jvmtiCapabilities capa;
memset(&capa, 0, sizeof(jvmtiCapabilities));
capa.can_tag_objects = 1;
jvmtiError error = new_jvmti_env->AddCapabilities(&capa);
CHECK_EQ(JVMTI_ERROR_NONE, error);
return new_jvmti_env;
}
static void SetTag(jvmtiEnv* env, jobject obj, jlong tag) {
jvmtiError ret = env->SetTag(obj, tag);
CHECK_EQ(JVMTI_ERROR_NONE, ret);
}
static jlong GetTag(jvmtiEnv* env, jobject obj) {
jlong tag;
jvmtiError ret = env->GetTag(obj, &tag);
CHECK_EQ(JVMTI_ERROR_NONE, ret);
return tag;
}
extern "C" JNIEXPORT jlongArray JNICALL Java_art_Test903_testTagsInDifferentEnvs(
JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject obj, jlong base_tag, jint count) {
std::unique_ptr<jvmtiEnv*[]> envs = std::unique_ptr<jvmtiEnv*[]>(new jvmtiEnv*[count]);
envs[0] = jvmti_env;
for (int32_t i = 1; i != count; ++i) {
envs[i] = CreateJvmtiEnv(env);
}
for (int32_t i = 0; i != count; ++i) {
SetTag(envs[i], obj, base_tag + i);
}
std::unique_ptr<jlong[]> vals = std::unique_ptr<jlong[]>(new jlong[count]);
for (int32_t i = 0; i != count; ++i) {
vals[i] = GetTag(envs[i], obj);
}
for (int32_t i = 1; i != count; ++i) {
CHECK_EQ(JVMTI_ERROR_NONE, envs[i]->DisposeEnvironment());
}
jlongArray res = env->NewLongArray(count);
if (res == nullptr) {
return nullptr;
}
env->SetLongArrayRegion(res, 0, count, vals.get());
return res;
}
} // namespace Test903HelloTagging
} // namespace art