While this has been an experiment, it turns out that Android native stack is too:
- Stripped out of all linux debuging goodies
- dependant of processor & driver implementations.
Moreover, the use of this library usually makes matter worse when the app crashes inside a thread : The Android system tries to restart the faulting activity making the app having an infinite crash loop.
This project remains here for reference, but I advise against using it until some better way to debug native code is found.
A simple set of code to enable JNI Native crashes to be caught by java and throw a meaningfull exception
While java exceptions are a tremendous tool for Android JAVA developers to investigate crashes, JNI C / C++ crashes terminate the process and are not catchable by the system.
Can it work with crash reporters like ACRA ?
Yes ! I'm glad you ask : That's the reason why it was developped in the first place ! As this throws an exception, a crash handler can intercept it and handle it the way it needs.
On the JNI side, it registers sigaction callbacks to be called when the program receives an error signal (such as SIGSEGV: segmentation fault). It then generates an exception to capture the stack trace and then launches a new activity in a new process (because the current is to be killed by the signal) which will throw an exception.
This tool has been largely inspired by Chris Boyle's great SO answer.
-
Download the archive and extract it
-
Put NativeCrashHandler-vX.Y.jar in the libs/ directory of your project
-
Put NativeCrashHandler.h and NativeCrashHandler.cpp in the jni/ directory of your project
-
Add NativeCrashHandler.cpp to the LOCAL_SRC_FILES variable in your jni/Android.mk file
-
In your application class, in onCreate, add:
new NativeCrashHandler().registerForNativeCrash(this);
- In The JNI_OnLoad JNI C function, add:
#include "NativeCrashHandler.h"
/*...*/
nativeCrashHandler_onLoad(jvm);
- In your AndroidManifest.xml, add:
<activity
android:name="com.github.nativehandler.NativeCrashActivity"
android:configChanges="keyboard|keyboardHidden|orientation"
android:exported="false"
android:process=":CrashHandler"
android:stateNotNeeded="true"
android:theme="@android:style/Theme.Translucent.NoTitleBar" />
- You're done
Sometimes, you need to register the NativeCrashACRAHandler with an Activity Context instead of an Application Context. Please remember that this should not be done if you can use the Application Context.
When you use NativeCrashACRAHandler.registerForNativeCrash with an Activity Context, the activity is retained by the NativeCrashACRAHandler JNI code. Therefore, you need to call NativeCrashACRAHandler.unregisterForNativeCrash when the activity is paused, otherwise the application will never be relased by the system. Remember that only ONE context needs to be registered at a time, so you need to unregister an activity before register a new one.