在写JNI方法时有两种方法:
一种是通过javah生成头文件,然后自己实现;
一种在JNI_OnLoad函数中进行RegisterNatives
第一个没什么要说的。第二个基本的操作流程就是找到的对应的Java class然后进行Runtime的注册,具体怎么写;
//注册函数映射
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved) {
JNIEnv *pEnv = NULL;
jint ret = vm->GetEnv((void**) &pEnv, JNI_VERSION_1_6);
if (ret != JNI_OK) {
LOGE("[-]ERROR:GetEnv");
return -1;
}
JNINativeMethod g_Methods[] = {{"radamNum", "()V", (void*) radamNum_func_addr},
{"sign", "Ljava/lang/String;", (jstring*)sign_func_addr}
};
jclass cls = pEnv->FindClass("com/example/NativeExample/jni_entry");
if (cls == NULL) {
LOGE("[-]ERROR:FindClass");
return -1;
}
ret = pEnv->RegisterNatives(cls, g_Methods,sizeof(g_Methods) / sizeof(g_Methods[0]));
if (ret != JNI_OK) {
LOGE("[-]ERROR:RegisterNatives");
return -1;
}
return JNI_VERSION_1_6;
}
总结&Tips:
1.其中RegisterNatives可以进行二次注册;二次注册为第二次replace的函数;CTF遇到过这种把戏
2.注意在attach JVM thread时候注意拿到运行时的指针,用完记得deattachthread;
3.其中从JVM中使用他的字符串要进行复制,当然从C的runtime也要进行复制到JVM中;
4.两个运行时的内存是独立的,注意记得要释放内存;