Blame view

lib/find_next_bit.c 5.26 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
  /* find_next_bit.c: fallback find next bit implementation
   *
   * 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.
   */
  
  #include <linux/bitops.h>
402344012   David Howells   [PATCH] frv: impl...
13
  #include <linux/module.h>
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
14
  #include <asm/types.h>
930ae745f   Akinobu Mita   [PATCH] bitops: g...
15
  #include <asm/byteorder.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
16

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
17
18
19
20
21
22
23
24
25
26
  #define BITOP_WORD(nr)		((nr) / BITS_PER_LONG)
  
  /**
   * 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 maximum size to search
   */
  unsigned long find_next_bit(const unsigned long *addr, unsigned long size,
  		unsigned long offset)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
27
  {
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
28
29
  	const unsigned long *p = addr + BITOP_WORD(offset);
  	unsigned long result = offset & ~(BITS_PER_LONG-1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
30
  	unsigned long tmp;
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
31
32
33
34
  	if (offset >= size)
  		return size;
  	size -= result;
  	offset %= BITS_PER_LONG;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
  	if (offset) {
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
36
37
38
39
40
41
42
43
44
45
46
47
48
49
  		tmp = *(p++);
  		tmp &= (~0UL << offset);
  		if (size < BITS_PER_LONG)
  			goto found_first;
  		if (tmp)
  			goto found_middle;
  		size -= BITS_PER_LONG;
  		result += BITS_PER_LONG;
  	}
  	while (size & ~(BITS_PER_LONG-1)) {
  		if ((tmp = *(p++)))
  			goto found_middle;
  		result += BITS_PER_LONG;
  		size -= BITS_PER_LONG;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
50
  	}
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
51
52
53
  	if (!size)
  		return result;
  	tmp = *p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
54

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
55
56
57
58
59
60
61
  found_first:
  	tmp &= (~0UL >> (BITS_PER_LONG - size));
  	if (tmp == 0UL)		/* Are any bits set? */
  		return result + size;	/* Nope. */
  found_middle:
  	return result + __ffs(tmp);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
62

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
63
  EXPORT_SYMBOL(find_next_bit);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
64

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
65
66
67
68
69
70
71
72
73
74
  /*
   * This implementation of find_{first,next}_zero_bit was stolen from
   * Linus' asm-alpha/bitops.h.
   */
  unsigned long find_next_zero_bit(const unsigned long *addr, unsigned long size,
  		unsigned long offset)
  {
  	const unsigned long *p = addr + BITOP_WORD(offset);
  	unsigned long result = offset & ~(BITS_PER_LONG-1);
  	unsigned long tmp;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
75

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
  	if (offset >= size)
  		return size;
  	size -= result;
  	offset %= BITS_PER_LONG;
  	if (offset) {
  		tmp = *(p++);
  		tmp |= ~0UL >> (BITS_PER_LONG - offset);
  		if (size < BITS_PER_LONG)
  			goto found_first;
  		if (~tmp)
  			goto found_middle;
  		size -= BITS_PER_LONG;
  		result += BITS_PER_LONG;
  	}
  	while (size & ~(BITS_PER_LONG-1)) {
  		if (~(tmp = *(p++)))
  			goto found_middle;
  		result += BITS_PER_LONG;
  		size -= BITS_PER_LONG;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
95
  	}
c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
96
97
98
  	if (!size)
  		return result;
  	tmp = *p;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
99

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
100
101
102
103
104
105
  found_first:
  	tmp |= ~0UL << size;
  	if (tmp == ~0UL)	/* Are any bits zero? */
  		return result + size;	/* Nope. */
  found_middle:
  	return result + ffz(tmp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
106
  }
402344012   David Howells   [PATCH] frv: impl...
107

c7f612cdf   Akinobu Mita   [PATCH] bitops: g...
108
  EXPORT_SYMBOL(find_next_zero_bit);
930ae745f   Akinobu Mita   [PATCH] bitops: g...
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
  
  #ifdef __BIG_ENDIAN
  
  /* include/linux/byteorder does not support "unsigned long" type */
  static inline unsigned long ext2_swabp(const unsigned long * x)
  {
  #if BITS_PER_LONG == 64
  	return (unsigned long) __swab64p((u64 *) x);
  #elif BITS_PER_LONG == 32
  	return (unsigned long) __swab32p((u32 *) x);
  #else
  #error BITS_PER_LONG not defined
  #endif
  }
  
  /* include/linux/byteorder doesn't support "unsigned long" type */
  static inline unsigned long ext2_swab(const unsigned long y)
  {
  #if BITS_PER_LONG == 64
  	return (unsigned long) __swab64((u64) y);
  #elif BITS_PER_LONG == 32
  	return (unsigned long) __swab32((u32) y);
  #else
  #error BITS_PER_LONG not defined
  #endif
  }
  
  unsigned long generic_find_next_zero_le_bit(const unsigned long *addr, unsigned
  		long size, unsigned long offset)
  {
  	const unsigned long *p = addr + BITOP_WORD(offset);
  	unsigned long result = offset & ~(BITS_PER_LONG - 1);
  	unsigned long tmp;
  
  	if (offset >= size)
  		return size;
  	size -= result;
  	offset &= (BITS_PER_LONG - 1UL);
  	if (offset) {
  		tmp = ext2_swabp(p++);
  		tmp |= (~0UL >> (BITS_PER_LONG - offset));
  		if (size < BITS_PER_LONG)
  			goto found_first;
  		if (~tmp)
  			goto found_middle;
  		size -= BITS_PER_LONG;
  		result += BITS_PER_LONG;
  	}
  
  	while (size & ~(BITS_PER_LONG - 1)) {
  		if (~(tmp = *(p++)))
  			goto found_middle_swap;
  		result += BITS_PER_LONG;
  		size -= BITS_PER_LONG;
  	}
  	if (!size)
  		return result;
  	tmp = ext2_swabp(p);
  found_first:
  	tmp |= ~0UL << size;
  	if (tmp == ~0UL)	/* Are any bits zero? */
  		return result + size; /* Nope. Skip ffz */
  found_middle:
  	return result + ffz(tmp);
  
  found_middle_swap:
  	return result + ffz(ext2_swab(tmp));
  }
  
  EXPORT_SYMBOL(generic_find_next_zero_le_bit);
aa02ad67d   Aneesh Kumar K.V   ext4: Add ext4_fi...
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
  unsigned long generic_find_next_le_bit(const unsigned long *addr, unsigned
  		long size, unsigned long offset)
  {
  	const unsigned long *p = addr + BITOP_WORD(offset);
  	unsigned long result = offset & ~(BITS_PER_LONG - 1);
  	unsigned long tmp;
  
  	if (offset >= size)
  		return size;
  	size -= result;
  	offset &= (BITS_PER_LONG - 1UL);
  	if (offset) {
  		tmp = ext2_swabp(p++);
  		tmp &= (~0UL << offset);
  		if (size < BITS_PER_LONG)
  			goto found_first;
  		if (tmp)
  			goto found_middle;
  		size -= BITS_PER_LONG;
  		result += BITS_PER_LONG;
  	}
  
  	while (size & ~(BITS_PER_LONG - 1)) {
  		tmp = *(p++);
  		if (tmp)
  			goto found_middle_swap;
  		result += BITS_PER_LONG;
  		size -= BITS_PER_LONG;
  	}
  	if (!size)
  		return result;
  	tmp = ext2_swabp(p);
  found_first:
  	tmp &= (~0UL >> (BITS_PER_LONG - size));
  	if (tmp == 0UL)		/* Are any bits set? */
  		return result + size; /* Nope. */
  found_middle:
  	return result + __ffs(tmp);
  
  found_middle_swap:
  	return result + __ffs(ext2_swab(tmp));
  }
  EXPORT_SYMBOL(generic_find_next_le_bit);
930ae745f   Akinobu Mita   [PATCH] bitops: g...
222
  #endif /* __BIG_ENDIAN */