10.1、android输入系统_必备Linux编程知识_inotify和epoll
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的更多相关文章
- 10.2、android输入系统_必备Linux编程知识_双向通信(scoketpair)
2. 双向通信(socketpair) 输入系统肯定涉及进程通讯:进程A读取/分发输入事件,APP处理输入事件,进程A给APP发送输入事件,APP处理完事件回复信息给进程A,APP关闭的时候也要发信息 ...
- 10.3、android输入系统_必备Linux编程知识_任意进程双向通信(scoketpair+binder)
3. 任意进程间通信(socketpair_binder) 进程每执行一次open打开文件,都会在内核中有一个file结构体表示它: 对每一个进程在内核中都会有一个task_struct表示进程,这个 ...
- 10.11 android输入系统_补充知识_activity_window_decor_view关系
android里:1个application, 有1个或多个activity(比如支付宝有:首页.财富.口碑.朋友.我的,这些就是activity)1个activity, 有1个window(每个ac ...
- 10.4 android输入系统_框架、编写一个万能模拟输入驱动程序、reader/dispatcher线程启动过程源码分析
1. 输入系统框架 android输入系统官方文档 // 需FQhttp://source.android.com/devices/input/index.html <深入理解Android 卷 ...
- 10.13 android输入系统_多点触摸驱动理论与框架
1.多点触摸驱动理论 驱动程序仅上报多个触点的位置就可以,是放大还是缩小由应用程序控制 对于多点触摸驱动在linux系统中有个输入子系统,其已经实现了open/read/write等接口 我们只需要实 ...
- 10.14 android输入系统_多点触摸驱动测试及Reader线程、InputStage分析
21. 多点触摸_电容屏驱动程序_实践_tiny4412 tiny4412触摸屏: 分辨率为800 x 480http://wiki.friendlyarm.com/wiki/index.php/LC ...
- 10.8 android输入系统_实战_使用GlobalKey一键启动程序
11. 实战_使用GlobalKey一键启动程序参考文章:Android 两种注册(动态注册和静态注册).发送广播的区别http://www.jianshu.com/p/ea5e233d9f43 [A ...
- 10.9 android输入系统_APP跟输入系统建立联系和Dispatcher线程_分发dispatch
12. 输入系统_APP跟输入系统建立联系_InputChannel和Connection核心: socketpair // 第9课第3节_输入系统_必备Linux编程知识_任意进程双向通信(scok ...
- 10.5 android输入系统_Reader线程_使用EventHub读取事件和核心类及配置文件_实验_分析
4. Reader线程_使用EventHub读取事件 使用inotify监测/dev/input下文件的创建和删除 使用epoll监测有无数据上报 细节: a.fd1 = inotify_init(& ...
随机推荐
- CentOS 开启 IPV6
编辑网卡地址:#vi /etc/sysconfig/network-scripts/ifcfg-eth0IPV6INIT=yesIPV6FORWARDING=yesIPV6ADDR=2607:9000 ...
- vim基础学习之自动补全功能
本章我们学习自动补全功能1.自动补全优先从当前的编辑区获得补全列表例如:我们写下如下内容 aaaaa aabbb aaab 当我们再次输入aa,然后我们按下Tab的时候,会弹出一个包含 aaaaa a ...
- js插件---10个免费开源的JS音乐播放器插件
js插件---10个免费开源的JS音乐播放器插件 一.总结 一句话总结:各种插件都有很多,多去找. 二.js插件---10个免费开源的JS音乐播放器插件 亲测可用 音乐播放器在网页设计中有时候会用到, ...
- javafx style and cssFile
public class EffectTest extends Application { public static void main(String[] args) { launch(args); ...
- [lougu2243]双端队列搜索
正统双端队列搜索 回顾:普通队列进行边权为定值的最短路 每次到达都是最优的(意味着不用取min) why? 因为所有状态按照 入队的先后顺序 具有 层次单调性,每次扩展,都往外走一步,满足从起始到该状 ...
- gpasswd---指定要管理的工作组,及更改密码
gpasswd 命令详解 gpasswd命令是Linux下工作组文件/etc/group和/etc/gshadow的管理工具,用于指定要管理的工作组. 2.选项详解: -a : 添加用户到组 -d : ...
- jquery的图片轮播 模板类型
先说一下用到的几个重要的事件 j jQuery on()方法是官方推荐的绑定事件的一个方法. $(selector).on(event,childSelector,data,function,map) ...
- 负载均衡器&http正向代理
透明的负载均衡器&http正向代理 * master-workers架构,http正向代理由独立的dns请求以及缓冲进程 * 使用epoll(ET)模式,採用全异步方式(双缓存,实现双向同一 ...
- 在带(继承)TextView的控件中,在代码中动态更改TextView的文字颜色
今天由于公司项目需求,须要实现一种类似tab的选项卡,当时直接想到的就是使用RadioGroup和RadioButton来实现. 这种方法全然没问题.可是在后来的开发过程中,却遇到了一些困扰非常久的小 ...
- android 弹幕评论效果
纯粹依照自己的想法仿照b站的弹幕写的一个demo,不知道正确的姿势怎么样的. demo下载地址 首先.一条弹幕就是一个textview public abstract class Danmu exte ...