xref: /aosp_15_r20/art/test/2005-pause-all-redefine-multithreaded/pause-all.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
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