blob: 395e9763b00e09a639fc68fe947f9fa2aea9e712 [file] [log] [blame]
// Copyright 2015 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/java_exception_reporter.h"
#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/android/scoped_java_ref.h"
#include "base/base_jni_headers/JavaExceptionReporter_jni.h"
#include "base/debug/dump_without_crashing.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/lazy_instance.h"
using base::android::JavaParamRef;
using base::android::JavaRef;
namespace base {
namespace android {
namespace {
void (*g_java_exception_callback)(const char*);
using JavaExceptionFilter =
base::RepeatingCallback<bool(const JavaRef<jthrowable>&)>;
LazyInstance<JavaExceptionFilter>::Leaky g_java_exception_filter =
LAZY_INSTANCE_INITIALIZER;
} // namespace
void InitJavaExceptionReporter() {
JNIEnv* env = base::android::AttachCurrentThread();
// Since JavaExceptionReporter#installHandler will chain through to the
// default handler, the default handler should cause a crash as if it's a
// normal java exception. Prefer to crash the browser process in java rather
// than native since for webview, the embedding app may have installed its
// own JavaExceptionReporter handler and would expect it to be called.
constexpr bool crash_after_report = false;
SetJavaExceptionFilter(
base::BindRepeating([](const JavaRef<jthrowable>&) { return true; }));
Java_JavaExceptionReporter_installHandler(env, crash_after_report);
}
void InitJavaExceptionReporterForChildProcess() {
JNIEnv* env = base::android::AttachCurrentThread();
constexpr bool crash_after_report = true;
SetJavaExceptionFilter(
base::BindRepeating([](const JavaRef<jthrowable>&) { return true; }));
Java_JavaExceptionReporter_installHandler(env, crash_after_report);
}
void SetJavaExceptionFilter(JavaExceptionFilter java_exception_filter) {
g_java_exception_filter.Get() = std::move(java_exception_filter);
}
void SetJavaExceptionCallback(void (*callback)(const char*)) {
DCHECK(!g_java_exception_callback);
g_java_exception_callback = callback;
}
void SetJavaException(const char* exception) {
// No need to print exception because they are already logged via
// env->ExceptionDescribe() within jni_android.cc.
if (g_java_exception_callback) {
g_java_exception_callback(exception);
}
}
void JNI_JavaExceptionReporter_ReportJavaException(
JNIEnv* env,
jboolean crash_after_report,
const JavaParamRef<jthrowable>& e) {
std::string exception_info = base::android::GetJavaExceptionInfo(env, e);
bool should_report_exception = g_java_exception_filter.Get().Run(e);
if (should_report_exception) {
SetJavaException(exception_info.c_str());
}
if (crash_after_report) {
LOG(ERROR) << exception_info;
LOG(FATAL) << "Uncaught exception";
}
if (should_report_exception) {
base::debug::DumpWithoutCrashing();
SetJavaException(nullptr);
}
}
void JNI_JavaExceptionReporter_ReportJavaStackTrace(
JNIEnv* env,
const JavaParamRef<jstring>& stack_trace) {
SetJavaException(ConvertJavaStringToUTF8(stack_trace).c_str());
base::debug::DumpWithoutCrashing();
SetJavaException(nullptr);
}
} // namespace android
} // namespace base