// Copyright 2012 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/android/jni_android.h" #include #include #include "base/android/java_exception_reporter.h" #include "base/at_exit.h" #include "base/base_unittest_support_jni/JniAndroidTestUtils_jni.h" #include "base/functional/bind.h" #include "base/logging.h" #include "base/memory/raw_ptr.h" #include "base/test/scoped_feature_list.h" #include "base/threading/thread.h" #include "base/time/time.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" using ::testing::Eq; using ::testing::Optional; using ::testing::StartsWith; namespace base { namespace android { namespace { class JniAndroidExceptionTestContext final { public: JniAndroidExceptionTestContext() { CHECK(instance == nullptr); instance = this; SetJavaExceptionCallback(CapturingExceptionCallback); g_log_fatal_callback_for_testing = CapturingLogFatalCallback; } ~JniAndroidExceptionTestContext() { g_log_fatal_callback_for_testing = nullptr; SetJavaExceptionCallback(prev_exception_callback_); env->ExceptionClear(); Java_JniAndroidTestUtils_restoreGlobalExceptionHandler(env); instance = nullptr; } private: static void CapturingLogFatalCallback(const char* message) { auto* self = instance; CHECK(self); // Capture only the first one (can be called multiple times due to // LOG(FATAL) not terminating). if (!self->assertion_message) { self->assertion_message = message; } } static void CapturingExceptionCallback(const char* message) { auto* self = instance; CHECK(self); if (self->throw_in_exception_callback) { self->throw_in_exception_callback = false; Java_JniAndroidTestUtils_throwRuntimeException(self->env); } else if (self->throw_oom_in_exception_callback) { self->throw_oom_in_exception_callback = false; Java_JniAndroidTestUtils_throwOutOfMemoryError(self->env); } else { self->last_java_exception = message; } } static JniAndroidExceptionTestContext* instance; const JavaExceptionCallback prev_exception_callback_ = GetJavaExceptionCallback(); public: const raw_ptr env = base::android::AttachCurrentThread(); bool throw_in_exception_callback = false; bool throw_oom_in_exception_callback = false; std::optional assertion_message; std::optional last_java_exception; }; JniAndroidExceptionTestContext* JniAndroidExceptionTestContext::instance = nullptr; std::atomic g_atomic_id(nullptr); int LazyMethodIDCall(JNIEnv* env, jclass clazz, int p) { jmethodID id = base::android::MethodID::LazyGet< base::android::MethodID::TYPE_STATIC>( env, clazz, "abs", "(I)I", &g_atomic_id); return env->CallStaticIntMethod(clazz, id, p); } int MethodIDCall(JNIEnv* env, jclass clazz, jmethodID id, int p) { return env->CallStaticIntMethod(clazz, id, p); } } // namespace TEST(JNIAndroidMicrobenchmark, MethodId) { JNIEnv* env = AttachCurrentThread(); ScopedJavaLocalRef clazz(GetClass(env, "java/lang/Math")); base::Time start_lazy = base::Time::Now(); int o = 0; for (int i = 0; i < 1024; ++i) o += LazyMethodIDCall(env, clazz.obj(), i); base::Time end_lazy = base::Time::Now(); jmethodID id = g_atomic_id; base::Time start = base::Time::Now(); for (int i = 0; i < 1024; ++i) o += MethodIDCall(env, clazz.obj(), id, i); base::Time end = base::Time::Now(); // On a Galaxy Nexus, results were in the range of: // JNI LazyMethodIDCall (us) 1984 // JNI MethodIDCall (us) 1861 LOG(ERROR) << "JNI LazyMethodIDCall (us) " << base::TimeDelta(end_lazy - start_lazy).InMicroseconds(); LOG(ERROR) << "JNI MethodIDCall (us) " << base::TimeDelta(end - start).InMicroseconds(); LOG(ERROR) << "JNI " << o; } TEST(JniAndroidTest, GetJavaStackTraceIfPresent_Normal) { // The main thread should always have Java frames in it. EXPECT_THAT(GetJavaStackTraceIfPresent(), StartsWith("\tat")); } TEST(JniAndroidTest, GetJavaStackTraceIfPresent_NoEnv) { class HelperThread : public Thread { public: HelperThread() : Thread("TestThread"), java_stack_1_("X"), java_stack_2_("X") {} void Init() override { // Test without a JNIEnv. java_stack_1_ = GetJavaStackTraceIfPresent(); // Test with a JNIEnv but no Java frames. AttachCurrentThread(); java_stack_2_ = GetJavaStackTraceIfPresent(); } std::string java_stack_1_; std::string java_stack_2_; }; HelperThread t; t.StartAndWaitForTesting(); EXPECT_EQ(t.java_stack_1_, ""); EXPECT_EQ(t.java_stack_2_, ""); } TEST(JniAndroidTest, GetJavaStackTraceIfPresent_PendingException) { JNIEnv* env = base::android::AttachCurrentThread(); Java_JniAndroidTestUtils_throwRuntimeExceptionUnchecked(env); std::string result = GetJavaStackTraceIfPresent(); env->ExceptionClear(); EXPECT_EQ(result, kUnableToGetStackTraceMessage); } TEST(JniAndroidTest, GetJavaStackTraceIfPresent_OutOfMemoryError) { JNIEnv* env = base::android::AttachCurrentThread(); Java_JniAndroidTestUtils_setSimulateOomInSanitizedStacktrace(env, true); std::string result = GetJavaStackTraceIfPresent(); Java_JniAndroidTestUtils_setSimulateOomInSanitizedStacktrace(env, false); EXPECT_EQ(result, ""); } TEST(JniAndroidExceptionTest, HandleExceptionInNative) { JniAndroidExceptionTestContext ctx; test::ScopedFeatureList feature_list; feature_list.InitFromCommandLine("", "HandleJniExceptionsInJava"); // Do not call setGlobalExceptionHandlerAsNoOp(). Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); EXPECT_THAT(ctx.last_java_exception, Optional(StartsWith("java.lang.RuntimeException"))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kUncaughtExceptionMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_NoOpHandler) { JniAndroidExceptionTestContext ctx; Java_JniAndroidTestUtils_setGlobalExceptionHandlerAsNoOp(ctx.env); Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); EXPECT_THAT(ctx.last_java_exception, Optional(StartsWith("java.lang.RuntimeException"))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kUncaughtExceptionHandlerFailedMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_ThrowingHandler) { JniAndroidExceptionTestContext ctx; Java_JniAndroidTestUtils_setGlobalExceptionHandlerToThrow(ctx.env); Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); EXPECT_THAT(ctx.last_java_exception, Optional(StartsWith("java.lang.IllegalStateException"))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kUncaughtExceptionHandlerFailedMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_OomThrowingHandler) { JniAndroidExceptionTestContext ctx; Java_JniAndroidTestUtils_setGlobalExceptionHandlerToThrowOom(ctx.env); Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); // Should still report the original exception when the global exception // handler throws an OutOfMemoryError. EXPECT_THAT(ctx.last_java_exception, Optional(StartsWith("java.lang.RuntimeException"))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kUncaughtExceptionHandlerFailedMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_OomInGetJavaExceptionInfo) { JniAndroidExceptionTestContext ctx; Java_JniAndroidTestUtils_setGlobalExceptionHandlerToThrowOom(ctx.env); Java_JniAndroidTestUtils_setSimulateOomInSanitizedStacktrace(ctx.env, true); Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); Java_JniAndroidTestUtils_setSimulateOomInSanitizedStacktrace(ctx.env, false); EXPECT_THAT(ctx.last_java_exception, Optional(Eq(kOomInGetJavaExceptionInfoMessage))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kUncaughtExceptionHandlerFailedMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_Reentrant) { JniAndroidExceptionTestContext ctx; // Use the SetJavaException() callback to trigger re-entrancy. Java_JniAndroidTestUtils_setGlobalExceptionHandlerToThrow(ctx.env); ctx.throw_in_exception_callback = true; Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); EXPECT_THAT(ctx.last_java_exception, Optional(Eq(kReetrantExceptionMessage))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kReetrantExceptionMessage))); } TEST(JniAndroidExceptionTest, HandleExceptionInJava_ReentrantOom) { JniAndroidExceptionTestContext ctx; // Use the SetJavaException() callback to trigger re-entrancy. Java_JniAndroidTestUtils_setGlobalExceptionHandlerToThrow(ctx.env); ctx.throw_oom_in_exception_callback = true; Java_JniAndroidTestUtils_throwRuntimeException(ctx.env); EXPECT_THAT(ctx.last_java_exception, Optional(Eq(kReetrantOutOfMemoryMessage))); EXPECT_THAT(ctx.assertion_message, Optional(Eq(kReetrantOutOfMemoryMessage))); } } // namespace android } // namespace base