【发布时间】:2011-08-24 21:34:04
【问题描述】:
我对 NDK 有疑问。
在我的 JNI_OnLoad 方法中,我缓存了 JavaVm 指针、调用该方法的类以及稍后使用的方法 ID:
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved){
JNIEnv *env;
cachedJVM = jvm;
if((*jvm)->GetEnv(jvm, (void**)&env, JNI_VERSION_1_6)){
LOG_ERROR("Could not get JNIEnv*");
return JNI_ERR;
}
javaClass = (*env)->FindClass(env, "org/test/opensl/AudioProcessor");
if(javaClass == NULL){
LOG_ERROR("Could not get java class");
return JNI_ERR;
}
javaCallbackMID = (*env)->GetMethodID(env, javaClass, "enqueueAudio", "([B)V");
if(javaCallbackMID == NULL){
LOG_ERROR("Could not get method identifier");
return JNI_ERR;
}
return JNI_VERSION_1_6;
}
我有一个如下定义的小实用程序方法,它应该让我得到一个指向 JNIEnv 的指针:
JNIEnv* JNU_GetEnv(){
JNIEnv* env;
(*cachedJVM)->GetEnv(cachedJVM, (void**)&env, JNI_VERSION_1_6);
return env;
}
最后,我有一个来自 OpenSL ES SLAndroidSimpleBufferQueueItf 的回调,我想处理来自 SLRecordItf 的录制音频:
void recorderCallback(SLAndroidSimpleBufferQueueItf bq, void *context){
SLresult result;
JNIEnv* env;
recorderContext* thisContext = (recorderContext*)context;
env = JNU_GetEnv();
if(env == NULL){
LOG_ERROR("Could not get JNIEnv*");
return;
}
jbyteArray data = (*env)->NewByteArray(env, MAX_PACKET_SIZE);
if(data == NULL){
LOG_ERROR("No memory could be allocated for buffer");
return;
}
(*env)->SetByteArrayRegion(env, data, 0, MAX_PACKET_SIZE, recorderBuffer);
(*env)->CallByteMethodA(env, thisContext->caller, javaCallbackMID, data);
(*env)->DeleteLocalRef(env, data);
result = (*bq)->Enqueue(bq, recorderBuffer,
RECORDER_FRAMES * sizeof(jbyte));
checkError(result, "Unable to enqueue new buffer");
}
回调方法的上下文参数只保存对调用本机方法的对象的引用。它是一个像这样的自定义结构:
typedef struct recorderContext{
jobject caller;
} recorderContext;
但是,每次我尝试运行它时,我都会从回调方法中收到 "Could not get JNIEnv*" 错误消息。
我的问题基本上归结为:为什么我可以在 JNI_OnLoad 方法中获取指向 JNIEnv 的指针,但在 recorderCallback 中却不能,因为两者都使用相同的 Java VM 指针来获取 JNIEnv?
我需要此回调将录制的音频传回我的 Java 层以进行进一步处理...
【问题讨论】:
标签: android c java-native-interface android-ndk