1. inotify和epoll

怎么监测键盘接入与拔出?

(1)hotplug机制:内核发现键盘接入/拔出==>启动hotplug进程==>发消息给输入系统

(2)inotify机制:输入系统使用inotify来监测目录/dev/input

android使用inofity机制

当插入多个键盘时,系统怎么知道哪个键盘被按下?

android下使用epoll,可以同时监控多个文件,当文件发生改变,其会知道谁变化了

参考代码:
frameworks\native\services\inputflinger\EventHub.cpp

参考文章:
《深入理解Android 卷III》第五章 深入理解Android输入系统 
http://blog.csdn.net/innost/article/details/47660387

inotify的使用(监测目录或者文件的变化)

(1)fd = inotify_init()

(2)inotify_add_watch(目录名字/文件名字,创建/删除)

(3)read(fd),平时目录和文件没有创建或者删除时,会休眠,发生变化后read返回多个inotify_event结构体

inotify_event.name保存了名字,inotify_event.len表示名字的长度,inotify_event.mask表示发生了说明变化(创建还是删除)

inotify.c编写(Usage:inotify <dir> 这个目录下发生的变化)

#include <unistd.h>

#include <stdio.h>

#include <sys/inotify.h>

#include <string.h>

#include <errno.h>

int read_process_inotify_fd(int fd){

  int res;

  char event_buf[512];

  int event_size;

  int event_pos = 0;

  struct inotify_event *event;

  res = read(fd,event_buf,sizeof(event_buf));

  if(res < (int)sizeof(*event)){

    if(errno == EINTR)

      return 0;

    printf("could not get event ,%s\n",strerror(errno));

    return -1;

  }

  //处理数据,读到的数据是一个或多个inotify_event,他们len不一样,逐个处理

  while(res >= (int)sizeof(*event)){

    event = (struct inotify_event *)(event_buf+event_pos);

    if(event->len){

      if(event->mask & IN_CREATE){

          printf("create file : %s\n",event->name);

      }else{

          printf("delete file : %s\n",event->name);

      }

    }

    event_size = sizeof(*event)+event->len;

    res -= event_size;

    event_pos += event_size;

  }

  return 0;

}

int main(int argc,char **argv)

{

  int mINotifyFd;

  int result;

  if(argc != 2)

  {

    printf("Usage:%s <dir>\n",argv[0]);

    return -1;

  }

  mINotifyFd = inotify_init(argv[0]);

  result = inotify_add_watch(mINotifyFd,argv[1],IN_DELETE | IN_CREATE);

  while(1)

  {

    read_process_inotify_fd(mINotifyFd);

  }

  return 0;

}
gcc -o inotify inotify.c
mkdir tmp
./inotify tmp &

echo > tmp/1
echo > tmp/2
rm tmp/1 tmp/2

epoll用来检测多个文件有无数据供读出、有无空间供写入

(1)epoll_create//创建fd

(2)对每个文件执行epoll_ctl(......,EPOLL_CTL_ADD,) 表示要监测它

(3)epoll_wait//等待某个文件可用

(4)不在想监测某文件可用执行epoll_ctl(......,EPOLL_CTL_DEL,)

epoll , fifo :
http://stackoverflow.com/questions/15055065/o-rdwr-on-named-pipes-with-poll

使用fifo是, 我们的epoll程序是reader
echo aa > tmp/1 是writer
a.
如果reader以 O_RDONLY|O_NONBLOCK打开FIFO文件,
当writer写入数据时, epoll_wait会立刻返回;
当writer关闭FIFO之后, reader再次调用epoll_wait, 它也会立刻返回(原因是EPPLLHUP, 描述符被挂断)
b.
如果reader以 O_RDWR打开FIFO文件
当writer写入数据时, epoll_wait会立刻返回;
当writer关闭FIFO之后, reader再次调用epoll_wait, 它并不会立刻返回, 而是继续等待有数据

epoll.c

/*Usage:epoll <file1> [file2] [file3]*/

#include <sys/epoll.h>

#include <unistd.h>

#include <stdio.h>

#include <sys/types.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <string.h>

#define DATA_MAX_LEN 500

int add_to_epoll(int fd,int epollFd)

{

  int result;

  struct epoll_event eventItem;

  memset(&eventItem,0,sizeof(eventItem));

  eventItem.events = EPOLLIN;//表示监测其有数据

  eventItem.data.fd=fd;

  result = epoll_ctl(epollFd,EPOLL_CTL_ADD,fd,&eventItem);

  return result;

}

void rm_from_epoll(int fd,int epollFd)

{

  result = epoll_ctl(epollFd,EPOLL_CTL_DEL,fd,NULL);

}

int main(int argc,char **argv)

{

  int mEpollFd;

  int i;

  char buf[DATA_MAX_LEN];

  static const int EPOLL_MAX_EVENTS = 16;//epoll_wait一次最大监测事件数

  struct epoll_event mPendingEventItems[EPOLL_MAX_EVENTS];

  if(argc < 2)

  {

    printf("Usage:%s<file1> [file2] [file3] \n",argv[0]);

    return -1;

  }

  mEpollFd = epoll_create(8);

  /*for each file:open it /add it to epoll*/

  for(i = 1;i < argc;i++)

  {

    int tmpFd = open(argv[i],O_RDWR);

    add_to_epoll(tmpFd,mEpollFd);

  }

  /*epoll_wait*/

  while(1){

    int pollResult = epoll_wait(mEpollFd ,mPendingEventItems,EPOLL_MAX_EVENTS ,-1);//-1表示永远监测不退出

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

    {

      int len =read(mPendingEventItems[i].data.fd,buf,DATA_MAX_LEN);

      buf[len] = '\0';

      printf("get data:%s\n",buf);

    }

  }

  return 0;

}

gcc -o epoll epoll.c
mkdir tmp
mkfifo tmp/1 tmp/2 tmp/3
./epoll tmp/1 tmp/2 tmp/3 &
echo aaa > tmp/1
echo bbb > tmp/2

课后作业:
编写 inotify_epoll.c, 用它来监测tmp/目录: 有文件被创建/删除, 有文件可读出数据
a. 当在tmp/下创建文件时, 会立刻监测到,并且使用epoll监测该文件
b. 当文件有数据时,读出数据
c. 当tmp/下文件被删除时,会立刻监测到,并且把它从epoll中移除不再监测

inotify_epoll.c
gcc -o inotify_epoll inotify_epoll.c
mkdir tmp
./inotify_epoll tmp/ &
mkfifo tmp/1 tmp/2 tmp/3
echo aaa > tmp/1
echo bbb > tmp/2
rm tmp/3

10.1、android输入系统_必备Linux编程知识_inotify和epoll的更多相关文章

  1. 10.2、android输入系统_必备Linux编程知识_双向通信(scoketpair)

    2. 双向通信(socketpair) 输入系统肯定涉及进程通讯:进程A读取/分发输入事件,APP处理输入事件,进程A给APP发送输入事件,APP处理完事件回复信息给进程A,APP关闭的时候也要发信息 ...

  2. 10.3、android输入系统_必备Linux编程知识_任意进程双向通信(scoketpair+binder)

    3. 任意进程间通信(socketpair_binder) 进程每执行一次open打开文件,都会在内核中有一个file结构体表示它: 对每一个进程在内核中都会有一个task_struct表示进程,这个 ...

  3. 10.11 android输入系统_补充知识_activity_window_decor_view关系

    android里:1个application, 有1个或多个activity(比如支付宝有:首页.财富.口碑.朋友.我的,这些就是activity)1个activity, 有1个window(每个ac ...

  4. 10.4 android输入系统_框架、编写一个万能模拟输入驱动程序、reader/dispatcher线程启动过程源码分析

    1. 输入系统框架 android输入系统官方文档 // 需FQhttp://source.android.com/devices/input/index.html <深入理解Android 卷 ...

  5. 10.13 android输入系统_多点触摸驱动理论与框架

    1.多点触摸驱动理论 驱动程序仅上报多个触点的位置就可以,是放大还是缩小由应用程序控制 对于多点触摸驱动在linux系统中有个输入子系统,其已经实现了open/read/write等接口 我们只需要实 ...

  6. 10.14 android输入系统_多点触摸驱动测试及Reader线程、InputStage分析

    21. 多点触摸_电容屏驱动程序_实践_tiny4412 tiny4412触摸屏: 分辨率为800 x 480http://wiki.friendlyarm.com/wiki/index.php/LC ...

  7. 10.8 android输入系统_实战_使用GlobalKey一键启动程序

    11. 实战_使用GlobalKey一键启动程序参考文章:Android 两种注册(动态注册和静态注册).发送广播的区别http://www.jianshu.com/p/ea5e233d9f43 [A ...

  8. 10.9 android输入系统_APP跟输入系统建立联系和Dispatcher线程_分发dispatch

    12. 输入系统_APP跟输入系统建立联系_InputChannel和Connection核心: socketpair // 第9课第3节_输入系统_必备Linux编程知识_任意进程双向通信(scok ...

  9. 10.5 android输入系统_Reader线程_使用EventHub读取事件和核心类及配置文件_实验_分析

    4. Reader线程_使用EventHub读取事件 使用inotify监测/dev/input下文件的创建和删除 使用epoll监测有无数据上报 细节: a.fd1 = inotify_init(& ...

随机推荐

  1. 给Linux添加新用户,新建用户,新建帐号

    给Linux添加新用户,新建用户,新建帐号 添加用户组 sudo groupadd groupname 添加用户 sudo useradd username -m -s /sbin/nologin - ...

  2. php实现简单算法3

    php实现简单算法3 这篇文章主要介绍了PHP经典算法集锦,整理了各种常见的算法,包括排序.查找.遍历.运算等各种常见算法原理与实现技巧,需要的朋友可以参考下 1.首先来画个菱形玩玩,很多人学C时在书 ...

  3. 41.内存函数实现(memcpy,memset,memmove,memicmp,memchr.memccpy)

    memcpy #include <stdio.h> #include <stdlib.h> #include <memory.h> void * mymemcpy( ...

  4. 【基础篇】点击Button按钮更换图片

    我们在开发的过程中,往往为了美化界面的需要,会修改按钮的默认外观,而因为Android中的按钮有三种状态—默认,被点击,被选中.所以,如果要改变按钮的外观,需要对这三种情况都做出修改,也许在以往,我们 ...

  5. Python Web框架要点

    1. Web应用程序处理流程 前端客户端(浏览器.APP.ajax.爬虫程序)>>>--http的请求与响应--<<<服务器程序(Gunicorn.uwsig)&l ...

  6. csdn课堂学习

    http://edu.csdn.net/course/detail/2495?ref=blog&loc=0 http://edu.csdn.net/course/detail/2140/336 ...

  7. 深入理解javascript之原型

    理解原型 原型是一个对象.其它对象能够通过它实现属性继承. 不论什么一个对象都能够成为继承,全部对象在默认的情况下都有一个原型.由于原型本身也是对象,所以每一个原型自身又有一个原型. 不论什么一个对象 ...

  8. Android Support 包里到底有什么

    大家假设喜欢我的博客,请关注一下我的微博,请点击这里(http://weibo.com/kifile),谢谢 转载请标明出处(http://blog.csdn.net/kifile),再次感谢 随着 ...

  9. python使用大漠插件进行脚本开发的尝试(一)

    关于游戏脚本是纯然的小白,记一下学习过程遇到的问题.是在win10系统下对PC端的游戏进行脚本编辑,不知道会不会半途放弃. 一.大漠插件 大漠插件在游戏脚本编辑过程中是比较常见的工具,按我理解大致做的 ...

  10. codeforces 204E. Little Elephant and Strings(广义后缀自动机,Parent树)

    传送门在这里. 大意: 给一堆字符串,询问每个字符串有多少子串在所有字符串中出现K次以上. 解题思路: 这种子串问题一定要见后缀自动机Parent树Dfs序统计出现次数都是套路了吧. 这道题统计子串个 ...