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 #include <nativehelper/scoped_local_ref.h>
18*795d594fSAndroid Build Coastguard Worker
19*795d594fSAndroid Build Coastguard Worker #include <atomic>
20*795d594fSAndroid Build Coastguard Worker #include <iomanip>
21*795d594fSAndroid Build Coastguard Worker #include <iostream>
22*795d594fSAndroid Build Coastguard Worker #include <istream>
23*795d594fSAndroid Build Coastguard Worker #include <jni.h>
24*795d594fSAndroid Build Coastguard Worker #include <jvmti.h>
25*795d594fSAndroid Build Coastguard Worker #include <memory>
26*795d594fSAndroid Build Coastguard Worker #include <sstream>
27*795d594fSAndroid Build Coastguard Worker #include <string.h>
28*795d594fSAndroid Build Coastguard Worker #include <string>
29*795d594fSAndroid Build Coastguard Worker #include <unordered_map>
30*795d594fSAndroid Build Coastguard Worker #include <vector>
31*795d594fSAndroid Build Coastguard Worker
32*795d594fSAndroid Build Coastguard Worker namespace fieldnull {
33*795d594fSAndroid Build Coastguard Worker
34*795d594fSAndroid Build Coastguard Worker #define CHECK_JVMTI(x) CHECK_EQ((x), JVMTI_ERROR_NONE)
35*795d594fSAndroid Build Coastguard Worker
36*795d594fSAndroid Build Coastguard Worker // Special art ti-version number. We will use this as a fallback if we cannot get a regular JVMTI
37*795d594fSAndroid Build Coastguard Worker // env.
38*795d594fSAndroid Build Coastguard Worker static constexpr jint kArtTiVersion = JVMTI_VERSION_1_2 | 0x40000000;
39*795d594fSAndroid Build Coastguard Worker
40*795d594fSAndroid Build Coastguard Worker static JavaVM* java_vm = nullptr;
41*795d594fSAndroid Build Coastguard Worker
42*795d594fSAndroid Build Coastguard Worker // Field is "Lclass/name/here;.field_name:Lfield/type/here;"
SplitField(JNIEnv * env,const std::string & field_id)43*795d594fSAndroid Build Coastguard Worker static std::pair<jclass, jfieldID> SplitField(JNIEnv* env, const std::string& field_id) {
44*795d594fSAndroid Build Coastguard Worker CHECK_EQ(field_id[0], 'L');
45*795d594fSAndroid Build Coastguard Worker env->PushLocalFrame(1);
46*795d594fSAndroid Build Coastguard Worker std::istringstream is(field_id);
47*795d594fSAndroid Build Coastguard Worker std::string class_name;
48*795d594fSAndroid Build Coastguard Worker std::string field_name;
49*795d594fSAndroid Build Coastguard Worker std::string field_type;
50*795d594fSAndroid Build Coastguard Worker
51*795d594fSAndroid Build Coastguard Worker std::getline(is, class_name, '.');
52*795d594fSAndroid Build Coastguard Worker std::getline(is, field_name, ':');
53*795d594fSAndroid Build Coastguard Worker std::getline(is, field_type, '\0');
54*795d594fSAndroid Build Coastguard Worker
55*795d594fSAndroid Build Coastguard Worker jclass klass = reinterpret_cast<jclass>(
56*795d594fSAndroid Build Coastguard Worker env->NewGlobalRef(env->FindClass(class_name.substr(1, class_name.size() - 2).c_str())));
57*795d594fSAndroid Build Coastguard Worker CHECK(klass != nullptr) << class_name;
58*795d594fSAndroid Build Coastguard Worker jfieldID field = env->GetFieldID(klass, field_name.c_str(), field_type.c_str());
59*795d594fSAndroid Build Coastguard Worker CHECK(field != nullptr) << field_name;
60*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "listing field " << field_id;
61*795d594fSAndroid Build Coastguard Worker env->PopLocalFrame(nullptr);
62*795d594fSAndroid Build Coastguard Worker return std::make_pair(klass, field);
63*795d594fSAndroid Build Coastguard Worker }
64*795d594fSAndroid Build Coastguard Worker
GetRequestedFields(JNIEnv * env,const std::string & args)65*795d594fSAndroid Build Coastguard Worker static std::vector<std::pair<jclass, jfieldID>> GetRequestedFields(JNIEnv* env,
66*795d594fSAndroid Build Coastguard Worker const std::string& args) {
67*795d594fSAndroid Build Coastguard Worker std::vector<std::pair<jclass, jfieldID>> res;
68*795d594fSAndroid Build Coastguard Worker std::stringstream args_stream(args);
69*795d594fSAndroid Build Coastguard Worker std::string item;
70*795d594fSAndroid Build Coastguard Worker while (std::getline(args_stream, item, ',')) {
71*795d594fSAndroid Build Coastguard Worker if (item == "") {
72*795d594fSAndroid Build Coastguard Worker continue;
73*795d594fSAndroid Build Coastguard Worker }
74*795d594fSAndroid Build Coastguard Worker res.push_back(SplitField(env, item));
75*795d594fSAndroid Build Coastguard Worker }
76*795d594fSAndroid Build Coastguard Worker return res;
77*795d594fSAndroid Build Coastguard Worker }
78*795d594fSAndroid Build Coastguard Worker
SetupJvmtiEnv(JavaVM * vm,jvmtiEnv ** jvmti)79*795d594fSAndroid Build Coastguard Worker static jint SetupJvmtiEnv(JavaVM* vm, jvmtiEnv** jvmti) {
80*795d594fSAndroid Build Coastguard Worker jint res = 0;
81*795d594fSAndroid Build Coastguard Worker res = vm->GetEnv(reinterpret_cast<void**>(jvmti), JVMTI_VERSION_1_1);
82*795d594fSAndroid Build Coastguard Worker
83*795d594fSAndroid Build Coastguard Worker if (res != JNI_OK || *jvmti == nullptr) {
84*795d594fSAndroid Build Coastguard Worker LOG(ERROR) << "Unable to access JVMTI, error code " << res;
85*795d594fSAndroid Build Coastguard Worker return vm->GetEnv(reinterpret_cast<void**>(jvmti), kArtTiVersion);
86*795d594fSAndroid Build Coastguard Worker }
87*795d594fSAndroid Build Coastguard Worker return res;
88*795d594fSAndroid Build Coastguard Worker }
89*795d594fSAndroid Build Coastguard Worker
90*795d594fSAndroid Build Coastguard Worker struct RequestList {
91*795d594fSAndroid Build Coastguard Worker std::vector<std::pair<jclass, jfieldID>> fields_;
92*795d594fSAndroid Build Coastguard Worker };
93*795d594fSAndroid Build Coastguard Worker
DataDumpRequestCb(jvmtiEnv * jvmti)94*795d594fSAndroid Build Coastguard Worker static void DataDumpRequestCb(jvmtiEnv* jvmti) {
95*795d594fSAndroid Build Coastguard Worker JNIEnv* env = nullptr;
96*795d594fSAndroid Build Coastguard Worker CHECK_EQ(java_vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6), JNI_OK);
97*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "Dumping counts of fields.";
98*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "\t" << "Field name"
99*795d594fSAndroid Build Coastguard Worker << "\t" << "Type"
100*795d594fSAndroid Build Coastguard Worker << "\t" << "Count"
101*795d594fSAndroid Build Coastguard Worker << "\t" << "TotalSize";
102*795d594fSAndroid Build Coastguard Worker RequestList* list;
103*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&list)));
104*795d594fSAndroid Build Coastguard Worker for (std::pair<jclass, jfieldID>& p : list->fields_) {
105*795d594fSAndroid Build Coastguard Worker jclass klass = p.first;
106*795d594fSAndroid Build Coastguard Worker jfieldID field = p.second;
107*795d594fSAndroid Build Coastguard Worker // Make sure all instances of the class are tagged with the klass ptr value. Since this is a
108*795d594fSAndroid Build Coastguard Worker // global ref it's guaranteed to be unique.
109*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->IterateOverInstancesOfClass(
110*795d594fSAndroid Build Coastguard Worker p.first,
111*795d594fSAndroid Build Coastguard Worker // We need to do this to all objects every time since we might be looking for multiple
112*795d594fSAndroid Build Coastguard Worker // fields in classes that are subtypes of each other.
113*795d594fSAndroid Build Coastguard Worker JVMTI_HEAP_OBJECT_EITHER,
114*795d594fSAndroid Build Coastguard Worker /* class_tag, size, tag_ptr, user_data*/
115*795d594fSAndroid Build Coastguard Worker [](jlong, jlong, jlong* tag_ptr, void* klass) -> jvmtiIterationControl {
116*795d594fSAndroid Build Coastguard Worker *tag_ptr = static_cast<jlong>(reinterpret_cast<intptr_t>(klass));
117*795d594fSAndroid Build Coastguard Worker return JVMTI_ITERATION_CONTINUE;
118*795d594fSAndroid Build Coastguard Worker },
119*795d594fSAndroid Build Coastguard Worker klass));
120*795d594fSAndroid Build Coastguard Worker jobject* obj_list;
121*795d594fSAndroid Build Coastguard Worker jint obj_len;
122*795d594fSAndroid Build Coastguard Worker jlong tag = static_cast<jlong>(reinterpret_cast<intptr_t>(klass));
123*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetObjectsWithTags(1, &tag, &obj_len, &obj_list, nullptr));
124*795d594fSAndroid Build Coastguard Worker
125*795d594fSAndroid Build Coastguard Worker std::unordered_map<std::string, size_t> class_sizes;
126*795d594fSAndroid Build Coastguard Worker std::unordered_map<std::string, size_t> class_counts;
127*795d594fSAndroid Build Coastguard Worker size_t total_size = 0;
128*795d594fSAndroid Build Coastguard Worker // Mark all the referenced objects with a single tag value, this way we can dedup them.
129*795d594fSAndroid Build Coastguard Worker jlong referenced_object_tag = static_cast<jlong>(reinterpret_cast<intptr_t>(klass) + 1);
130*795d594fSAndroid Build Coastguard Worker std::string null_class_name("<null>");
131*795d594fSAndroid Build Coastguard Worker class_counts[null_class_name] = 0;
132*795d594fSAndroid Build Coastguard Worker class_sizes[null_class_name] = 0;
133*795d594fSAndroid Build Coastguard Worker for (jint i = 0; i < obj_len; i++) {
134*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jobject> cur_thiz(env, obj_list[i]);
135*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jobject> obj(env, env->GetObjectField(cur_thiz.get(), field));
136*795d594fSAndroid Build Coastguard Worker std::string class_name(null_class_name);
137*795d594fSAndroid Build Coastguard Worker if (obj == nullptr) {
138*795d594fSAndroid Build Coastguard Worker class_counts[null_class_name]++;
139*795d594fSAndroid Build Coastguard Worker } else {
140*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetTag(obj.get(), referenced_object_tag));
141*795d594fSAndroid Build Coastguard Worker jlong size = 0;
142*795d594fSAndroid Build Coastguard Worker if (obj.get() != nullptr) {
143*795d594fSAndroid Build Coastguard Worker char* class_name_tmp;
144*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> obj_klass(env, env->GetObjectClass(obj.get()));
145*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetClassSignature(obj_klass.get(), &class_name_tmp, nullptr));
146*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetObjectSize(obj.get(), &size));
147*795d594fSAndroid Build Coastguard Worker class_name = class_name_tmp;
148*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(class_name_tmp)));
149*795d594fSAndroid Build Coastguard Worker }
150*795d594fSAndroid Build Coastguard Worker if (class_sizes.find(class_name) == class_counts.end()) {
151*795d594fSAndroid Build Coastguard Worker class_sizes[class_name] = 0;
152*795d594fSAndroid Build Coastguard Worker class_counts[class_name] = 0;
153*795d594fSAndroid Build Coastguard Worker }
154*795d594fSAndroid Build Coastguard Worker class_counts[class_name]++;
155*795d594fSAndroid Build Coastguard Worker }
156*795d594fSAndroid Build Coastguard Worker }
157*795d594fSAndroid Build Coastguard Worker jobject* ref_list;
158*795d594fSAndroid Build Coastguard Worker jint ref_len;
159*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetObjectsWithTags(1, &referenced_object_tag, &ref_len, &ref_list, nullptr));
160*795d594fSAndroid Build Coastguard Worker for (jint i = 0; i < ref_len; i++) {
161*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jobject> obj(env, ref_list[i]);
162*795d594fSAndroid Build Coastguard Worker std::string class_name(null_class_name);
163*795d594fSAndroid Build Coastguard Worker jlong size = 0;
164*795d594fSAndroid Build Coastguard Worker if (obj.get() != nullptr) {
165*795d594fSAndroid Build Coastguard Worker char* class_name_tmp;
166*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> obj_klass(env, env->GetObjectClass(obj.get()));
167*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetClassSignature(obj_klass.get(), &class_name_tmp, nullptr));
168*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetObjectSize(obj.get(), &size));
169*795d594fSAndroid Build Coastguard Worker class_name = class_name_tmp;
170*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(class_name_tmp)));
171*795d594fSAndroid Build Coastguard Worker }
172*795d594fSAndroid Build Coastguard Worker total_size += static_cast<size_t>(size);
173*795d594fSAndroid Build Coastguard Worker class_sizes[class_name] += static_cast<size_t>(size);
174*795d594fSAndroid Build Coastguard Worker }
175*795d594fSAndroid Build Coastguard Worker
176*795d594fSAndroid Build Coastguard Worker char* field_name;
177*795d594fSAndroid Build Coastguard Worker char* field_sig;
178*795d594fSAndroid Build Coastguard Worker char* field_class_name;
179*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetFieldName(klass, field, &field_name, &field_sig, nullptr));
180*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetClassSignature(klass, &field_class_name, nullptr));
181*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "\t" << field_class_name << "." << field_name << ":" << field_sig
182*795d594fSAndroid Build Coastguard Worker << "\t" << "<ALL_TYPES>"
183*795d594fSAndroid Build Coastguard Worker << "\t" << obj_len
184*795d594fSAndroid Build Coastguard Worker << "\t" << total_size;
185*795d594fSAndroid Build Coastguard Worker for (const std::pair<std::string, size_t> sz : class_sizes) {
186*795d594fSAndroid Build Coastguard Worker size_t count = class_counts[sz.first];
187*795d594fSAndroid Build Coastguard Worker LOG(INFO) << "\t" << field_class_name << "." << field_name << ":" << field_sig
188*795d594fSAndroid Build Coastguard Worker << "\t" << sz.first
189*795d594fSAndroid Build Coastguard Worker << "\t" << count
190*795d594fSAndroid Build Coastguard Worker << "\t" << sz.second;
191*795d594fSAndroid Build Coastguard Worker }
192*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(field_name)));
193*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(field_sig)));
194*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(field_class_name)));
195*795d594fSAndroid Build Coastguard Worker }
196*795d594fSAndroid Build Coastguard Worker }
197*795d594fSAndroid Build Coastguard Worker
VMDeathCb(jvmtiEnv * jvmti,JNIEnv * env)198*795d594fSAndroid Build Coastguard Worker static void VMDeathCb(jvmtiEnv* jvmti, [[maybe_unused]] JNIEnv* env) {
199*795d594fSAndroid Build Coastguard Worker DataDumpRequestCb(jvmti);
200*795d594fSAndroid Build Coastguard Worker RequestList* list = nullptr;
201*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&list)));
202*795d594fSAndroid Build Coastguard Worker delete list;
203*795d594fSAndroid Build Coastguard Worker }
204*795d594fSAndroid Build Coastguard Worker
CreateFieldList(jvmtiEnv * jvmti,JNIEnv * env,const std::string & args)205*795d594fSAndroid Build Coastguard Worker static void CreateFieldList(jvmtiEnv* jvmti, JNIEnv* env, const std::string& args) {
206*795d594fSAndroid Build Coastguard Worker RequestList* list = nullptr;
207*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Allocate(sizeof(*list), reinterpret_cast<unsigned char**>(&list)));
208*795d594fSAndroid Build Coastguard Worker new (list) RequestList{
209*795d594fSAndroid Build Coastguard Worker .fields_ = GetRequestedFields(env, args),
210*795d594fSAndroid Build Coastguard Worker };
211*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(list));
212*795d594fSAndroid Build Coastguard Worker }
213*795d594fSAndroid Build Coastguard Worker
VMInitCb(jvmtiEnv * jvmti,JNIEnv * env,jobject thr)214*795d594fSAndroid Build Coastguard Worker static void VMInitCb(jvmtiEnv* jvmti, JNIEnv* env, [[maybe_unused]] jobject thr) {
215*795d594fSAndroid Build Coastguard Worker char* args = nullptr;
216*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&args)));
217*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(nullptr));
218*795d594fSAndroid Build Coastguard Worker CreateFieldList(jvmti, env, args);
219*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_DEATH, nullptr));
220*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(
221*795d594fSAndroid Build Coastguard Worker jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_DATA_DUMP_REQUEST, nullptr));
222*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Deallocate(reinterpret_cast<unsigned char*>(args)));
223*795d594fSAndroid Build Coastguard Worker }
224*795d594fSAndroid Build Coastguard Worker
AgentStart(JavaVM * vm,char * options,bool is_onload)225*795d594fSAndroid Build Coastguard Worker static jint AgentStart(JavaVM* vm, char* options, bool is_onload) {
226*795d594fSAndroid Build Coastguard Worker android::base::InitLogging(/* argv= */ nullptr);
227*795d594fSAndroid Build Coastguard Worker java_vm = vm;
228*795d594fSAndroid Build Coastguard Worker jvmtiEnv* jvmti = nullptr;
229*795d594fSAndroid Build Coastguard Worker if (SetupJvmtiEnv(vm, &jvmti) != JNI_OK) {
230*795d594fSAndroid Build Coastguard Worker LOG(ERROR) << "Could not get JVMTI env or ArtTiEnv!";
231*795d594fSAndroid Build Coastguard Worker return JNI_ERR;
232*795d594fSAndroid Build Coastguard Worker }
233*795d594fSAndroid Build Coastguard Worker jvmtiCapabilities caps{
234*795d594fSAndroid Build Coastguard Worker .can_tag_objects = 1,
235*795d594fSAndroid Build Coastguard Worker };
236*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->AddCapabilities(&caps));
237*795d594fSAndroid Build Coastguard Worker jvmtiEventCallbacks cb{
238*795d594fSAndroid Build Coastguard Worker .VMInit = VMInitCb,
239*795d594fSAndroid Build Coastguard Worker .VMDeath = VMDeathCb,
240*795d594fSAndroid Build Coastguard Worker .DataDumpRequest = DataDumpRequestCb,
241*795d594fSAndroid Build Coastguard Worker };
242*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEventCallbacks(&cb, sizeof(cb)));
243*795d594fSAndroid Build Coastguard Worker if (is_onload) {
244*795d594fSAndroid Build Coastguard Worker unsigned char* ptr = nullptr;
245*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->Allocate(strlen(options) + 1, &ptr));
246*795d594fSAndroid Build Coastguard Worker strcpy(reinterpret_cast<char*>(ptr), options);
247*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEnvironmentLocalStorage(ptr));
248*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_INIT, nullptr));
249*795d594fSAndroid Build Coastguard Worker } else {
250*795d594fSAndroid Build Coastguard Worker JNIEnv* env = nullptr;
251*795d594fSAndroid Build Coastguard Worker CHECK_EQ(vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6), JNI_OK);
252*795d594fSAndroid Build Coastguard Worker CreateFieldList(jvmti, env, options);
253*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_DEATH, nullptr));
254*795d594fSAndroid Build Coastguard Worker CHECK_JVMTI(
255*795d594fSAndroid Build Coastguard Worker jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_DATA_DUMP_REQUEST, nullptr));
256*795d594fSAndroid Build Coastguard Worker }
257*795d594fSAndroid Build Coastguard Worker return JNI_OK;
258*795d594fSAndroid Build Coastguard Worker }
259*795d594fSAndroid Build Coastguard Worker
260*795d594fSAndroid Build Coastguard Worker // Late attachment (e.g. 'am attach-agent').
Agent_OnAttach(JavaVM * vm,char * options,void * reserved)261*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT jint JNICALL Agent_OnAttach(JavaVM* vm,
262*795d594fSAndroid Build Coastguard Worker char* options,
263*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] void* reserved) {
264*795d594fSAndroid Build Coastguard Worker return AgentStart(vm, options, /*is_onload=*/false);
265*795d594fSAndroid Build Coastguard Worker }
266*795d594fSAndroid Build Coastguard Worker
267*795d594fSAndroid Build Coastguard Worker // Early attachment
Agent_OnLoad(JavaVM * jvm,char * options,void * reserved)268*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM* jvm,
269*795d594fSAndroid Build Coastguard Worker char* options,
270*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] void* reserved) {
271*795d594fSAndroid Build Coastguard Worker return AgentStart(jvm, options, /*is_onload=*/true);
272*795d594fSAndroid Build Coastguard Worker }
273*795d594fSAndroid Build Coastguard Worker
274*795d594fSAndroid Build Coastguard Worker } // namespace fieldnull
275