2008-11-03 23:34:17 +03:00
|
|
|
#ifndef _SPL_ATOMIC_COMPAT_H
|
|
|
|
#define _SPL_ATOMIC_COMPAT_H
|
|
|
|
|
|
|
|
#include <asm/atomic.h>
|
2009-03-18 00:55:59 +03:00
|
|
|
#include <spl_config.h>
|
2008-11-03 23:34:17 +03:00
|
|
|
|
|
|
|
#ifndef HAVE_ATOMIC64_T
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
spinlock_t lock;
|
|
|
|
__s64 val;
|
|
|
|
} atomic64_t;
|
|
|
|
|
|
|
|
#define ATOMIC64_INIT(i) { .lock = SPIN_LOCK_UNLOCKED, .val = (i) }
|
|
|
|
|
|
|
|
static inline void atomic64_add(__s64 i, atomic64_t *v)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
v->val += i;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic64_sub(__s64 i, atomic64_t *v)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
v->val -= i;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
}
|
|
|
|
|
2009-12-01 21:15:27 +03:00
|
|
|
#define atomic64_inc(v) (atomic64_add(1, (v)))
|
|
|
|
#define atomic64_dec(v) (atomic64_sub(1, (v)))
|
|
|
|
|
|
|
|
static inline __s64 atomic64_add_return(__s64 i, atomic64_t *v)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
__s64 ret;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
v->val += i;
|
|
|
|
ret = v->val;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline __s64 atomic64_sub_return(__s64 i, atomic64_t *v)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
__s64 ret;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
v->val -= i;
|
|
|
|
ret = v->val;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define atomic64_inc_return(v) (atomic64_add_return(1, (v)))
|
|
|
|
#define atomic64_dec_return(v) (atomic64_sub_return(1, (v)))
|
|
|
|
|
2008-11-03 23:34:17 +03:00
|
|
|
static inline __s64 atomic64_read(atomic64_t *v)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
__s64 r;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
r = v->val;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void atomic64_set(atomic64_t *v, __s64 i)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&v->lock, flags);
|
|
|
|
v->val = i;
|
|
|
|
spin_unlock_irqrestore(&v->lock, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* HAVE_ATOMIC64_T */
|
|
|
|
|
2009-10-30 23:53:17 +03:00
|
|
|
#ifndef HAVE_ATOMIC64_CMPXCHG
|
|
|
|
#define atomic64_cmpxchg(v, o, n) (cmpxchg(&((v)->counter), (o), (n)))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef HAVE_ATOMIC64_XCHG
|
|
|
|
#define atomic64_xchg(v, n) (xchg(&((v)->counter), n))
|
|
|
|
#endif
|
|
|
|
|
2008-11-03 23:34:17 +03:00
|
|
|
#endif /* _SPL_ATOMIC_COMPAT_H */
|
|
|
|
|