atomic.h

来自「linux 内核源代码」· C头文件 代码 · 共 284 行

H
284
字号
/* atomic.h: atomic operation emulation for FR-V * * For an explanation of how atomic ops work in this arch, see: *   Documentation/fujitsu/frv/atomic-ops.txt * * Copyright (C) 2004 Red Hat, Inc. All Rights Reserved. * Written by David Howells (dhowells@redhat.com) * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version * 2 of the License, or (at your option) any later version. */#ifndef _ASM_ATOMIC_H#define _ASM_ATOMIC_H#include <linux/types.h>#include <asm/spr-regs.h>#include <asm/system.h>#ifdef CONFIG_SMP#error not SMP safe#endif/* * Atomic operations that C can't guarantee us.  Useful for * resource counting etc.. * * We do not have SMP systems, so we don't have to deal with that. *//* Atomic operations are already serializing */#define smp_mb__before_atomic_dec()	barrier()#define smp_mb__after_atomic_dec()	barrier()#define smp_mb__before_atomic_inc()	barrier()#define smp_mb__after_atomic_inc()	barrier()typedef struct {	int counter;} atomic_t;#define ATOMIC_INIT(i)		{ (i) }#define atomic_read(v)		((v)->counter)#define atomic_set(v, i)	(((v)->counter) = (i))#ifndef CONFIG_FRV_OUTOFLINE_ATOMIC_OPSstatic inline int atomic_add_return(int i, atomic_t *v){	unsigned long val;	asm("0:						\n"	    "	orcc		gr0,gr0,gr0,icc3	\n"	/* set ICC3.Z */	    "	ckeq		icc3,cc7		\n"	    "	ld.p		%M0,%1			\n"	/* LD.P/ORCR must be atomic */	    "	orcr		cc7,cc7,cc3		\n"	/* set CC3 to true */	    "	add%I2		%1,%2,%1		\n"	    "	cst.p		%1,%M0		,cc3,#1	\n"	    "	corcc		gr29,gr29,gr0	,cc3,#1	\n"	/* clear ICC3.Z if store happens */	    "	beq		icc3,#0,0b		\n"	    : "+U"(v->counter), "=&r"(val)	    : "NPr"(i)	    : "memory", "cc7", "cc3", "icc3"	    );	return val;}static inline int atomic_sub_return(int i, atomic_t *v){	unsigned long val;	asm("0:						\n"	    "	orcc		gr0,gr0,gr0,icc3	\n"	/* set ICC3.Z */	    "	ckeq		icc3,cc7		\n"	    "	ld.p		%M0,%1			\n"	/* LD.P/ORCR must be atomic */	    "	orcr		cc7,cc7,cc3		\n"	/* set CC3 to true */	    "	sub%I2		%1,%2,%1		\n"	    "	cst.p		%1,%M0		,cc3,#1	\n"	    "	corcc		gr29,gr29,gr0	,cc3,#1	\n"	/* clear ICC3.Z if store happens */	    "	beq		icc3,#0,0b		\n"	    : "+U"(v->counter), "=&r"(val)	    : "NPr"(i)	    : "memory", "cc7", "cc3", "icc3"	    );	return val;}#elseextern int atomic_add_return(int i, atomic_t *v);extern int atomic_sub_return(int i, atomic_t *v);#endifstatic inline int atomic_add_negative(int i, atomic_t *v){	return atomic_add_return(i, v) < 0;}static inline void atomic_add(int i, atomic_t *v){	atomic_add_return(i, v);}static inline void atomic_sub(int i, atomic_t *v){	atomic_sub_return(i, v);}static inline void atomic_inc(atomic_t *v){	atomic_add_return(1, v);}static inline void atomic_dec(atomic_t *v){	atomic_sub_return(1, v);}#define atomic_dec_return(v)		atomic_sub_return(1, (v))#define atomic_inc_return(v)		atomic_add_return(1, (v))#define atomic_sub_and_test(i,v)	(atomic_sub_return((i), (v)) == 0)#define atomic_dec_and_test(v)		(atomic_sub_return(1, (v)) == 0)#define atomic_inc_and_test(v)		(atomic_add_return(1, (v)) == 0)#ifndef CONFIG_FRV_OUTOFLINE_ATOMIC_OPSstatic inlineunsigned long atomic_test_and_ANDNOT_mask(unsigned long mask, volatile unsigned long *v){	unsigned long old, tmp;	asm volatile(		"0:						\n"		"	orcc		gr0,gr0,gr0,icc3	\n"	/* set ICC3.Z */		"	ckeq		icc3,cc7		\n"		"	ld.p		%M0,%1			\n"	/* LD.P/ORCR are atomic */		"	orcr		cc7,cc7,cc3		\n"	/* set CC3 to true */		"	and%I3		%1,%3,%2		\n"		"	cst.p		%2,%M0		,cc3,#1	\n"	/* if store happens... */		"	corcc		gr29,gr29,gr0	,cc3,#1	\n"	/* ... clear ICC3.Z */		"	beq		icc3,#0,0b		\n"		: "+U"(*v), "=&r"(old), "=r"(tmp)		: "NPr"(~mask)		: "memory", "cc7", "cc3", "icc3"		);	return old;}static inlineunsigned long atomic_test_and_OR_mask(unsigned long mask, volatile unsigned long *v){	unsigned long old, tmp;	asm volatile(		"0:						\n"		"	orcc		gr0,gr0,gr0,icc3	\n"	/* set ICC3.Z */		"	ckeq		icc3,cc7		\n"		"	ld.p		%M0,%1			\n"	/* LD.P/ORCR are atomic */		"	orcr		cc7,cc7,cc3		\n"	/* set CC3 to true */		"	or%I3		%1,%3,%2		\n"		"	cst.p		%2,%M0		,cc3,#1	\n"	/* if store happens... */		"	corcc		gr29,gr29,gr0	,cc3,#1	\n"	/* ... clear ICC3.Z */		"	beq		icc3,#0,0b		\n"		: "+U"(*v), "=&r"(old), "=r"(tmp)		: "NPr"(mask)		: "memory", "cc7", "cc3", "icc3"		);	return old;}static inlineunsigned long atomic_test_and_XOR_mask(unsigned long mask, volatile unsigned long *v){	unsigned long old, tmp;	asm volatile(		"0:						\n"		"	orcc		gr0,gr0,gr0,icc3	\n"	/* set ICC3.Z */		"	ckeq		icc3,cc7		\n"		"	ld.p		%M0,%1			\n"	/* LD.P/ORCR are atomic */		"	orcr		cc7,cc7,cc3		\n"	/* set CC3 to true */		"	xor%I3		%1,%3,%2		\n"		"	cst.p		%2,%M0		,cc3,#1	\n"	/* if store happens... */		"	corcc		gr29,gr29,gr0	,cc3,#1	\n"	/* ... clear ICC3.Z */		"	beq		icc3,#0,0b		\n"		: "+U"(*v), "=&r"(old), "=r"(tmp)		: "NPr"(mask)		: "memory", "cc7", "cc3", "icc3"		);	return old;}#elseextern unsigned long atomic_test_and_ANDNOT_mask(unsigned long mask, volatile unsigned long *v);extern unsigned long atomic_test_and_OR_mask(unsigned long mask, volatile unsigned long *v);extern unsigned long atomic_test_and_XOR_mask(unsigned long mask, volatile unsigned long *v);#endif#define atomic_clear_mask(mask, v)	atomic_test_and_ANDNOT_mask((mask), (v))#define atomic_set_mask(mask, v)	atomic_test_and_OR_mask((mask), (v))/*****************************************************************************//* * exchange value with memory */#ifndef CONFIG_FRV_OUTOFLINE_ATOMIC_OPS#define xchg(ptr, x)								\({										\	__typeof__(ptr) __xg_ptr = (ptr);					\	__typeof__(*(ptr)) __xg_orig;						\										\	switch (sizeof(__xg_orig)) {						\	case 4:									\		asm volatile(							\			"swap%I0 %M0,%1"					\			: "+m"(*__xg_ptr), "=r"(__xg_orig)			\			: "1"(x)						\			: "memory"						\			);							\		break;								\										\	default:								\		__xg_orig = (__typeof__(__xg_orig))0;				\		asm volatile("break");						\		break;								\	}									\										\	__xg_orig;								\})#elseextern uint32_t __xchg_32(uint32_t i, volatile void *v);#define xchg(ptr, x)										\({												\	__typeof__(ptr) __xg_ptr = (ptr);							\	__typeof__(*(ptr)) __xg_orig;								\												\	switch (sizeof(__xg_orig)) {								\	case 4: __xg_orig = (__typeof__(*(ptr))) __xchg_32((uint32_t) x, __xg_ptr);	break;	\	default:										\		__xg_orig = (__typeof__(__xg_orig))0;									\		asm volatile("break");								\		break;										\	}											\	__xg_orig;										\})#endif#define tas(ptr) (xchg((ptr), 1))#define atomic_cmpxchg(v, old, new) (cmpxchg(&((v)->counter), old, new))#define atomic_xchg(v, new) (xchg(&((v)->counter), new))static __inline__ int atomic_add_unless(atomic_t *v, int a, int u){	int c, old;	c = atomic_read(v);	for (;;) {		if (unlikely(c == (u)))			break;		old = atomic_cmpxchg((v), c, c + (a));		if (likely(old == c))			break;		c = old;	}	return c != (u);}#define atomic_inc_not_zero(v) atomic_add_unless((v), 1, 0)#include <asm-generic/atomic.h>#endif /* _ASM_ATOMIC_H */

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?