]> git.proxmox.com Git - mirror_ubuntu-artful-kernel.git/blame - lib/find_bit.c
Merge tag 'mmc-v4.13-rc7' of git://git.kernel.org/pub/scm/linux/kernel/git/ulfh/mmc
[mirror_ubuntu-artful-kernel.git] / lib / find_bit.c
CommitLineData
8f6f19dd 1/* bit search implementation
1da177e4
LT
2 *
3 * Copyright (C) 2004 Red Hat, Inc. All Rights Reserved.
4 * Written by David Howells (dhowells@redhat.com)
5 *
8f6f19dd
YN
6 * Copyright (C) 2008 IBM Corporation
7 * 'find_last_bit' is written by Rusty Russell <rusty@rustcorp.com.au>
8 * (Inspired by David Howell's find_next_bit implementation)
9 *
2c57a0e2
YN
10 * Rewritten by Yury Norov <yury.norov@gmail.com> to decrease
11 * size and improve performance, 2015.
12 *
1da177e4
LT
13 * This program is free software; you can redistribute it and/or
14 * modify it under the terms of the GNU General Public License
15 * as published by the Free Software Foundation; either version
16 * 2 of the License, or (at your option) any later version.
17 */
18
19#include <linux/bitops.h>
8f6f19dd 20#include <linux/bitmap.h>
8bc3bcc9 21#include <linux/export.h>
2c57a0e2 22#include <linux/kernel.h>
1da177e4 23
2c57a0e2 24#if !defined(find_next_bit) || !defined(find_next_zero_bit)
c7f612cd 25
64970b68 26/*
2c57a0e2
YN
27 * This is a common helper function for find_next_bit and
28 * find_next_zero_bit. The difference is the "invert" argument, which
29 * is XORed with each fetched word before searching it for one bits.
c7f612cd 30 */
2c57a0e2
YN
31static unsigned long _find_next_bit(const unsigned long *addr,
32 unsigned long nbits, unsigned long start, unsigned long invert)
1da177e4 33{
1da177e4
LT
34 unsigned long tmp;
35
e4afd2e5 36 if (unlikely(start >= nbits))
2c57a0e2
YN
37 return nbits;
38
39 tmp = addr[start / BITS_PER_LONG] ^ invert;
40
41 /* Handle 1st word. */
42 tmp &= BITMAP_FIRST_WORD_MASK(start);
43 start = round_down(start, BITS_PER_LONG);
44
45 while (!tmp) {
46 start += BITS_PER_LONG;
47 if (start >= nbits)
48 return nbits;
49
50 tmp = addr[start / BITS_PER_LONG] ^ invert;
1da177e4
LT
51 }
52
2c57a0e2 53 return min(start + __ffs(tmp), nbits);
c7f612cd 54}
19de85ef 55#endif
1da177e4 56
2c57a0e2 57#ifndef find_next_bit
c7f612cd 58/*
2c57a0e2 59 * Find the next set bit in a memory region.
c7f612cd 60 */
2c57a0e2
YN
61unsigned long find_next_bit(const unsigned long *addr, unsigned long size,
62 unsigned long offset)
63{
64 return _find_next_bit(addr, size, offset, 0UL);
65}
66EXPORT_SYMBOL(find_next_bit);
67#endif
68
69#ifndef find_next_zero_bit
fee4b19f
TG
70unsigned long find_next_zero_bit(const unsigned long *addr, unsigned long size,
71 unsigned long offset)
c7f612cd 72{
2c57a0e2 73 return _find_next_bit(addr, size, offset, ~0UL);
1da177e4 74}
fee4b19f 75EXPORT_SYMBOL(find_next_zero_bit);
19de85ef 76#endif
77b9bd9c 77
19de85ef 78#ifndef find_first_bit
77b9bd9c
AH
79/*
80 * Find the first set bit in a memory region.
81 */
fee4b19f 82unsigned long find_first_bit(const unsigned long *addr, unsigned long size)
77b9bd9c 83{
2c57a0e2 84 unsigned long idx;
77b9bd9c 85
2c57a0e2
YN
86 for (idx = 0; idx * BITS_PER_LONG < size; idx++) {
87 if (addr[idx])
88 return min(idx * BITS_PER_LONG + __ffs(addr[idx]), size);
77b9bd9c 89 }
77b9bd9c 90
2c57a0e2 91 return size;
77b9bd9c 92}
fee4b19f 93EXPORT_SYMBOL(find_first_bit);
19de85ef 94#endif
77b9bd9c 95
19de85ef 96#ifndef find_first_zero_bit
77b9bd9c
AH
97/*
98 * Find the first cleared bit in a memory region.
99 */
fee4b19f 100unsigned long find_first_zero_bit(const unsigned long *addr, unsigned long size)
77b9bd9c 101{
2c57a0e2 102 unsigned long idx;
77b9bd9c 103
2c57a0e2
YN
104 for (idx = 0; idx * BITS_PER_LONG < size; idx++) {
105 if (addr[idx] != ~0UL)
106 return min(idx * BITS_PER_LONG + ffz(addr[idx]), size);
77b9bd9c 107 }
77b9bd9c 108
2c57a0e2 109 return size;
77b9bd9c 110}
fee4b19f 111EXPORT_SYMBOL(find_first_zero_bit);
19de85ef 112#endif
930ae745 113
8f6f19dd
YN
114#ifndef find_last_bit
115unsigned long find_last_bit(const unsigned long *addr, unsigned long size)
116{
117 if (size) {
118 unsigned long val = BITMAP_LAST_WORD_MASK(size);
119 unsigned long idx = (size-1) / BITS_PER_LONG;
120
121 do {
122 val &= addr[idx];
123 if (val)
124 return idx * BITS_PER_LONG + __fls(val);
125
126 val = ~0ul;
127 } while (idx--);
128 }
129 return size;
130}
131EXPORT_SYMBOL(find_last_bit);
132#endif
133
930ae745
AM
134#ifdef __BIG_ENDIAN
135
136/* include/linux/byteorder does not support "unsigned long" type */
930ae745
AM
137static inline unsigned long ext2_swab(const unsigned long y)
138{
139#if BITS_PER_LONG == 64
140 return (unsigned long) __swab64((u64) y);
141#elif BITS_PER_LONG == 32
142 return (unsigned long) __swab32((u32) y);
143#else
144#error BITS_PER_LONG not defined
145#endif
146}
147
2c57a0e2
YN
148#if !defined(find_next_bit_le) || !defined(find_next_zero_bit_le)
149static unsigned long _find_next_bit_le(const unsigned long *addr,
150 unsigned long nbits, unsigned long start, unsigned long invert)
930ae745 151{
930ae745
AM
152 unsigned long tmp;
153
e4afd2e5 154 if (unlikely(start >= nbits))
2c57a0e2
YN
155 return nbits;
156
157 tmp = addr[start / BITS_PER_LONG] ^ invert;
158
159 /* Handle 1st word. */
160 tmp &= ext2_swab(BITMAP_FIRST_WORD_MASK(start));
161 start = round_down(start, BITS_PER_LONG);
930ae745 162
2c57a0e2
YN
163 while (!tmp) {
164 start += BITS_PER_LONG;
165 if (start >= nbits)
166 return nbits;
167
168 tmp = addr[start / BITS_PER_LONG] ^ invert;
930ae745 169 }
930ae745 170
2c57a0e2
YN
171 return min(start + __ffs(ext2_swab(tmp)), nbits);
172}
173#endif
174
175#ifndef find_next_zero_bit_le
176unsigned long find_next_zero_bit_le(const void *addr, unsigned
177 long size, unsigned long offset)
178{
179 return _find_next_bit_le(addr, size, offset, ~0UL);
930ae745 180}
c4945b9e 181EXPORT_SYMBOL(find_next_zero_bit_le);
19de85ef 182#endif
930ae745 183
19de85ef 184#ifndef find_next_bit_le
a56560b3 185unsigned long find_next_bit_le(const void *addr, unsigned
aa02ad67
AK
186 long size, unsigned long offset)
187{
2c57a0e2 188 return _find_next_bit_le(addr, size, offset, 0UL);
aa02ad67 189}
c4945b9e 190EXPORT_SYMBOL(find_next_bit_le);
19de85ef 191#endif
0664996b 192
930ae745 193#endif /* __BIG_ENDIAN */