1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2019 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker *
4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker *
8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker *
10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker */
16*795d594fSAndroid Build Coastguard Worker
17*795d594fSAndroid Build Coastguard Worker #include <stdio.h>
18*795d594fSAndroid Build Coastguard Worker
19*795d594fSAndroid Build Coastguard Worker #include <vector>
20*795d594fSAndroid Build Coastguard Worker
21*795d594fSAndroid Build Coastguard Worker #include "android-base/logging.h"
22*795d594fSAndroid Build Coastguard Worker #include "android-base/macros.h"
23*795d594fSAndroid Build Coastguard Worker #include "jni.h"
24*795d594fSAndroid Build Coastguard Worker #include "jvmti.h"
25*795d594fSAndroid Build Coastguard Worker
26*795d594fSAndroid Build Coastguard Worker // Test infrastructure
27*795d594fSAndroid Build Coastguard Worker #include "jvmti_helper.h"
28*795d594fSAndroid Build Coastguard Worker #include "scoped_local_ref.h"
29*795d594fSAndroid Build Coastguard Worker #include "test_env.h"
30*795d594fSAndroid Build Coastguard Worker
31*795d594fSAndroid Build Coastguard Worker namespace art {
32*795d594fSAndroid Build Coastguard Worker namespace Test2005PauseAllRedefineMultithreaded {
33*795d594fSAndroid Build Coastguard Worker
34*795d594fSAndroid Build Coastguard Worker static constexpr jlong kRedefinedObjectTag = 0xDEADBEEF;
35*795d594fSAndroid Build Coastguard Worker
36*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL
Java_art_Test2005_UpdateFieldValuesAndResumeThreads(JNIEnv * env,jclass klass,jobjectArray threads_arr,jclass redefined_class,jobjectArray new_fields,jstring default_val)37*795d594fSAndroid Build Coastguard Worker Java_art_Test2005_UpdateFieldValuesAndResumeThreads(JNIEnv* env,
38*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] jclass klass,
39*795d594fSAndroid Build Coastguard Worker jobjectArray threads_arr,
40*795d594fSAndroid Build Coastguard Worker jclass redefined_class,
41*795d594fSAndroid Build Coastguard Worker jobjectArray new_fields,
42*795d594fSAndroid Build Coastguard Worker jstring default_val) {
43*795d594fSAndroid Build Coastguard Worker std::vector<jthread> threads;
44*795d594fSAndroid Build Coastguard Worker threads.reserve(env->GetArrayLength(threads_arr));
45*795d594fSAndroid Build Coastguard Worker for (jint i = 0; i < env->GetArrayLength(threads_arr); i++) {
46*795d594fSAndroid Build Coastguard Worker threads.push_back(env->GetObjectArrayElement(threads_arr, i));
47*795d594fSAndroid Build Coastguard Worker }
48*795d594fSAndroid Build Coastguard Worker std::vector<jfieldID> fields;
49*795d594fSAndroid Build Coastguard Worker fields.reserve(env->GetArrayLength(new_fields));
50*795d594fSAndroid Build Coastguard Worker for (jint i = 0; i < env->GetArrayLength(new_fields); i++) {
51*795d594fSAndroid Build Coastguard Worker fields.push_back(env->FromReflectedField(env->GetObjectArrayElement(new_fields, i)));
52*795d594fSAndroid Build Coastguard Worker }
53*795d594fSAndroid Build Coastguard Worker // Tag every instance of the redefined class with kRedefinedObjectTag
54*795d594fSAndroid Build Coastguard Worker CHECK_EQ(jvmti_env->IterateOverInstancesOfClass(
55*795d594fSAndroid Build Coastguard Worker redefined_class,
56*795d594fSAndroid Build Coastguard Worker JVMTI_HEAP_OBJECT_EITHER,
57*795d594fSAndroid Build Coastguard Worker []([[maybe_unused]] jlong class_tag,
58*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] jlong size,
59*795d594fSAndroid Build Coastguard Worker jlong* tag_ptr,
60*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] void* user_data) -> jvmtiIterationControl {
61*795d594fSAndroid Build Coastguard Worker *tag_ptr = kRedefinedObjectTag;
62*795d594fSAndroid Build Coastguard Worker return JVMTI_ITERATION_CONTINUE;
63*795d594fSAndroid Build Coastguard Worker },
64*795d594fSAndroid Build Coastguard Worker nullptr),
65*795d594fSAndroid Build Coastguard Worker JVMTI_ERROR_NONE);
66*795d594fSAndroid Build Coastguard Worker jobject* objs;
67*795d594fSAndroid Build Coastguard Worker jint cnt;
68*795d594fSAndroid Build Coastguard Worker // Get the objects.
69*795d594fSAndroid Build Coastguard Worker CHECK_EQ(jvmti_env->GetObjectsWithTags(1, &kRedefinedObjectTag, &cnt, &objs, nullptr),
70*795d594fSAndroid Build Coastguard Worker JVMTI_ERROR_NONE);
71*795d594fSAndroid Build Coastguard Worker // Set every field that's null
72*795d594fSAndroid Build Coastguard Worker for (jint i = 0; i < cnt; i++) {
73*795d594fSAndroid Build Coastguard Worker jobject obj = objs[i];
74*795d594fSAndroid Build Coastguard Worker for (jfieldID field : fields) {
75*795d594fSAndroid Build Coastguard Worker if (ScopedLocalRef<jobject>(env, env->GetObjectField(obj, field)).get() == nullptr) {
76*795d594fSAndroid Build Coastguard Worker env->SetObjectField(obj, field, default_val);
77*795d594fSAndroid Build Coastguard Worker }
78*795d594fSAndroid Build Coastguard Worker }
79*795d594fSAndroid Build Coastguard Worker }
80*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "Setting " << cnt << " objects with default values";
81*795d594fSAndroid Build Coastguard Worker if (!threads.empty()) {
82*795d594fSAndroid Build Coastguard Worker std::vector<jvmtiError> errs(threads.size(), JVMTI_ERROR_NONE);
83*795d594fSAndroid Build Coastguard Worker CHECK_EQ(jvmti_env->ResumeThreadList(threads.size(), threads.data(), errs.data()),
84*795d594fSAndroid Build Coastguard Worker JVMTI_ERROR_NONE);
85*795d594fSAndroid Build Coastguard Worker }
86*795d594fSAndroid Build Coastguard Worker jvmti_env->Deallocate(reinterpret_cast<unsigned char*>(objs));
87*795d594fSAndroid Build Coastguard Worker }
88*795d594fSAndroid Build Coastguard Worker
89*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT jobject JNICALL
Java_Main_fastNativeSleepAndReturnInteger42(JNIEnv * env,jclass klass)90*795d594fSAndroid Build Coastguard Worker Java_Main_fastNativeSleepAndReturnInteger42(JNIEnv* env, [[maybe_unused]] jclass klass) {
91*795d594fSAndroid Build Coastguard Worker jclass integer_class = env->FindClass("java/lang/Integer");
92*795d594fSAndroid Build Coastguard Worker CHECK(integer_class != nullptr);
93*795d594fSAndroid Build Coastguard Worker jmethodID integer_value_of =
94*795d594fSAndroid Build Coastguard Worker env->GetStaticMethodID(integer_class, "valueOf", "(I)Ljava/lang/Integer;");
95*795d594fSAndroid Build Coastguard Worker CHECK(integer_value_of != nullptr);
96*795d594fSAndroid Build Coastguard Worker jobject value = env->CallStaticObjectMethod(integer_class, integer_value_of, 42);
97*795d594fSAndroid Build Coastguard Worker CHECK(value != nullptr);
98*795d594fSAndroid Build Coastguard Worker // Sleep for 500ms, blocking thread suspension (this method is @FastNative).
99*795d594fSAndroid Build Coastguard Worker // Except for some odd thread timing, this should ensure that the suspend
100*795d594fSAndroid Build Coastguard Worker // request from the redefinition thread is seen by the suspend check in the
101*795d594fSAndroid Build Coastguard Worker // JNI stub when we exit this function and then processed with the JNI stub
102*795d594fSAndroid Build Coastguard Worker // still on the stack. The instrumentation previously erroneously
103*795d594fSAndroid Build Coastguard Worker // intercepted returning to the JNI stub and the "instrumentation exit"
104*795d594fSAndroid Build Coastguard Worker // handler treated the return value `jobject` as `mirror::Object*`.
105*795d594fSAndroid Build Coastguard Worker usleep(500000);
106*795d594fSAndroid Build Coastguard Worker return value;
107*795d594fSAndroid Build Coastguard Worker }
108*795d594fSAndroid Build Coastguard Worker
109*795d594fSAndroid Build Coastguard Worker } // namespace Test2005PauseAllRedefineMultithreaded
110*795d594fSAndroid Build Coastguard Worker } // namespace art
111