aboutsummaryrefslogtreecommitdiffstats
path: root/include
diff options
context:
space:
mode:
Diffstat (limited to 'include')
-rw-r--r--include/asm-generic/bitops/find.h2
-rw-r--r--include/asm-x86/bitops.h6
-rw-r--r--include/asm-x86/bitops_64.h10
-rw-r--r--include/linux/bitops.h77
4 files changed, 79 insertions, 16 deletions
diff --git a/include/asm-generic/bitops/find.h b/include/asm-generic/bitops/find.h
index 72a51e5a12ef..1914e9742512 100644
--- a/include/asm-generic/bitops/find.h
+++ b/include/asm-generic/bitops/find.h
@@ -1,11 +1,13 @@
1#ifndef _ASM_GENERIC_BITOPS_FIND_H_ 1#ifndef _ASM_GENERIC_BITOPS_FIND_H_
2#define _ASM_GENERIC_BITOPS_FIND_H_ 2#define _ASM_GENERIC_BITOPS_FIND_H_
3 3
4#ifndef CONFIG_GENERIC_FIND_NEXT_BIT
4extern unsigned long find_next_bit(const unsigned long *addr, unsigned long 5extern unsigned long find_next_bit(const unsigned long *addr, unsigned long
5 size, unsigned long offset); 6 size, unsigned long offset);
6 7
7extern unsigned long find_next_zero_bit(const unsigned long *addr, unsigned 8extern unsigned long find_next_zero_bit(const unsigned long *addr, unsigned
8 long size, unsigned long offset); 9 long size, unsigned long offset);
10#endif
9 11
10#define find_first_bit(addr, size) find_next_bit((addr), (size), 0) 12#define find_first_bit(addr, size) find_next_bit((addr), (size), 0)
11#define find_first_zero_bit(addr, size) find_next_zero_bit((addr), (size), 0) 13#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 31e408de90c6..1ae7b270a1ef 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);
306#undef BIT_ADDR 306#undef BIT_ADDR
307#undef ADDR 307#undef ADDR
308 308
309unsigned long find_next_bit(const unsigned long *addr,
310 unsigned long size, unsigned long offset);
311unsigned long find_next_zero_bit(const unsigned long *addr,
312 unsigned long size, unsigned long offset);
313
314
315#ifdef CONFIG_X86_32 309#ifdef CONFIG_X86_32
316# include "bitops_32.h" 310# include "bitops_32.h"
317#else 311#else
diff --git a/include/asm-x86/bitops_64.h b/include/asm-x86/bitops_64.h
index 65b20fb2ae78..7118ef2cc4ec 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)
15 return val; 15 return val;
16} 16}
17 17
18#define find_next_bit(addr,size,off) \
19((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? \
20 ((off) + (__scanbit((*(unsigned long *)addr) >> (off),(size)-(off)))) : \
21 find_next_bit(addr,size,off)))
22
23#define find_next_zero_bit(addr,size,off) \
24((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? \
25 ((off)+(__scanbit(~(((*(unsigned long *)addr)) >> (off)),(size)-(off)))) : \
26 find_next_zero_bit(addr,size,off)))
27
28#define find_first_bit(addr, size) \ 18#define find_first_bit(addr, size) \
29 ((__builtin_constant_p((size)) && (size) <= BITS_PER_LONG \ 19 ((__builtin_constant_p((size)) && (size) <= BITS_PER_LONG \
30 ? (__scanbit(*(unsigned long *)(addr), (size))) \ 20 ? (__scanbit(*(unsigned long *)(addr), (size))) \
diff --git a/include/linux/bitops.h b/include/linux/bitops.h
index 40d54731de7e..3865f2c93bd8 100644
--- a/include/linux/bitops.h
+++ b/include/linux/bitops.h
@@ -112,4 +112,81 @@ static inline unsigned fls_long(unsigned long l)
112 return fls64(l); 112 return fls64(l);
113} 113}
114 114
115#ifdef __KERNEL__
116#ifdef CONFIG_GENERIC_FIND_NEXT_BIT
117extern unsigned long __find_next_bit(const unsigned long *addr,
118 unsigned long size, unsigned long offset);
119
120/**
121 * find_next_bit - find the next set bit in a memory region
122 * @addr: The address to base the search on
123 * @offset: The bitnumber to start searching at
124 * @size: The bitmap size in bits
125 */
126static __always_inline unsigned long
127find_next_bit(const unsigned long *addr, unsigned long size,
128 unsigned long offset)
129{
130 unsigned long value;
131
132 /* Avoid a function call if the bitmap size is a constant */
133 /* and not bigger than BITS_PER_LONG. */
134
135 /* insert a sentinel so that __ffs returns size if there */
136 /* are no set bits in the bitmap */
137 if (__builtin_constant_p(size) && (size < BITS_PER_LONG)) {
138 value = (*addr) & ((~0ul) << offset);
139 value |= (1ul << size);
140 return __ffs(value);
141 }
142
143 /* the result of __ffs(0) is undefined, so it needs to be */
144 /* handled separately */
145 if (__builtin_constant_p(size) && (size == BITS_PER_LONG)) {
146 value = (*addr) & ((~0ul) << offset);
147 return (value == 0) ? BITS_PER_LONG : __ffs(value);
148 }
149
150 /* size is not constant or too big */
151 return __find_next_bit(addr, size, offset);
152}
153
154extern unsigned long __find_next_zero_bit(const unsigned long *addr,
155 unsigned long size, unsigned long offset);
156
157/**
158 * find_next_zero_bit - find the next cleared bit in a memory region
159 * @addr: The address to base the search on
160 * @offset: The bitnumber to start searching at
161 * @size: The bitmap size in bits
162 */
163static __always_inline unsigned long
164find_next_zero_bit(const unsigned long *addr, unsigned long size,
165 unsigned long offset)
166{
167 unsigned long value;
168
169 /* Avoid a function call if the bitmap size is a constant */
170 /* and not bigger than BITS_PER_LONG. */
171
172 /* insert a sentinel so that __ffs returns size if there */
173 /* are no set bits in the bitmap */
174 if (__builtin_constant_p(size) && (size < BITS_PER_LONG)) {
175 value = (~(*addr)) & ((~0ul) << offset);
176 value |= (1ul << size);
177 return __ffs(value);
178 }
179
180 /* the result of __ffs(0) is undefined, so it needs to be */
181 /* handled separately */
182 if (__builtin_constant_p(size) && (size == BITS_PER_LONG)) {
183 value = (~(*addr)) & ((~0ul) << offset);
184 return (value == 0) ? BITS_PER_LONG : __ffs(value);
185 }
186
187 /* size is not constant or too big */
188 return __find_next_zero_bit(addr, size, offset);
189}
190#endif /* CONFIG_GENERIC_FIND_NEXT_BIT */
191#endif /* __KERNEL__ */
115#endif 192#endif