2008-02-08 15:19:31 +03:00
/* MN10300 bit operations
*
* Copyright ( C ) 2007 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 Licence
* as published by the Free Software Foundation ; either version
* 2 of the Licence , or ( at your option ) any later version .
*
* These have to be done with inline assembly : that way the bit - setting
* is guaranteed to be atomic . All bit operations return 0 if the bit
* was cleared before the operation and ! = 0 if it was not .
*
* bit 0 is the LSB of addr ; bit 32 is the LSB of ( addr + 1 ) .
*/
# ifndef __ASM_BITOPS_H
# define __ASM_BITOPS_H
# include <asm/cpu-regs.h>
# define smp_mb__before_clear_bit() barrier()
# define smp_mb__after_clear_bit() barrier()
/*
* set bit
*/
# define __set_bit(nr, addr) \
( { \
volatile unsigned char * _a = ( unsigned char * ) ( addr ) ; \
const unsigned shift = ( nr ) & 7 ; \
_a + = ( nr ) > > 3 ; \
\
asm volatile ( " bset %2,(%1) # set_bit reg " \
: " =m " ( * _a ) \
: " a " ( _a ) , " d " ( 1 < < shift ) , " m " ( * _a ) \
: " memory " , " cc " ) ; \
} )
# define set_bit(nr, addr) __set_bit((nr), (addr))
/*
* clear bit
*/
# define ___clear_bit(nr, addr) \
( { \
volatile unsigned char * _a = ( unsigned char * ) ( addr ) ; \
const unsigned shift = ( nr ) & 7 ; \
_a + = ( nr ) > > 3 ; \
\
asm volatile ( " bclr %2,(%1) # clear_bit reg " \
: " =m " ( * _a ) \
: " a " ( _a ) , " d " ( 1 < < shift ) , " m " ( * _a ) \
: " memory " , " cc " ) ; \
} )
# define clear_bit(nr, addr) ___clear_bit((nr), (addr))
2010-10-27 20:28:34 +04:00
static inline void __clear_bit ( unsigned long nr , volatile void * addr )
2008-02-08 15:19:31 +03:00
{
unsigned int * a = ( unsigned int * ) addr ;
int mask ;
a + = nr > > 5 ;
mask = 1 < < ( nr & 0x1f ) ;
* a & = ~ mask ;
}
/*
* test bit
*/
2010-10-27 20:28:34 +04:00
static inline int test_bit ( unsigned long nr , const volatile void * addr )
2008-02-08 15:19:31 +03:00
{
2010-10-27 20:28:36 +04:00
return 1UL & ( ( ( const volatile unsigned int * ) addr ) [ nr > > 5 ] > > ( nr & 31 ) ) ;
2008-02-08 15:19:31 +03:00
}
/*
* change bit
*/
2010-10-27 20:28:34 +04:00
static inline void __change_bit ( unsigned long nr , volatile void * addr )
2008-02-08 15:19:31 +03:00
{
int mask ;
unsigned int * a = ( unsigned int * ) addr ;
a + = nr > > 5 ;
mask = 1 < < ( nr & 0x1f ) ;
* a ^ = mask ;
}
2010-10-27 20:28:34 +04:00
extern void change_bit ( unsigned long nr , volatile void * addr ) ;
2008-02-08 15:19:31 +03:00
/*
* test and set bit
*/
# define __test_and_set_bit(nr,addr) \
( { \
volatile unsigned char * _a = ( unsigned char * ) ( addr ) ; \
const unsigned shift = ( nr ) & 7 ; \
unsigned epsw ; \
_a + = ( nr ) > > 3 ; \
\
asm volatile ( " bset %3,(%2) # test_set_bit reg \n " \
" mov epsw,%1 " \
: " =m " ( * _a ) , " =d " ( epsw ) \
: " a " ( _a ) , " d " ( 1 < < shift ) , " m " ( * _a ) \
: " memory " , " cc " ) ; \
\
! ( epsw & EPSW_FLAG_Z ) ; \
} )
# define test_and_set_bit(nr, addr) __test_and_set_bit((nr), (addr))
/*
* test and clear bit
*/
# define __test_and_clear_bit(nr, addr) \
( { \
volatile unsigned char * _a = ( unsigned char * ) ( addr ) ; \
const unsigned shift = ( nr ) & 7 ; \
unsigned epsw ; \
_a + = ( nr ) > > 3 ; \
\
asm volatile ( " bclr %3,(%2) # test_clear_bit reg \n " \
" mov epsw,%1 " \
: " =m " ( * _a ) , " =d " ( epsw ) \
: " a " ( _a ) , " d " ( 1 < < shift ) , " m " ( * _a ) \
: " memory " , " cc " ) ; \
\
! ( epsw & EPSW_FLAG_Z ) ; \
} )
# define test_and_clear_bit(nr, addr) __test_and_clear_bit((nr), (addr))
/*
* test and change bit
*/
2010-10-27 20:28:34 +04:00
static inline int __test_and_change_bit ( unsigned long nr , volatile void * addr )
2008-02-08 15:19:31 +03:00
{
int mask , retval ;
unsigned int * a = ( unsigned int * ) addr ;
a + = nr > > 5 ;
mask = 1 < < ( nr & 0x1f ) ;
retval = ( mask & * a ) ! = 0 ;
* a ^ = mask ;
return retval ;
}
2010-10-27 20:28:34 +04:00
extern int test_and_change_bit ( unsigned long nr , volatile void * addr ) ;
2008-02-08 15:19:31 +03:00
# include <asm-generic/bitops/lock.h>
# ifdef __KERNEL__
/**
* __ffs - find first bit set
* @ x : the word to search
*
* - return 31. .0 to indicate bit 31. .0 most least significant bit set
* - if no bits are set in x , the result is undefined
*/
static inline __attribute__ ( ( const ) )
unsigned long __ffs ( unsigned long x )
{
int bit ;
2010-01-09 01:43:16 +03:00
asm ( " bsch %2,%0 " : " =r " ( bit ) : " 0 " ( 0 ) , " r " ( x & - x ) : " cc " ) ;
2008-02-08 15:19:31 +03:00
return bit ;
}
/*
* special slimline version of fls ( ) for calculating ilog2_u32 ( )
* - note : no protection against n = = 0
*/
static inline __attribute__ ( ( const ) )
int __ilog2_u32 ( u32 n )
{
int bit ;
2010-01-09 01:43:16 +03:00
asm ( " bsch %2,%0 " : " =r " ( bit ) : " 0 " ( 0 ) , " r " ( n ) : " cc " ) ;
2008-02-08 15:19:31 +03:00
return bit ;
}
/**
* fls - find last bit set
* @ x : the word to search
*
* This is defined the same way as ffs :
* - return 32. .1 to indicate bit 31. .0 most significant bit set
* - return 0 to indicate no bits set
*/
static inline __attribute__ ( ( const ) )
int fls ( int x )
{
return ( x ! = 0 ) ? __ilog2_u32 ( x ) + 1 : 0 ;
}
2009-01-03 08:49:03 +03:00
/**
* __fls - find last ( most - significant ) set bit in a long word
* @ word : the word to search
*
* Undefined if no set bit exists , so code should check against 0 first .
*/
static inline unsigned long __fls ( unsigned long word )
{
return __ilog2_u32 ( word ) ;
}
2008-02-08 15:19:31 +03:00
/**
* ffs - find first bit set
* @ x : the word to search
*
* - return 32. .1 to indicate bit 31. .0 most least significant bit set
* - return 0 to indicate no bits set
*/
static inline __attribute__ ( ( const ) )
int ffs ( int x )
{
/* Note: (x & -x) gives us a mask that is the least significant
* ( rightmost ) 1 - bit of the value in x .
*/
return fls ( x & - x ) ;
}
# include <asm-generic/bitops/ffz.h>
# include <asm-generic/bitops/fls64.h>
# include <asm-generic/bitops/find.h>
# include <asm-generic/bitops/sched.h>
# include <asm-generic/bitops/hweight.h>
2011-07-27 03:09:04 +04:00
# include <asm-generic/bitops/ext2-atomic-setbit.h>
bitops: introduce little-endian bitops for most architectures
Introduce little-endian bit operations to the big-endian architectures
which do not have native little-endian bit operations and the
little-endian architectures. (alpha, avr32, blackfin, cris, frv, h8300,
ia64, m32r, mips, mn10300, parisc, sh, sparc, tile, x86, xtensa)
These architectures can just include generic implementation
(asm-generic/bitops/le.h).
Signed-off-by: Akinobu Mita <akinobu.mita@gmail.com>
Cc: Richard Henderson <rth@twiddle.net>
Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
Cc: Mikael Starvik <starvik@axis.com>
Cc: David Howells <dhowells@redhat.com>
Cc: Yoshinori Sato <ysato@users.sourceforge.jp>
Cc: "Luck, Tony" <tony.luck@intel.com>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Matthew Wilcox <willy@debian.org>
Cc: Grant Grundler <grundler@parisc-linux.org>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Kazumoto Kojima <kkojima@rr.iij4u.or.jp>
Cc: Hirokazu Takata <takata@linux-m32r.org>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Chris Zankel <chris@zankel.net>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Thomas Gleixner <tglx@linutronix.de>
Acked-by: Hans-Christian Egtvedt <hans-christian.egtvedt@atmel.com>
Acked-by: "H. Peter Anvin" <hpa@zytor.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-03-24 02:42:02 +03:00
# include <asm-generic/bitops/le.h>
2008-02-08 15:19:31 +03:00
# endif /* __KERNEL__ */
# endif /* __ASM_BITOPS_H */