/* * Copyright (C) 2015 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 "jni.h" #include #include "base/mutex.h" #include "dex/dex_file-inl.h" #include "jni_internal.h" #include "mirror/class-inl.h" #include "nth_caller_visitor.h" #include "oat_file.h" #include "runtime.h" #include "scoped_thread_state_change-inl.h" #include "stack.h" #include "thread-current-inl.h" namespace art { static bool asserts_enabled = true; // public static native void disableStackFrameAsserts(); // Note: to globally disable asserts in unsupported configurations. extern "C" JNIEXPORT void JNICALL Java_Main_disableStackFrameAsserts(JNIEnv* env ATTRIBUTE_UNUSED, jclass cls ATTRIBUTE_UNUSED) { asserts_enabled = false; } static jboolean IsInterpreted(JNIEnv* env, jclass, size_t level) { ScopedObjectAccess soa(env); NthCallerVisitor caller(soa.Self(), level, false); caller.WalkStack(); CHECK(caller.caller != nullptr); return caller.GetCurrentShadowFrame() != nullptr ? JNI_TRUE : JNI_FALSE; } // public static native boolean isInterpreted(); extern "C" JNIEXPORT jboolean JNICALL Java_Main_isInterpreted(JNIEnv* env, jclass klass) { return IsInterpreted(env, klass, 1); } // public static native boolean isInterpreted(int depth); extern "C" JNIEXPORT jboolean JNICALL Java_Main_isInterpretedAt(JNIEnv* env, jclass klass, jint depth) { return IsInterpreted(env, klass, depth); } // public static native boolean isInterpretedFunction(String smali); // TODO Remove 'allow_runtime_frames' option once we have deoptimization through runtime frames. struct MethodIsInterpretedVisitor : public StackVisitor { public: MethodIsInterpretedVisitor(Thread* thread, ArtMethod* goal, bool require_deoptable) : StackVisitor(thread, nullptr, StackVisitor::StackWalkKind::kIncludeInlinedFrames), goal_(goal), method_is_interpreted_(true), method_found_(false), prev_was_runtime_(true), require_deoptable_(require_deoptable) {} virtual bool VisitFrame() OVERRIDE REQUIRES_SHARED(Locks::mutator_lock_) { if (goal_ == GetMethod()) { method_is_interpreted_ = (require_deoptable_ && prev_was_runtime_) || IsShadowFrame(); method_found_ = true; return false; } prev_was_runtime_ = GetMethod()->IsRuntimeMethod(); return true; } bool IsInterpreted() { return method_is_interpreted_; } bool IsFound() { return method_found_; } private: const ArtMethod* goal_; bool method_is_interpreted_; bool method_found_; bool prev_was_runtime_; bool require_deoptable_; }; // TODO Remove 'require_deoptimizable' option once we have deoptimization through runtime frames. extern "C" JNIEXPORT jboolean JNICALL Java_Main_isInterpretedFunction( JNIEnv* env, jclass klass ATTRIBUTE_UNUSED, jobject method, jboolean require_deoptimizable) { // Return false if this seems to not be an ART runtime. if (Runtime::Current() == nullptr) { return JNI_FALSE; } if (method == nullptr) { env->ThrowNew(env->FindClass("java/lang/NullPointerException"), "method is null!"); return JNI_FALSE; } jmethodID id = env->FromReflectedMethod(method); if (id == nullptr) { env->ThrowNew(env->FindClass("java/lang/Error"), "Unable to interpret method argument!"); return JNI_FALSE; } bool result; bool found; { ScopedObjectAccess soa(env); ArtMethod* goal = jni::DecodeArtMethod(id); MethodIsInterpretedVisitor v(soa.Self(), goal, require_deoptimizable); v.WalkStack(); bool enters_interpreter = Runtime::Current()->GetClassLinker()->IsQuickToInterpreterBridge( goal->GetEntryPointFromQuickCompiledCode()); result = (v.IsInterpreted() || enters_interpreter); found = v.IsFound(); } if (!found) { env->ThrowNew(env->FindClass("java/lang/Error"), "Unable to find given method in stack!"); return JNI_FALSE; } return result; } // public static native void assertIsInterpreted(); extern "C" JNIEXPORT void JNICALL Java_Main_assertIsInterpreted(JNIEnv* env, jclass klass) { if (asserts_enabled) { CHECK(Java_Main_isInterpreted(env, klass)); } } static jboolean IsManaged(JNIEnv* env, jclass, size_t level) { ScopedObjectAccess soa(env); NthCallerVisitor caller(soa.Self(), level, false); caller.WalkStack(); CHECK(caller.caller != nullptr); return caller.GetCurrentShadowFrame() != nullptr ? JNI_FALSE : JNI_TRUE; } // public static native boolean isManaged(); extern "C" JNIEXPORT jboolean JNICALL Java_Main_isManaged(JNIEnv* env, jclass cls) { return IsManaged(env, cls, 1); } // public static native void assertIsManaged(); extern "C" JNIEXPORT void JNICALL Java_Main_assertIsManaged(JNIEnv* env, jclass cls) { if (asserts_enabled) { CHECK(Java_Main_isManaged(env, cls)); } } // public static native boolean isCallerInterpreted(); extern "C" JNIEXPORT jboolean JNICALL Java_Main_isCallerInterpreted(JNIEnv* env, jclass klass) { return IsInterpreted(env, klass, 2); } // public static native void assertCallerIsInterpreted(); extern "C" JNIEXPORT void JNICALL Java_Main_assertCallerIsInterpreted(JNIEnv* env, jclass klass) { if (asserts_enabled) { CHECK(Java_Main_isCallerInterpreted(env, klass)); } } // public static native boolean isCallerManaged(); extern "C" JNIEXPORT jboolean JNICALL Java_Main_isCallerManaged(JNIEnv* env, jclass cls) { return IsManaged(env, cls, 2); } // public static native void assertCallerIsManaged(); extern "C" JNIEXPORT void JNICALL Java_Main_assertCallerIsManaged(JNIEnv* env, jclass cls) { if (asserts_enabled) { CHECK(Java_Main_isCallerManaged(env, cls)); } } } // namespace art