带有JNI的ioctl():损坏的文件描述符

时间:2017-05-29 15:31:15

标签: java linux java-native-interface ioctl

我正在尝试使用Java中的Linux tun驱动程序进行交互,如此处所述。

How to interface with the Linux tun driver

但是由于你不能用java调用ioctl(),我使用的是Java Native Interface。只要我不在同一个文件中读写,它就能正常工作。

如果我这样做,我会得到这个例外,我将翻译为" FileDescriptor处于破坏状态" :

java.io.IOException: Le descripteur du fichier est dans un mauvais état
    at java.io.FileOutputStream.writeBytes(Native Method)
    at java.io.FileOutputStream.write(FileOutputStream.java:326)
    at WriterThread.main(WriterThread.java:54)

这是java代码:

public static void main(String[] arg){
        File tunFile = new File("/dev/net/tun");
        FileOutputStream outStream;
        FileInputStream inStream;

        try {

            inStream = new FileInputStream(tunFile);
            outStream = new FileOutputStream(tunFile);
            FileDescriptor fd = inStream.getFD();

            //getting the file descriptor

            Field f = fd.getClass().getDeclaredField("fd");
            f.setAccessible(true);
            int descriptor = f.getInt(fd);


            //use of Java Native Interface
            new TestOuvertureFichier().ioctl(descriptor);

            while(true){
                System.out.println("reading");
                byte[] bytes = new byte[500];
                int l = 0;
                l = inStream.read(bytes);

                //the problem seems to come from here
                outStream.write(bytes,0,l);

            }

        } catch (Exception e) {
            e.printStackTrace();
        }

    }

这是C代码:

JNIEXPORT void JNICALL Java_TestOuvertureFichier_ioctl(JNIEnv *env,jobject obj, jint descriptor){
      struct ifreq ifr;
      memset(&ifr, 0, sizeof(ifr));
      ifr.ifr_flags = IFF_TUN;
      strncpy(ifr.ifr_name, "tun0", IFNAMSIZ);
      int err;

      if ( (err = ioctl(descriptor, TUNSETIFF, (void *) &ifr)) == -1 ) {
          perror("ioctl TUNSETIFF");exit(1);
      }
      return;
}

3 个答案:

答案 0 :(得分:1)

-G。 Fiedler是对的,读取应该至少与接口MTU一样大,并且写入不应该超过MTU。除此之外,我会检查:

  • 在您尝试读取或写入之前,接口已启动(ip addr add x.x.x.x / xx dev tun0,ip link set tun0 up)
  • 只打开一次tun设备,例如使用RandomAccessFile。在这里,我不确定inStream和outStream是否具有相同的文件描述符。

答案 1 :(得分:0)

注意bytes至少应该是接口的MTU大小,例如1500字节。每次调用时,tun fd上的read()都会读取整个数据包。

在写入tun设备之前,您应该操作IP标头,尤其是接收数据包的源和目标地址。

答案 2 :(得分:0)

文件描述符不是由new File()调用创建的,而是在创建FileInputStreamFileOutputStream对象时创建的。这意味着您的代码会打开/ dev / net / tun文件两次(创建两个不同的文件描述符)。

inStream = new FileInputStream(tunFile);
outStream = new FileOutputStream(tunFile);

因此,ioctl仅适用于inStream,而不适用于outStream。 尝试使用与FileOutputStream相同的文件描述符创建FileInputStream

outStream = new FileOutputStream(inStream.getFD());    

编辑:FileInputStream很可能会打开一个只读FD。正如JayTE提出的那样,最好先创建一个RandomAccessFile,然后使用FD来创建两个流。