diff options
Diffstat (limited to 'include')
| -rw-r--r-- | include/asm-generic/bitops/find.h | 2 | ||||
| -rw-r--r-- | include/asm-x86/bitops.h | 6 | ||||
| -rw-r--r-- | include/asm-x86/bitops_64.h | 10 | ||||
| -rw-r--r-- | include/linux/bitops.h | 77 | 
4 files changed, 79 insertions, 16 deletions
diff --git a/include/asm-generic/bitops/find.h b/include/asm-generic/bitops/find.h index 72a51e5a12e..1914e974251 100644 --- a/include/asm-generic/bitops/find.h +++ b/include/asm-generic/bitops/find.h @@ -1,11 +1,13 @@  #ifndef _ASM_GENERIC_BITOPS_FIND_H_  #define _ASM_GENERIC_BITOPS_FIND_H_ +#ifndef CONFIG_GENERIC_FIND_NEXT_BIT  extern unsigned long find_next_bit(const unsigned long *addr, unsigned long  		size, unsigned long offset);  extern unsigned long find_next_zero_bit(const unsigned long *addr, unsigned  		long size, unsigned long offset); +#endif  #define find_first_bit(addr, size) find_next_bit((addr), (size), 0)  #define find_first_zero_bit(addr, size) find_next_zero_bit((addr), (size), 0) diff --git a/include/asm-x86/bitops.h b/include/asm-x86/bitops.h index 31e408de90c..1ae7b270a1e 100644 --- a/include/asm-x86/bitops.h +++ b/include/asm-x86/bitops.h @@ -306,12 +306,6 @@ static int test_bit(int nr, const volatile unsigned long *addr);  #undef BIT_ADDR  #undef ADDR -unsigned long find_next_bit(const unsigned long *addr, -		unsigned long size, unsigned long offset); -unsigned long find_next_zero_bit(const unsigned long *addr, -		unsigned long size, unsigned long offset); - -  #ifdef CONFIG_X86_32  # include "bitops_32.h"  #else diff --git a/include/asm-x86/bitops_64.h b/include/asm-x86/bitops_64.h index 65b20fb2ae7..7118ef2cc4e 100644 --- a/include/asm-x86/bitops_64.h +++ b/include/asm-x86/bitops_64.h @@ -15,16 +15,6 @@ static inline long __scanbit(unsigned long val, unsigned long max)  	return val;  } -#define find_next_bit(addr,size,off) \ -((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? 	  \ -  ((off) + (__scanbit((*(unsigned long *)addr) >> (off),(size)-(off)))) : \ -	find_next_bit(addr,size,off))) - -#define find_next_zero_bit(addr,size,off) \ -((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? 	  \ -  ((off)+(__scanbit(~(((*(unsigned long *)addr)) >> (off)),(size)-(off)))) : \ -	find_next_zero_bit(addr,size,off))) -  #define find_first_bit(addr, size)					\  	((__builtin_constant_p((size)) && (size) <= BITS_PER_LONG	\  	  ? (__scanbit(*(unsigned long *)(addr), (size)))		\ diff --git a/include/linux/bitops.h b/include/linux/bitops.h index 40d54731de7..3865f2c93bd 100644 --- a/include/linux/bitops.h +++ b/include/linux/bitops.h @@ -112,4 +112,81 @@ static inline unsigned fls_long(unsigned long l)  	return fls64(l);  } +#ifdef __KERNEL__ +#ifdef CONFIG_GENERIC_FIND_NEXT_BIT +extern unsigned long __find_next_bit(const unsigned long *addr, +		unsigned long size, unsigned long offset); + +/** + * find_next_bit - find the next set bit in a memory region + * @addr: The address to base the search on + * @offset: The bitnumber to start searching at + * @size: The bitmap size in bits + */ +static __always_inline unsigned long +find_next_bit(const unsigned long *addr, unsigned long size, +		unsigned long offset) +{ +	unsigned long value; + +	/* Avoid a function call if the bitmap size is a constant */ +	/* and not bigger than BITS_PER_LONG. */ + +	/* insert a sentinel so that __ffs returns size if there */ +	/* are no set bits in the bitmap */ +	if (__builtin_constant_p(size) && (size < BITS_PER_LONG)) { +		value = (*addr) & ((~0ul) << offset); +		value |= (1ul << size); +		return __ffs(value); +	} + +	/* the result of __ffs(0) is undefined, so it needs to be */ +	/* handled separately */ +	if (__builtin_constant_p(size) && (size == BITS_PER_LONG)) { +		value = (*addr) & ((~0ul) << offset); +		return (value == 0) ? BITS_PER_LONG : __ffs(value); +	} + +	/* size is not constant or too big */ +	return __find_next_bit(addr, size, offset); +} + +extern unsigned long __find_next_zero_bit(const unsigned long *addr, +		unsigned long size, unsigned long offset); + +/** + * find_next_zero_bit - find the next cleared bit in a memory region + * @addr: The address to base the search on + * @offset: The bitnumber to start searching at + * @size: The bitmap size in bits + */ +static __always_inline unsigned long +find_next_zero_bit(const unsigned long *addr, unsigned long size, +		unsigned long offset) +{ +	unsigned long value; + +	/* Avoid a function call if the bitmap size is a constant */ +	/* and not bigger than BITS_PER_LONG. */ + +	/* insert a sentinel so that __ffs returns size if there */ +	/* are no set bits in the bitmap */ +	if (__builtin_constant_p(size) && (size < BITS_PER_LONG)) { +		value = (~(*addr)) & ((~0ul) << offset); +		value |= (1ul << size); +		return __ffs(value); +	} + +	/* the result of __ffs(0) is undefined, so it needs to be */ +	/* handled separately */ +	if (__builtin_constant_p(size) && (size == BITS_PER_LONG)) { +		value = (~(*addr)) & ((~0ul) << offset); +		return (value == 0) ? BITS_PER_LONG : __ffs(value); +	} + +	/* size is not constant or too big */ +	return __find_next_zero_bit(addr, size, offset); +} +#endif /* CONFIG_GENERIC_FIND_NEXT_BIT */ +#endif /* __KERNEL__ */  #endif  |