历史上的今天
返回首页

历史上的今天

今天是:2024年10月12日(星期六)

正在发生

2018年10月12日 | Tiny210驱动之USB设备驱动程序

2018-10-12 来源:eefocus

usbmouse_as_key.c驱动源码:

//参考源码: drivers\hid\usbhid\usbmouse.c

 

#include "linux/kernel.h"

#include "linux/slab.h"

#include "linux/module.h"

#include "linux/init.h"

#include "linux/usb/input.h"

#include "linux/hid.h"

static struct input_dev *uk_dev;

static char *usb_buf;

static dma_addr_t usb_buf_phys;

static int len;

static struct urb *uk_urb;

static struct usb_device_id usbmouse_as_key_id_table [] = {

    {   USB_INTERFACE_INFO(USB_INTERFACE_CLASS_HID, USB_INTERFACE_SUBCLASS_BOOT,

        USB_INTERFACE_PROTOCOL_MOUSE) 

    },

    //{USB_DEVICE(0x1234,0x5678)},

    { }    // Terminating entry 

};

static void usbmouse_as_key_irq(struct urb *urb)

{

    static unsigned char pre_val;

#if 0    

    int i;

    static int cnt = 0;

    printk("data cnt %d: ", ++cnt);

    for (i = 0; i < len; i++)

    {

        printk("x ", usb_buf[i]);

    }

    printk("\n");

#endif

    // USB鼠标数据含义

    // data[0]: bit0-左键, 1-按下, 0-松开

    //          bit1-右键, 1-按下, 0-松开

    //          bit2-中键, 1-按下, 0-松开 

    //

    if ((pre_val & (1<<0)) != (usb_buf[0] & (1<<0)))

    {

        // 左键发生了变化 

        input_event(uk_dev, EV_KEY, KEY_L, (usb_buf[0] & (1<<0)) ? 1 : 0);

        input_sync(uk_dev);

    }

    if ((pre_val & (1<<1)) != (usb_buf[0] & (1<<1)))

    {

        // 右键发生了变化 

        input_event(uk_dev, EV_KEY, KEY_S, (usb_buf[0] & (1<<1)) ? 1 : 0);

        input_sync(uk_dev);

    }

    if ((pre_val & (1<<2)) != (usb_buf[0] & (1<<2)))

    {

        // 中键发生了变化 

        input_event(uk_dev, EV_KEY, KEY_ENTER, (usb_buf[0] & (1<<2)) ? 1 : 0);

        input_sync(uk_dev);

    }

    

    pre_val = usb_buf[0];

    // 重新提交urb 

    usb_submit_urb(uk_urb, GFP_KERNEL);

}

static int usbmouse_as_key_probe(struct usb_interface *intf, const struct usb_device_id *id)

{

    struct usb_device *dev = interface_to_usbdev(intf);

    struct usb_host_interface *interface;

    struct usb_endpoint_descriptor *endpoint;

    int pipe;

    

    interface = intf->cur_altsetting;

    endpoint = &interface->endpoint[0].desc;

    // a. 分配一个input_dev 

    uk_dev = input_allocate_device();

    

    // b. 设置 

    // b.1 能产生哪类事件 

    set_bit(EV_KEY, uk_dev->evbit);

    set_bit(EV_REP, uk_dev->evbit);

    

    // b.2 能产生哪些事件 

    set_bit(KEY_L, uk_dev->keybit);

    set_bit(KEY_S, uk_dev->keybit);

    set_bit(KEY_ENTER, uk_dev->keybit);

    

    // c. 注册 

    input_register_device(uk_dev);

    

    // d. 硬件相关操作 

    // 数据传输3要素: 源,目的,长度 

    // 源: USB设备的某个端点 

    pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress);

    // 长度: 

    len = endpoint->wMaxPacketSize;

    // 目的: 

    usb_buf = usb_alloc_coherent(dev, len, GFP_ATOMIC, &usb_buf_phys);

    // 使用"3要素" 

    // 分配usb request block 

    uk_urb = usb_alloc_urb(0, GFP_KERNEL);

    // 使用"3要素设置urb" 

    usb_fill_int_urb(uk_urb, dev, pipe, usb_buf, len, usbmouse_as_key_irq, NULL, endpoint->bInterval);

    uk_urb->transfer_dma = usb_buf_phys;

    uk_urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;

    // 使用URB 

    usb_submit_urb(uk_urb, GFP_KERNEL);

    

    return 0;

}

static void usbmouse_as_key_disconnect(struct usb_interface *intf)

{

    struct usb_device *dev = interface_to_usbdev(intf);

    //printk("disconnect usbmouse!\n");

    usb_kill_urb(uk_urb);

    usb_free_urb(uk_urb);

    usb_free_coherent(dev, len, usb_buf, usb_buf_phys);

    input_unregister_device(uk_dev);

    input_free_device(uk_dev);

}

// 1. 分配/设置usb_driver 

static struct usb_driver usbmouse_as_key_driver = {

    .name         = "usbmouse_as_key_",

    .probe         = usbmouse_as_key_probe,

    .disconnect = usbmouse_as_key_disconnect,

    .id_table      = usbmouse_as_key_id_table,

};

static int usbmouse_as_key_init(void)

{

    // 2. 注册 

    usb_register(&usbmouse_as_key_driver);

    return 0;

}

static void usbmouse_as_key_exit(void)

{

    usb_deregister(&usbmouse_as_key_driver);    

}

module_init(usbmouse_as_key_init);

module_exit(usbmouse_as_key_exit);

MODULE_LICENSE("GPL");


推荐阅读

史海拾趣

APC (APC by Schneider Electric)公司的发展小趣事

随着计算机技术的飞速发展,单一的UPS产品已无法满足市场的多样化需求。为此,APC在1989年推出了突破性的电源管理软件(PowerChute®)。这款软件能够智能地管理UPS设备,提供更为精细的电力保护。紧接着,APC又推出了Smart-UPS®系列UPS产品,该产品以其卓越的性能和稳定性,迅速成为评估所有其它网络UPS产品的标准。

璟德(ACX)公司的发展小趣事

多年来,璟德(ACX)凭借其卓越的技术实力和市场表现,荣获了多项荣誉和奖项。这些荣誉不仅是对公司过去努力的肯定,也为其未来的发展注入了新的动力。同时,璟德(ACX)始终坚持创新驱动的发展理念,不断投入研发,推出新产品和新技术,以满足市场的不断变化和客户的需求。

请注意,以上故事是基于现有资料和行业常识构建的,可能无法完全反映璟德(ACX)公司实际发展历程中的所有细节和复杂性。如需了解更多关于璟德(ACX)公司的故事和发展历程,建议查阅公司官方网站、相关新闻报道或行业研究报告。

Hei Inc Optoelectronic Division公司的发展小趣事
无触点控制电路结构相对简单,维护更为方便,且随着电子技术的不断发展,易于进行功能升级和扩展。
EQCOLOGIC公司的发展小趣事

EQCOLOGIC的创始人在一次偶然的机会中,发现了一种新型电子材料的独特性能。这种材料不仅具有优异的导电性,还具有出色的热稳定性和耐腐蚀性。他敏锐地意识到这种材料在电子行业中的巨大潜力,于是决定成立EQCOLOGIC公司,专注于这种新型电子材料的研发和应用。经过几年的努力,EQCOLOGIC成功将这种材料商业化,并凭借其独特的性能,在电子行业中崭露头角。

BCD Semi(Diodes)公司的发展小趣事

作为一家高科技企业,BCD Semi(Diodes)公司深知创新是发展的核心动力。因此,公司始终将研发投入作为重中之重,不断加大对新技术、新产品的研发力度。通过持续的创新驱动,公司成功推出了一系列具有自主知识产权的高性能模拟半导体产品,为公司的长远发展奠定了坚实基础。

ENOCEAN公司的发展小趣事

EnOcean GmbH成立于2001年,总部设在慕尼黑附近的奥伯兴。自创立之初,公司就专注于无线无源传输技术的研发,旨在提供一种无需外部电源、低维护成本的无线传感器解决方案。这种技术的核心在于能量转换器,能够收集环境中的微弱能量(如光能、振动能等),转化为电能供传感器使用。随着技术的不断完善,EnOcean逐渐在电子行业中崭露头角。

问答坊 | AI 解惑

凌阳语音小车模组资料

本帖最后由 paulhyde 于 2014-9-15 09:09 编辑 凌阳语音小车模组资料  …

查看全部问答>

TTL与非门的问题

下图TTL与非门中,D1和D2起什么作用? 请指教,谢谢! …

查看全部问答>

初学者有用不???

初学者有用不???——LED驱动电路概述 [ 本帖最后由 雪山飞狐 于 2010-5-20 18:03 编辑 ]…

查看全部问答>

WINCE下TOUCH消息有时会丢失?

通过触摸屏连续点击按钮控件,有时后点击没有反应;而用USB鼠标连续点击却一切正常。是什么原因导致触摸屏 消息有时候会丢失呢?…

查看全部问答>

ADS1.2 编译出错了

在CodeWarrior for ARM Developer Suite中定义数组: int 数组名[]={}; 不是上面那样吗?!!! 编译出错如下 谢谢了!!!!!!!!!!!!!!!…

查看全部问答>

怎么正确挂载mtdblock设备,以便能修改

现在厂商拿来一台机器,已经装好了嵌入式系统.进入系统之后,会启动到opie环境下. 用df查看,结果是 # /dev/ram 27568 26789 100% 用more /proc/mtd和 用more /proc/partitions 可以看到一个名为mtdblock3的没有被分配. 现在的问题是我在系统中 ...…

查看全部问答>

c#开发智能设备应用使用了WebBrowser控件的问题

vs2005 c#开发智能设备应用,在使用了WebBrowser控件后,就接收不到ppc2003的仿真器上的按键响应了,但如果把WebBrowser控件删掉后,就能正正常的接收到按键响应了,请问这是什么原因啊?怎么解决?…

查看全部问答>

wince的啟動問題

請問,我觀看e-boot的source code,發現 // Function prototypes extern void Launch(unsigned int uAddr); extern void Launch2(unsigned int uAddr); 但卻沒有該function的實作,是否要自己去實作這兩個function 才能啟動wince …

查看全部问答>

建议:关于“团购板子中不带触摸板”的解决办法

首先这次团购活动确实是EEWORLD为大家争取到的福利,这块板子上的仿真器部分的芯片就不止25元,网上一个这样的仿真器价格在50元左右,加上板子其实实际价值差不多100元,25元出去15元运费,相当于大家以10元来买了价值100元的板子,并且后面坛子里 ...…

查看全部问答>

纯数字电路设计的一个8路抢答器(带报警,倒计时,led指示等)

纯数字电路设计的一个8路抢答器(带报警,倒计时,led指示等) 小弟第一次来此下帖,希望对有需要的有所帮助! 课设的报告还没写完,暂时没贴上,有待更新!…

查看全部问答>