原子操作简介:
原子操作就是:指不能再进一步分割的操作
一般原子操作用于:<整形操作
>或者<位操作
>。
假如现在要对无符号整形变量 a 赋值,值为3,对于 C 语言来讲很简单,直接就是:a = 3
但是 C 语言要先编译为成汇编指令,ARM 架构不支持直接对寄存器进行读写操作,比如 要借助寄存器 R0、R1 等来完成赋值操作。假设变量 a 的地址为 0X3000000,“a=3”这一行 C 语言可能会被编译为如下所示的汇编代码:
ldr r0, =0X30000000 /* 变量 a 地址 /
ldr r1, = 3 / 要写入的值 /
str r1, [r0] / 将 3 写入到 a 变量中 */
假设现在线程 A
要向 a 变量写入 10 这个值,而线程 B
也要向 a 变量写入 20 这个值,我们理想中的执行顺序:
确实可以实现线程 A
将 a 变量设置为 10,线程 B
将 a 变量设置为 20。但是实际上的执行流程可能如图 下 所示:
线程 A
最终将变量 a 设置为了 20,而并不是要求的 10!线程B
没有问题。这就是一个最简单的设置变量值的并发与竞争的例子,要解决这个问题就要保证代码a=10
中的三行汇编指令作为一个整体运行,也就是作为一个原子存在。Linux内核提供了一组原子操作 API
函数来完成此功能,Linux 内核提供了两组原子操作 API 函数
,一组是对整形变量进行操作的,一组是对位进行操作的,我们接下来看一下这些 API 函数。
1、原子整形操作 API 函数:
Linux 内核定义了叫做
atomic_t
的结构体来完成整形数据
的原子操作,在使用中用原子变量
来代替整形变量
,此结构体定义在include/linux/types.h
文件中,定义如下:
typedef struct {
int counter;
}atomic_t;
如果要使用原子操作 API 函数,首先要先定义一个 atomic_t
类型的变量,如下所示:
atomic_t a; //定义 a
也可以在定义原子变量的时候通过宏 ATOMIC_INIT()
向原子变量赋初值给原子变量赋初值,如下所示:
atomic_t b = ATOMIC_INIT(0); //定义原子变量 b 并赋初值为 0
原子变量有了,接下来就是对原子变量进行操作,比如读、写、增加、减少
等等,Linux 内
核提供了大量的原子操作 API 函数
,如下所示:
注意:
如果使用64 位的 SOC
的话,就要用到64
位的原子变量,Linux 内核也定义了 64 位原子 结构体,如下所示:
typedef struct {
long long counter;
} atomic64_t;
相应的也提供了 64 位原子变量的操作 API 函数,这里就不讲了,和表中的 API
函数有用法一样,只是将“atomic_
”前缀换为“atomic64_
”,将int
换为long long
。
32 位
的架构原子变量和相应的 API 函数使用起来很简单,参考如下示例:
atomic_t v = ATOMIC_INIT(0); /* 定义并初始化原子变零 v=0 */
atomic_set(10); /* 设置 v=10 */
atomic_read(&v); /* 读取 v 的值,肯定是 10 */
atomic_inc(&v); /* v 的值加 1,v=11 */
2、原子位操作 API 函数:
位操作也是很常用的操作,Linux 内核也提供了一系列的
原子位操作 API 函数
,只不过原 子位操作不像原子整形变量那样有个atomic_t
的数据结构,原子位操作
是直接对内存
进行操作, API 函数如表 下 所示: