xref: /aosp_15_r20/art/tools/jvmti-agents/field-null-percent/fieldnull.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker // Copyright (C) 2018 The Android Open Source Project
2*795d594fSAndroid Build Coastguard Worker //
3*795d594fSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*795d594fSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*795d594fSAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*795d594fSAndroid Build Coastguard Worker //
7*795d594fSAndroid Build Coastguard Worker //      http://www.apache.org/licenses/LICENSE-2.0
8*795d594fSAndroid Build Coastguard Worker //
9*795d594fSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*795d594fSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*795d594fSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*795d594fSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*795d594fSAndroid Build Coastguard Worker // limitations under the License.
14*795d594fSAndroid Build Coastguard Worker //
15*795d594fSAndroid Build Coastguard Worker 
16*795d594fSAndroid Build Coastguard Worker #include <android-base/logging.h>
17*795d594fSAndroid Build Coastguard Worker 
18*795d594fSAndroid Build Coastguard Worker #include <atomic>
19*795d594fSAndroid Build Coastguard Worker #include <iomanip>
20*795d594fSAndroid Build Coastguard Worker #include <iostream>
21*795d594fSAndroid Build Coastguard Worker #include <istream>
22*795d594fSAndroid Build Coastguard Worker #include <jni.h>
23*795d594fSAndroid Build Coastguard Worker #include <jvmti.h>
24*795d594fSAndroid Build Coastguard Worker #include <memory>
25*795d594fSAndroid Build Coastguard Worker #include <sstream>
26*795d594fSAndroid Build Coastguard Worker #include <string.h>
27*795d594fSAndroid Build Coastguard Worker #include <string>
28*795d594fSAndroid Build Coastguard Worker #include <vector>
29*795d594fSAndroid Build Coastguard Worker 
30*795d594fSAndroid Build Coastguard Worker namespace fieldnull {
31*795d594fSAndroid Build Coastguard Worker 
32*795d594fSAndroid Build Coastguard Worker #define CHECK_JVMTI(x) CHECK_EQ((x), JVMTI_ERROR_NONE)
33*795d594fSAndroid Build Coastguard Worker 
34*795d594fSAndroid Build Coastguard Worker // Special art ti-version number. We will use this as a fallback if we cannot get a regular JVMTI
35*795d594fSAndroid Build Coastguard Worker // env.
36*795d594fSAndroid Build Coastguard Worker static constexpr jint kArtTiVersion = JVMTI_VERSION_1_2 | 0x40000000;
37*795d594fSAndroid Build Coastguard Worker 
38*795d594fSAndroid Build Coastguard Worker static JavaVM* java_vm = nullptr;
39*795d594fSAndroid Build Coastguard Worker 
40*795d594fSAndroid Build Coastguard Worker // Field is "Lclass/name/here;.field_name:Lfield/type/here;"
SplitField(JNIEnv * env,const std::string & field_id)41*795d594fSAndroid Build Coastguard Worker static std::pair<jclass, jfieldID> SplitField(JNIEnv* env, const std::string& field_id) {
42*795d594fSAndroid Build Coastguard Worker   CHECK_EQ(field_id[0], 'L');
43*795d594fSAndroid Build Coastguard Worker   env->PushLocalFrame(1);
44*795d594fSAndroid Build Coastguard Worker   std::istringstream is(field_id);
45*795d594fSAndroid Build Coastguard Worker   std::string class_name;
46*795d594fSAndroid Build Coastguard Worker   std::string field_name;
47*795d594fSAndroid Build Coastguard Worker   std::string field_type;
48*795d594fSAndroid Build Coastguard Worker 
49*795d594fSAndroid Build Coastguard Worker   std::getline(is, class_name, '.');
50*795d594fSAndroid Build Coastguard Worker   std::getline(is, field_name, ':');
51*795d594fSAndroid Build Coastguard Worker   std::getline(is, field_type, '\0');
52*795d594fSAndroid Build Coastguard Worker 
53*795d594fSAndroid Build Coastguard Worker   jclass klass = reinterpret_cast<jclass>(
54*795d594fSAndroid Build Coastguard Worker       env->NewGlobalRef(env->FindClass(class_name.substr(1, class_name.size() - 2).c_str())));
55*795d594fSAndroid Build Coastguard Worker   jfieldID field = env->GetFieldID(klass, field_name.c_str(), field_type.c_str());
56*795d594fSAndroid Build Coastguard Worker   CHECK(klass != nullptr);
57*795d594fSAndroid Build Coastguard Worker   CHECK(field != nullptr);
58*795d594fSAndroid Build Coastguard Worker   LOG(INFO) << "listing field " << field_id;
59*795d594fSAndroid Build Coastguard Worker   env->PopLocalFrame(nullptr);
60*795d594fSAndroid Build Coastguard Worker   return std::make_pair(klass, field);
61*795d594fSAndroid Build Coastguard Worker }
62*795d594fSAndroid Build Coastguard Worker 
GetRequestedFields(JNIEnv * env,const std::string & args)63*795d594fSAndroid Build Coastguard Worker static std::vector<std::pair<jclass, jfieldID>> GetRequestedFields(JNIEnv* env,
64*795d594fSAndroid Build Coastguard Worker                                                                    const std::string& args) {
65*795d594fSAndroid Build Coastguard Worker   std::vector<std::pair<jclass, jfieldID>> res;
66*795d594fSAndroid Build Coastguard Worker   std::stringstream args_stream(args);
67*795d594fSAndroid Build Coastguard Worker   std::string item;
68*795d594fSAndroid Build Coastguard Worker   while (std::getline(args_stream, item, ',')) {
69*795d594fSAndroid Build Coastguard Worker     if (item == "") {
70*795d594fSAndroid Build Coastguard Worker       continue;
71*795d594fSAndroid Build Coastguard Worker     }
72*795d594fSAndroid Build Coastguard Worker     res.push_back(SplitField(env, item));
73*795d594fSAndroid Build Coastguard Worker   }
74*795d594fSAndroid Build Coastguard Worker   return res;
75*795d594fSAndroid Build Coastguard Worker }
76*795d594fSAndroid Build Coastguard Worker 
SetupJvmtiEnv(JavaVM * vm,jvmtiEnv ** jvmti)77*795d594fSAndroid Build Coastguard Worker static jint SetupJvmtiEnv(JavaVM* vm, jvmtiEnv** jvmti) {
78*795d594fSAndroid Build Coastguard Worker   jint res = 0;
79*795d594fSAndroid Build Coastguard Worker   res = vm->GetEnv(reinterpret_cast<void**>(jvmti), JVMTI_VERSION_1_1);
80*795d594fSAndroid Build Coastguard Worker 
81*795d594fSAndroid Build Coastguard Worker   if (res != JNI_OK || *jvmti == nullptr) {
82*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << "Unable to access JVMTI, error code " << res;
83*795d594fSAndroid Build Coastguard Worker     return vm->GetEnv(reinterpret_cast<void**>(jvmti), kArtTiVersion);
84*795d594fSAndroid Build Coastguard Worker   }
85*795d594fSAndroid Build Coastguard Worker   return res;
86*795d594fSAndroid Build Coastguard Worker }
87*795d594fSAndroid Build Coastguard Worker 
88*795d594fSAndroid Build Coastguard Worker struct RequestList {
89*795d594fSAndroid Build Coastguard Worker   std::vector<std::pair<jclass, jfieldID>> fields_;
90*795d594fSAndroid Build Coastguard Worker };
91*795d594fSAndroid Build Coastguard Worker 
DataDumpRequestCb(jvmtiEnv * jvmti)92*795d594fSAndroid Build Coastguard Worker static void DataDumpRequestCb(jvmtiEnv* jvmti) {
93*795d594fSAndroid Build Coastguard Worker   JNIEnv* env = nullptr;
94*795d594fSAndroid Build Coastguard Worker   CHECK_EQ(java_vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6), JNI_OK);
95*795d594fSAndroid Build Coastguard Worker   LOG(INFO) << "Dumping counts of null fields.";
96*795d594fSAndroid Build Coastguard Worker   LOG(INFO) << "\t" << "Field name"
97*795d594fSAndroid Build Coastguard Worker             << "\t" << "null count"
98*795d594fSAndroid Build Coastguard Worker             << "\t" << "total count";
99*795d594fSAndroid Build Coastguard Worker   RequestList* list;
100*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&list)));
101*795d594fSAndroid Build Coastguard Worker   for (std::pair<jclass, jfieldID>& p : list->fields_) {
102*795d594fSAndroid Build Coastguard Worker     jclass klass = p.first;
103*795d594fSAndroid Build Coastguard Worker     jfieldID field = p.second;
104*795d594fSAndroid Build Coastguard Worker     // Make sure all instances of the class are tagged with the klass ptr value. Since this is a
105*795d594fSAndroid Build Coastguard Worker     // global ref it's guaranteed to be unique.
106*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->IterateOverInstancesOfClass(
107*795d594fSAndroid Build Coastguard Worker         p.first,
108*795d594fSAndroid Build Coastguard Worker         // We need to do this to all objects every time since we might be looking for multiple
109*795d594fSAndroid Build Coastguard Worker         // fields in classes that are subtypes of each other.
110*795d594fSAndroid Build Coastguard Worker         JVMTI_HEAP_OBJECT_EITHER,
111*795d594fSAndroid Build Coastguard Worker         /* class_tag, size, tag_ptr, user_data*/
112*795d594fSAndroid Build Coastguard Worker         [](jlong, jlong, jlong* tag_ptr, void* klass) -> jvmtiIterationControl {
113*795d594fSAndroid Build Coastguard Worker           *tag_ptr = static_cast<jlong>(reinterpret_cast<intptr_t>(klass));
114*795d594fSAndroid Build Coastguard Worker           return JVMTI_ITERATION_CONTINUE;
115*795d594fSAndroid Build Coastguard Worker         },
116*795d594fSAndroid Build Coastguard Worker         klass));
117*795d594fSAndroid Build Coastguard Worker     jobject* obj_list;
118*795d594fSAndroid Build Coastguard Worker     jint obj_len;
119*795d594fSAndroid Build Coastguard Worker     jlong tag = static_cast<jlong>(reinterpret_cast<intptr_t>(klass));
120*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->GetObjectsWithTags(1, &tag, &obj_len, &obj_list, nullptr));
121*795d594fSAndroid Build Coastguard Worker 
122*795d594fSAndroid Build Coastguard Worker     uint64_t null_cnt = 0;
123*795d594fSAndroid Build Coastguard Worker     for (jint i = 0; i < obj_len; i++) {
124*795d594fSAndroid Build Coastguard Worker       if (env->GetObjectField(obj_list[i], field) == nullptr) {
125*795d594fSAndroid Build Coastguard Worker         null_cnt++;
126*795d594fSAndroid Build Coastguard Worker       }
127*795d594fSAndroid Build Coastguard Worker     }
128*795d594fSAndroid Build Coastguard Worker 
129*795d594fSAndroid Build Coastguard Worker     char* field_name;
130*795d594fSAndroid Build Coastguard Worker     char* field_sig;
131*795d594fSAndroid Build Coastguard Worker     char* class_name;
132*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->GetFieldName(klass, field, &field_name, &field_sig, nullptr));
133*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->GetClassSignature(klass, &class_name, nullptr));
134*795d594fSAndroid Build Coastguard Worker     LOG(INFO) << "\t" << class_name << "." << field_name << ":" << field_sig
135*795d594fSAndroid Build Coastguard Worker               << "\t" << null_cnt
136*795d594fSAndroid Build Coastguard Worker               << "\t" << obj_len;
137*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(field_name)));
138*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(field_sig)));
139*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(class_name)));
140*795d594fSAndroid Build Coastguard Worker   }
141*795d594fSAndroid Build Coastguard Worker }
142*795d594fSAndroid Build Coastguard Worker 
VMDeathCb(jvmtiEnv * jvmti,JNIEnv * env)143*795d594fSAndroid Build Coastguard Worker static void VMDeathCb(jvmtiEnv* jvmti, [[maybe_unused]] JNIEnv* env) {
144*795d594fSAndroid Build Coastguard Worker   DataDumpRequestCb(jvmti);
145*795d594fSAndroid Build Coastguard Worker   RequestList* list = nullptr;
146*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&list)));
147*795d594fSAndroid Build Coastguard Worker   delete list;
148*795d594fSAndroid Build Coastguard Worker }
149*795d594fSAndroid Build Coastguard Worker 
CreateFieldList(jvmtiEnv * jvmti,JNIEnv * env,const std::string & args)150*795d594fSAndroid Build Coastguard Worker static void CreateFieldList(jvmtiEnv* jvmti, JNIEnv* env, const std::string& args) {
151*795d594fSAndroid Build Coastguard Worker   RequestList* list = nullptr;
152*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->Allocate(sizeof(*list), reinterpret_cast<unsigned char**>(&list)));
153*795d594fSAndroid Build Coastguard Worker   new (list) RequestList { .fields_ = GetRequestedFields(env, args), };
154*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(list));
155*795d594fSAndroid Build Coastguard Worker }
156*795d594fSAndroid Build Coastguard Worker 
VMInitCb(jvmtiEnv * jvmti,JNIEnv * env,jobject thr)157*795d594fSAndroid Build Coastguard Worker static void VMInitCb(jvmtiEnv* jvmti, JNIEnv* env, [[maybe_unused]] jobject thr) {
158*795d594fSAndroid Build Coastguard Worker   char* args = nullptr;
159*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&args)));
160*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(nullptr));
161*795d594fSAndroid Build Coastguard Worker   CreateFieldList(jvmti, env, args);
162*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_DEATH, nullptr));
163*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE,
164*795d594fSAndroid Build Coastguard Worker                                               JVMTI_EVENT_DATA_DUMP_REQUEST,
165*795d594fSAndroid Build Coastguard Worker                                               nullptr));
166*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(args)));
167*795d594fSAndroid Build Coastguard Worker }
168*795d594fSAndroid Build Coastguard Worker 
AgentStart(JavaVM * vm,char * options,bool is_onload)169*795d594fSAndroid Build Coastguard Worker static jint AgentStart(JavaVM* vm, char* options, bool is_onload) {
170*795d594fSAndroid Build Coastguard Worker   android::base::InitLogging(/* argv= */nullptr);
171*795d594fSAndroid Build Coastguard Worker   java_vm = vm;
172*795d594fSAndroid Build Coastguard Worker   jvmtiEnv* jvmti = nullptr;
173*795d594fSAndroid Build Coastguard Worker   if (SetupJvmtiEnv(vm, &jvmti) != JNI_OK) {
174*795d594fSAndroid Build Coastguard Worker     LOG(ERROR) << "Could not get JVMTI env or ArtTiEnv!";
175*795d594fSAndroid Build Coastguard Worker     return JNI_ERR;
176*795d594fSAndroid Build Coastguard Worker   }
177*795d594fSAndroid Build Coastguard Worker   jvmtiCapabilities caps { .can_tag_objects = 1, };
178*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->AddCapabilities(&caps));
179*795d594fSAndroid Build Coastguard Worker   jvmtiEventCallbacks cb {
180*795d594fSAndroid Build Coastguard Worker     .VMInit = VMInitCb,
181*795d594fSAndroid Build Coastguard Worker     .VMDeath = VMDeathCb,
182*795d594fSAndroid Build Coastguard Worker     .DataDumpRequest = DataDumpRequestCb,
183*795d594fSAndroid Build Coastguard Worker   };
184*795d594fSAndroid Build Coastguard Worker   CHECK_JVMTI(jvmti->SetEventCallbacks(&cb, sizeof(cb)));
185*795d594fSAndroid Build Coastguard Worker   if (is_onload) {
186*795d594fSAndroid Build Coastguard Worker     unsigned char* ptr = nullptr;
187*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->Allocate(strlen(options) + 1, &ptr));
188*795d594fSAndroid Build Coastguard Worker     strcpy(reinterpret_cast<char*>(ptr), options);
189*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(ptr));
190*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_INIT, nullptr));
191*795d594fSAndroid Build Coastguard Worker   } else {
192*795d594fSAndroid Build Coastguard Worker     JNIEnv* env = nullptr;
193*795d594fSAndroid Build Coastguard Worker     CHECK_EQ(vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6), JNI_OK);
194*795d594fSAndroid Build Coastguard Worker     CreateFieldList(jvmti, env, options);
195*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_DEATH, nullptr));
196*795d594fSAndroid Build Coastguard Worker     CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE,
197*795d594fSAndroid Build Coastguard Worker                                                 JVMTI_EVENT_DATA_DUMP_REQUEST,
198*795d594fSAndroid Build Coastguard Worker                                                 nullptr));
199*795d594fSAndroid Build Coastguard Worker   }
200*795d594fSAndroid Build Coastguard Worker   return JNI_OK;
201*795d594fSAndroid Build Coastguard Worker }
202*795d594fSAndroid Build Coastguard Worker 
203*795d594fSAndroid Build Coastguard Worker // Late attachment (e.g. 'am attach-agent').
Agent_OnAttach(JavaVM * vm,char * options,void * reserved)204*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT jint JNICALL Agent_OnAttach(JavaVM* vm,
205*795d594fSAndroid Build Coastguard Worker                                                  char* options,
206*795d594fSAndroid Build Coastguard Worker                                                  [[maybe_unused]] void* reserved) {
207*795d594fSAndroid Build Coastguard Worker   return AgentStart(vm, options, /*is_onload=*/false);
208*795d594fSAndroid Build Coastguard Worker }
209*795d594fSAndroid Build Coastguard Worker 
210*795d594fSAndroid Build Coastguard Worker // Early attachment
Agent_OnLoad(JavaVM * jvm,char * options,void * reserved)211*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM* jvm,
212*795d594fSAndroid Build Coastguard Worker                                                char* options,
213*795d594fSAndroid Build Coastguard Worker                                                [[maybe_unused]] void* reserved) {
214*795d594fSAndroid Build Coastguard Worker   return AgentStart(jvm, options, /*is_onload=*/true);
215*795d594fSAndroid Build Coastguard Worker }
216*795d594fSAndroid Build Coastguard Worker 
217*795d594fSAndroid Build Coastguard Worker }  // namespace fieldnull
218*795d594fSAndroid Build Coastguard Worker 
219