如何通过 JNI 从 Android 中的信号处理函数调用 Java 函数

2024-06-19

我的目标是将信号从内核发送到在用户空间中运行的 Android 服务。收到此信号后,服务应该向内核发出 IOCTL 回调。通过 ioctl 调用从内核获取数据后,它必须显示给用户。为此,我从 Java 服务调用本机方法,该方法注册 sigaction 结构,其中包括该信号的处理程序函数。该处理函数将进行 IOCTL 调用并调用 Java 函数以将字符串传递给 Java 服务。

这是 signal.java 类

public class signal {
static{
    System.loadLibrary("signal");
}
public native String hello();

public String messageMe(String s)
{
    if(null != MainActivity.mUiHandler)
    {
        Message msgToActivity = new Message();
        msgToActivity.what = 0;

            msgToActivity.obj  = s; // you can put extra message here

        MainActivity.mUiHandler.sendMessage(msgToActivity);
    }

    System.out.println(s);
        return s;
}

}

我需要从信号处理程序调用这个“messageMe”函数。 这是注册 sigaction 结构的本机 hello() 函数。

JNIEXPORT jstring JNICALL Java_com_example_service_signal_hello
(JNIEnv * env, jobject obj) {

 int configfd;
char buf[10];
/* setup the signal handler for SIG_TEST
 * SA_SIGINFO -> we want the signal handler function with 3 arguments
 */
struct sigaction sig;
sig.sa_sigaction = receiveData;
sig.sa_flags = SA_SIGINFO;
sigaction(SIG_TEST, &sig,NULL);
}

receiveData 是处理函数。就这样吧。

void receiveData(int n, siginfo_t *info,void* unused) {
    char buf[200];
    char *msg = "hello world";
    int fd = -1;
    if ((fd = open("/dev/my_device", O_RDWR)) < 0) {
            perror("on");
            return;
    }
    if(ioctl(fd, READ_IOCTL, buf) < 0)
            perror("second ioctl");

     jstring jstr = (*env)->NewStringUTF(env, buf);
     jclass *clazz = (*env)->GetObjectClass(env, obj);
     jmethodID messageMe = (*env)->GetMethodID(env,clazz, "messageMe", "(Ljava/lang/String;)Ljava/lang/String;");
     jobject result = (*env)->CallObjectMethod(env,obj, messageMe, jstr);

    printf("message: %s\n", buf);

} 该处理函数应该调用 Java 函数“messageMe”并将其从内核读取的字符串作为参数传递。但它没有环境变量 JNIEnv * 和 jobject obj。当我在 hello() 函数中维护指向这些变量的全局指针并在此处使用它们时,它给出了 NoClassDefFoundError。 现在如何将字符串从该信号处理程序发送到 Java 函数?


我有以下全局变量。我缓存了这些对象以供将来使用。

JavaVM* vm;
static jclass cl;
static jobject ob;

在 hello 函数中,我对上述变量进行了全局引用。

JNIEXPORT jstring JNICALL Java_com_example_service_signal_hello
(JNIEnv * env, jobject obj) {
(*env)->GetJavaVM(env,&vm);
ob = (*env)->NewGlobalRef(env,obj);
jclass clazz = (*env)->FindClass(env,"com/example/service/signal");
cl = (*env)->NewGlobalRef(env,clazz);
}

现在我在处理函数中使用了这些

void receiveData(int n, siginfo_t *info,void* unused) {
//other code
JNIEnv* env = NULL;
    if(vm==NULL)
        return;
    (*vm)->AttachCurrentThread(vm,&env,0);
 jmethodID messageMe = (*env)->GetStaticMethodID(env,cl, "messageMe", "(Ljava/lang/String;)Ljava/lang/String;");
jobject result = (*env)->CallStaticObjectMethod(env,cl, messageMe, jstr);
}

我所做的一项更改是使用静态方法而不是普通方法。所以我不必使用该类的特定实例。如果该方法不是静态的,则可以在 CallObjectMethod 中使用全局变量“ob”。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何通过 JNI 从 Android 中的信号处理函数调用 Java 函数 的相关文章

随机推荐