嵌入式linux驱动-usb实现按键功能笔记
2017-04-12 20:37
1291 查看
一、开发环境
1、内核:Linux 2.6.22.6;
2、JZ2440v3
3、ubuntu 9.10
二、过程
三、程序
#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 *usb_key_dev;
static char *usb_buff;
static dma_addr_t usb_buff_phys;
static int len;
static struct urb *usb_key_urb;
static void usbmouse_as_key_irq(struct urb *urb)
{
int i;
static int cnt;
static unsigned char pre_val;
/*
printk("data cnt = %d ",++cnt);
for(i=0;i<len;i++)
{
printk("%02x ",usb_buff[i]);
}
printk("\n");
*/
//usb_buff[0]:bit0 左键1-按下0-松开
//bit1 右键1-按下0-松开
//bit2 中键1-按下0-松开
if ((pre_val & (1<<0)) != (usb_buff[0] & (1<<0)))
{
/* 左键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_L, (usb_buff[0] & (1<<0)) ? 1 : 0);
input_sync(usb_key_dev);
}
if ((pre_val & (1<<1)) != (usb_buff[0] & (1<<1)))
{
/* 右键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_S, (usb_buff[0] & (1<<1)) ? 1 : 0);
input_sync(usb_key_dev);
}
if ((pre_val & (1<<2)) != (usb_buff[0] & (1<<2)))
{
/* 中键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_ENTER, (usb_buff[0] & (1<<2)) ? 1 : 0);
input_sync(usb_key_dev);
}
pre_val = usb_buff[0];
//重新提交urb
usb_submit_urb(usb_key_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;
/*printk("fount usb mouse \n");
printk("bcdUSB = %x \n",dev->descriptor.bcdUSB);
printk("VID = %x \n",dev->descriptor.idVendor);
printk("PID = %x \n",dev->descriptor.idProduct);*/
interface = intf->cur_altsetting;
endpoint = &interface->endpoint[0].desc;
//1.分配一个input_dev 结构体
usb_key_dev=input_allocate_device();
//2.设置
//2.1能长生那类事件
set_bit(EV_KEY,usb_key_dev->evbit);
set_bit(EV_REP,usb_key_dev->evbit);
//2.2 能产生这类事件的那些事件
set_bit(KEY_L,usb_key_dev->keybit);
set_bit(KEY_S,usb_key_dev->keybit);
set_bit(KEY_ENTER,usb_key_dev->keybit);
//3.注册
input_register_device(usb_key_dev);
//4.硬件相关的操作
//数据传输3要素:源、目的、长度
//源:usb设备的某个端点
pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress);
//长度:
len=endpoint->wMaxPacketSize;
//目的:
usb_buff = usb_buffer_alloc(dev, 8, GFP_ATOMIC, &usb_buff_phys);
//使用3要素
//分配usb request block
usb_key_urb=usb_alloc_urb(0, GFP_KERNEL);
//使用3要素设置urb
usb_fill_int_urb(usb_key_urb, dev, pipe, usb_buff,len,usbmouse_as_key_irq,NULL , endpoint->bInterval);
usb_key_urb->transfer_dma = usb_buff_phys;
usb_key_urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
//使用urb
usb_submit_urb(usb_key_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 usb mouse \n");
usb_kill_urb(usb_key_urb);
usb_free_urb(usb_key_urb);
usb_buffer_free(dev, len,usb_buff,usb_buff_phys);
input_unregister_device(usb_key_dev);
input_free_device(usb_key_dev);
}
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) },
{ } /* Terminating entry */
};
//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_driver
usb_register(&usbmouse_as_key_driver);
return 0;
}
static void usbmouse_as_key_exit(void)
{
//卸载usb_driver
usb_deregister(&usbmouse_as_key_driver);
}
module_init(usbmouse_as_key_init);
module_exit(usbmouse_as_key_exit);
MODULE_LICENSE("GPL");
1、内核:Linux 2.6.22.6;
2、JZ2440v3
3、ubuntu 9.10
二、过程
三、程序
#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 *usb_key_dev;
static char *usb_buff;
static dma_addr_t usb_buff_phys;
static int len;
static struct urb *usb_key_urb;
static void usbmouse_as_key_irq(struct urb *urb)
{
int i;
static int cnt;
static unsigned char pre_val;
/*
printk("data cnt = %d ",++cnt);
for(i=0;i<len;i++)
{
printk("%02x ",usb_buff[i]);
}
printk("\n");
*/
//usb_buff[0]:bit0 左键1-按下0-松开
//bit1 右键1-按下0-松开
//bit2 中键1-按下0-松开
if ((pre_val & (1<<0)) != (usb_buff[0] & (1<<0)))
{
/* 左键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_L, (usb_buff[0] & (1<<0)) ? 1 : 0);
input_sync(usb_key_dev);
}
if ((pre_val & (1<<1)) != (usb_buff[0] & (1<<1)))
{
/* 右键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_S, (usb_buff[0] & (1<<1)) ? 1 : 0);
input_sync(usb_key_dev);
}
if ((pre_val & (1<<2)) != (usb_buff[0] & (1<<2)))
{
/* 中键发生了变化 */
input_event(usb_key_dev, EV_KEY, KEY_ENTER, (usb_buff[0] & (1<<2)) ? 1 : 0);
input_sync(usb_key_dev);
}
pre_val = usb_buff[0];
//重新提交urb
usb_submit_urb(usb_key_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;
/*printk("fount usb mouse \n");
printk("bcdUSB = %x \n",dev->descriptor.bcdUSB);
printk("VID = %x \n",dev->descriptor.idVendor);
printk("PID = %x \n",dev->descriptor.idProduct);*/
interface = intf->cur_altsetting;
endpoint = &interface->endpoint[0].desc;
//1.分配一个input_dev 结构体
usb_key_dev=input_allocate_device();
//2.设置
//2.1能长生那类事件
set_bit(EV_KEY,usb_key_dev->evbit);
set_bit(EV_REP,usb_key_dev->evbit);
//2.2 能产生这类事件的那些事件
set_bit(KEY_L,usb_key_dev->keybit);
set_bit(KEY_S,usb_key_dev->keybit);
set_bit(KEY_ENTER,usb_key_dev->keybit);
//3.注册
input_register_device(usb_key_dev);
//4.硬件相关的操作
//数据传输3要素:源、目的、长度
//源:usb设备的某个端点
pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress);
//长度:
len=endpoint->wMaxPacketSize;
//目的:
usb_buff = usb_buffer_alloc(dev, 8, GFP_ATOMIC, &usb_buff_phys);
//使用3要素
//分配usb request block
usb_key_urb=usb_alloc_urb(0, GFP_KERNEL);
//使用3要素设置urb
usb_fill_int_urb(usb_key_urb, dev, pipe, usb_buff,len,usbmouse_as_key_irq,NULL , endpoint->bInterval);
usb_key_urb->transfer_dma = usb_buff_phys;
usb_key_urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
//使用urb
usb_submit_urb(usb_key_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 usb mouse \n");
usb_kill_urb(usb_key_urb);
usb_free_urb(usb_key_urb);
usb_buffer_free(dev, len,usb_buff,usb_buff_phys);
input_unregister_device(usb_key_dev);
input_free_device(usb_key_dev);
}
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) },
{ } /* Terminating entry */
};
//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_driver
usb_register(&usbmouse_as_key_driver);
return 0;
}
static void usbmouse_as_key_exit(void)
{
//卸载usb_driver
usb_deregister(&usbmouse_as_key_driver);
}
module_init(usbmouse_as_key_init);
module_exit(usbmouse_as_key_exit);
MODULE_LICENSE("GPL");
相关文章推荐
- 使用u-boot的USB下载功能烧写程序到Nand Flash ——韦东山嵌入式Linux学习笔记06
- 基于嵌入式linux的usb摄像头的驱动及采集程序的实现
- OK6410之LINUX2.6.39.2之USB-HOST,MMC/SD,GIPO-KEYS,I2C,USB-STORAGE功能实现
- 嵌入式LINUX驱动笔记1---创世纪
- 转: 嵌入式linux下usb驱动开发方法--看完少走弯路
- Linux环境下USB的原理、驱动和配置--本文由CSDN 特别约稿,作者为北京中科红旗软件技术有限公司 嵌入式工程师 梁国军
- 嵌入式Linux下PCI设备驱动的设计与实现
- 基于嵌入式Linux的MPC850&nbspUSB 驱动程序的实现
- linux设备驱动(四)--利用linux设备模型实现按键驱动
- USB学习总结3—USB gadget设备驱动实现(usb串口功能)
- 嵌入式Linux之我行——2440按键驱动实例开发详解(带去抖动)
- Linux环境下USB的原理、驱动和配置--本文由CSDN 特别约稿,作者为北京中科红旗软件技术有限公司 嵌入式工程师 梁国军
- 【嵌入式Linux学习七步曲之第五篇 Linux内核及驱动编程】Linux内核抢占实现机制分析
- linux驱动学习笔记:USB README
- 嵌入式Linux下高速USB主控制器的设计与实现
- 有限状态机的嵌入式Linux按键驱动设计
- 转 Linux环境下USB的原理、驱动和配置--本文由CSDN 特别约稿,作者为北京中科红旗软件技术有限公司 嵌入式工程师 梁国军
- 如何实现Linux下的U盘(USB Mass Storage)驱动
- 一个嵌入式Linux系统的键盘驱动实现
- 嵌入式Linux USB WIFI驱动的移植