xref: /freebsd/sys/compat/linuxkpi/common/include/linux/bitops.h (revision 6132212808e8dccedc9e5d85fea4390c2f38059a)
1 /*-
2  * Copyright (c) 2010 Isilon Systems, Inc.
3  * Copyright (c) 2010 iX Systems, Inc.
4  * Copyright (c) 2010 Panasas, Inc.
5  * Copyright (c) 2013-2017 Mellanox Technologies, Ltd.
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice unmodified, this list of conditions, and the following
13  *    disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  *
29  * $FreeBSD$
30  */
31 #ifndef	_LINUX_BITOPS_H_
32 #define	_LINUX_BITOPS_H_
33 
34 #include <sys/param.h>
35 #include <sys/types.h>
36 #include <sys/systm.h>
37 #include <sys/errno.h>
38 #include <sys/libkern.h>
39 
40 #define	BIT(nr)			(1UL << (nr))
41 #define	BIT_ULL(nr)		(1ULL << (nr))
42 #ifdef __LP64__
43 #define	BITS_PER_LONG		64
44 #else
45 #define	BITS_PER_LONG		32
46 #endif
47 
48 #define	BITS_PER_LONG_LONG	64
49 
50 #define	BITMAP_FIRST_WORD_MASK(start)	(~0UL << ((start) % BITS_PER_LONG))
51 #define	BITMAP_LAST_WORD_MASK(n)	(~0UL >> (BITS_PER_LONG - (n)))
52 #define	BITS_TO_LONGS(n)	howmany((n), BITS_PER_LONG)
53 #define	BIT_MASK(nr)		(1UL << ((nr) & (BITS_PER_LONG - 1)))
54 #define	BIT_WORD(nr)		((nr) / BITS_PER_LONG)
55 #define	GENMASK(h, l)		(((~0UL) >> (BITS_PER_LONG - (h) - 1)) & ((~0UL) << (l)))
56 #define	GENMASK_ULL(h, l)	(((~0ULL) >> (BITS_PER_LONG_LONG - (h) - 1)) & ((~0ULL) << (l)))
57 #define	BITS_PER_BYTE		8
58 #define	BITS_PER_TYPE(t)	(sizeof(t) * BITS_PER_BYTE)
59 
60 #define	hweight8(x)	bitcount((uint8_t)(x))
61 #define	hweight16(x)	bitcount16(x)
62 #define	hweight32(x)	bitcount32(x)
63 #define	hweight64(x)	bitcount64(x)
64 #define	hweight_long(x)	bitcountl(x)
65 
66 static inline int
67 __ffs(int mask)
68 {
69 	return (ffs(mask) - 1);
70 }
71 
72 static inline int
73 __fls(int mask)
74 {
75 	return (fls(mask) - 1);
76 }
77 
78 static inline int
79 __ffsl(long mask)
80 {
81 	return (ffsl(mask) - 1);
82 }
83 
84 static inline int
85 __flsl(long mask)
86 {
87 	return (flsl(mask) - 1);
88 }
89 
90 static inline int
91 fls64(uint64_t mask)
92 {
93 	return (flsll(mask));
94 }
95 
96 static inline uint32_t
97 ror32(uint32_t word, unsigned int shift)
98 {
99 	return ((word >> shift) | (word << (32 - shift)));
100 }
101 
102 #define	ffz(mask)	__ffs(~(mask))
103 
104 static inline int get_count_order(unsigned int count)
105 {
106         int order;
107 
108         order = fls(count) - 1;
109         if (count & (count - 1))
110                 order++;
111         return order;
112 }
113 
114 static inline unsigned long
115 find_first_bit(const unsigned long *addr, unsigned long size)
116 {
117 	long mask;
118 	int bit;
119 
120 	for (bit = 0; size >= BITS_PER_LONG;
121 	    size -= BITS_PER_LONG, bit += BITS_PER_LONG, addr++) {
122 		if (*addr == 0)
123 			continue;
124 		return (bit + __ffsl(*addr));
125 	}
126 	if (size) {
127 		mask = (*addr) & BITMAP_LAST_WORD_MASK(size);
128 		if (mask)
129 			bit += __ffsl(mask);
130 		else
131 			bit += size;
132 	}
133 	return (bit);
134 }
135 
136 static inline unsigned long
137 find_first_zero_bit(const unsigned long *addr, unsigned long size)
138 {
139 	long mask;
140 	int bit;
141 
142 	for (bit = 0; size >= BITS_PER_LONG;
143 	    size -= BITS_PER_LONG, bit += BITS_PER_LONG, addr++) {
144 		if (~(*addr) == 0)
145 			continue;
146 		return (bit + __ffsl(~(*addr)));
147 	}
148 	if (size) {
149 		mask = ~(*addr) & BITMAP_LAST_WORD_MASK(size);
150 		if (mask)
151 			bit += __ffsl(mask);
152 		else
153 			bit += size;
154 	}
155 	return (bit);
156 }
157 
158 static inline unsigned long
159 find_last_bit(const unsigned long *addr, unsigned long size)
160 {
161 	long mask;
162 	int offs;
163 	int bit;
164 	int pos;
165 
166 	pos = size / BITS_PER_LONG;
167 	offs = size % BITS_PER_LONG;
168 	bit = BITS_PER_LONG * pos;
169 	addr += pos;
170 	if (offs) {
171 		mask = (*addr) & BITMAP_LAST_WORD_MASK(offs);
172 		if (mask)
173 			return (bit + __flsl(mask));
174 	}
175 	while (pos--) {
176 		addr--;
177 		bit -= BITS_PER_LONG;
178 		if (*addr)
179 			return (bit + __flsl(*addr));
180 	}
181 	return (size);
182 }
183 
184 static inline unsigned long
185 find_next_bit(const unsigned long *addr, unsigned long size, unsigned long offset)
186 {
187 	long mask;
188 	int offs;
189 	int bit;
190 	int pos;
191 
192 	if (offset >= size)
193 		return (size);
194 	pos = offset / BITS_PER_LONG;
195 	offs = offset % BITS_PER_LONG;
196 	bit = BITS_PER_LONG * pos;
197 	addr += pos;
198 	if (offs) {
199 		mask = (*addr) & ~BITMAP_LAST_WORD_MASK(offs);
200 		if (mask)
201 			return (bit + __ffsl(mask));
202 		if (size - bit <= BITS_PER_LONG)
203 			return (size);
204 		bit += BITS_PER_LONG;
205 		addr++;
206 	}
207 	for (size -= bit; size >= BITS_PER_LONG;
208 	    size -= BITS_PER_LONG, bit += BITS_PER_LONG, addr++) {
209 		if (*addr == 0)
210 			continue;
211 		return (bit + __ffsl(*addr));
212 	}
213 	if (size) {
214 		mask = (*addr) & BITMAP_LAST_WORD_MASK(size);
215 		if (mask)
216 			bit += __ffsl(mask);
217 		else
218 			bit += size;
219 	}
220 	return (bit);
221 }
222 
223 static inline unsigned long
224 find_next_zero_bit(const unsigned long *addr, unsigned long size,
225     unsigned long offset)
226 {
227 	long mask;
228 	int offs;
229 	int bit;
230 	int pos;
231 
232 	if (offset >= size)
233 		return (size);
234 	pos = offset / BITS_PER_LONG;
235 	offs = offset % BITS_PER_LONG;
236 	bit = BITS_PER_LONG * pos;
237 	addr += pos;
238 	if (offs) {
239 		mask = ~(*addr) & ~BITMAP_LAST_WORD_MASK(offs);
240 		if (mask)
241 			return (bit + __ffsl(mask));
242 		if (size - bit <= BITS_PER_LONG)
243 			return (size);
244 		bit += BITS_PER_LONG;
245 		addr++;
246 	}
247 	for (size -= bit; size >= BITS_PER_LONG;
248 	    size -= BITS_PER_LONG, bit += BITS_PER_LONG, addr++) {
249 		if (~(*addr) == 0)
250 			continue;
251 		return (bit + __ffsl(~(*addr)));
252 	}
253 	if (size) {
254 		mask = ~(*addr) & BITMAP_LAST_WORD_MASK(size);
255 		if (mask)
256 			bit += __ffsl(mask);
257 		else
258 			bit += size;
259 	}
260 	return (bit);
261 }
262 
263 #define	__set_bit(i, a)							\
264     atomic_set_long(&((volatile unsigned long *)(a))[BIT_WORD(i)], BIT_MASK(i))
265 
266 #define	set_bit(i, a)							\
267     atomic_set_long(&((volatile unsigned long *)(a))[BIT_WORD(i)], BIT_MASK(i))
268 
269 #define	__clear_bit(i, a)						\
270     atomic_clear_long(&((volatile unsigned long *)(a))[BIT_WORD(i)], BIT_MASK(i))
271 
272 #define	clear_bit(i, a)							\
273     atomic_clear_long(&((volatile unsigned long *)(a))[BIT_WORD(i)], BIT_MASK(i))
274 
275 #define	clear_bit_unlock(i, a)						\
276     atomic_clear_rel_long(&((volatile unsigned long *)(a))[BIT_WORD(i)], BIT_MASK(i))
277 
278 #define	test_bit(i, a)							\
279     !!(READ_ONCE(((volatile const unsigned long *)(a))[BIT_WORD(i)]) & BIT_MASK(i))
280 
281 static inline int
282 test_and_clear_bit(long bit, volatile unsigned long *var)
283 {
284 	long val;
285 
286 	var += BIT_WORD(bit);
287 	bit %= BITS_PER_LONG;
288 	bit = (1UL << bit);
289 
290 	val = *var;
291 	while (!atomic_fcmpset_long(var, &val, val & ~bit))
292 		;
293 	return !!(val & bit);
294 }
295 
296 static inline int
297 __test_and_clear_bit(long bit, volatile unsigned long *var)
298 {
299 	long val;
300 
301 	var += BIT_WORD(bit);
302 	bit %= BITS_PER_LONG;
303 	bit = (1UL << bit);
304 
305 	val = *var;
306 	*var &= ~bit;
307 
308 	return !!(val & bit);
309 }
310 
311 static inline int
312 test_and_set_bit(long bit, volatile unsigned long *var)
313 {
314 	long val;
315 
316 	var += BIT_WORD(bit);
317 	bit %= BITS_PER_LONG;
318 	bit = (1UL << bit);
319 
320 	val = *var;
321 	while (!atomic_fcmpset_long(var, &val, val | bit))
322 		;
323 	return !!(val & bit);
324 }
325 
326 static inline int
327 __test_and_set_bit(long bit, volatile unsigned long *var)
328 {
329 	long val;
330 
331 	var += BIT_WORD(bit);
332 	bit %= BITS_PER_LONG;
333 	bit = (1UL << bit);
334 
335 	val = *var;
336 	*var |= bit;
337 
338 	return !!(val & bit);
339 }
340 
341 enum {
342         REG_OP_ISFREE,
343         REG_OP_ALLOC,
344         REG_OP_RELEASE,
345 };
346 
347 static inline int
348 linux_reg_op(unsigned long *bitmap, int pos, int order, int reg_op)
349 {
350         int nbits_reg;
351         int index;
352         int offset;
353         int nlongs_reg;
354         int nbitsinlong;
355         unsigned long mask;
356         int i;
357         int ret = 0;
358 
359         nbits_reg = 1 << order;
360         index = pos / BITS_PER_LONG;
361         offset = pos - (index * BITS_PER_LONG);
362         nlongs_reg = BITS_TO_LONGS(nbits_reg);
363         nbitsinlong = min(nbits_reg,  BITS_PER_LONG);
364 
365         mask = (1UL << (nbitsinlong - 1));
366         mask += mask - 1;
367         mask <<= offset;
368 
369         switch (reg_op) {
370         case REG_OP_ISFREE:
371                 for (i = 0; i < nlongs_reg; i++) {
372                         if (bitmap[index + i] & mask)
373                                 goto done;
374                 }
375                 ret = 1;
376                 break;
377 
378         case REG_OP_ALLOC:
379                 for (i = 0; i < nlongs_reg; i++)
380                         bitmap[index + i] |= mask;
381                 break;
382 
383         case REG_OP_RELEASE:
384                 for (i = 0; i < nlongs_reg; i++)
385                         bitmap[index + i] &= ~mask;
386                 break;
387         }
388 done:
389         return ret;
390 }
391 
392 #define for_each_set_bit(bit, addr, size) \
393 	for ((bit) = find_first_bit((addr), (size));		\
394 	     (bit) < (size);					\
395 	     (bit) = find_next_bit((addr), (size), (bit) + 1))
396 
397 #define	for_each_clear_bit(bit, addr, size) \
398 	for ((bit) = find_first_zero_bit((addr), (size));		\
399 	     (bit) < (size);						\
400 	     (bit) = find_next_zero_bit((addr), (size), (bit) + 1))
401 
402 static inline uint64_t
403 sign_extend64(uint64_t value, int index)
404 {
405 	uint8_t shift = 63 - index;
406 
407 	return ((int64_t)(value << shift) >> shift);
408 }
409 
410 #endif	/* _LINUX_BITOPS_H_ */
411