#include "base/android/java_exception_reporter.h"
#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/debug/dump_without_crashing.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/logging.h"
#include "base/no_destructor.h"
#include "base/base_minimal_jni/JavaExceptionReporter_jni.h"
using jni_zero::JavaRef;
namespace base {
namespace android {
namespace {
JavaExceptionCallback g_java_exception_callback;
using JavaExceptionFilter =
base::RepeatingCallback<bool(const JavaRef<jthrowable>&)>;
JavaExceptionFilter& GetJavaExceptionFilter() {
static base::NoDestructor<JavaExceptionFilter> java_exception_filter;
return *java_exception_filter;
}
}
void InitJavaExceptionReporter() {
JNIEnv* env = jni_zero::AttachCurrentThread();
constexpr bool crash_after_report = false;
SetJavaExceptionFilter(
base::BindRepeating([](const JavaRef<jthrowable>&) { return true; }));
Java_JavaExceptionReporter_installHandler(env, crash_after_report);
}
void InitJavaExceptionReporterForChildProcess() {
if (!base::android::IsJavaAvailable()) {
return;
}
JNIEnv* env = jni_zero::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) {
GetJavaExceptionFilter() = std::move(java_exception_filter);
}
void SetJavaExceptionCallback(JavaExceptionCallback callback) {
DCHECK(!g_java_exception_callback || !callback);
g_java_exception_callback = callback;
}
JavaExceptionCallback GetJavaExceptionCallback() {
return g_java_exception_callback;
}
void SetJavaException(const char* exception) {
if (g_java_exception_callback) {
g_java_exception_callback(exception);
}
}
static void JNI_JavaExceptionReporter_ReportJavaException(
JNIEnv* env,
jboolean crash_after_report,
const JavaRef<jthrowable>& e) {
std::string exception_info = base::android::GetJavaExceptionInfo(env, e);
bool should_report_exception = GetJavaExceptionFilter().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);
}
}
static void JNI_JavaExceptionReporter_ReportJavaStackTrace(
JNIEnv* env,
std::string& stack_trace) {
SetJavaException(stack_trace.c_str());
base::debug::DumpWithoutCrashing();
SetJavaException(nullptr);
}
}
}
DEFINE_JNI(JavaExceptionReporter)