blob: a46f4919158bf5aab73835d19de01077ce79a7ca [file] [log] [blame]
Andreas Gampee205aab2017-04-03 17:27:12 -07001/*
2 * Copyright (C) 2017 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 <mutex>
18
19#include "jni.h"
20#include "jvmti.h"
21
22#include "android-base/logging.h"
23#include "android-base/stringprintf.h"
24#include "common.h"
25#include "jvmti_helper.h"
26#include "scoped_local_ref.h"
27#include "scoped_utf_chars.h"
28
29namespace cts {
30namespace jvmti {
31namespace allocation_tracking {
32
33static std::string GetClassName(JNIEnv* jni_env, jclass cls) {
34 ScopedLocalRef<jclass> class_class(jni_env, jni_env->GetObjectClass(cls));
35 jmethodID mid = jni_env->GetMethodID(class_class.get(), "getName", "()Ljava/lang/String;");
36 ScopedLocalRef<jstring> str(
37 jni_env, reinterpret_cast<jstring>(jni_env->CallObjectMethod(cls, mid)));
38 ScopedUtfChars utf_chars(jni_env, str.get());
39 return utf_chars.c_str();
40}
41
42static std::mutex gLock;
43static std::string gCollection;
44
45static void JNICALL ObjectAllocated(jvmtiEnv* ti_env ATTRIBUTE_UNUSED,
46 JNIEnv* jni_env,
47 jthread thread ATTRIBUTE_UNUSED,
48 jobject object,
49 jclass object_klass,
50 jlong size) {
51 std::string object_klass_descriptor = GetClassName(jni_env, object_klass);
52 ScopedLocalRef<jclass> object_klass2(jni_env, jni_env->GetObjectClass(object));
53 std::string object_klass_descriptor2 = GetClassName(jni_env, object_klass2.get());
54 std::string result = android::base::StringPrintf("ObjectAllocated type %s/%s size %zu",
55 object_klass_descriptor.c_str(),
56 object_klass_descriptor2.c_str(),
57 static_cast<size_t>(size));
58 std::unique_lock<std::mutex> mu(gLock);
59 gCollection += result + "#";
60}
61
62extern "C" JNIEXPORT void JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_setupObjectAllocCallback(
63 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jboolean enable) {
64 jvmtiEventCallbacks callbacks;
65 memset(&callbacks, 0, sizeof(jvmtiEventCallbacks));
66 callbacks.VMObjectAlloc = enable ? ObjectAllocated : nullptr;
67
68 jvmtiError ret = GetJvmtiEnv()->SetEventCallbacks(&callbacks, sizeof(callbacks));
69 JvmtiErrorToException(env, GetJvmtiEnv(), ret);
70}
71
72extern "C" JNIEXPORT void JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_enableAllocationTracking(
73 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jthread thread, jboolean enable) {
74 jvmtiError ret = GetJvmtiEnv()->SetEventNotificationMode(
75 enable ? JVMTI_ENABLE : JVMTI_DISABLE,
76 JVMTI_EVENT_VM_OBJECT_ALLOC,
77 thread);
78 JvmtiErrorToException(env, GetJvmtiEnv(), ret);
79}
80
81extern "C" JNIEXPORT
82jstring JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_getAndResetAllocationTrackingString(
83 JNIEnv* env, jclass klass ATTRIBUTE_UNUSED) {
84 // We will have a string allocation. So only do the C++ string retrieval under lock.
85 std::string result;
86 {
87 std::unique_lock<std::mutex> mu(gLock);
88 result.swap(gCollection);
89 }
90
91 if (result.empty()) {
92 return nullptr;
93 }
94
95 return env->NewStringUTF(result.c_str());
96}
97
98} // namespace allocation_tracking
99} // namespace jvmti
100} // namespace cts