storage R&D guy.
全部博文(1000)
分类: 服务器与存储
2015-05-31 11:30:20
最近在研究hid通信,下面总结下供以后参考,同时也希望能给正在研究这个的同胞提供一些帮助。
有关自定义hid设备在linux内核源码下的Documentation/usb/gadget_hid.txt中有相关说明:如何添加自定义描述符,以及应用程序,首先对这个说明做一些补充.
下面我以s3c2416进行说明,内核源码linux3.1.0
hid相关源码在linux内核源码下的drivers/usg/gadget里,主要包括的文件的有:hid.c,f_hid.c,composite.c,s3c-hsudc.c.
首先在hid.c里添加关于你自定义的设备的描述符:
static int __init hidg_init(void)
中添加如下代码:
在s3c-hsudc.c中找到S3C2443_PHYCTRL
然后直接给他写成0
然后在f_hid.c中找到:static int hidg_setup(struct usb_function *f,const struct usb_ctrlrequest *ctrl)函数,文件第324行
在变量声明后加入如下代码(第338行加入)::
修改完成后pc就可以识别它了。经过测试这个只能用于小数据的通信,就是说用pc每次向2416发送64Bytes是没有问题的,但是要是传文件的话就会丢包,而且很严重,如果pc每次向2416发64Bytes,然后2416回一个收到的指令给pc,反复这样传估计是没问题的,具体我没测试。
我要的效果是pc能给2416传文件,把文件封装成由多个9点几K组成的包然后传给2416,即pc每给2416传9点几K,然后2416给pc回一个指令,然后pc接着传,通过上面这样修改后这个效果是达不到的,所以我又做了如下修改,主要修改文件为f_hid.c.
Linux里有一种循环缓冲区kfifo,这个网上有很多说明,我这里就不说了,我主要说下我增加了些什么.首先增加一个kfifo.定义buffer大小,buffer大小16K,丢包的原因是因为pc端数据发送太快,而2416这边应用程序需要对数据进行处理反应不过来,所以需要增加这个动态缓冲区
static struct kfifo recv_fifo;
#define RECV_LEN (16*1024)
定义在文件开头即可.
然后在static int __init hidg_bind(struct usb_configuration *c, struct usb_function *f)函数中添加如下代码:
kfifo_in(&recv_fifo, req->buf, req->actual);位置如下:
#define READ_COND (hidg->set_report_buff != NULL)这个定义修改为:
#define READ_COND (!kfifo_is_empty(&recv_fifo))
同时增加如下代码,首先增加声明:
unsigned int copied;
修改完成后重新编译。
经过测试传送2M以下的文件是没有问题的,但是2M以上传送失败,速度大概在40-50KB/s,理论速度应该是64KB/s但是实际上受你的应用程序还有其它一些因素的影响不可能达到这么多.
对于为什么传2M以上的文件不行,继续纠结中。希望高手指点.
补充:最开始的时候还发现:
s3c-hsudc setup failed, returned -95的错误但是没影响正常使用 所以我就没说 觉得还是说一下好
可能是哪个请求没有完成,打开f_hid.c里的hidg_setup中的default里的
VDBG(cdev, "Unknown request 0x%x\n",ctrl->bRequest);
编译后重新烧写你会发现提示
Unknown request 0xa果然和s3c6410一样,这个是host向设备发送get_interface设备没有回所以出现这个错误
具体定义在linux内核源码include/linux/usb/ch9.ch查找0x0A你就会找到
#define USB_REQ_GET_INTERFACE 0x0A
所以我们在hidg_setup中加入对get_interface的处理:
这样问题就解决了
2M文件上传不成功,经过加打印信息,发现是PC端程序的问题,但是我这改不了,所以没办法验证