输入子系统由驱动层、输入子系统核心、事件处理层三部分组成。一个输入事件,如鼠标移动、键盘按下等通过Driver->Inputcore->Event handler->userspace的顺序到达用户控件的应用程序。

其中核心层提供一些设备层与事件层公用的函数,比如说注册函数、反注册函数、事件到来的处理函数等等;事件层其实在Linux内核里面已经帮我们写好了很多有关的事件;而设备层就跟我们新添加到输入系统的具体设备相关了。这里以JZ2440开发板上的4个按键作为输入子系统的按键:它定义的功能分别为:KEY_L、KEY_S、KEY_ENTER、KEY_LEFTSHIFT。这几个值是在include\linux\input.h中被定义的。

button_drv.c文件:

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <asm/io.h> 
#include <asm/uaccess.h>
#include <linux/device.h>/
#include <asm/arch/regs-gpio.h>
#include <linux/irq.h>
#include <asm-arm/irq.h>
#include <linux/interrupt.h>
#include <linux/poll.h>
#include <asm-generic/errno-base.h>
#include <linux/input.h>

struct pin_desc
{
  char * name;            //名称
  unsigned int pin;       //管脚定义
  unsigned int irq;       //中断号
  unsigned int key_val;   //按键值
};

static struct pin_desc pins_desc[4] = 
{
  {"S1",S3C2410_GPF0,IRQ_EINT0,KEY_L},
  {"S2",S3C2410_GPF2,IRQ_EINT2,KEY_S},
  {"S3",S3C2410_GPG3,IRQ_EINT11,KEY_ENTER},
  {"S4",S3C2410_GPG11,IRQ_EINT19,KEY_LEFTSHIFT}
};

static struct pin_desc *pinss_desc=NULL;

static struct timer_list timer_button;                //新建一个定时器

static struct input_dev *buttons_input;               //新建一个输入子系统的设备层结构

static irqreturn_t buttons_irq(int irq, void *dev_id)
{
  pinss_desc = (struct pin_desc *)dev_id;            //取得哪个按键被按下的状态
  mod_timer(&inputbuttons_timer, jiffies+HZ/100);    //10ms之后调用定时器处理函数
  return IRQ_HANDLED;
}

static void timer_button_timeout(unsigned long a)
{
  unsigned int pin_val;

  if(pinss_desc==NULL)
    return;
  else
  {
    pin_val = s3c2410_gpio_getpin(pinss_desc->pin);

    if(pin_val)
    {
      input_event(buttons_input,EV_KEY, pinss_desc->key_val, 0);
    }
    else
    {
      input_event(buttons_input,EV_KEY, pinss_desc->key_val, 1);
    }
  }
}

static int button_drv_init(void)
{
  unsigned char i;
  int ret;
  buttons_input = input_allocate_device();        //分配input_dev结构体
  if (!buttons_input)
    return -ENOMEM;

  set_bit(EV_KEY, buttons_input->evbit);          //按键事件
  set_bit(EV_REP, buttons_input->evbit);          //重复事件类型
  set_bit(KEY_L, buttons_input->keybit);
  set_bit(KEY_S, buttons_input->keybit);
  set_bit(KEY_ENTER, buttons_input->keybit);
  set_bit(KEY_LEFTSHIFT, buttons_input->keybit);

  input_register_device(buttons_input);           //注册设备驱动

  init_timer(&timer_button);
  inputbuttons_timer.expires = 0;
  inputbuttons_timer.function = timer_button_timeout;
  add_timer(&timer_button);
                                                                                                                                     //注意这里没有创建设备类和设备文件(设备节点)

                                                              //因为输入子系统的设备文件本来就存在为/dev/event0
  for(i=0;i<4;i++)
  {
    ret = request_irq(pins_desc[i].irq, buttons_irq, IRQT_BOTHEDGE, pins_desc[i].name, (void * )&pins_desc[i]);
    if(ret)
    {
      printk("open failed %d\n",i);
      return -(i+1);
    }
  }
  return 0;
}

static void button_drv_exit(void)
{
  unsigned char i;

  input_unregister_device(buttons_input);
  input_free_device(buttons_input);
  del_timer(&timer_button);

  for(i=0;i<4;i++)
  {
    free_irq(pins_desc[i].irq, (void * )&pins_desc[i]);
  }
}

module_init(seven_drv_init);
module_exit(seven_drv_exit);

MODULE_LICENSE("GPL");

Makefile文件:

obj-m += button_drv.o

KERN_DIR = /work/system/linux-2.6.22.6

all:
make -C $(KERN_DIR) M=`pwd` modules 
clean:
rm -rf *.o *.ko *.order *.symvers *.mod.c

button_app.c文件:

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

static int fd;

int main(int argc, char **argv)
{
  char* filename = argv[1];
  int oflags, ret=0;
  unsigned char key_val[16];

  fd = open(filename, O_RDWR);//打开设备文件,阻塞方式打开
  if (fd < 0)
  {
    printf("error, can't open %s\n", filename);
    return 0;
  }
  while(1)
  {
    ret = read(fd, key_val, 16);
    printf("ret: %d, code: %02d, value: %d \n", ret, key_val[10], key_val[12]);
  }
  return 0;
}

app_read->evdev_read->kbtab_irq->input_report_key->input_event->evdev_event->evdev_read

应用层      事件层        设备层        核心层               核心层          事件层          事件层

应用函数最终会通过事件层的evdev_event调用evdev_read

而在linux中evdev_event结构体共16字节,所以app中的read函数也应该读取16字节

编译生成button_drv.ko和button_app文件,运行./button_app /dev/event0

Linux 驱动——Button8(输入子系统)的更多相关文章

  1. linux驱动模型<输入子系统>

    在linux中提供一种输入子系统的驱动模型,其主要是实现在input.c中. 在输入子系统这套模型中,他把驱动分层分类.首先分为上下两层,上层为input.c .下层为驱动的实现,下层分为两部分,一部 ...

  2. Linux驱动之输入子系统简析

    输入子系统由驱动层.输入子系统核心.事件处理层三部分组成.一个输入事件,如鼠标移动.键盘按下等通过Driver->Inputcore->Event handler->userspac ...

  3. Linux 驱动框架---input子系统框架

    前面从具体(Linux 驱动框架---input子系统)的工作过程学习了Linux的input子系统相关的架构知识,但是前面的学习比较实际缺少总结,所以今天就来总结一下输入子系统的架构分层,站到远处来 ...

  4. Linux 驱动框架---input子系统

    input 子系统也是作为内核的一个字符设备模块存在的,所以他也是字符设备自然也会有字符设备的文件接口.input子系统的注册过程主要分为两步,先注册了一个input class然后再注册一个字符设备 ...

  5. ARM Linux内核Input输入子系统浅解

    --以触摸屏驱动为例 第一章.了解linux input子系统         Linux输入设备总类繁杂,常见的包括有按键.键盘.触摸屏.鼠标.摇杆等等,他们本身就是字符设备,而linux内核将这些 ...

  6. Linux驱动之一个简单的输入子系统程序编写

    的在Linux驱动之输入子系统简析已经分析过了输入子系统的构成,它是由设备层.核心层.事件层共同组成的.其中核心层提供一些设备层与事件层公用的函数,比如说注册函数.反注册函数.事件到来的处理函数等等: ...

  7. Linux输入子系统 转载

    NQian 记录成长~ 首页 新随笔 联系 订阅 管理 随笔 - 305  文章 - 0  评论 - 254 12.Linux之输入子系统分析(详解)   在此节之前,我们学的都是简单的字符驱动,涉及 ...

  8. Linux驱动之触摸屏程序编写

    本篇博客分以下几部分讲解 1.介绍电阻式触摸屏的原理 2.介绍触摸屏驱动的框架(输入子系统) 3.介绍程序用到的结构体 4.介绍程序用到的函数 5.编写程序 6.测试程序 1.介绍电阻式触摸屏的原理 ...

  9. linux 输入子系统之电阻式触摸屏驱动

    一.输入子系统情景回忆ING...... 在Linux中,输入子系统是由输入子系统设备驱动层.输入子系统核心层(Input Core)和输入子系统事件处理层(Event Handler)组成.其中设备 ...

随机推荐

  1. c#+Unity3D

    Windows系统unity3d中的快捷键 自己整理常用; f2 改名 shift + 三维轴中间:45度 点击左键时 F:聚焦 V顶点贴合 ws拉近拉远 ad左右 qe 上下 长按右键 自由调换视角 ...

  2. Vant-Weapp小程序+商城案例

    功能还在进一步完善中,欢迎扫一扫提出宝贵意见! 详细信息可进群沟通:

  3. Windows10 下安装 Apache2.4+PHP7.1+MySQL5.7

    这个教程主要是分享如何快速组建WAMP开发环境,对于软件的详细配置,自行参考文档或搜索. Visual C++ Redistributable for Visual Studio 2015 下载地址: ...

  4. Treap标准模板

    这是Treap的模板程序,支持Left/Right Rotate,Find the maxnum/minnum,Find the predecessor/successor of a node,Add ...

  5. Linux内存管理(一)

    Linux内存管理之一:基本概念篇 物理地址.线性地址(虚拟地址)和逻辑地址:阐述段式管理和页式管理基本概念:Linux操作系统内存管理和虚拟内存概念:为内核开发做一个基础铺垫. 内存是linux内核 ...

  6. fflush 和 fsync 的区别

    int fflush(FILE *stream);If stream points to an output stream or an update stream in which the most ...

  7. A context-aware personalized travel recommendation system based on geotagged social media data mining

    文章简介:利用社交网站Flickr上照片的geotag信息将这些照片聚类发现城市里的旅游景点,通过各照片的拍照时间得到用户访问某景点时的时间上下文和天气上下文(利用时间和public API of W ...

  8. linux-centos系统下安装python3.5.4步骤

    查看当前python版本:python -V 查看Python可执行文件位置:which python [root@localhost bin]# which python/usr/bin/pytho ...

  9. java8中optional和.stream().map()

    使用optional的好处:是一个可以包含或不可以包含非空值的容器对象,更加友好的处理程序中的空对象. Optional<T>有方法 isPresent() 和 get() 是用来检查其包 ...

  10. 第一章 Python基本语法

    寒假不能荒废,终于静下心来认真地开始学习Python,在这里与大家分享一下所学知识,希望能对像我这样的小白有所帮助,如有错误之处,谢大佬不吝赐教!!    编程语言包括机器语言.汇编语言.高级语言.超 ...