Linux系统处于一个高并发的运行环境,不管是系统调用还是中断都要求可重入,但是有一些系统资源处于临界区,因此,必须保证临界区资源访问的原子性。

对于临界区资源被占用时,发起访问的进程,有三种处理方法——睡眠、阻塞以及撤销。

Linux驱动编程中,通常不建议使用锁机制,因为容易导致死锁问题。不使用锁的场景,尽量使用kfifo缓冲队列来存取数据;在必须使用锁的场景,建议使用信号量和自旋锁。

信号量通常用在可以睡眠的场景,如进程上下文;而自旋锁通常用在不可睡眠的场景,如中断上下文。

1、信号量

#include <linux/semaphore.h>

信号量初始化
void sema_init(struct semaphore *sem, int val); 互斥信号量
DECLARE_MUTEX(name);
DECLARE_MUTEX_LOCKED(name); void init_MUTEX(struct semaphore *sem);
void init_MUTEX_LOCKED(struct semaphore *sem); 获取信号量
void down(struct semaphore *sem);
int down_interruptible(struct semaphore *sem); /* 可中断,即进入睡眠状态等待信号量,需要一直检查返回值并且针对性地响应 */
int down_trylock(struct semaphore *sem); /* 从不睡眠,如果信号量被占用,立刻返回 */ 释放信号量
void up(struct semaphore *sem);

示例:

if (down_interruptible(&sem))
return -ERESTARTSYS; /* 临界区资源访问 */ up(&sem);

1.1、读写信号量

由信号量衍生,可以允许多个用户并发读,一个用户互斥写。

#include <linux/rwsem.h>

/* 初始化 */
void init_rwsem(struct rw_semaphore *sem); /* 读者接口 */
void down_read(struct rw_semaphore *sem);
void down_read_trylock(struct rw_semaphore *sem);
void up_read(struct rw_semaphore *sem); /* 写者接口 */
void down_write(struct rw_semaphore *sem);
void down_write_trylock(struct rw_semaphore *sem);
void up_write(struct rw_semaphore *sem);

2、自旋锁

#include <linux/spinlock.h>

初始化
spinlock_t my_lock = SPIN_LOCK_UNLOCKED;
或者
void spin_lock_init(spinlock_t *lock); 获取自旋锁
void spin_lock(spinlock_t *lock);
释放自旋锁
void spin_unlock(spinlock_t *lock); 获取自旋锁时保存中断状态,并禁止本地处理器中断,释放时打开中断
void spin_lock_irqsave(spinlock_t *lock, unsigned long flags);
void spin_unlock_irqrestore(spinlock_t *lock, unsigned long flags); 获取自旋锁时禁止中断,释放时打开中断
void spin_lock_irq(spinlock_t *lock);
void spin_unlock_irq(spinlock_t *lock);

2.1、读写自旋锁

类似于读写信号量,读写自旋锁是自旋锁的衍生。

3、原子量(atomic)

4、顺序锁(seqlock)

顺序锁用来保护较小的资源,快速存取的场景。它允许读者释放对资源的存取,但是要求读者检查与写者的冲突,如果发生冲突,重试存取操作。

seqlock通常不能用在保护包含指针的数据结构,因为读者可能跟随一个无效指针而写者在改变数据结构。

#include <linux/seqlock.h>

seqlock_init(seqlock_t *lock);

5、读取-拷贝-更新(RCU)

一种高级的互斥方法

Linux驱动开发4——并发和竞态的更多相关文章

  1. Linux内核中的并发与竞态概述

    1.前言 众所周知,Linux系统是一个多任务的操作系统,当多个任务同时访问同一片内存区域的时候,这些任务可能会相互覆盖内存中数据,从而造成内存中的数据混乱,问题严重的话,还可能会导致系统崩溃. 2. ...

  2. linux设备驱动程序之并发和竞态(二)

    事实上这blog都是阅读ldd3时的一些总结,巩固自己的学习.也方便后期的使用.大家也能够直接阅读ldd3原文. 锁陷阱         所谓的锁陷阱就是防止死锁.         不明白的规则:   ...

  3. 漫画|Linux 并发、竞态、互斥锁、自旋锁、信号量都是什么鬼?(转)

    知乎链接:https://zhuanlan.zhihu.com/p/57354304 1. 锁的由来? 学习linux的时候,肯定会遇到各种和锁相关的知识,有时候自己学好了一点,感觉半桶水的自己已经可 ...

  4. linux设备驱动第五篇:驱动中的并发与竟态

    综述 在上一篇介绍了linux驱动的调试方法,这一篇介绍一下在驱动编程中会遇到的并发和竟态以及如何处理并发和竞争. 首先什么是并发与竟态呢?并发(concurrency)指的是多个执行单元同时.并行被 ...

  5. 《Linux 设备驱动程序》读后感。 并发,竞态,死锁。

    1. 概念 并发:在操作系统中,是指一个时间段中有几个程序都处于已启动运行到运行完毕之间,且这几个程序都是在同一个处理机上运行,但任一个时刻点上只有一个程序在处理机上运行. 来源: 1. Linux ...

  6. Linux内核分析(七)----并发与竞态

    原文:Linux内核分析(七)----并发与竞态 Linux内核分析(七) 这两天家里的事好多,我们今天继续接着上一次的内容学习,上次我们完善了字符设备控制方法,并深入分析了系统调用的实质,今天我们主 ...

  7. Linux 驱动开发

    linux驱动开发总结(一) 基础性总结 1, linux驱动一般分为3大类: * 字符设备 * 块设备 * 网络设备 2, 开发环境构建: * 交叉工具链构建 * NFS和tftp服务器安装 3, ...

  8. Linux驱动开发必看详解神秘内核(完全转载)

    Linux驱动开发必看详解神秘内核 完全转载-链接:http://blog.chinaunix.net/uid-21356596-id-1827434.html   IT168 技术文档]在开始步入L ...

  9. LDD3之并发和竞态-completion(完毕量)的学习和验证

    LDD3之并发和竞态-completion(完毕量)的学习和验证 首先说下測试环境: Linux2.6.32.2 Mini2440开发板 一開始难以理解书上的书面语言,这里<linux中同步样例 ...

随机推荐

  1. html5酷炫效果链接收集

    HTML5 3D图片相册图片轮播动画   http://www.html5tricks.com/demo/html5-3d-gallery-animation/index.html 36种漂亮的CSS ...

  2. C#设计模式:访问者模式(Vistor Pattern)

    一,访问者模式是用来封装一些施加于某种数据结构之上的操作.它使得可以在不改变元素本身的前提下增加作用于这些元素的新操作,访问者模式的目的是把操作从数据结构中分离出来. 二,代码 using Syste ...

  3. 猿题库 iOS 客户端架构设计(原文地址:http://gracelancy.com/blog/2016/01/06/ape-ios-arch-design/)

    猿题库 iOS 客户端架构设计 序 猿题库是一个拥有数千万用户的创业公司,从2013年题库项目起步到2015年,团队保持了极高的生产效率,使我们的产品完成了五个大版本和数十个小版本的高速迭代.在如此快 ...

  4. MySQL中Innodb的聚簇索引和非聚簇索引

    聚簇索引 数据库表的索引从数据存储方式上可以分为聚簇索引和非聚簇索引(又叫二级索引)两种.Innodb的聚簇索引在同一个B-Tree中保存了索引列和具体的数据,在聚簇索引中,实际的数据保存在叶子页中, ...

  5. python并发编程中的多进程(代码实现)

    一.multipricessing模块的介绍 python中的多线程无法利用多核优势,如果想要充分的使用多核CPU资源,在python中大部分情况下需要用多线程,python提供了multiproce ...

  6. json反序列化与pickle的用法

    json反序列化与pickle 一.定义 序列化:将内存中的不可持久化和传输对象转换为可方便持久化和传输对象的过程. 反序列化:将可持久化和传输对象转换为不可持久化和传输对象的过程. 二. 应用场景 ...

  7. PAT Advanced 1036 Boys vs Girls (25 分)

    This time you are asked to tell the difference between the lowest grade of all the male students and ...

  8. PAT Advanced 1042 Shuffling Machine (20 分)(知识点:利用sstream进行转换int和string)

    Shuffling is a procedure used to randomize a deck of playing cards. Because standard shuffling techn ...

  9. Listview使用

    ListView控件比前面几种控件要复杂一些,通过此控件,可将项目组成带有或不带有列标头的列,并显示伴随的图标和文本.ListView控件是由ColumnHeader和ListItem对象所组成的,其 ...

  10. GUI学习之十二——QTextEdit学习总结

    在学习了QFrame和QAbstractScrollArea两个父类后,接下来是一个重头戏:QTextEdit. 一.QTextEdit特性 QTextEdit是一个高级的WYSIWYG(What Y ...