历史上的今天

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

正在发生

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

发布者:码梦创想 来源: eefocus关键字:Tiny210驱动  USB设备  驱动程序 手机看文章 扫描二维码
随时随地手机看文章

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");


关键字:Tiny210驱动  USB设备  驱动程序 引用地址:Tiny210驱动之USB设备驱动程序

上一篇:Tiny210块设备驱动之内存模拟磁盘
下一篇:Tiny210触摸屏之一线触摸屏驱动

推荐阅读

Teledyne Technologies旗下的全球成像解决方案技术创新公司, Teledyne e2v宣布针对光学检测和工厂自动化开发的Emerald 12M和16M CMOS成像传感器现已开始量产並接受大量采购。 这两款传感器隶属于Teledyne e2v公司的Emerald产品系列,采用小规格的 2.8μm 低噪全局快门像素,使用特种制造全球领军企业TowerJazz (TSEM)的110纳米晶圆制造工艺...
笔者读了本版有关PIC 8位单片机的产品性能和相应的封装引脚介绍后,认为对初学者而言还需了解各引脚符号的意义,才能进一步学习和使用它。笔者为此作相关的说明,以便和初学者共同提高。  一、关于I/O口符号 PIC单片机系列封装引脚最少的是8引脚(如PIC12C5XX和PIC12C6XX),多的可达84引脚(如PIC17C76X),其中I/O(输入/输出)口线按PIC单片机产品...
10月12日,华润微发布投资者关系活动记录,回答了投资者关心的IGBT、MOSFET、MCU等方面业务问题。对于投资者提出的“华润微的IGBT是否在自己的产线上生产,是在六吋还是八吋线上生产?”华润微表示,公司IGBT芯片研发和生产是独立自主的,目前主要是六吋产线,正逐步往8吋产线转移。同时,华润微在代工业务与自有产品业务之间有严密的防火墙设置,切实保护...
数字源表作为电学测量的常用仪器,在高校和研究所的相关实验室内几乎都能看到,源表集多表合一、配上专业软件可以实现各种定制测量,使用非常广泛。数字源表的型号很多,以泰克旗下KEITHLEY品牌为例,就有以下几大源表产品线:吉时利源表热销型号KEITHLEY2400系列:型号有:2400、2450、2401、2410、2420、2440、2460、2461、2470等,广泛用于通信、半导体...

史海拾趣

小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved