blob: eefaabca9c79f89b755e69f289906675f4f38c79 [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 "base/logging.h"
#include "dex_file-inl.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-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);
struct MethodIsInterpretedVisitor : public StackVisitor {
public:
MethodIsInterpretedVisitor(Thread* thread, std::string shorty)
: StackVisitor(thread, nullptr, StackVisitor::StackWalkKind::kIncludeInlinedFrames),
shorty_(shorty),
method_is_interpreted_(false) {}
virtual bool VisitFrame() OVERRIDE REQUIRES_SHARED(Locks::mutator_lock_) {
if (!GetMethod()->IsRuntimeMethod() && shorty_ == GetMethod()->GetShorty()) {
method_is_interpreted_ = IsShadowFrame();
return false;
}
return true;
}
bool IsInterpreted() {
return method_is_interpreted_;
}
private:
const std::string shorty_;
bool method_is_interpreted_;
};
extern "C" JNIEXPORT jboolean JNICALL Java_Main_isInterpretedFunction(JNIEnv* env,
jclass klass ATTRIBUTE_UNUSED,
jstring name) {
if (Runtime::Current() == nullptr) {
return JNI_TRUE;
}
ScopedObjectAccess soa(env);
MethodIsInterpretedVisitor v(soa.Self(), soa.Decode<mirror::String>(name)->ToModifiedUtf8());
v.WalkStack();
return v.IsInterpreted();
}
// 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 cls, size_t level) {
ScopedObjectAccess soa(env);
ObjPtr<mirror::Class> klass = soa.Decode<mirror::Class>(cls);
const DexFile& dex_file = klass->GetDexFile();
const OatFile::OatDexFile* oat_dex_file = dex_file.GetOatDexFile();
if (oat_dex_file == nullptr) {
// No oat file, this must be a test configuration that doesn't compile at all. Ignore that the
// result will be that we're running the interpreter.
return JNI_FALSE;
}
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