2012-12-24 15:36:14 +04:00
|
|
|
#ifndef __CR_ATOMIC_H__
|
|
|
|
#define __CR_ATOMIC_H__
|
2011-11-18 16:09:01 +04:00
|
|
|
|
2016-10-24 14:58:06 +03:00
|
|
|
#include "common/arch/x86/asm/cmpxchg.h"
|
2011-12-26 20:33:09 +04:00
|
|
|
|
2012-09-11 17:55:15 +04:00
|
|
|
typedef struct {
|
2013-08-16 19:31:55 +04:00
|
|
|
int counter;
|
2012-09-11 17:55:15 +04:00
|
|
|
} atomic_t;
|
|
|
|
|
2013-08-16 19:31:55 +04:00
|
|
|
#define ATOMIC_INIT(i) { (i) }
|
|
|
|
|
|
|
|
static inline int atomic_read(const atomic_t *v)
|
|
|
|
{
|
|
|
|
return (*(volatile int *)&(v)->counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic_set(atomic_t *v, int i)
|
|
|
|
{
|
|
|
|
v->counter = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic_add(int i, atomic_t *v)
|
|
|
|
{
|
|
|
|
asm volatile(LOCK_PREFIX "addl %1,%0"
|
|
|
|
: "+m" (v->counter)
|
|
|
|
: "ir" (i));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic_sub(int i, atomic_t *v)
|
|
|
|
{
|
|
|
|
asm volatile(LOCK_PREFIX "subl %1,%0"
|
|
|
|
: "+m" (v->counter)
|
|
|
|
: "ir" (i));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic_inc(atomic_t *v)
|
|
|
|
{
|
|
|
|
asm volatile(LOCK_PREFIX "incl %0"
|
|
|
|
: "+m" (v->counter));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic_dec(atomic_t *v)
|
|
|
|
{
|
|
|
|
asm volatile(LOCK_PREFIX "decl %0"
|
|
|
|
: "+m" (v->counter));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int atomic_dec_and_test(atomic_t *v)
|
|
|
|
{
|
|
|
|
unsigned char c;
|
|
|
|
|
|
|
|
asm volatile(LOCK_PREFIX "decl %0; sete %1"
|
|
|
|
: "+m" (v->counter), "=qm" (c)
|
|
|
|
: : "memory");
|
|
|
|
return c != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int atomic_add_return(int i, atomic_t *v)
|
|
|
|
{
|
|
|
|
return i + xadd(&v->counter, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int atomic_sub_return(int i, atomic_t *v)
|
|
|
|
{
|
|
|
|
return atomic_add_return(-i, v);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define atomic_inc_return(v) (atomic_add_return(1, v))
|
|
|
|
#define atomic_dec_return(v) (atomic_sub_return(1, v))
|
2012-09-11 17:55:23 +04:00
|
|
|
|
2014-12-03 19:27:59 +02:00
|
|
|
static inline int atomic_cmpxchg(atomic_t *v, int old, int new)
|
|
|
|
{
|
|
|
|
return cmpxchg(&v->counter, old, new);
|
|
|
|
}
|
|
|
|
|
2012-12-25 22:40:24 +04:00
|
|
|
#endif /* __CR_ATOMIC_H__ */
|