/* * Copyright (C) 2008 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 "dalvik_system_VMStack.h" #include #include "nativehelper/jni_macros.h" #include "art_method-inl.h" #include "gc/task_processor.h" #include "jni/jni_internal.h" #include "mirror/class-inl.h" #include "mirror/class_loader.h" #include "mirror/object-inl.h" #include "native_util.h" #include "nth_caller_visitor.h" #include "scoped_fast_native_object_access-inl.h" #include "scoped_thread_state_change-inl.h" #include "thread_list.h" namespace art HIDDEN { template > static ResultT GetThreadStack(const ScopedFastNativeObjectAccess& soa, jobject peer, T fn) REQUIRES_SHARED(Locks::mutator_lock_) { ResultT trace = nullptr; ObjPtr decoded_peer = soa.Decode(peer); if (decoded_peer == soa.Self()->GetPeer()) { trace = fn(soa.Self(), soa); return trace; } // Suspend thread to build stack trace. ScopedThreadSuspension sts(soa.Self(), ThreadState::kNative); Runtime* runtime = Runtime::Current(); ThreadList* thread_list = runtime->GetThreadList(); Thread* thread = thread_list->SuspendThreadByPeer(peer, SuspendReason::kInternal); if (thread != nullptr) { // If we were asked for the HeapTaskDaemon's stack trace, we went ahead and suspended it. // It's usually already in a suspended state anyway. But we should immediately give up and // resume it, since we must be able to allocate while generating the stack trace. if (!runtime->GetHeap()->GetTaskProcessor()->IsRunningThread(thread, /*wait=*/true)) { { // Must be runnable to create returned array. ScopedObjectAccess soa2(soa.Self()); trace = fn(thread, soa); } // Else either thread is the HeapTaskDaemon, or we couldn't identify the thread yet. The // HeapTaskDaemon can appear in enumerations before it is registered with the task // processor, and we don't wait indefinitely, so there is a tiny chance of the latter. } // Restart suspended thread. bool resumed = thread_list->Resume(thread, SuspendReason::kInternal); DCHECK(resumed); } return trace; } static jint VMStack_fillStackTraceElements(JNIEnv* env, jclass, jobject javaThread, jobjectArray javaSteArray) { ScopedFastNativeObjectAccess soa(env); auto fn = [](Thread* thread, const ScopedFastNativeObjectAccess& soaa) REQUIRES_SHARED(Locks::mutator_lock_) -> jobject { return thread->CreateInternalStackTrace(soaa); }; jobject trace = GetThreadStack(soa, javaThread, fn); if (trace == nullptr) { return 0; } int32_t depth; Thread::InternalStackTraceToStackTraceElementArray(soa, trace, javaSteArray, &depth); return depth; } // Returns the defining class loader of the caller's caller. static jobject VMStack_getCallingClassLoader(JNIEnv* env, jclass) { ScopedFastNativeObjectAccess soa(env); NthCallerVisitor visitor(soa.Self(), 2); visitor.WalkStack(); if (UNLIKELY(visitor.caller == nullptr)) { // The caller is an attached native thread. return nullptr; } return soa.AddLocalReference(visitor.caller->GetDeclaringClass()->GetClassLoader()); } static jobject VMStack_getClosestUserClassLoader(JNIEnv* env, jclass) { struct ClosestUserClassLoaderVisitor : public StackVisitor { explicit ClosestUserClassLoaderVisitor(Thread* thread) : StackVisitor(thread, nullptr, StackVisitor::StackWalkKind::kIncludeInlinedFrames), class_loader(nullptr) {} bool VisitFrame() override REQUIRES_SHARED(Locks::mutator_lock_) { DCHECK(class_loader == nullptr); ObjPtr c = GetMethod()->GetDeclaringClass(); // c is null for runtime methods. if (c != nullptr) { ObjPtr cl = c->GetClassLoader(); if (cl != nullptr) { class_loader = cl; return false; } } return true; } ObjPtr class_loader; }; ScopedFastNativeObjectAccess soa(env); ClosestUserClassLoaderVisitor visitor(soa.Self()); visitor.WalkStack(); return soa.AddLocalReference(visitor.class_loader); } // Returns the class of the caller's caller's caller. static jclass VMStack_getStackClass2(JNIEnv* env, jclass) { ScopedFastNativeObjectAccess soa(env); NthCallerVisitor visitor(soa.Self(), 3); visitor.WalkStack(); if (UNLIKELY(visitor.caller == nullptr)) { // The caller is an attached native thread. return nullptr; } return soa.AddLocalReference(visitor.caller->GetDeclaringClass()); } static jobjectArray VMStack_getThreadStackTrace(JNIEnv* env, jclass, jobject javaThread) { ScopedFastNativeObjectAccess soa(env); auto fn = [](Thread* thread, const ScopedFastNativeObjectAccess& soaa) REQUIRES_SHARED(Locks::mutator_lock_) -> jobject { return thread->CreateInternalStackTrace(soaa); }; jobject trace = GetThreadStack(soa, javaThread, fn); if (trace == nullptr) { return nullptr; } return Thread::InternalStackTraceToStackTraceElementArray(soa, trace); } static jobjectArray VMStack_getAnnotatedThreadStackTrace(JNIEnv* env, jclass, jobject javaThread) { ScopedFastNativeObjectAccess soa(env); auto fn = [](Thread* thread, const ScopedFastNativeObjectAccess& soaa) REQUIRES_SHARED(Locks::mutator_lock_) -> jobjectArray { return thread->CreateAnnotatedStackTrace(soaa); }; return GetThreadStack(soa, javaThread, fn); } static JNINativeMethod gMethods[] = { FAST_NATIVE_METHOD(VMStack, fillStackTraceElements, "(Ljava/lang/Thread;[Ljava/lang/StackTraceElement;)I"), FAST_NATIVE_METHOD(VMStack, getCallingClassLoader, "()Ljava/lang/ClassLoader;"), FAST_NATIVE_METHOD(VMStack, getClosestUserClassLoader, "()Ljava/lang/ClassLoader;"), FAST_NATIVE_METHOD(VMStack, getStackClass2, "()Ljava/lang/Class;"), FAST_NATIVE_METHOD(VMStack, getThreadStackTrace, "(Ljava/lang/Thread;)[Ljava/lang/StackTraceElement;"), FAST_NATIVE_METHOD(VMStack, getAnnotatedThreadStackTrace, "(Ljava/lang/Thread;)[Ldalvik/system/AnnotatedStackTraceElement;"), }; void register_dalvik_system_VMStack(JNIEnv* env) { REGISTER_NATIVE_METHODS("dalvik/system/VMStack"); } } // namespace art