問題は、どの Java 関数が何らかの JNI 関数を呼び出しているかを見つけることです。Java では、これは で実現できますがnew Exception.printStackTrace()
、ネイティブ (JNI) 関数から実行する必要があります。
後で独自のコードを見つける最も簡単な方法は、ネットに公開することなので、質問と回答の両方を投稿します。
問題は、どの Java 関数が何らかの JNI 関数を呼び出しているかを見つけることです。Java では、これは で実現できますがnew Exception.printStackTrace()
、ネイティブ (JNI) 関数から実行する必要があります。
後で独自のコードを見つける最も簡単な方法は、ネットに公開することなので、質問と回答の両方を投稿します。
の JNI 類似物new Exception.printStackTrace()
は次のとおりです。
//#include <android/log.h>
//#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG , "~~~~~~", __VA_ARGS__)
//#define DLOG(...) __android_log_print(ANDROID_LOG_DEBUG , "~~~~~~", __VA_ARGS__)
void printStackTrace(JNIEnv *env) {
LOGD("###################################################################################printStackTrace{");
jclass cls = env->FindClass("java/lang/Exception");
if (cls != NULL) {
jmethodID constructor = env->GetMethodID(cls, "<init>", "()V");
if(constructor != NULL) {
jobject exc = env->NewObject(cls, constructor);
if(exc != NULL) {
jmethodID printStackTrace = env->GetMethodID(cls, "printStackTrace", "()V");
if(printStackTrace != NULL) {
env->CallObjectMethod(exc, printStackTrace);
} else { DLOG("err4"); }
} else { DLOG("err3"); }
env->DeleteLocalRef(exc);
} else { DLOG("err2"); }
} else { DLOG("err1"); }
/* free the local ref */
env->DeleteLocalRef(cls);
LOGD("###################################################################################printStackTrace}");
}
ところで、ネイティブ層から Java に例外をスローすることができます:) 次のようなもの:
jint throwOutOfMemoryError( JNIEnv *env, char *message ){
jclass exClass;
char *className = "java/lang/OutOfMemoryError" ;
exClass = (*env)->FindClass( env, className );
if ( exClass == NULL ){
return throwNoClassDefError( env, className );
}
return (*env)->ThrowNew( env, exClass, message );
}
または、例外のインスタンスがある場合は、それを Java レイヤーにスローしてから、Java でスタック トレースを取得します。