linux c 进程原子操作,Linux 设备驱动 ====> 并发控制 --- 原子操作

原子操作

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

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

整型原子

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

1. 设置源自变量的值

staticinlinevoidatomic_set(atomic_t *v,inti);//设置原子的值

atomic_t = ATOMIC_INIT(0);//定义原子变量并且初始化为02. 获取原子变量的值

#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. 操作并测试

intatomic_inc_and_test(atomic_t *v);

intatomic_dec_and_test(atomic_t *v);

intatomic_sub_and_test(inti, atomic_t *v);

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

5. 操作并返回

intatomic_add_return(inti,atomic_t *v);

intatomic_sub_return(inti,atomic_t *v);

intatomic_inc_return(atomic_t *v);

intatomic_dec_return(atomic_t *v);

返回新值。

位原子操作与整型雷同

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

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

#include 

#include 

#include 

intmain()

{

intfd;

fd = open("/dev/globalmem",O_RDWR);

if(fd == -1) {

printf("open error!!\n");

return-1;

}

sleep(10);

printf("close fd!\n");

close(fd);

return0;

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

root@www.linuxidc.com:/dev# cat globalmem

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

修改驱动代码如下

staticatomic_t globalmem_available = ATOMIC_INIT(1);//define atomic valiable

intglobalmem_open(structinode *inode,structfile *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;

return0;

}

intglobalmem_release(structinode *inode ,structfile *filp)

{

printk(KERN_INFO"globalmem release!\n");

atomic_inc(&globalmem_available);

return0;

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

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

root@www.linuxidc.com:/dev# cat globalmem

cat: globalmem: Device or resource busy

root@www.linuxidc.com:/dev#

原子的操作介绍到这里,结束。0b1331709591d260c1c78e86d0c51c18.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值