您的位置:首页 > 运维架构 > Linux

Linux驱动之输入子系统

2016-06-19 23:07 495 查看

一. 概述

1. 输入子系统的概念

输入子系统是对分散的、多种不同类别的输入设备(键盘、鼠标、触摸屏、加速计、跟踪球、操纵杆等)进行统一处理的驱动程序。


2. 输入子系统的优点

抽象底层形态各异的硬件(鼠标,键盘,触摸屏,游戏杆等)输入设备,为上层提供了统一的操作接口。


二. 输入子系统的分层结构

1.三层结构

事件驱动层:负责和应用程序的接口。
核心层:   提供事件驱动层和设备驱动层所需的函数接口
设备驱动层:负责和底层设备驱动通信


2.层次框架图



三. 设备驱动

1. 设备驱动的数据结构

对于系统的每个输入设备硬件,在输入子系统都要实现一个设备驱动。每个设备驱动都由input_dev的数据结构描述,部分成员定义如:


struct input_dev{
const char *name;
const char *phys;
const char *uniq;
struct input_id id;
unsigned longevbit[BITS_TO_LONGS(EV_CNT)];
unsigned longkeybit[BITS_TO_LONGS(KEY_CNT)];
}


name: 该成员的设备驱动名字。但与对于的设备文件名无任何关系。

id : 该成员的输入设备身份。

evbit :该成员会产生的输入事件类型。事件类型定义如下:

1.  #define EV_SYN          0x00    /*表示设备支持所有的事件*/
2.  #define EV_KEY          0x01    /*键盘或者按键,表示一个键码*/
3.  #define EV_REL          0x02    /*鼠标设备,表示一个相对的光标位置结果*/
4.  #define EV_ABS          0x03    /*手写板产生的值,其是一个绝对整数值*/
5.  #define EV_MSC          0x04    /*其他类型*/
6.  #define EV_LED          0x11    /*LED灯设备*/
7.  #define EV_SND          0x12    /*蜂鸣器,输入声音*/
8.  #define EV_REP          0x14    /*允许重复按键类型*/
9.  #define EV_PWR          0x16    /*电源管理事件*/


把指定的位置1,可以使用set_bit()函数:

set_bit( EV_KEY, input_dev->evbit);


keybit : 当设备驱动可以产生按键事件时,keybit成员表示设备驱动支持按键的键值。键值的定义:

#define     KEY_RESERVED    0
#define     KEY_ESC         1
#define     KEY_1           2
#define     KEY_2           3
......


2. 注册/注销设备驱动

当一个input_dev结构体被初始化完成后,就可以调用input_register_device()函数注册到输入子系统:


int input_register_device(struct input_dev *);


注销:

void input_unregister_device(struct input_dev *);


3. 报告按键值

void input_report_key(struct input_dev*dev,unsigned int code,int value);


value : 1为按下,0为提起;

4. 提交同步事件,防止数据混乱

void input_sync(struct input_dev *dev);


四. 驱动实现

1. 驱动函数event_drv.c

/*
*   event_drv.c
*
*   Program:   Key input event
*
*   Author:  Lin Xubin
*/

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/irq.h>
#include <asm/uaccess.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>
#include <linux/input.h>
#include <linux/interrupt.h>

struct input_dev *key_event_dev;

typedef struct
{
int irq;
char *devname;
unsigned int gpio_pin;
unsigned int key_val;
}KeyEvent_TypeDef;

KeyEvent_TypeDef KeyEvent[4]={
{IRQ_EINT0,  "S2", S3C2410_GPF0,  KEY_A},
{IRQ_EINT2,  "S3", S3C2410_GPF2,  KEY_B},
{IRQ_EINT11, "S4", S3C2410_GPG3,  KEY_C},
{IRQ_EINT19, "S5", S3C2410_GPG11, KEY_D},
};

static irqreturn_t key_drv_irq(int irq,void *dev_id)
{
KeyEvent_TypeDef *KeyPin = (KeyEvent_TypeDef *)dev_id;
static int PinState;
PinState = s3c2410_gpio_getpin(KeyPin->gpio_pin);

if(PinState)
{
input_event(key_event_dev, EV_KEY, KeyPin->key_val, 1);
input_sync(key_event_dev);
}
else
{
input_event(key_event_dev, EV_KEY, KeyPin->key_val, 0);
input_sync(key_event_dev);
}
return IRQ_RETVAL(IRQ_HANDLED);
}

int key_drv_init(void)
{
int i;
key_event_dev = input_allocate_device();  //分配input_dev结构体

set_bit(EV_KEY, key_event_dev->evbit);    //按键事件

for(i=0; i<4; i++)
{
set_bit(KeyEvent[i].key_val , key_event_dev->keybit);
}
input_register_device(key_event_dev);   //注册input事件

for(i=0; i<4; i++)
{
request_irq(KeyEvent[i].irq , key_drv_irq, IRQT_BOTHEDGE, KeyEvent[i].devname, &KeyEvent[i]);
}
return 0;
}

void key_drv_exit(void)
{
int i;
for(i=0; i<4; i++)
{
free_irq(KeyEvent[i].irq, &KeyEvent[i]);
}
input_unregister_device(key_event_dev);
input_free_device(key_event_dev);
}

module_init(key_drv_init);
module_exit(key_drv_exit);
MODULE_LICENSE("GPL");


2. 应用测试程序 event_drv_test.c

/*
*   event_drv_test.c
*
*   Author: Lin Xubin
*/

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <poll.h>
#include <signal.h>
#include <unistd.h>
#include <linux/input.h>

int main(char argc, char *argv[])
{
int fd,val;
struct input_event key_event_val;
fd = open("/dev/event1", O_RDWR);
if(fd < 0)
{
printf("open device failed\n");
return 0;
}
while(1)
{
val = read(fd,&key_event_val, sizeof(struct input_event));
if( val < 0)
{
printf("read input err\n");
return 0;
}
switch(val)
printf("%d,%d,%d\n", key_event_val.type, key_event_val.code, key_event_val.value);
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息