背景
熟悉我的老朋友可能都知道,之前为了应对crash与anr,开源过一个“民间偏方”的库Signal,用于解决在发生crash或者anr时进行应用的重启,从而最大程度减少其坏影响。
在维护的过程中,发生过这样一件趣事,就是有位朋友发现在遇到信号为SIGSEGV时,再调用信号处理函数的时候
void SigFunc(int sig_num, siginfo *info, void *ptr) {
// 这里判空并不代表这个对象就是安全的,因为有可能是脏内存
if (currentEnv == nullptr || currentObj == nullptr) {
return;
}
__android_log_print(ANDROID_LOG_INFO, TAG, "%d catch", sig_num);
__android_log_print(ANDROID_LOG_INFO, TAG, "crash info pid:%d ", info->si_pid);
jclass main = currentEnv->FindClass("com/example/lib_signal/SignalController");
jmethodID id = currentEnv->GetMethodID(main, "callNativeException", "(ILjava/lang/String;)V");
if (!id) {
__android_log_print(ANDROID_LOG_INFO, TAG, "%d !id!id!id!id!id!id!id", sig_num);
return;
}
__android_log_print(ANDROID_LOG_INFO, TAG, "%d 11111111111111111111", sig_num);
jstring nativeStackTrace = currentEnv->NewStringUTF(backtraceToLogcat().c_str());
__android_log_print(ANDROID_LOG_INFO, TAG, "%d 22222222222222222222", sig_num);
currentEnv->CallVoidMethod(currentObj, id, sig_num, nativeStackTrace);
__android_log_print(ANDROID_LOG_INFO, TAG, "%d 33333333333333333333", sig_num);
// 释放资源
currentEnv->DeleteGlobalRef(currentObj);
currentEnv->DeleteLocalRef(nativeStackTrace);
}
复制代码
会遇到
从上文打印中看到,SIGSEGV被抛出,之后被我们的信号处理函数抓到了,但是却没有被回调到java层,反而变成了SIGABRT。还有就是SIGSEGV被捕获后,却无法通过jni回调给java层的重启处理。本文将从这个例子出发,从踩坑的过程中去学习更多jni知识。