插播一条消息:火箭官网刚刚公布,姚明彻底告别本赛季。看来退役也是迟早的事情了,其实上大学就一直关注火箭队,一个主要原因是有姚明,谁让咱是中国人呢,中国人都是很爱国的,当然个别间谍分子除外。因为姚明,喜欢上了麦迪……如今二人都沦落这般境地,想想让人心寒。烟花易冷、人事易分啊。好了,不提这些伤心地事儿了。我们做点有劲的事儿,继续研究我们的evdev 设备美眉。
前面已经打开她的心扉了。某位脑残的兄弟问:“打开之后干嘛呢”,还能干嘛呢?当然是读了,好,下来我们分析操作集中的第二个函数:evdev_read()深入跟踪下去:
static ssize_t evdev_read(struct file *file, char __user *buffer,
size_t count, loff_t *ppos)
{
1 struct evdev_client *client = file->private_data;
2 struct evdev *evdev = client->evdev;
3 struct input_event event;
4 int retval;
5 if (count < input_event_size())
6 return -EINVAL;
7 if (client->head == client->tail && evdev->exist &&
8 (file->f_flags & O_NONBLOCK))
9 return -EAGAIN;
10 retval = wait_event_interruptible(evdev->wait,
11 client->head != client->tail || !evdev->exist);
12 if (retval)
13 return retval;
14 if (!evdev->exist)
15 return -ENODEV;
16 while (retval + input_event_size() <= count &&
17 evdev_fetch_next_event(client, &event)) {
18 if (input_event_to_user(buffer + retval, &event))
return -EFAULT;
19 retval += input_event_size();
}
20 return retval;
}
相信在座的各位,现在分析这个函数也不会有什么困难。
1行,把在evdev_open里那个私有数据放到struct evdev_client *client中。
5-6行,如果要copy的字节数少于一个event的大小,对不起,结束吧。不完整的event对我们来说没什么用,别浪费表情。
10-11两行,wait_event_interruptible(evdev->wait,client->head != client->tail || !evdev->exist);
还记得她吗?是的,她已经在这里睡了好久,等待她的的如意郎君把她唤醒。后面的client->head != client->tail || !evdev->exist为条件,条件必须满足,她才会醒来。
16-19行,每次从client的buffer中取出一个input event数据放到我们这里的event中,然后把它传到应用层的buffer中,retval记录总共返回的字节数。深入input_event_to_user(buffer + retval, &event):
int input_event_to_user(char __user *buffer,
const struct input_event *event)
{
if (copy_to_user(buffer, event, sizeof(struct input_event)))
return -EFAULT;
return 0;
}
函数copy_to_user就是把一个内核空间buffer放到我们的用户空间buffer,他的兄弟copy_from_user完成相反的动作。
好了,数据都已经到我们的用户空间了,现在我们想干嘛就可以干嘛了……
至于函数集中的evdev_write、evdev_ioctl等函数我就不一一分析了,怎么去往她心里面写东西,怎么去操作她,这些东西听起来是不是有点yellow,我就不好意思再和你一起探讨了。这些东西哥们你要自己慢慢去摸索,俗话说的好:“自己动手,丰衣足食”。
到此,整个系列都已分析完了,发现自己理解和分析代码是一回事,写出来是另外一回事。还是那句话:兄弟们,我们要多实践啊,无论是工作还是把妹,你不去实践,光yy的话,效果还是是相差的很远的。感谢大家的关注,有任何问题,私下找我讨论。