/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <mutex> #include "jni.h" #include "jvmti.h" #include "android-base/logging.h" #include "android-base/stringprintf.h" #include "jvmti_helper.h" #include "scoped_local_ref.h" #include "scoped_utf_chars.h" #include "test_env.h" namespace art { static std::string GetClassName(JNIEnv* jni_env, jclass cls) { ScopedLocalRef<jclass> class_class(jni_env, jni_env->GetObjectClass(cls)); jmethodID mid = jni_env->GetMethodID(class_class.get(), "getName", "()Ljava/lang/String;"); ScopedLocalRef<jstring> str( jni_env, reinterpret_cast<jstring>(jni_env->CallObjectMethod(cls, mid))); ScopedUtfChars utf_chars(jni_env, str.get()); return utf_chars.c_str(); } static std::mutex gLock; static std::string gCollection; static void JNICALL ObjectAllocated(jvmtiEnv* ti_env ATTRIBUTE_UNUSED, JNIEnv* jni_env, jthread thread ATTRIBUTE_UNUSED, jobject object, jclass object_klass, jlong size) { std::string object_klass_descriptor = GetClassName(jni_env, object_klass); ScopedLocalRef<jclass> object_klass2(jni_env, jni_env->GetObjectClass(object)); std::string object_klass_descriptor2 = GetClassName(jni_env, object_klass2.get()); std::string result = android::base::StringPrintf("ObjectAllocated type %s/%s size %zu", object_klass_descriptor.c_str(), object_klass_descriptor2.c_str(), static_cast<size_t>(size)); std::unique_lock<std::mutex> mu(gLock); gCollection += result + "#"; } extern "C" JNIEXPORT void JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_setupObjectAllocCallback( JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jboolean enable) { jvmtiEventCallbacks callbacks; memset(&callbacks, 0, sizeof(jvmtiEventCallbacks)); callbacks.VMObjectAlloc = enable ? ObjectAllocated : nullptr; jvmtiError ret = jvmti_env->SetEventCallbacks(&callbacks, sizeof(callbacks)); JvmtiErrorToException(env, jvmti_env, ret); } extern "C" JNIEXPORT void JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_enableAllocationTracking( JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jthread thread, jboolean enable) { jvmtiError ret = jvmti_env->SetEventNotificationMode( enable ? JVMTI_ENABLE : JVMTI_DISABLE, JVMTI_EVENT_VM_OBJECT_ALLOC, thread); JvmtiErrorToException(env, jvmti_env, ret); } extern "C" JNIEXPORT jstring JNICALL Java_android_jvmti_cts_JvmtiTrackingTest_getAndResetAllocationTrackingString( JNIEnv* env, jclass klass ATTRIBUTE_UNUSED) { // We will have a string allocation. So only do the C++ string retrieval under lock. std::string result; { std::unique_lock<std::mutex> mu(gLock); result.swap(gCollection); } if (result.empty()) { return nullptr; } return env->NewStringUTF(result.c_str()); } } // namespace art