- 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
信号量
什么是信号量
信号量的使川主要是川来保护井享资源,使得资源在一个时刻只有一个进程(线程)所 拥有。
信号量的位为正的时候,说明它空闲。所测试的线程可以锁定而使用它。若为0,说明 它被占用,测试的线程要进入睡眠队列中,等待被唤醒。
二.信号量的分类
在学习信号量之前,我们必须先知道一一Linux提供两种信号量:
(1) 内核信号量,由内核控制路径使用
(2) 用户态进程使用的信号量,这种信号量又分为POSIX信号量和SYSTEM V信 号量。
POSIX信号量又分为有名信号量和无名信号量。
有名信号量,其值保存在文件中,所以它可以用于线程也可以用于进程间的同步。无名 信号量,其值保存在内存中。
倘若对信号M没有以上的全而认识的话,你就会很快发现自己在信号S:的森林里迷失 了方向。
三.内核信号量
内核信号量的构成
内核信号量类似于自旋锁,因为当锁关闭着时,它不允许内核控制路径继续进行。然而, 当内核控制路径试图获取内核信号量锁保护的忙资源吋,相应的进程就被挂起。只有在资源 被释放时,进程才再次变为可运行。
只有可以睡眠的函数才能获取内核信号量;中断处理程序和可延迟函数都不能使用内核 信号量。
内核信号量是struct semaphore类型的对象,它在〈asm/semaphore.hM1定义:
struct semaphore { atomic t count;
int sleepers;
wait_queue_head_t wait;
}
count:相当于信号S的值,大于0,资源空闲;等于0,资源忙,但没有进程等待这个 保护的资源;小于0,资源不可用,并至少有一个进程等待资源。
wait:存放等待队列链表的地址,当前等待资源的所有睡眠进程都会放在这个链表中。 sleepers:存放一个标志,表示是否有一些进程在信号量上睡眠。
内核信号量中的等待队列(删除,没有联系)
上而已经提到了内核信号量使用了等待队列wait.queue来实现阻塞操作。
当某任务由于没有某种条件没有得到满足时,它就被挂到等待队列中睡眠。当条件得到满足 吋,该任务就被移出等待队列,此吋并不意味着该任务就被马上执行,因为它又被移进工作 队列中等待CPU资源,在适当的时机被调度。
内核信号量是在内部使用等待队列的,也就是说该等待队列对用户是隐藏的,无须用户 干涉。巾用户真正使用的等待队列我们将在另外的篇章进行详解。
内核信号量的相关函数
(1)初始化:
void sema_init (struct semaphore *sem, int val);
void init_MUTEX (struct semaphore *sem); //将 sem 的值置为 1,表示资源空闲 void init MUTEX LOCKED (struct semaphore *sem); //将 sem 的值罝为 0,表示资源忙
申请内核信号量所保护的资源:
void down(struct semaphore * sem); // 可引起睡眠
int down_interruptible(struct semaphore * sem); // down_interruptible 能被信号fj?断 int down_trylock(struct semaphore * sem); //非阻塞函数,不会睡眠。无法锁定资源则
马上返回
释放内核信号量所保护的资源: void up(struct semaphore * sem);
内核信号量的使用例程
在驱动程序巾,当多个线程同时访问相同的资源时(驱动巾的全局变量时一种典型的共 享资源),可能会引发“竞态“,因此我们必须对共享资源进行并发控制。Linux内核中解 决并发控制的最常用方法是自旋锁与信号S (绝大多数时候作为互斥锁使用)。 ssize_t globcilvar_write($truct file *filp,const char *buf,size_t len,loff_t *off)
{
//获得信号量
if (down_inteiTLiptible(sem))
{
return - ERESTARTSYS;
}
//将用户空间的数裾复制到内核空间的global.var if (copy_from_u$er(global_var, buf, $izeof(int)))
{
up(sem); return - EFAULT;
}
//释放信号量 up(sem); return sizeof(int);
}
U!POSIX
U!
POSIX信号量与SYSTEM V信号量的比较
对POSIX来说,信号釐是个非负整数。常用于线程间同步。
而SYSTEM V信号量则是一个或多个信号量的集合,它对应的是一个信号量结构体,
文档评论(0)