使用JNI通过UNIX域套接字发送Java FD

时间:2013-10-06 13:33:35

标签: android c sockets android-ndk java-native-interface

我正在编写一个Android应用程序,它传递一个取自ParcelFileDescriptor.getFd()的JAVA fd,根据[1]声明我返回的int是本机fd。

现在,有了这个fd,我试图通过一个unix域套接字将它写入正在侦听的现有进程。为此,我使用JNI并将上面收到的int作为名为 fdToSend 的参数传递给JNI函数。

当我的JNI代码尝试调用sendmsg()时,会发生错误,指出“文件编号错误”。

在google的帮助下,当我调用sendmsg()时,似乎套接字连接可能已关闭,但我看不出这是怎么回事。

sendfd()方法与[2]中的完全相同。

以下是我的桥接JNI功能:

JNIEXPORT jint JNICALL Java_com_example_myapp_AppManager_bridgeSendFd(JNIEnv *env, jint fdToSend) {
    int fd;
    struct sockaddr_un addr;

    if ( (fd = socket(AF_UNIX, SOCK_STREAM, 0)) == -1) {
        __android_log_print(ANDROID_LOG_ERROR, APPNAME, "socket() failed: %s (socket fd = %d)\n", strerror(errno), fd);
        return (jint)-1;
    }

    memset(&addr, 0, sizeof(addr));
    addr.sun_family = AF_UNIX;
    strncpy(addr.sun_path, "/data/data/com.example.myapp/sock_path", sizeof(addr.sun_path)-1);

    if (connect(fd, (struct sockaddr*)&addr, sizeof(addr)) == -1) {
        __android_log_print(ANDROID_LOG_ERROR, APPNAME, "connect() failed: %s (fd = %d)\n", strerror(errno), fd);
        return (jint)-1;
    }

    return (jint)sendfd(fd, (int)fdToSend);
}

[1] http://developer.android.com/reference/android/os/ParcelFileDescriptor.html#getFd()

[2] https://stackoverflow.com/a/4491203/2796346

1 个答案:

答案 0 :(得分:1)

原来问题是JNI函数声明。

第一个arg是JNIEnv,第二个是jclass,然后它是从Java传入的值(即对我来说,fdToSend)。

所以,我的猜测是,因为我使用的是第二个arg(应该是一个jclass),所以我可能会得到一个内存引用或类似的东西。

如下所示:https://stackoverflow.com/a/10743451/2796346