linux查看自定义HID,s3c2416 自定义hid设备通信

最近在研究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里添加关于你自定义的设备的描述符:

"font-size:18px;">#include

/* hid descriptor for a keyboard */

staticstructhidg_func_descriptor my_hid_data = {

.subclass       = 0, /* No subclass */

.protocol       = 0, /* User define */

.report_length      = 64,

.report_desc_length = 28,

.report_desc        = {

}

};

staticstructplatform_device my_hid = {

.name           = "hidg",

.id         = 0,

.num_resources      = 0,

.resource       = 0,

.dev.platform_data  = &my_hid_data,

};

具体内容我就不填进去了,添加这个后,在函数:

static int __init hidg_init(void)

中添加如下代码:

status = platform_device_register(&my_hid);

if(status

printk("++++++++++++++++reg failed\n");

platform_device_unregister(&my_hid);

returnstatus;

}

如果你的硬件部分没有问题,重新编译内核烧写电脑就应该可以识别了。但是2416的配制部分还有一点问题:

在s3c-hsudc.c中找到S3C2443_PHYCTRL

然后直接给他写成0

writel(0, S3C2443_PHYCTRL);

这样修改后还是不成功,你可以下载一个Bus Hound的软件进行分析,这里我就不具体说了。

然后在f_hid.c中找到:static int hidg_setup(struct usb_function *f,const struct usb_ctrlrequest *ctrl)函数,文件第324行

在变量声明后加入如下代码(第338行加入)::

if(((ctrl->bRequestType 

| HID_REQ_SET_REPORT)) {

mdelay(1);

}

else

udelay(100);

这里把set_report的时间间隔改成100us即你用2416给pc发数据的时间间隔,其它的设置成1ms,这里不一定要是1ms你可以改小一点。

修改完成后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)函数中添加如下代码:

//add by hclydao

status = kfifo_alloc(&recv_fifo, RECV_LEN, GFP_KERNEL);

if(status) {

printk(KERN_ERR "+++++++++++ error kfifo_alloc\n");

gotofail;

}

在static void hidg_unbind(struct usb_configuration *c, struct usb_function *f)中添加:

kfifo_free(&recv_fifo);

在static void hidg_set_report_complete(struct usb_ep *ep, struct usb_request *req)函数中添加:

kfifo_in(&recv_fifo, req->buf, req->actual);位置如下:

hidg->set_report_length = req->actual;

memcpy(hidg->set_report_buff, req->buf, req->actual);

//hclydao

//mutex_lock(&recv_lock);

kfifo_in(&recv_fifo, req->buf, req->actual);

//mutex_unlock(&recv_lock);

spin_unlock(&hidg->spinlock);

然后在static ssize_t f_hidg_read(struct file *file, char __user *buffer,

size_t count, loff_t *ptr)函数中将

#define READ_COND (hidg->set_report_buff != NULL)这个定义修改为:

#define READ_COND (!kfifo_is_empty(&recv_fifo))

同时增加如下代码,首先增加声明:

unsigned int copied;

if(tmp_buff != NULL) {

/* copy to user outside spinlock */

count -= copy_to_user(buffer, tmp_buff, count);

kfree(tmp_buff);

} else

count = -ENOMEM;

//add by hclydao

//mutex_lock(&recv_lock);

count = kfifo_to_user(&recv_fifo,buffer,hidg->set_report_length,&copied);

//mutex_unlock(&recv_lock);

//count = kfifo_out(&recv_fifo,buffer,count,&copied);

//return count;

returncopied;

//add by hclydao后面是增加的代码.

修改完成后重新编译。

经过测试传送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的处理:

case((USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE) <

| USB_REQ_GET_INTERFACE):

VDBG(cdev, "get_interface\n");

/* send an empty report */

length = min_t(unsigned, length, hidg->report_length);

memset(req->buf, 0x0, length);

gotorespond;

break;

当host发get_interface的时候,我们发个空包回去。

这样问题就解决了

2M文件上传不成功,经过加打印信息,发现是PC端程序的问题,但是我这改不了,所以没办法验证

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值