LDD3源码学习日记<四>

来源:互联网 发布:手机突然连不上4g网络 编辑:程序博客网 时间:2024/05/06 10:41

    日记三不小心被删了。。。哎,只能在回收站里面自己看了。这博客80%的功能还是写给自己看的,其他部分的功能应该是用来练习写作能力的,留下这一路走来的证据,两年后回过头来看看自己当初是怎么犯低级错误的。哈哈。。

第五章是并发与竞态,书上的内容讲的主要是信号量和自旋锁机制,还有completion,不过书上介绍这部分的内容不多,只用了短短的一页纸。信号量可以导致休眠,所以一般的可以用于临界保护区比较大的场合下,而自旋转锁锁住的进程会在原地打转,所以,如果临界保护区比较大的话,非常消耗系统资源。LDD3给出的源码里包含了信号量和completion,信号量是嵌在之前的代码里面的,一般而言,它总会出现在可能修改保护对象的代码前面。在使用信号量之前,首先要明确什么是需要用信号量保护的资源,
然后,我们才能用信号量保证对这些资源的互斥访问。对于scull设备来说,所有的信息都保存在scull_dev结构体中,因此,scull_dev就是我们要保护的资源。在main.c文件在有很多地方使用了信号量来保证对scull_dev的互斥访问。或者说,凡是要改变scull_dev结构休内容的地方,都必须加锁,防止竞态。

一、信号量

       信号量在使用之前必须先初始化,scull在模块初始化函数scull_init_module。中执行下面的循环完成对所有scull设备专用信号量的初始化,并且要注意,信号量必须在scull设备对系统其它部分可用前被初始化。因此,在下面的代码里,在scull_setup_cdev之前使用了iniy_MUTEX。

for (i = 0; i < scull_nr_devs; i++) {scull_devices[i].quantum = scull_quantum;scull_devices[i].qset = scull_qset;init_MUTEX(&scull_devices[i].sem);scull_setup_cdev(&scull_devices[i], i);}
在main.c中,涉及到信号量的函数有open和write、read函数,down_interruptible都是在一些局部变量初始化后执行的第一个动作。

if (down_interruptible(&dev->sem))return -ERESTARTSYS;
调用down_interruptible(&dev->sem)进行加锁,注意,要对down_interruptible的返回值进行检查,如果返回0,说明说明加锁成功了,可以开始操作受保护的资源scull_dev,反之,如果down_interruptible返回非0值,说明是在等待过程中被中断了,这时要退出并返回-ERESTARTSYS,交给系统处理。

给信号量加锁后,不管scull_write能否完成其工作,都必须释放信号量,代码如下:

out:up(&dev->sem);return retval;
二、completion

completion是一种轻量级机制,它允许一个线程告诉另一个线程某个工作已经完成。它包含在<linux/completion.h>里面。

如果有多个线程在等待同一个completion事件,complete函数只唤醒一个等待线程,而complete_all函数将唤醒所有等待线程。

等待completion使用如下函数:
void wait_for_completion(struct completion *c);
相应的,completion事件可以通过如下函数触发:
void complete(struct completion *c);
void complete_all(struct completion *c);

LDD3在misc-modules里给出了completion的演示,代码如下:

#include <linux/module.h>#include <linux/init.h>#include <linux/sched.h>  /* current and everything */#include <linux/kernel.h> /* printk() */#include <linux/fs.h>     /* everything... */#include <linux/types.h>  /* size_t */#include <linux/completion.h>MODULE_LICENSE("Dual BSD/GPL");static int complete_major = 0;DECLARE_COMPLETION(comp);//注册completionssize_t complete_read (struct file *filp, char __user *buf, size_t count, loff_t *pos){printk(KERN_DEBUG "process %i (%s) going to sleep\n",current->pid, current->comm);wait_for_completion(&comp);//等待com完成printk(KERN_DEBUG "awoken %i (%s)\n", current->pid, current->comm);return 0; /* EOF */}ssize_t complete_write (struct file *filp, const char __user *buf, size_t count,loff_t *pos){printk(KERN_DEBUG "process %i (%s) awakening the readers...\n",current->pid, current->comm);complete(&comp);//触发completion,唤醒在等待的com进程return count; /* succeed, to avoid retrial */}struct file_operations complete_fops = {.owner = THIS_MODULE,.read =  complete_read,.write = complete_write,};int complete_init(void){int result;/* * Register your major, and accept a dynamic number */result = register_chrdev(complete_major, "complete", &complete_fops);if (result < 0)return result;if (complete_major == 0)complete_major = result; /* dynamic */return 0;}void complete_cleanup(void){unregister_chrdev(complete_major, "complete");}module_init(complete_init);module_exit(complete_cleanup);
在这里主要需要关注read和write函数的实现。

下面看complete_read的实现:
在打印即将进入睡眠的信息后,complete_read在调用wait_for_completion(&comp),进入睡眠,即等待completion “comp”。”comp”是用DECLARE_COMPLETION(comp)创建的。如果等待的completion发生了,complete_read函数将再次打印已被唤醒相关信息。也就是说,任何进程读取模块设备文件,都会进入睡眠等待。
再来看complete_write的实现:
首先打印提示信息,然后在47行调用complete(&comp)触发completion事件,相应会唤醒一个在等待”comp”的进程。可以有多个进程进行读操作,这些读进程都会进入睡眠等待,当有执行写操作的进程时,只有一个等待进程会被唤醒,但是哪个进程,不能确定。


参考博客:http://blog.csdn.net/liuhaoyutz/article/details/7383653


原创粉丝点击