我正在尝试将用C编写的一些软件移植到Android平台。该软件具有可以从连接的USB设备读取和写入的组件。我想要做的是在Java中打开与设备的连接,然后将USB设备的文件描述符传递给JNI代码。
以下是我的应用程序lsof
的(相关)输出,显示我有两个USB设备描述符:
com.tim 8861 u0_a66 35 ??? ??? ??? ??? /dev/bus/usb/001/002 com.tim 8861 u0_a66 36 ??? ??? ??? ??? socket:[51170] com.tim 8861 u0_a66 37 ??? ??? ??? ??? socket:[51173] com.tim 8861 u0_a66 38 ??? ??? ??? ??? /dev/bus/usb/001/003
我已将两个描述符(上面为35和38)传递给我的本机方法,但是当我尝试写入任一文件描述符时,write()
返回-1
,然后我得到了一个EINVAL
错误。
这是我原生方法的主体:
char buff[1024] = {0};
jsize len = (*env)->GetArrayLength(env, fds);
jint *arr = (*env)->GetIntArrayElements(env, fds, 0);
int i;
char data[4] = {
0x09,
0x90,
0x50,
0x50,
};
for (i = 0; i < len; i++) {
int wrote = write(arr[i], data, 4);
int flags = fcntl(arr[i], F_GETFL);
char *err = strerror(errno);
sprintf(buff, "%sFD: %d \n"
"wrote: %d \n"
"(err: %d %s) \n"
"flags: %d \n"
"NBIO %d \n"
"readonly %d \n"
"writeonly %d \n"
"both %d \n"
"append %d \n"
"large file %d \n\n", buff, arr[i], wrote, errno, err, flags, flags & O_NONBLOCK,
flags & O_RDONLY, flags & O_WRONLY, flags & O_RDWR, flags & O_APPEND,
flags & O_LARGEFILE);
}
return (*env)->NewStringUTF(env, buff);
调用该方法时返回的字符串是:
FD: 35 wrote: -1 (err: 22 Invalid argument) flags: 32770 NBIO 0 readonly 0 writeonly 0 both 2 append 0 large file 32768 FD: 38 wrote: -1 (err: 22 Invalid argument) flags: 32770 NBIO 0 readonly 0 writeonly 0 both 2 append 0 large file 32768
写入USB设备确实可以通过Java工作,所以当尝试通过本机代码执行时,它似乎只是一个问题。
有没有人有这样的经历?
答案 0 :(得分:8)
似乎在USB文件描述符上使用write()
并不完全有效,因为USB设备上有多个可以写入数据的端点。
我可以使用ioctl()
功能批量转移到设备上的特定端点:
#include <linux/usbdevice_fs.h>
#include <sys/ioctl.h>
// ...
char data[4] = {0x09, 0x90, 0x50, 0x50};
struct usbdevfs_bulktransfer bt;
bt.ep = usb_endpoint; /* endpoint (received from Java) */
bt.len = 4; /* length of data */
bt.timeout = 100; /* timeout in ms */
bt.data = data; /* the data */
int rtn = ioctl(fd, USBDEVFS_BULK, &bt);