atomic_add实现原子加操作,代码如下,输入参数:couter地址(被加数),add加数,返回值:counter地址内原有输入
int atomic_add(volatile int *count, int add)
{
#ifdef __linux__
__asm__ __volatile__(
"lock xadd %0, (%1);"
: "=a"(add)
: "r"(count), "a"(add)
: "memory"
);
#else
#error "To-Do "
#endif
return add;
}
汇编部分的含义
lock xadd %0 (%1) 交换操作数,并将结果保存在dest操作数中。(AT&T xadd source, dest)
其中%0由 "=a"(add)得其为eax,%1由"r"(count)得知其为编译器为其非配的寄存器,未知,便于理解可以认为是ebx"=a"(add)含义是输出结果保存在add中,与之关联的是eax寄存器,相当于mov add %eax设传入的count地址内的数据为10,其地址为0x123456 add为3,
因此整个过程为:
1. eax = 3("a"(add)) ebx=0x123456
2. xadd %eax (0x123456) 根据xadd的语义,则eax最后为(0x123456)即为10,ebx=10+3=13;
3. mov add %eax 则最后add保存的是10
4. return add 则最后返回的是10,即未加之前的值。
注:"memory"表示操作过程中内存数据把发生变化,lock为对内存的原子操作