bitmap.h 18.4 KB
Newer Older
1
/* SPDX-License-Identifier: GPL-2.0 */
L
Linus Torvalds 已提交
2 3 4 5 6 7 8 9
#ifndef __LINUX_BITMAP_H
#define __LINUX_BITMAP_H

#ifndef __ASSEMBLY__

#include <linux/types.h>
#include <linux/bitops.h>
#include <linux/string.h>
J
Jiri Slaby 已提交
10
#include <linux/kernel.h>
L
Linus Torvalds 已提交
11 12 13 14 15 16 17 18 19 20 21 22 23 24

/*
 * bitmaps provide bit arrays that consume one or more unsigned
 * longs.  The bitmap interface and available operations are listed
 * here, in bitmap.h
 *
 * Function implementations generic to all architectures are in
 * lib/bitmap.c.  Functions implementations that are architecture
 * specific are in various include/asm-<arch>/bitops.h headers
 * and other arch/<arch> specific files.
 *
 * See lib/bitmap.c for more details.
 */

25 26 27
/**
 * DOC: bitmap overview
 *
L
Linus Torvalds 已提交
28 29 30
 * The available bitmap operations and their rough meaning in the
 * case that the bitmap is a single unsigned long are thus:
 *
31 32
 * The generated code is more efficient when nbits is known at
 * compile-time and at most BITS_PER_LONG.
33
 *
34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
 * ::
 *
 *  bitmap_zero(dst, nbits)                     *dst = 0UL
 *  bitmap_fill(dst, nbits)                     *dst = ~0UL
 *  bitmap_copy(dst, src, nbits)                *dst = *src
 *  bitmap_and(dst, src1, src2, nbits)          *dst = *src1 & *src2
 *  bitmap_or(dst, src1, src2, nbits)           *dst = *src1 | *src2
 *  bitmap_xor(dst, src1, src2, nbits)          *dst = *src1 ^ *src2
 *  bitmap_andnot(dst, src1, src2, nbits)       *dst = *src1 & ~(*src2)
 *  bitmap_complement(dst, src, nbits)          *dst = ~(*src)
 *  bitmap_equal(src1, src2, nbits)             Are *src1 and *src2 equal?
 *  bitmap_intersects(src1, src2, nbits)        Do *src1 and *src2 overlap?
 *  bitmap_subset(src1, src2, nbits)            Is *src1 a subset of *src2?
 *  bitmap_empty(src, nbits)                    Are all bits zero in *src?
 *  bitmap_full(src, nbits)                     Are all bits set in *src?
 *  bitmap_weight(src, nbits)                   Hamming Weight: number set bits
 *  bitmap_set(dst, pos, nbits)                 Set specified bit area
 *  bitmap_clear(dst, pos, nbits)               Clear specified bit area
 *  bitmap_find_next_zero_area(buf, len, pos, n, mask)  Find bit free area
 *  bitmap_find_next_zero_area_off(buf, len, pos, n, mask)  as above
 *  bitmap_shift_right(dst, src, n, nbits)      *dst = *src >> n
 *  bitmap_shift_left(dst, src, n, nbits)       *dst = *src << n
 *  bitmap_remap(dst, src, old, new, nbits)     *dst = map(old, new)(src)
 *  bitmap_bitremap(oldbit, old, new, nbits)    newbit = map(old, new)(oldbit)
 *  bitmap_onto(dst, orig, relmap, nbits)       *dst = orig relative to relmap
 *  bitmap_fold(dst, orig, sz, nbits)           dst bits = orig bits mod sz
 *  bitmap_parse(buf, buflen, dst, nbits)       Parse bitmap dst from kernel buf
 *  bitmap_parse_user(ubuf, ulen, dst, nbits)   Parse bitmap dst from user buf
 *  bitmap_parselist(buf, dst, nbits)           Parse bitmap dst from kernel buf
 *  bitmap_parselist_user(buf, dst, nbits)      Parse bitmap dst from user buf
 *  bitmap_find_free_region(bitmap, bits, order)  Find and allocate bit region
 *  bitmap_release_region(bitmap, pos, order)   Free specified bit region
 *  bitmap_allocate_region(bitmap, pos, order)  Allocate specified bit region
67 68
 *  bitmap_from_arr32(dst, buf, nbits)          Copy nbits from u32[] buf to dst
 *  bitmap_to_arr32(buf, src, nbits)            Copy nbits from buf to u32[] dst
69
 *
70 71 72 73 74
 * Note, bitmap_zero() and bitmap_fill() operate over the region of
 * unsigned longs, that is, bits behind bitmap till the unsigned long
 * boundary will be zeroed or filled as well. Consider to use
 * bitmap_clear() or bitmap_set() to make explicit zeroing or filling
 * respectively.
L
Linus Torvalds 已提交
75 76
 */

77 78 79 80 81 82 83 84 85 86 87 88 89 90
/**
 * DOC: bitmap bitops
 *
 * Also the following operations in asm/bitops.h apply to bitmaps.::
 *
 *  set_bit(bit, addr)                  *addr |= bit
 *  clear_bit(bit, addr)                *addr &= ~bit
 *  change_bit(bit, addr)               *addr ^= bit
 *  test_bit(bit, addr)                 Is bit set in *addr?
 *  test_and_set_bit(bit, addr)         Set bit and return old value
 *  test_and_clear_bit(bit, addr)       Clear bit and return old value
 *  test_and_change_bit(bit, addr)      Change bit and return old value
 *  find_first_zero_bit(addr, nbits)    Position first zero bit in *addr
 *  find_first_bit(addr, nbits)         Position first set bit in *addr
91 92
 *  find_next_zero_bit(addr, nbits, bit)
 *                                      Position next zero bit in *addr >= bit
93
 *  find_next_bit(addr, nbits, bit)     Position next set bit in *addr >= bit
94 95 96
 *  find_next_and_bit(addr1, addr2, nbits, bit)
 *                                      Same as find_next_bit, but in
 *                                      (*addr1 & *addr2)
L
Linus Torvalds 已提交
97 98 99
 *
 */

100 101
/**
 * DOC: declare bitmap
L
Linus Torvalds 已提交
102 103 104 105 106
 * The DECLARE_BITMAP(name,bits) macro, in linux/types.h, can be used
 * to declare an array named 'name' of just enough unsigned longs to
 * contain all bit positions from 0 to 'bits' - 1.
 */

107 108 109 110 111 112 113 114
/*
 * Allocation and deallocation of bitmap.
 * Provided in lib/bitmap.c to avoid circular dependency.
 */
extern unsigned long *bitmap_alloc(unsigned int nbits, gfp_t flags);
extern unsigned long *bitmap_zalloc(unsigned int nbits, gfp_t flags);
extern void bitmap_free(const unsigned long *bitmap);

L
Linus Torvalds 已提交
115 116 117 118
/*
 * lib/bitmap.c provides these functions:
 */

119
extern int __bitmap_empty(const unsigned long *bitmap, unsigned int nbits);
120
extern int __bitmap_full(const unsigned long *bitmap, unsigned int nbits);
L
Linus Torvalds 已提交
121
extern int __bitmap_equal(const unsigned long *bitmap1,
122
			  const unsigned long *bitmap2, unsigned int nbits);
123 124 125 126
extern bool __pure __bitmap_or_equal(const unsigned long *src1,
				     const unsigned long *src2,
				     const unsigned long *src3,
				     unsigned int nbits);
L
Linus Torvalds 已提交
127
extern void __bitmap_complement(unsigned long *dst, const unsigned long *src,
128
			unsigned int nbits);
129 130
extern void __bitmap_shift_right(unsigned long *dst, const unsigned long *src,
				unsigned int shift, unsigned int nbits);
131 132
extern void __bitmap_shift_left(unsigned long *dst, const unsigned long *src,
				unsigned int shift, unsigned int nbits);
133
extern int __bitmap_and(unsigned long *dst, const unsigned long *bitmap1,
134
			const unsigned long *bitmap2, unsigned int nbits);
L
Linus Torvalds 已提交
135
extern void __bitmap_or(unsigned long *dst, const unsigned long *bitmap1,
136
			const unsigned long *bitmap2, unsigned int nbits);
L
Linus Torvalds 已提交
137
extern void __bitmap_xor(unsigned long *dst, const unsigned long *bitmap1,
138
			const unsigned long *bitmap2, unsigned int nbits);
139
extern int __bitmap_andnot(unsigned long *dst, const unsigned long *bitmap1,
140
			const unsigned long *bitmap2, unsigned int nbits);
L
Linus Torvalds 已提交
141
extern int __bitmap_intersects(const unsigned long *bitmap1,
142
			const unsigned long *bitmap2, unsigned int nbits);
L
Linus Torvalds 已提交
143
extern int __bitmap_subset(const unsigned long *bitmap1,
144
			const unsigned long *bitmap2, unsigned int nbits);
145
extern int __bitmap_weight(const unsigned long *bitmap, unsigned int nbits);
146 147
extern void __bitmap_set(unsigned long *map, unsigned int start, int len);
extern void __bitmap_clear(unsigned long *map, unsigned int start, int len);
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

extern unsigned long bitmap_find_next_zero_area_off(unsigned long *map,
						    unsigned long size,
						    unsigned long start,
						    unsigned int nr,
						    unsigned long align_mask,
						    unsigned long align_offset);

/**
 * bitmap_find_next_zero_area - find a contiguous aligned zero area
 * @map: The address to base the search on
 * @size: The bitmap size in bits
 * @start: The bitnumber to start searching at
 * @nr: The number of zeroed bits we're looking for
 * @align_mask: Alignment mask for zero area
 *
 * The @align_mask should be one less than a power of 2; the effect is that
 * the bit offset of all zero areas this function finds is multiples of that
 * power of 2. A @align_mask of 0 means no alignment is required.
 */
static inline unsigned long
bitmap_find_next_zero_area(unsigned long *map,
			   unsigned long size,
			   unsigned long start,
			   unsigned int nr,
			   unsigned long align_mask)
{
	return bitmap_find_next_zero_area_off(map, size, start, nr,
					      align_mask, 0);
}
178

179 180 181
extern int __bitmap_parse(const char *buf, unsigned int buflen, int is_user,
			unsigned long *dst, int nbits);
extern int bitmap_parse_user(const char __user *ubuf, unsigned int ulen,
L
Linus Torvalds 已提交
182 183 184
			unsigned long *dst, int nbits);
extern int bitmap_parselist(const char *buf, unsigned long *maskp,
			int nmaskbits);
185 186
extern int bitmap_parselist_user(const char __user *ubuf, unsigned int ulen,
			unsigned long *dst, int nbits);
187
extern void bitmap_remap(unsigned long *dst, const unsigned long *src,
188
		const unsigned long *old, const unsigned long *new, unsigned int nbits);
189 190
extern int bitmap_bitremap(int oldbit,
		const unsigned long *old, const unsigned long *new, int bits);
191
extern void bitmap_onto(unsigned long *dst, const unsigned long *orig,
192
		const unsigned long *relmap, unsigned int bits);
193
extern void bitmap_fold(unsigned long *dst, const unsigned long *orig,
194
		unsigned int sz, unsigned int nbits);
195 196 197
extern int bitmap_find_free_region(unsigned long *bitmap, unsigned int bits, int order);
extern void bitmap_release_region(unsigned long *bitmap, unsigned int pos, int order);
extern int bitmap_allocate_region(unsigned long *bitmap, unsigned int pos, int order);
198

199
#ifdef __BIG_ENDIAN
200
extern void bitmap_copy_le(unsigned long *dst, const unsigned long *src, unsigned int nbits);
201 202 203
#else
#define bitmap_copy_le bitmap_copy
#endif
204
extern unsigned int bitmap_ord_to_pos(const unsigned long *bitmap, unsigned int ord, unsigned int nbits);
205 206
extern int bitmap_print_to_pagebuf(bool list, char *buf,
				   const unsigned long *maskp, int nmaskbits);
L
Linus Torvalds 已提交
207

208 209
#define BITMAP_FIRST_WORD_MASK(start) (~0UL << ((start) & (BITS_PER_LONG - 1)))
#define BITMAP_LAST_WORD_MASK(nbits) (~0UL >> (-(nbits) & (BITS_PER_LONG - 1)))
L
Linus Torvalds 已提交
210

211 212 213 214 215
/*
 * The static inlines below do not handle constant nbits==0 correctly,
 * so make such users (should any ever turn up) call the out-of-line
 * versions.
 */
216
#define small_const_nbits(nbits) \
217
	(__builtin_constant_p(nbits) && (nbits) <= BITS_PER_LONG && (nbits) > 0)
218

219
static inline void bitmap_zero(unsigned long *dst, unsigned int nbits)
L
Linus Torvalds 已提交
220
{
221 222
	unsigned int len = BITS_TO_LONGS(nbits) * sizeof(unsigned long);
	memset(dst, 0, len);
L
Linus Torvalds 已提交
223 224
}

225
static inline void bitmap_fill(unsigned long *dst, unsigned int nbits)
L
Linus Torvalds 已提交
226
{
227 228
	unsigned int len = BITS_TO_LONGS(nbits) * sizeof(unsigned long);
	memset(dst, 0xff, len);
L
Linus Torvalds 已提交
229 230 231
}

static inline void bitmap_copy(unsigned long *dst, const unsigned long *src,
232
			unsigned int nbits)
L
Linus Torvalds 已提交
233
{
234 235
	unsigned int len = BITS_TO_LONGS(nbits) * sizeof(unsigned long);
	memcpy(dst, src, len);
L
Linus Torvalds 已提交
236 237
}

238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266
/*
 * Copy bitmap and clear tail bits in last word.
 */
static inline void bitmap_copy_clear_tail(unsigned long *dst,
		const unsigned long *src, unsigned int nbits)
{
	bitmap_copy(dst, src, nbits);
	if (nbits % BITS_PER_LONG)
		dst[nbits / BITS_PER_LONG] &= BITMAP_LAST_WORD_MASK(nbits);
}

/*
 * On 32-bit systems bitmaps are represented as u32 arrays internally, and
 * therefore conversion is not needed when copying data from/to arrays of u32.
 */
#if BITS_PER_LONG == 64
extern void bitmap_from_arr32(unsigned long *bitmap, const u32 *buf,
							unsigned int nbits);
extern void bitmap_to_arr32(u32 *buf, const unsigned long *bitmap,
							unsigned int nbits);
#else
#define bitmap_from_arr32(bitmap, buf, nbits)			\
	bitmap_copy_clear_tail((unsigned long *) (bitmap),	\
			(const unsigned long *) (buf), (nbits))
#define bitmap_to_arr32(buf, bitmap, nbits)			\
	bitmap_copy_clear_tail((unsigned long *) (buf),		\
			(const unsigned long *) (bitmap), (nbits))
#endif

267
static inline int bitmap_and(unsigned long *dst, const unsigned long *src1,
268
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
269
{
270
	if (small_const_nbits(nbits))
271
		return (*dst = *src1 & *src2 & BITMAP_LAST_WORD_MASK(nbits)) != 0;
272
	return __bitmap_and(dst, src1, src2, nbits);
L
Linus Torvalds 已提交
273 274 275
}

static inline void bitmap_or(unsigned long *dst, const unsigned long *src1,
276
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
277
{
278
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
279 280 281 282 283 284
		*dst = *src1 | *src2;
	else
		__bitmap_or(dst, src1, src2, nbits);
}

static inline void bitmap_xor(unsigned long *dst, const unsigned long *src1,
285
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
286
{
287
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
288 289 290 291 292
		*dst = *src1 ^ *src2;
	else
		__bitmap_xor(dst, src1, src2, nbits);
}

293
static inline int bitmap_andnot(unsigned long *dst, const unsigned long *src1,
294
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
295
{
296
	if (small_const_nbits(nbits))
297
		return (*dst = *src1 & ~(*src2) & BITMAP_LAST_WORD_MASK(nbits)) != 0;
298
	return __bitmap_andnot(dst, src1, src2, nbits);
L
Linus Torvalds 已提交
299 300 301
}

static inline void bitmap_complement(unsigned long *dst, const unsigned long *src,
302
			unsigned int nbits)
L
Linus Torvalds 已提交
303
{
304
	if (small_const_nbits(nbits))
305
		*dst = ~(*src);
L
Linus Torvalds 已提交
306 307 308 309
	else
		__bitmap_complement(dst, src, nbits);
}

310 311 312 313 314 315 316
#ifdef __LITTLE_ENDIAN
#define BITMAP_MEM_ALIGNMENT 8
#else
#define BITMAP_MEM_ALIGNMENT (8 * sizeof(unsigned long))
#endif
#define BITMAP_MEM_MASK (BITMAP_MEM_ALIGNMENT - 1)

L
Linus Torvalds 已提交
317
static inline int bitmap_equal(const unsigned long *src1,
318
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
319
{
320
	if (small_const_nbits(nbits))
A
Andrew Morton 已提交
321
		return !((*src1 ^ *src2) & BITMAP_LAST_WORD_MASK(nbits));
322 323
	if (__builtin_constant_p(nbits & BITMAP_MEM_MASK) &&
	    IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT))
324
		return !memcmp(src1, src2, nbits / 8);
A
Andrew Morton 已提交
325
	return __bitmap_equal(src1, src2, nbits);
L
Linus Torvalds 已提交
326 327
}

328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346
/**
 * bitmap_or_equal - Check whether the or of two bitnaps is equal to a third
 * @src1:	Pointer to bitmap 1
 * @src2:	Pointer to bitmap 2 will be or'ed with bitmap 1
 * @src3:	Pointer to bitmap 3. Compare to the result of *@src1 | *@src2
 *
 * Returns: True if (*@src1 | *@src2) == *@src3, false otherwise
 */
static inline bool bitmap_or_equal(const unsigned long *src1,
				   const unsigned long *src2,
				   const unsigned long *src3,
				   unsigned int nbits)
{
	if (!small_const_nbits(nbits))
		return __bitmap_or_equal(src1, src2, src3, nbits);

	return !(((*src1 | *src2) ^ *src3) & BITMAP_LAST_WORD_MASK(nbits));
}

L
Linus Torvalds 已提交
347
static inline int bitmap_intersects(const unsigned long *src1,
348
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
349
{
350
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
351 352 353 354 355 356
		return ((*src1 & *src2) & BITMAP_LAST_WORD_MASK(nbits)) != 0;
	else
		return __bitmap_intersects(src1, src2, nbits);
}

static inline int bitmap_subset(const unsigned long *src1,
357
			const unsigned long *src2, unsigned int nbits)
L
Linus Torvalds 已提交
358
{
359
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
360 361 362 363 364
		return ! ((*src1 & ~(*src2)) & BITMAP_LAST_WORD_MASK(nbits));
	else
		return __bitmap_subset(src1, src2, nbits);
}

365
static inline int bitmap_empty(const unsigned long *src, unsigned nbits)
L
Linus Torvalds 已提交
366
{
367
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
368
		return ! (*src & BITMAP_LAST_WORD_MASK(nbits));
369 370

	return find_first_bit(src, nbits) == nbits;
L
Linus Torvalds 已提交
371 372
}

373
static inline int bitmap_full(const unsigned long *src, unsigned int nbits)
L
Linus Torvalds 已提交
374
{
375
	if (small_const_nbits(nbits))
L
Linus Torvalds 已提交
376
		return ! (~(*src) & BITMAP_LAST_WORD_MASK(nbits));
377 378

	return find_first_zero_bit(src, nbits) == nbits;
L
Linus Torvalds 已提交
379 380
}

381
static __always_inline int bitmap_weight(const unsigned long *src, unsigned int nbits)
L
Linus Torvalds 已提交
382
{
383
	if (small_const_nbits(nbits))
384
		return hweight_long(*src & BITMAP_LAST_WORD_MASK(nbits));
L
Linus Torvalds 已提交
385 386 387
	return __bitmap_weight(src, nbits);
}

388 389 390 391 392
static __always_inline void bitmap_set(unsigned long *map, unsigned int start,
		unsigned int nbits)
{
	if (__builtin_constant_p(nbits) && nbits == 1)
		__set_bit(start, map);
393 394 395 396
	else if (__builtin_constant_p(start & BITMAP_MEM_MASK) &&
		 IS_ALIGNED(start, BITMAP_MEM_ALIGNMENT) &&
		 __builtin_constant_p(nbits & BITMAP_MEM_MASK) &&
		 IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT))
397
		memset((char *)map + start / 8, 0xff, nbits / 8);
398 399 400 401 402 403 404 405 406
	else
		__bitmap_set(map, start, nbits);
}

static __always_inline void bitmap_clear(unsigned long *map, unsigned int start,
		unsigned int nbits)
{
	if (__builtin_constant_p(nbits) && nbits == 1)
		__clear_bit(start, map);
407 408 409 410
	else if (__builtin_constant_p(start & BITMAP_MEM_MASK) &&
		 IS_ALIGNED(start, BITMAP_MEM_ALIGNMENT) &&
		 __builtin_constant_p(nbits & BITMAP_MEM_MASK) &&
		 IS_ALIGNED(nbits, BITMAP_MEM_ALIGNMENT))
411
		memset((char *)map + start / 8, 0, nbits / 8);
412 413 414 415
	else
		__bitmap_clear(map, start, nbits);
}

416
static inline void bitmap_shift_right(unsigned long *dst, const unsigned long *src,
417
				unsigned int shift, unsigned int nbits)
L
Linus Torvalds 已提交
418
{
419
	if (small_const_nbits(nbits))
420
		*dst = (*src & BITMAP_LAST_WORD_MASK(nbits)) >> shift;
L
Linus Torvalds 已提交
421
	else
422
		__bitmap_shift_right(dst, src, shift, nbits);
L
Linus Torvalds 已提交
423 424
}

425 426
static inline void bitmap_shift_left(unsigned long *dst, const unsigned long *src,
				unsigned int shift, unsigned int nbits)
L
Linus Torvalds 已提交
427
{
428
	if (small_const_nbits(nbits))
429
		*dst = (*src << shift) & BITMAP_LAST_WORD_MASK(nbits);
L
Linus Torvalds 已提交
430
	else
431
		__bitmap_shift_left(dst, src, shift, nbits);
L
Linus Torvalds 已提交
432 433
}

434 435 436 437 438 439
static inline int bitmap_parse(const char *buf, unsigned int buflen,
			unsigned long *maskp, int nmaskbits)
{
	return __bitmap_parse(buf, buflen, 0, maskp, nmaskbits);
}

440
/**
Y
Yury Norov 已提交
441
 * BITMAP_FROM_U64() - Represent u64 value in the format suitable for bitmap.
442
 * @n: u64 value
Y
Yury Norov 已提交
443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472
 *
 * Linux bitmaps are internally arrays of unsigned longs, i.e. 32-bit
 * integers in 32-bit environment, and 64-bit integers in 64-bit one.
 *
 * There are four combinations of endianness and length of the word in linux
 * ABIs: LE64, BE64, LE32 and BE32.
 *
 * On 64-bit kernels 64-bit LE and BE numbers are naturally ordered in
 * bitmaps and therefore don't require any special handling.
 *
 * On 32-bit kernels 32-bit LE ABI orders lo word of 64-bit number in memory
 * prior to hi, and 32-bit BE orders hi word prior to lo. The bitmap on the
 * other hand is represented as an array of 32-bit words and the position of
 * bit N may therefore be calculated as: word #(N/32) and bit #(N%32) in that
 * word.  For example, bit #42 is located at 10th position of 2nd word.
 * It matches 32-bit LE ABI, and we can simply let the compiler store 64-bit
 * values in memory as it usually does. But for BE we need to swap hi and lo
 * words manually.
 *
 * With all that, the macro BITMAP_FROM_U64() does explicit reordering of hi and
 * lo parts of u64.  For LE32 it does nothing, and for BE environment it swaps
 * hi and lo words, as is expected by bitmap.
 */
#if __BITS_PER_LONG == 64
#define BITMAP_FROM_U64(n) (n)
#else
#define BITMAP_FROM_U64(n) ((unsigned long) ((u64)(n) & ULONG_MAX)), \
				((unsigned long) ((u64)(n) >> 32))
#endif

473
/**
474 475 476 477
 * bitmap_from_u64 - Check and swap words within u64.
 *  @mask: source bitmap
 *  @dst:  destination bitmap
 *
478
 * In 32-bit Big Endian kernel, when using ``(u32 *)(&val)[*]``
479
 * to read u64 mask, we will get the wrong word.
480
 * That is ``(u32 *)(&val)[0]`` gets the upper 32 bits,
481 482 483 484 485 486 487 488 489 490
 * but we expect the lower 32-bits of u64.
 */
static inline void bitmap_from_u64(unsigned long *dst, u64 mask)
{
	dst[0] = mask & ULONG_MAX;

	if (sizeof(mask) > sizeof(unsigned long))
		dst[1] = mask >> 32;
}

L
Linus Torvalds 已提交
491 492 493
#endif /* __ASSEMBLY__ */

#endif /* __LINUX_BITMAP_H */