Linux驱动之原子变量

使用原子变量可以避免竞争问题。其实,很多的同步技术都使用了原子变量,例如自旋锁。

它的名称是 atomic_t, 实际上是一个int类型。不过由于某些处理器上这种数据类型的工作方式有些限制,因此不能使用完整的整数范围(只能使用24位)


内核提供了一系列的API,基本形式都是atomic_xxx()

这些操作都非常快,原因是他们被编译成单个机器指令。

以atomic_inc接口来说:

Arm架构如下:


static inline void atomic_add(int i, atomic_t *v)
{
	unsigned long tmp;
	int result;

	__asm__ __volatile__("@ atomic_add\n"
"1:	ldrex	%0, [%2]\n"
"	add	%0, %0, %3\n"
"	strex	%1, %0, [%2]\n"
"	teq	%1, #0\n"
"	bne	1b"
	: "=&r" (result), "=&r" (tmp)
	: "r" (&v->counter), "Ir" (i)
	: "cc");
}

mips架构:

/*
 * atomic_add - add integer to atomic variable
 * @i: integer value to add
 * @v: pointer of type atomic_t
 *
 * Atomically adds @i to @v.
 */
static __inline__ void atomic_add(int i, atomic_t * v)
{
	if (cpu_has_llsc && R10000_LLSC_WAR) {
		int temp;

		__asm__ __volatile__(
		"	.set	mips3					\n"
		"1:	ll	%0, %1		# atomic_add		\n"
		"	addu	%0, %2					\n"
		"	sc	%0, %1					\n"
		"	beqzl	%0, 1b					\n"
		"	.set	mips0					\n"
		: "=&r" (temp), "=m" (v->counter)
		: "Ir" (i), "m" (v->counter));
	} else if (cpu_has_llsc) {
		int temp;

		__asm__ __volatile__(
		"	.set	mips3					\n"
		"1:	ll	%0, %1		# atomic_add		\n"
		"	addu	%0, %2					\n"
		"	sc	%0, %1					\n"
		"	beqz	%0, 2f					\n"
		"	.subsection 2					\n"
		"2:	b	1b					\n"
		"	.previous					\n"
		"	.set	mips0					\n"
		: "=&r" (temp), "=m" (v->counter)
		: "Ir" (i), "m" (v->counter));
	} else {
		unsigned long flags;

		raw_local_irq_save(flags);
		v->counter += i;
		raw_local_irq_restore(flags);
	}
}

解释C语言嵌入汇编的格式:

__asm__ __volatile__(

汇编语句

输出寄存器

输入寄存器

会被修改的寄存器

)

除了汇编语句,其他部分都能省略。

输出寄存器:C语言表达式值或者内存地址

输入寄存器:C语言变量或者常数

会被修改的寄存器:汇编语句中修改的寄存器(除了输入输出用到的寄存器)。编译器需要知道这些寄存器,从而在这个汇编代码结束后,执行其他代码时遇到这些寄存器,需要重新加载更新寄存器的值。



另外还有一个位操作

这是为了解决以原子操作单个位,简称原子位。接口的形式:xxx_bit。

一般这些函数用来访问和修改一个共享标志时非常有用。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值