blob: 365f8207ea59d517849c9940e07a1f7b9b153332 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001#ifndef _X86_64_BITOPS_H
2#define _X86_64_BITOPS_H
3
4/*
5 * Copyright 1992, Linus Torvalds.
6 */
7
Randy Dunlapade8c562007-10-25 14:27:24 -07008extern long find_first_zero_bit(const unsigned long *addr, unsigned long size);
9extern long find_next_zero_bit(const unsigned long *addr, long size, long offset);
10extern long find_first_bit(const unsigned long *addr, unsigned long size);
11extern long find_next_bit(const unsigned long *addr, long size, long offset);
Linus Torvalds1da177e2005-04-16 15:20:36 -070012
13/* return index of first bet set in val or max when no bit is set */
Chuck Leverd2ccc3f2007-10-17 18:04:38 +020014static inline long __scanbit(unsigned long val, unsigned long max)
Linus Torvalds1da177e2005-04-16 15:20:36 -070015{
16 asm("bsfq %1,%0 ; cmovz %2,%0" : "=&r" (val) : "r" (val), "r" (max));
17 return val;
18}
19
Linus Torvalds1da177e2005-04-16 15:20:36 -070020#define find_next_bit(addr,size,off) \
21((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? \
22 ((off) + (__scanbit((*(unsigned long *)addr) >> (off),(size)-(off)))) : \
23 find_next_bit(addr,size,off)))
24
Linus Torvalds1da177e2005-04-16 15:20:36 -070025#define find_next_zero_bit(addr,size,off) \
26((__builtin_constant_p(size) && (size) <= BITS_PER_LONG ? \
27 ((off)+(__scanbit(~(((*(unsigned long *)addr)) >> (off)),(size)-(off)))) : \
28 find_next_zero_bit(addr,size,off)))
29
Joe Perches49f74942008-03-23 01:01:44 -070030#define find_first_bit(addr, size) \
31 ((__builtin_constant_p((size)) && (size) <= BITS_PER_LONG \
32 ? (__scanbit(*(unsigned long *)(addr), (size))) \
33 : find_first_bit((addr), (size))))
34
35#define find_first_zero_bit(addr, size) \
36 ((__builtin_constant_p((size)) && (size) <= BITS_PER_LONG \
37 ? (__scanbit(~*(unsigned long *)(addr), (size))) \
38 : find_first_zero_bit((addr), (size))))
39
40static inline void set_bit_string(unsigned long *bitmap, unsigned long i,
41 int len)
42{
43 unsigned long end = i + len;
Linus Torvalds1da177e2005-04-16 15:20:36 -070044 while (i < end) {
Joe Perches49f74942008-03-23 01:01:44 -070045 __set_bit(i, bitmap);
Linus Torvalds1da177e2005-04-16 15:20:36 -070046 i++;
47 }
Joe Perches49f74942008-03-23 01:01:44 -070048}
Linus Torvalds1da177e2005-04-16 15:20:36 -070049
Linus Torvalds1da177e2005-04-16 15:20:36 -070050/**
51 * ffz - find first zero in word.
52 * @word: The word to search
53 *
54 * Undefined if no zero exists, so code should check against ~0UL first.
55 */
Randy Dunlapade8c562007-10-25 14:27:24 -070056static inline unsigned long ffz(unsigned long word)
Linus Torvalds1da177e2005-04-16 15:20:36 -070057{
58 __asm__("bsfq %1,%0"
59 :"=r" (word)
60 :"r" (~word));
61 return word;
62}
63
64/**
65 * __ffs - find first bit in word.
66 * @word: The word to search
67 *
68 * Undefined if no bit exists, so code should check against 0 first.
69 */
Randy Dunlapade8c562007-10-25 14:27:24 -070070static inline unsigned long __ffs(unsigned long word)
Linus Torvalds1da177e2005-04-16 15:20:36 -070071{
72 __asm__("bsfq %1,%0"
73 :"=r" (word)
74 :"rm" (word));
75 return word;
76}
77
Stephen Hemminger90933fc2005-12-21 19:31:36 -080078/*
79 * __fls: find last bit set.
80 * @word: The word to search
81 *
82 * Undefined if no zero exists, so code should check against ~0UL first.
83 */
Randy Dunlapade8c562007-10-25 14:27:24 -070084static inline unsigned long __fls(unsigned long word)
Stephen Hemminger90933fc2005-12-21 19:31:36 -080085{
86 __asm__("bsrq %1,%0"
87 :"=r" (word)
88 :"rm" (word));
89 return word;
90}
91
Linus Torvalds1da177e2005-04-16 15:20:36 -070092#ifdef __KERNEL__
93
Akinobu Mitaf33e2fb2006-03-26 01:39:42 -080094#include <asm-generic/bitops/sched.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070095
96/**
97 * ffs - find first bit set
98 * @x: the word to search
99 *
100 * This is defined the same way as
101 * the libc and compiler builtin ffs routines, therefore
102 * differs in spirit from the above ffz (man ffs).
103 */
Randy Dunlapade8c562007-10-25 14:27:24 -0700104static inline int ffs(int x)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700105{
106 int r;
107
108 __asm__("bsfl %1,%0\n\t"
109 "cmovzl %2,%0"
110 : "=r" (r) : "rm" (x), "r" (-1));
111 return r+1;
112}
113
114/**
Stephen Hemminger90933fc2005-12-21 19:31:36 -0800115 * fls64 - find last bit set in 64 bit word
116 * @x: the word to search
117 *
118 * This is defined the same way as fls.
119 */
Randy Dunlapade8c562007-10-25 14:27:24 -0700120static inline int fls64(__u64 x)
Stephen Hemminger90933fc2005-12-21 19:31:36 -0800121{
122 if (x == 0)
123 return 0;
124 return __fls(x) + 1;
125}
126
127/**
Stephen Hemminger636dd2b2006-01-11 22:43:24 +0100128 * fls - find last bit set
129 * @x: the word to search
130 *
131 * This is defined the same way as ffs.
132 */
Randy Dunlapade8c562007-10-25 14:27:24 -0700133static inline int fls(int x)
Stephen Hemminger636dd2b2006-01-11 22:43:24 +0100134{
135 int r;
136
137 __asm__("bsrl %1,%0\n\t"
138 "cmovzl %2,%0"
139 : "=&r" (r) : "rm" (x), "rm" (-1));
140 return r+1;
141}
142
Andi Kleen01366112006-09-26 10:52:38 +0200143#define ARCH_HAS_FAST_MULTIPLIER 1
144
Akinobu Mitaf33e2fb2006-03-26 01:39:42 -0800145#include <asm-generic/bitops/hweight.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -0700146
147#endif /* __KERNEL__ */
148
149#ifdef __KERNEL__
150
Akinobu Mitaf33e2fb2006-03-26 01:39:42 -0800151#include <asm-generic/bitops/ext2-non-atomic.h>
152
Joe Perches49f74942008-03-23 01:01:44 -0700153#define ext2_set_bit_atomic(lock, nr, addr) \
154 test_and_set_bit((nr), (unsigned long *)(addr))
155#define ext2_clear_bit_atomic(lock, nr, addr) \
156 test_and_clear_bit((nr), (unsigned long *)(addr))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157
Akinobu Mitaf33e2fb2006-03-26 01:39:42 -0800158#include <asm-generic/bitops/minix.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -0700159
Linus Torvalds1da177e2005-04-16 15:20:36 -0700160#endif /* __KERNEL__ */
161
162#endif /* _X86_64_BITOPS_H */