blob: 8be355005a7b69a28f86bf530c1b074b3fb104e3 [file] [log] [blame]
/*
* 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 <android-base/logging.h>
#include "arch/context.h"
#include "base/mutex.h"
#include "dex/dex_file-inl.h"
#include "jni/jni_internal.h"
#include "mirror/class-inl.h"
#include "nth_caller_visitor.h"
#include "oat/oat_file.h"
#include "oat/oat_quick_method_header.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([[maybe_unused]] JNIEnv* env,
[[maybe_unused]] jclass cls) {
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);
bool is_shadow_frame = (caller.GetCurrentShadowFrame() != nullptr);
bool is_nterp_frame = (caller.GetCurrentQuickFrame() != nullptr) &&
(caller.GetCurrentOatQuickMethodHeader()->IsNterpMethodHeader());
return (is_shadow_frame || is_nterp_frame) ? 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);
static bool IsMethodInterpreted(Thread* self,
const ArtMethod* goal,
const bool require_deoptable,
/* out */ bool* method_is_interpreted)
REQUIRES_SHARED(Locks::mutator_lock_) {
*method_is_interpreted = true;
bool method_found = false;
bool prev_was_runtime = true;
StackVisitor::WalkStack(
[&](const art::StackVisitor* stack_visitor) REQUIRES_SHARED(Locks::mutator_lock_) {
if (goal == stack_visitor->GetMethod()) {
// We don't deoptimize beyond a runtime frame. So if we need the method to be
// deoptimizeable we cannot allow the previous frame to be a runtime frame.
*method_is_interpreted =
(require_deoptable && prev_was_runtime) || stack_visitor->IsShadowFrame();
method_found = true;
return false;
}
prev_was_runtime = stack_visitor->GetMethod()->IsRuntimeMethod();
return true;
},
self,
/* context= */ nullptr,
art::StackVisitor::StackWalkKind::kIncludeInlinedFrames);
return method_found;
}
// TODO Remove 'require_deoptimizable' option once we have deoptimization through runtime frames.
extern "C" JNIEXPORT jboolean JNICALL Java_Main_isInterpretedFunction(
JNIEnv* env, [[maybe_unused]] jclass klass, 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;
}
{
ScopedObjectAccess soa(env);
ArtMethod* goal = jni::DecodeArtMethod(id);
bool is_interpreted;
if (!IsMethodInterpreted(soa.Self(), goal, require_deoptimizable, &is_interpreted)) {
env->ThrowNew(env->FindClass("java/lang/Error"), "Unable to find given method in stack!");
return JNI_FALSE;
}
bool enters_interpreter = Runtime::Current()->GetClassLinker()->IsQuickToInterpreterBridge(
goal->GetEntryPointFromQuickCompiledCode());
return (is_interpreted || enters_interpreter);
}
}
// 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));
}
}
extern "C" JNIEXPORT jobject JNICALL Java_Main_getThisOfCaller(
JNIEnv* env, [[maybe_unused]] jclass cls) {
ScopedObjectAccess soa(env);
std::unique_ptr<art::Context> context(art::Context::Create());
jobject result = nullptr;
StackVisitor::WalkStack(
[&](const art::StackVisitor* stack_visitor) REQUIRES_SHARED(Locks::mutator_lock_) {
// Discard stubs and Main.getThisOfCaller and methods without vreg info.
if (stack_visitor->GetMethod() == nullptr ||
stack_visitor->GetMethod()->IsNative() ||
(stack_visitor->GetCurrentShadowFrame() == nullptr &&
!Runtime::Current()->IsAsyncDeoptimizeable(stack_visitor->GetOuterMethod(),
stack_visitor->GetCurrentQuickFramePc()))) {
return true;
}
result = soa.AddLocalReference<jobject>(stack_visitor->GetThisObject());
return false;
},
soa.Self(),
context.get(),
art::StackVisitor::StackWalkKind::kIncludeInlinedFrames);
return result;
}
} // namespace art