1 /*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <inttypes.h>
18 #include <pthread.h>
19
20 #include <cstdio>
21 #include <iomanip>
22 #include <iostream>
23 #include <sstream>
24 #include <vector>
25
26 #include "android-base/logging.h"
27 #include "android-base/stringprintf.h"
28
29 #include "jni.h"
30 #include "jvmti.h"
31 #include "scoped_primitive_array.h"
32
33 // Test infrastructure
34 #include "jvmti_helper.h"
35 #include "test_env.h"
36 #include "ti_macros.h"
37 #include "ti_utf.h"
38
39 namespace art {
40 namespace Test906IterateHeap {
41
42 class IterationConfig {
43 public:
IterationConfig()44 IterationConfig() {}
~IterationConfig()45 virtual ~IterationConfig() {}
46
47 virtual jint Handle(jlong class_tag, jlong size, jlong* tag_ptr, jint length) = 0;
48 };
49
HeapIterationCallback(jlong class_tag,jlong size,jlong * tag_ptr,jint length,void * user_data)50 static jint JNICALL HeapIterationCallback(jlong class_tag,
51 jlong size,
52 jlong* tag_ptr,
53 jint length,
54 void* user_data) {
55 IterationConfig* config = reinterpret_cast<IterationConfig*>(user_data);
56 return config->Handle(class_tag, size, tag_ptr, length);
57 }
58
Run(JNIEnv * env,jint heap_filter,jclass klass_filter,IterationConfig * config)59 static bool Run(JNIEnv* env, jint heap_filter, jclass klass_filter, IterationConfig* config) {
60 jvmtiHeapCallbacks callbacks;
61 memset(&callbacks, 0, sizeof(jvmtiHeapCallbacks));
62 callbacks.heap_iteration_callback = HeapIterationCallback;
63
64 jvmtiError ret = jvmti_env->IterateThroughHeap(heap_filter,
65 klass_filter,
66 &callbacks,
67 config);
68 if (JvmtiErrorToException(env, jvmti_env, ret)) {
69 return false;
70 }
71 return true;
72 }
73
Java_art_Test906_iterateThroughHeapCount(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jint heap_filter,jclass klass_filter,jint stop_after)74 extern "C" JNIEXPORT jint JNICALL Java_art_Test906_iterateThroughHeapCount(
75 JNIEnv* env,
76 jclass klass ATTRIBUTE_UNUSED,
77 jint heap_filter,
78 jclass klass_filter,
79 jint stop_after) {
80 class CountIterationConfig : public IterationConfig {
81 public:
82 CountIterationConfig(jint _counter, jint _stop_after)
83 : counter(_counter),
84 stop_after(_stop_after) {
85 }
86
87 jint Handle(jlong class_tag ATTRIBUTE_UNUSED,
88 jlong size ATTRIBUTE_UNUSED,
89 jlong* tag_ptr ATTRIBUTE_UNUSED,
90 jint length ATTRIBUTE_UNUSED) OVERRIDE {
91 counter++;
92 if (counter == stop_after) {
93 return JVMTI_VISIT_ABORT;
94 }
95 return 0;
96 }
97
98 jint counter;
99 const jint stop_after;
100 };
101
102 CountIterationConfig config(0, stop_after);
103 Run(env, heap_filter, klass_filter, &config);
104
105 if (config.counter > config.stop_after) {
106 printf("Error: more objects visited than signaled.");
107 }
108
109 return config.counter;
110 }
111
Java_art_Test906_iterateThroughHeapData(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jint heap_filter,jclass klass_filter,jlongArray class_tags,jlongArray sizes,jlongArray tags,jintArray lengths)112 extern "C" JNIEXPORT jint JNICALL Java_art_Test906_iterateThroughHeapData(
113 JNIEnv* env,
114 jclass klass ATTRIBUTE_UNUSED,
115 jint heap_filter,
116 jclass klass_filter,
117 jlongArray class_tags,
118 jlongArray sizes,
119 jlongArray tags,
120 jintArray lengths) {
121 class DataIterationConfig : public IterationConfig {
122 public:
123 jint Handle(jlong class_tag, jlong size, jlong* tag_ptr, jint length) OVERRIDE {
124 class_tags_.push_back(class_tag);
125 sizes_.push_back(size);
126 tags_.push_back(*tag_ptr);
127 lengths_.push_back(length);
128
129 return 0; // Continue.
130 }
131
132 std::vector<jlong> class_tags_;
133 std::vector<jlong> sizes_;
134 std::vector<jlong> tags_;
135 std::vector<jint> lengths_;
136 };
137
138 DataIterationConfig config;
139 if (!Run(env, heap_filter, klass_filter, &config)) {
140 return -1;
141 }
142
143 ScopedLongArrayRW s_class_tags(env, class_tags);
144 ScopedLongArrayRW s_sizes(env, sizes);
145 ScopedLongArrayRW s_tags(env, tags);
146 ScopedIntArrayRW s_lengths(env, lengths);
147
148 for (size_t i = 0; i != config.class_tags_.size(); ++i) {
149 s_class_tags[i] = config.class_tags_[i];
150 s_sizes[i] = config.sizes_[i];
151 s_tags[i] = config.tags_[i];
152 s_lengths[i] = config.lengths_[i];
153 }
154
155 return static_cast<jint>(config.class_tags_.size());
156 }
157
Java_art_Test906_iterateThroughHeapAdd(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jint heap_filter,jclass klass_filter)158 extern "C" JNIEXPORT void JNICALL Java_art_Test906_iterateThroughHeapAdd(
159 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jint heap_filter, jclass klass_filter) {
160 class AddIterationConfig : public IterationConfig {
161 public:
162 AddIterationConfig() {}
163
164 jint Handle(jlong class_tag ATTRIBUTE_UNUSED,
165 jlong size ATTRIBUTE_UNUSED,
166 jlong* tag_ptr,
167 jint length ATTRIBUTE_UNUSED) OVERRIDE {
168 jlong current_tag = *tag_ptr;
169 if (current_tag != 0) {
170 *tag_ptr = current_tag + 10;
171 }
172 return 0;
173 }
174 };
175
176 AddIterationConfig config;
177 Run(env, heap_filter, klass_filter, &config);
178 }
179
Java_art_Test906_iterateThroughHeapString(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jlong tag)180 extern "C" JNIEXPORT jstring JNICALL Java_art_Test906_iterateThroughHeapString(
181 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jlong tag) {
182 struct FindStringCallbacks {
183 explicit FindStringCallbacks(jlong t) : tag_to_find(t) {}
184
185 static jint JNICALL HeapIterationCallback(jlong class_tag ATTRIBUTE_UNUSED,
186 jlong size ATTRIBUTE_UNUSED,
187 jlong* tag_ptr ATTRIBUTE_UNUSED,
188 jint length ATTRIBUTE_UNUSED,
189 void* user_data ATTRIBUTE_UNUSED) {
190 return 0;
191 }
192
193 static jint JNICALL StringValueCallback(jlong class_tag,
194 jlong size,
195 jlong* tag_ptr,
196 const jchar* value,
197 jint value_length,
198 void* user_data) {
199 FindStringCallbacks* p = reinterpret_cast<FindStringCallbacks*>(user_data);
200 if (*tag_ptr == p->tag_to_find) {
201 size_t utf_byte_count = ti::CountUtf8Bytes(value, value_length);
202 std::unique_ptr<char[]> mod_utf(new char[utf_byte_count + 1]);
203 memset(mod_utf.get(), 0, utf_byte_count + 1);
204 ti::ConvertUtf16ToModifiedUtf8(mod_utf.get(), utf_byte_count, value, value_length);
205 if (!p->data.empty()) {
206 p->data += "\n";
207 }
208 p->data += android::base::StringPrintf("%" PRId64 "@%" PRId64 " (%" PRId64 ", '%s')",
209 *tag_ptr,
210 class_tag,
211 size,
212 mod_utf.get());
213 // Update the tag to test whether that works.
214 *tag_ptr = *tag_ptr + 1;
215 }
216 return 0;
217 }
218
219 std::string data;
220 const jlong tag_to_find;
221 };
222
223 jvmtiHeapCallbacks callbacks;
224 memset(&callbacks, 0, sizeof(jvmtiHeapCallbacks));
225 callbacks.heap_iteration_callback = FindStringCallbacks::HeapIterationCallback;
226 callbacks.string_primitive_value_callback = FindStringCallbacks::StringValueCallback;
227
228 FindStringCallbacks fsc(tag);
229 jvmtiError ret = jvmti_env->IterateThroughHeap(0, nullptr, &callbacks, &fsc);
230 if (JvmtiErrorToException(env, jvmti_env, ret)) {
231 return nullptr;
232 }
233 return env->NewStringUTF(fsc.data.c_str());
234 }
235
Java_art_Test906_iterateThroughHeapPrimitiveArray(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jlong tag)236 extern "C" JNIEXPORT jstring JNICALL Java_art_Test906_iterateThroughHeapPrimitiveArray(
237 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jlong tag) {
238 struct FindArrayCallbacks {
239 explicit FindArrayCallbacks(jlong t) : tag_to_find(t) {}
240
241 static jint JNICALL HeapIterationCallback(jlong class_tag ATTRIBUTE_UNUSED,
242 jlong size ATTRIBUTE_UNUSED,
243 jlong* tag_ptr ATTRIBUTE_UNUSED,
244 jint length ATTRIBUTE_UNUSED,
245 void* user_data ATTRIBUTE_UNUSED) {
246 return 0;
247 }
248
249 static jint JNICALL ArrayValueCallback(jlong class_tag,
250 jlong size,
251 jlong* tag_ptr,
252 jint element_count,
253 jvmtiPrimitiveType element_type,
254 const void* elements,
255 void* user_data) {
256 FindArrayCallbacks* p = reinterpret_cast<FindArrayCallbacks*>(user_data);
257 if (*tag_ptr == p->tag_to_find) {
258 std::ostringstream oss;
259 oss << *tag_ptr
260 << '@'
261 << class_tag
262 << " ("
263 << size
264 << ", "
265 << element_count
266 << "x"
267 << static_cast<char>(element_type)
268 << " '";
269 size_t element_size;
270 switch (element_type) {
271 case JVMTI_PRIMITIVE_TYPE_BOOLEAN:
272 case JVMTI_PRIMITIVE_TYPE_BYTE:
273 element_size = 1;
274 break;
275 case JVMTI_PRIMITIVE_TYPE_CHAR:
276 case JVMTI_PRIMITIVE_TYPE_SHORT:
277 element_size = 2;
278 break;
279 case JVMTI_PRIMITIVE_TYPE_INT:
280 case JVMTI_PRIMITIVE_TYPE_FLOAT:
281 element_size = 4;
282 break;
283 case JVMTI_PRIMITIVE_TYPE_LONG:
284 case JVMTI_PRIMITIVE_TYPE_DOUBLE:
285 element_size = 8;
286 break;
287 default:
288 LOG(FATAL) << "Unknown type " << static_cast<size_t>(element_type);
289 UNREACHABLE();
290 }
291 const uint8_t* data = reinterpret_cast<const uint8_t*>(elements);
292 for (size_t i = 0; i != element_size * element_count; ++i) {
293 oss << android::base::StringPrintf("%02x", data[i]);
294 }
295 oss << "')";
296
297 if (!p->data.empty()) {
298 p->data += "\n";
299 }
300 p->data += oss.str();
301 // Update the tag to test whether that works.
302 *tag_ptr = *tag_ptr + 1;
303 }
304 return 0;
305 }
306
307 std::string data;
308 const jlong tag_to_find;
309 };
310
311 jvmtiHeapCallbacks callbacks;
312 memset(&callbacks, 0, sizeof(jvmtiHeapCallbacks));
313 callbacks.heap_iteration_callback = FindArrayCallbacks::HeapIterationCallback;
314 callbacks.array_primitive_value_callback = FindArrayCallbacks::ArrayValueCallback;
315
316 FindArrayCallbacks fac(tag);
317 jvmtiError ret = jvmti_env->IterateThroughHeap(0, nullptr, &callbacks, &fac);
318 if (JvmtiErrorToException(env, jvmti_env, ret)) {
319 return nullptr;
320 }
321 return env->NewStringUTF(fac.data.c_str());
322 }
323
GetPrimitiveTypeName(jvmtiPrimitiveType type)324 static constexpr const char* GetPrimitiveTypeName(jvmtiPrimitiveType type) {
325 switch (type) {
326 case JVMTI_PRIMITIVE_TYPE_BOOLEAN:
327 return "boolean";
328 case JVMTI_PRIMITIVE_TYPE_BYTE:
329 return "byte";
330 case JVMTI_PRIMITIVE_TYPE_CHAR:
331 return "char";
332 case JVMTI_PRIMITIVE_TYPE_SHORT:
333 return "short";
334 case JVMTI_PRIMITIVE_TYPE_INT:
335 return "int";
336 case JVMTI_PRIMITIVE_TYPE_FLOAT:
337 return "float";
338 case JVMTI_PRIMITIVE_TYPE_LONG:
339 return "long";
340 case JVMTI_PRIMITIVE_TYPE_DOUBLE:
341 return "double";
342 }
343 LOG(FATAL) << "Unknown type " << static_cast<size_t>(type);
344 UNREACHABLE();
345 }
346
Java_art_Test906_iterateThroughHeapPrimitiveFields(JNIEnv * env,jclass klass ATTRIBUTE_UNUSED,jlong tag)347 extern "C" JNIEXPORT jstring JNICALL Java_art_Test906_iterateThroughHeapPrimitiveFields(
348 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jlong tag) {
349 struct FindFieldCallbacks {
350 explicit FindFieldCallbacks(jlong t) : tag_to_find(t) {}
351
352 static jint JNICALL HeapIterationCallback(jlong class_tag ATTRIBUTE_UNUSED,
353 jlong size ATTRIBUTE_UNUSED,
354 jlong* tag_ptr ATTRIBUTE_UNUSED,
355 jint length ATTRIBUTE_UNUSED,
356 void* user_data ATTRIBUTE_UNUSED) {
357 return 0;
358 }
359
360 static jint JNICALL PrimitiveFieldValueCallback(jvmtiHeapReferenceKind kind,
361 const jvmtiHeapReferenceInfo* info,
362 jlong class_tag,
363 jlong* tag_ptr,
364 jvalue value,
365 jvmtiPrimitiveType value_type,
366 void* user_data) {
367 FindFieldCallbacks* p = reinterpret_cast<FindFieldCallbacks*>(user_data);
368 if (*tag_ptr >= p->tag_to_find) {
369 std::ostringstream oss;
370 oss << *tag_ptr
371 << '@'
372 << class_tag
373 << " ("
374 << (kind == JVMTI_HEAP_REFERENCE_FIELD ? "instance, " : "static, ")
375 << GetPrimitiveTypeName(value_type)
376 << ", index="
377 << info->field.index
378 << ") ";
379 // Be lazy, always print eight bytes.
380 static_assert(sizeof(jvalue) == sizeof(uint64_t), "Unexpected jvalue size");
381 uint64_t val;
382 memcpy(&val, &value, sizeof(uint64_t)); // To avoid undefined behavior.
383 oss << android::base::StringPrintf("%016" PRIx64, val);
384
385 if (!p->data.empty()) {
386 p->data += "\n";
387 }
388 p->data += oss.str();
389 *tag_ptr = *tag_ptr + 1;
390 }
391 return 0;
392 }
393
394 std::string data;
395 const jlong tag_to_find;
396 };
397
398 jvmtiHeapCallbacks callbacks;
399 memset(&callbacks, 0, sizeof(jvmtiHeapCallbacks));
400 callbacks.heap_iteration_callback = FindFieldCallbacks::HeapIterationCallback;
401 callbacks.primitive_field_callback = FindFieldCallbacks::PrimitiveFieldValueCallback;
402
403 FindFieldCallbacks ffc(tag);
404 jvmtiError ret = jvmti_env->IterateThroughHeap(0, nullptr, &callbacks, &ffc);
405 if (JvmtiErrorToException(env, jvmti_env, ret)) {
406 return nullptr;
407 }
408 return env->NewStringUTF(ffc.data.c_str());
409 }
410
Java_art_Test906_checkInitialized(JNIEnv * env,jclass,jclass c)411 extern "C" JNIEXPORT jboolean JNICALL Java_art_Test906_checkInitialized(
412 JNIEnv* env, jclass, jclass c) {
413 jint status;
414 jvmtiError error = jvmti_env->GetClassStatus(c, &status);
415 if (JvmtiErrorToException(env, jvmti_env, error)) {
416 return false;
417 }
418 return (status & JVMTI_CLASS_STATUS_INITIALIZED) != 0;
419 }
420
421 } // namespace Test906IterateHeap
422 } // namespace art
423