2011-12-22 15:19:25 +00:00
|
|
|
/*
|
|
|
|
* \brief Atomic operations for x86
|
|
|
|
* \author Norman Feske
|
|
|
|
* \date 2006-07-26
|
|
|
|
*
|
|
|
|
* Based on l4util/include/ARCH-x86/atomic_arch.h.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2013-01-10 20:44:47 +00:00
|
|
|
* Copyright (C) 2006-2013 Genode Labs GmbH
|
2011-12-22 15:19:25 +00:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
|
|
* under the terms of the GNU General Public License version 2.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _INCLUDE__X86__CPU__ATOMIC_H_
|
|
|
|
#define _INCLUDE__X86__CPU__ATOMIC_H_
|
|
|
|
|
|
|
|
namespace Genode {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Atomic compare and exchange
|
|
|
|
*
|
|
|
|
* This function compares the value at dest with cmp_val.
|
|
|
|
* If both values are equal, dest is set to new_val. If
|
|
|
|
* both values are different, the value at dest remains
|
|
|
|
* unchanged.
|
|
|
|
*
|
2014-11-14 09:44:56 +00:00
|
|
|
* Note, that cmpxchg() represents a memory barrier.
|
|
|
|
*
|
2011-12-22 15:19:25 +00:00
|
|
|
* \return 1 if the value was successfully changed to new_val,
|
|
|
|
* 0 if cmp_val and the value at dest differ.
|
|
|
|
*/
|
|
|
|
inline int cmpxchg(volatile int *dest, int cmp_val, int new_val)
|
|
|
|
{
|
|
|
|
int tmp;
|
|
|
|
|
|
|
|
__asm__ __volatile__
|
|
|
|
(
|
|
|
|
"lock cmpxchgl %1, %3 \n\t"
|
|
|
|
:
|
|
|
|
"=a" (tmp) /* 0 EAX, return val */
|
|
|
|
:
|
|
|
|
"r" (new_val), /* 1 reg, new value */
|
|
|
|
"0" (cmp_val), /* 2 EAX, compare value */
|
|
|
|
"m" (*dest) /* 3 mem, destination operand */
|
|
|
|
:
|
|
|
|
"memory", "cc"
|
|
|
|
);
|
|
|
|
|
|
|
|
return tmp == cmp_val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__X86__CPU__ATOMIC_H_ */
|