Linux 设备驱动 ====> 并发控制 --- 原子操作

原创 2012年03月27日 22:18:39

原子操作

原子的操作指的就是在执行过程中不会被别的代码所中断的操作。

在Linux中原子操作的方法有很多,有整型原子和位原子,他们在任何情况下操作都是原子的,这些原子操作的实现都是依赖CPU来实现的,因此这些函数都与CPU架构密切相关。


整型原子

我们arm架构的原子实现在kernel/arch/arm/include/asm/atomic.h

1. 设置源自变量的值

static inline void atomic_set(atomic_t *v, int i);            //设置原子的值
atomic_t = ATOMIC_INIT(0);                                    //定义原子变量并且初始化为0

2. 获取原子变量的值

#define atomic_read(v)	((v)->counter)                         //返回原子变量的值

3. 原子变量加减,自增自减

#define atomic_add(i, v)	(void) atomic_add_return(i, v)
#define atomic_inc(v)		(void) atomic_add_return(1, v)
#define atomic_sub(i, v)	(void) atomic_sub_return(i, v)
#define atomic_dec(v)		(void) atomic_sub_return(1, v)

4. 操作并测试

int atomic_inc_and_test(atomic_t *v);
int atomic_dec_and_test(atomic_t *v);
int atomic_sub_and_test(int i, atomic_t *v);

上述操作对原子变量执行自增、自减和减操作后测试其是否为0,为-则返回true,否则返回false。

5. 操作并返回

int atomic_add_return(int i,atomic_t *v);
int atomic_sub_return(int i,atomic_t *v);
int atomic_inc_return(atomic_t *v);
int atomic_dec_return(atomic_t *v);



返回新值。

位原子操作与整型雷同


举个例子---使用原子变量实现设备只能被一个进程打开。

我们写一个小小的应用程序,打开之前我们的字符设备,然后sleep 10秒钟,然后再close,

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>

int main()
{
	int fd;
	fd = open("/dev/globalmem",O_RDWR);
	if(fd == -1) {
		printf("open error!!\n");
		return -1;
	}
	sleep(10);
	printf("close fd!\n");
	close(fd);
	return 0;
}

我们在这sleep的10秒内去cat /dev/globalmem

root@jay-LJ:/dev# cat globalmem 
Hello globalmem driver

还是可以cat出来的,如果有好多个进程都要读写我们的globalmem的话就会产生竞态,会导致读出来的数据有问题,所以这里我们让我们的驱动在同一时间只能由一个进程来访问。

修改驱动代码如下

static atomic_t globalmem_available = ATOMIC_INIT(1);	//define atomic valiable

int globalmem_open(struct inode *inode, struct file *filp)
{
	if(!atomic_dec_and_test(&globalmem_available)) {
		printk(KERN_ERR "already open!\n");
		atomic_inc(&globalmem_available);
		return -EBUSY;	//already open
	}
	printk(KERN_INFO "globalmem open!\n");
	filp->private_data = globalmem_devp;	
	return 0;
}

int globalmem_release(struct inode *inode ,struct file *filp)
{
	printk(KERN_INFO "globalmem release!\n");
	atomic_inc(&globalmem_available);
	return 0;
}

很简单,使用我们的atomic_dec_and_test来进行原子的测试并返回,然后检测返回值来查看该设备是否已经被打开,最后在close的时候再自加1.

我们重新编译驱动,然后加载,并跟之前一样来测试,发现在打开之后还没关闭的时候,我们去cat会发生错误,提示设备忙。

root@jay-LJ:/dev# cat globalmem 
cat: globalmem: Device or resource busy
root@jay-LJ:/dev# 


原子的操作介绍到这里,结束。
=================================================================

mail & MSN :zhangjie201412@live.com

=========================================================


相关文章推荐

LDD3源码分析之按页分配内存

作者:刘昊昱  博客:http://blog.csdn.net/liuhaoyutz 编译环境:Ubuntu 10.10 内核版本:2.6.32-38-generic-pae LDD3源...

LDD3源码分析之slab高速缓存

作者:刘昊昱  博客:http://blog.csdn.net/liuhaoyutz 编译环境:Ubuntu 10.10 内核版本:2.6.32-38-generic-pae LDD3源...

linux设备驱动--并发与竞态之原子操作

原子操作指的是在执行过程中不会被别的代码路径所中断的操作。        linux下面有两类原子操作,一类是整形原子操作,一类是位原子操作。        原子操作目前仅仅做个测试验证学习,至于...
  • dndxhej
  • dndxhej
  • 2012年03月10日 00:15
  • 1933

Linux 设备驱动--- 并发 与 竞态 --- atomic_t --- atomic_dec_and_test --- 原子操作

并发:           多个执行单元同时被执行. 竞态:           并发的执行单元对资源 ( 硬件资源和软件上的全局变量等 ) 的访问导致的竞争状态.        ...

07-S3C2440驱动学习(一)嵌入式linux字符设备驱动-按键驱动程序之异步通知机制+原子操作+互斥信号量+阻塞与非阻塞+定时器去抖

一、异步通知机制 从按键的实现方式来说,可以分为以下几种方式 查询方式,极度耗费CPU资源中断方式,平时休眠,按键按下,唤醒休眠poll机制,不需要一直read,根据poll返回值来决定是否rea...

linux 设备驱动中的并发控制

  • 2011年09月19日 23:44
  • 7KB
  • 下载

linux驱动开发--字符设备:原子操作

解决竞态问题的途径是保证对共享资源的互斥访问,所谓互斥访问是指一个执行单元在访问共享资源的时候,其他的执行单元被禁止访问。 访问共享资源的代码区域称谓临界区(critical sections),临...
  • waldmer
  • waldmer
  • 2014年01月08日 16:04
  • 1764

对linux设备驱动中的并发控制相关内容的理解

这篇笔记主要是针对linux设备驱动中的并发控制内容的学习后,存在的一些问题的补充学习和调查结果,路过的大神们也可以帮我看看理解的是否正确,有问题的话欢迎大家帮我指出来,小弟在此谢过啦! 问题一 ...

Linux设备驱动的并发控制总结

1.linux驱动为什么存在并发控制? 答:并发( concurrency)指的是多个执行单元同时、并行被执行,而并发的执行单元对共享资源 (硬件资源和软件上的全局变量、静态变量等)的访问则很容易导致...
  • tsb151
  • tsb151
  • 2017年02月24日 15:17
  • 75

Linux 设备驱动 ====> 并发控制 --- 信号量与互斥体

信号量 信号量的使用 信号量(semaphore)是用于保护临界区的一种常用方法,他的用法和自旋锁类似,但是,与自旋锁不同的是,当获取不到信号量时,进程不会原地打转,而是进入休眠等状态。 Lin...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Linux 设备驱动 ====> 并发控制 --- 原子操作
举报原因:
原因补充:

(最多只允许输入30个字)