-解决竞态的一种操作--->原子操作
解决竞态的途径是“保证对共享资源的互斥访问
原子操作
原子的操作指的就是在执行过程中不会被别的代码所中断的操作。
在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) //返回原子变量的值 (*(volatile int *)&(v)->counter)
-
3. 原子变量加减,自增自减
-
#define atomic_add(i, v) (void) atomic_add_return(i, v) //把v的值加 i #define atomic_inc(v) (void) atomic_add_return(1, v) //vz自加 #define atomic_sub(i, v) (void) atomic_sub_return(i, v) #define atomic_dec(v) (void) atomic_sub_return(1, v) //v自减
-
4. 操作并测试
open_atomic_int_one以及open_atomic_int_two两个程序
要对/dev/atomic_int设备节点镜像操作
先运行的程序1,将变量赋值为1,释放的时候赋值为0
如果程序1在没有释放的情况下,程序2调用设备节点则会直接返回,无法调用。
#include <linux/init.h> #include <linux/module.h> /*驱动注册的头文件,包含驱动的结构体和注册和卸载的函数*/ #include <linux/platform_device.h> /*注册杂项设备头文件*/ #include <linux/miscdevice.h> /*注册设备节点的文件结构体*/ #include <linux/fs.h> //原子操作的函数头文件 #include <asm/atomic.h> #include <asm/types.h> #define DRIVER_NAME "atomic_int" #define DEVICE_NAME "atomic_int" MODULE_LICENSE("Dual BSD/GPL"); MODULE_AUTHOR("HKY"); //定义原子变量,并初始化为0 static atomic_t value_atomic = ATOMIC_INIT(0); static int atomic_int_open(struct inode *inode, struct file *file){ printk(KERN_EMERG "atomic_int open in!\n"); if(atomic_read(&value_atomic)){ return -EBUSY; } atomic_inc(&value_atomic); printk(KERN_EMERG "atomic_int open success!\n"); return 0; } static int atomic_int_release(struct inode *inode, struct file *file){ printk(KERN_EMERG "atomic_int release\n"); atomic_dec(&value_atomic); return 0; } static struct file_operations atomic_int_ops = { .owner = THIS_MODULE, .open = atomic_int_open, .release = atomic_int_release, }; static struct miscdevice atomic_int_dev = { .minor = MISC_DYNAMIC_MINOR, .name = DEVICE_NAME, .fops = &atomic_int_ops, }; static int atomic_int_probe(struct platform_device *pdv){ printk(KERN_EMERG "\tinitialized\n"); misc_register(&atomic_int_dev); return 0; } static int atomic_int_remove(struct platform_device *pdv){ printk(KERN_EMERG "\tremove\n"); misc_deregister(&atomic_int_dev); return 0; } struct platform_driver atomic_int_driver = { .probe = atomic_int_probe, .remove = atomic_int_remove, .driver = { .name = DRIVER_NAME, .owner = THIS_MODULE, } }; static int atomic_int_init(void) { int DriverState; printk(KERN_EMERG "HELLO WORLD enter!\n"); DriverState = platform_driver_register(&atomic_int_driver); printk(KERN_EMERG "\tDriverState is %d\n",DriverState); return 0; } static void atomic_int_exit(void) { printk(KERN_EMERG "HELLO WORLD exit!\n"); platform_driver_unregister(&atomic_int_driver); } module_init(atomic_int_init); module_exit(atomic_int_exit);