【Linux驱动开发】017 信号量与互斥体

一、信号量

1、 信号量定义

举一个很常见的例子,某个停车场有 100 个停车位,这 100 个停车位大家都可以用,对于大家来说这100 个停车位就是共享资源。假设现在这个停车场正常运行,你要把车停到这个这个停车场肯
定要先看一下现在停了多少车了?还有没有停车位?当前停车数量就是一个信号量,具体的停车数量就是这个信号量值,当这个值到 100 的时候说明停车场满了。

相比于自旋锁,信号量可以使线程进入休眠状态。

比如 A 与 B、C 合租了一套房子,这个房子只有一个厕所,一次只能一个人使用。某一天早上 A 去上厕所了,过了一会 B 也想用厕所,因为 A 在厕所里面,所以 B 只能等到 A 用来了才能进去。B 要么就一直在厕所门口等着,等 A 出来,这个时候就相当于自旋锁。B 也可以告诉 A,让 A 出来以后通知他一下,然后 B 继续回房间睡觉,这个时候相当于信号量。可以看出,使用信号量会提高处理器的使用效率,毕竟不用一直傻乎乎的在那里“自旋”等待。但是,信号量的开销要比自旋锁大,因为信号量使线程进入休眠状态以后会切换线程,切换线程就会有开销。

可以通过信号量来控制访问共享资源的访问数量。

一个房子有 10 把钥匙,这 10 把钥匙就相当于信号量值为10。因此,可以通过信号量来控制访问共享资源的访问数量,如果要想进房间,那就要先获取一把钥匙,信号量值减 1,直到 10 把钥匙都被拿走,信号量值为 0,这个时候就不允许任何人进入房间了,因为没钥匙了。如果有人从房间出来,那他要归还他所持有的那把钥匙,信号量值加 1,此时有 1 把钥匙了,那么可以允许进去一个人。

相当于通过信号量控制访问资源的线程数,在初始化的时候将信号量值设置的大于 1,那么这个信号量就是计数型信号量,计数型信号量不能用于互斥访问,因为它允许多个线程同时访问共享资源。如果要互斥的访问共享资源那么信号量的值就不能大于 1,此时的信号量就是一个二值信号量。 

2、 信号量的特点

  1. 因为信号量可以使等待资源线程进入休眠状态,因此适用于那些占用资源比较久的场合。
  2. 信号量不能用于中断中因为信号量会引起休眠,中断不能休眠。 
  3. 如果共享资源的持有时间比较短,那就不适合使用信号量了,因为频繁的休眠、切换线程引起的开销要远大于信号量带来的那点优势。 

3、信号量 API 

Linux 内核使用 semaphore 结构体表示信号量:

struct semaphore { 
    raw_spinlock_t      lock; 
    unsigned int        count; 
    struct list_head    wait_list; 
}; 

要想使用信号量就得先定义,然后初始化信号量。信号量相关的 API 函数如下表:

函数描述
DEFINE_SEAMPHORE(name) 定义一个信号量,并且设置信号量的值为 1。 
void sema_init(struct semaphore *sem, int val)  初始化信号量 sem,设置信号量值为 val。
void down(struct semaphore *sem) 获取信号量,因为会导致休眠,因此不能在中断中使用。 
int down_trylock(struct semaphore *sem); 尝试获取信号量,如果能获取到信号量就获取,并且返回 0。如果不能就返回非 0,并且不会进入休眠。 
int down_interruptible(struct semaphore *sem)获取信号量,和 down 类似,只是使用 down 进入休眠状态的线程不能被信号打断。而使用此函数进入休眠以后是可以被信号打断的。 
void up(struct semaphore *sem) 释放信号量

4、信号量使用示例

信号量的使用示例为:

struct semaphore sem;     /* 定义信号量   */ 
 
sema_init(&sem, 1);      /* 初始化信号量 */ 
 
down(&sem);             /* 申请信号量   */ 
/* 临界区 */ 
up(&sem);              /* 释放信号量   */ 

二、互斥体

1、互斥体定义

将信号量的值设置为 1 就可以使用信号量进行互斥访问了,虽然可以通过信号量实现互斥,但是 Linux 提供了一个比信号量更专业的机制来进行互斥,它就是互斥体—mutex。

互斥访问表示一次只有一个线程可以访问共享资源,不能递归申请互斥体。

2、互斥体的特点

  • mutex 可以导致休眠(使等待资源线程进入休眠状态),因此不能在中断中使用 mutex,中断中只能使用自旋锁。 
  • 和信号量一样,mutex 保护的临界区可以调用引起阻塞的 API 函数。 (因为等待线程可以休眠,因此即便调用引起阻塞的API,也不会引起线程的旋转等待)
  • 因为一次只有一个线程可以持有 mutex,因此,必须由 mutex 的持有者释放 mutex。并且 mutex 不能递归上锁和解锁。

3、互斥体 API 函数

Linux 内核使用 mutex 结构体表示互斥体:

struct mutex { 
    /* 1: unlocked, 0: locked, negative: locked, possible waiters */ 
    atomic_t        count; 
    spinlock_t      wait_lock; 
}; 
函数描述
DEFINE_MUTEX(name) 定义并初始化一个 mutex 变量。 
void mutex_init(mutex *lock) 初始化 mutex。 
void mutex_lock(struct mutex *lock) 获取 mutex,也就是给 mutex 上锁。如果获
取不到就进休眠。
void mutex_unlock(struct mutex *lock) 释放 mutex,也就给 mutex 解锁。 
int mutex_trylock(struct mutex *lock) 尝试获取 mutex,如果成功就返回 1,如果失败就返回 0。
int mutex_is_locked(struct mutex *lock) 判断 mutex 是否被获取,如果是的话就返回1,否则返回 0。 
int mutex_lock_interruptible(struct mutex *lock) 使用此函数获取信号量失败进入休眠以后可以被信号打断。 

4、信号量使用示例

struct mutex lock;    /* 定义一个互斥体   */
mutex_init(&lock);    /* 初始化互斥体     */

mutex_lock(&lock);    /* 上锁        */
/* 临界区 */
mutex_unlock(&lock);   /* 解锁         */

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Kashine

你的鼓励将是我创作的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值