|
@@ -242,3 +242,169 @@ static inline int __test_bit(unsigned long nr, const volatile void *addr)
|
|
|
__test_bit((nr),(addr)))
|
|
|
|
|
|
#include <asm-generic/bitops/find.h>
|
|
|
+
|
|
|
+/**
|
|
|
+ * fls - find last bit set
|
|
|
+ * @x: the word to search
|
|
|
+ *
|
|
|
+ * This is defined the same way as ffs:
|
|
|
+ * - return 32..1 to indicate bit 31..0 most significant bit set
|
|
|
+ * - return 0 to indicate no bits set
|
|
|
+ */
|
|
|
+#define fls(x) \
|
|
|
+({ \
|
|
|
+ int bit; \
|
|
|
+ \
|
|
|
+ asm(" subcc %1,gr0,gr0,icc0 \n" \
|
|
|
+ " ckne icc0,cc4 \n" \
|
|
|
+ " cscan.p %1,gr0,%0 ,cc4,#1 \n" \
|
|
|
+ " csub %0,%0,%0 ,cc4,#0 \n" \
|
|
|
+ " csub %2,%0,%0 ,cc4,#1 \n" \
|
|
|
+ : "=&r"(bit) \
|
|
|
+ : "r"(x), "r"(32) \
|
|
|
+ : "icc0", "cc4" \
|
|
|
+ ); \
|
|
|
+ \
|
|
|
+ bit; \
|
|
|
+})
|
|
|
+
|
|
|
+/**
|
|
|
+ * fls64 - find last bit set in a 64-bit value
|
|
|
+ * @n: the value to search
|
|
|
+ *
|
|
|
+ * This is defined the same way as ffs:
|
|
|
+ * - return 64..1 to indicate bit 63..0 most significant bit set
|
|
|
+ * - return 0 to indicate no bits set
|
|
|
+ */
|
|
|
+static inline __attribute__((const))
|
|
|
+int fls64(u64 n)
|
|
|
+{
|
|
|
+ union {
|
|
|
+ u64 ll;
|
|
|
+ struct { u32 h, l; };
|
|
|
+ } _;
|
|
|
+ int bit, x, y;
|
|
|
+
|
|
|
+ _.ll = n;
|
|
|
+
|
|
|
+ asm(" subcc.p %3,gr0,gr0,icc0 \n"
|
|
|
+ " subcc %4,gr0,gr0,icc1 \n"
|
|
|
+ " ckne icc0,cc4 \n"
|
|
|
+ " ckne icc1,cc5 \n"
|
|
|
+ " norcr cc4,cc5,cc6 \n"
|
|
|
+ " csub.p %0,%0,%0 ,cc6,1 \n"
|
|
|
+ " orcr cc5,cc4,cc4 \n"
|
|
|
+ " andcr cc4,cc5,cc4 \n"
|
|
|
+ " cscan.p %3,gr0,%0 ,cc4,0 \n"
|
|
|
+ " setlos #64,%1 \n"
|
|
|
+ " cscan.p %4,gr0,%0 ,cc4,1 \n"
|
|
|
+ " setlos #32,%2 \n"
|
|
|
+ " csub.p %1,%0,%0 ,cc4,0 \n"
|
|
|
+ " csub %2,%0,%0 ,cc4,1 \n"
|
|
|
+ : "=&r"(bit), "=r"(x), "=r"(y)
|
|
|
+ : "0r"(_.h), "r"(_.l)
|
|
|
+ : "icc0", "icc1", "cc4", "cc5", "cc6"
|
|
|
+ );
|
|
|
+ return bit;
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * ffs - find first bit set
|
|
|
+ * @x: the word to search
|
|
|
+ *
|
|
|
+ * - return 32..1 to indicate bit 31..0 most least significant bit set
|
|
|
+ * - return 0 to indicate no bits set
|
|
|
+ */
|
|
|
+static inline __attribute__((const))
|
|
|
+int ffs(int x)
|
|
|
+{
|
|
|
+ /* Note: (x & -x) gives us a mask that is the least significant
|
|
|
+ * (rightmost) 1-bit of the value in x.
|
|
|
+ */
|
|
|
+ return fls(x & -x);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * __ffs - find first bit set
|
|
|
+ * @x: the word to search
|
|
|
+ *
|
|
|
+ * - return 31..0 to indicate bit 31..0 most least significant bit set
|
|
|
+ * - if no bits are set in x, the result is undefined
|
|
|
+ */
|
|
|
+static inline __attribute__((const))
|
|
|
+int __ffs(unsigned long x)
|
|
|
+{
|
|
|
+ int bit;
|
|
|
+ asm("scan %1,gr0,%0" : "=r"(bit) : "r"(x & -x));
|
|
|
+ return 31 - bit;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * __fls - find last (most-significant) set bit in a long word
|
|
|
+ * @word: the word to search
|
|
|
+ *
|
|
|
+ * Undefined if no set bit exists, so code should check against 0 first.
|
|
|
+ */
|
|
|
+static inline unsigned long __fls(unsigned long word)
|
|
|
+{
|
|
|
+ unsigned long bit;
|
|
|
+ asm("scan %1,gr0,%0" : "=r"(bit) : "r"(word));
|
|
|
+ return bit;
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * special slimline version of fls() for calculating ilog2_u32()
|
|
|
+ * - note: no protection against n == 0
|
|
|
+ */
|
|
|
+#define ARCH_HAS_ILOG2_U32
|
|
|
+static inline __attribute__((const))
|
|
|
+int __ilog2_u32(u32 n)
|
|
|
+{
|
|
|
+ int bit;
|
|
|
+ asm("scan %1,gr0,%0" : "=r"(bit) : "r"(n));
|
|
|
+ return 31 - bit;
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * special slimline version of fls64() for calculating ilog2_u64()
|
|
|
+ * - note: no protection against n == 0
|
|
|
+ */
|
|
|
+#define ARCH_HAS_ILOG2_U64
|
|
|
+static inline __attribute__((const))
|
|
|
+int __ilog2_u64(u64 n)
|
|
|
+{
|
|
|
+ union {
|
|
|
+ u64 ll;
|
|
|
+ struct { u32 h, l; };
|
|
|
+ } _;
|
|
|
+ int bit, x, y;
|
|
|
+
|
|
|
+ _.ll = n;
|
|
|
+
|
|
|
+ asm(" subcc %3,gr0,gr0,icc0 \n"
|
|
|
+ " ckeq icc0,cc4 \n"
|
|
|
+ " cscan.p %3,gr0,%0 ,cc4,0 \n"
|
|
|
+ " setlos #63,%1 \n"
|
|
|
+ " cscan.p %4,gr0,%0 ,cc4,1 \n"
|
|
|
+ " setlos #31,%2 \n"
|
|
|
+ " csub.p %1,%0,%0 ,cc4,0 \n"
|
|
|
+ " csub %2,%0,%0 ,cc4,1 \n"
|
|
|
+ : "=&r"(bit), "=r"(x), "=r"(y)
|
|
|
+ : "0r"(_.h), "r"(_.l)
|
|
|
+ : "icc0", "cc4"
|
|
|
+ );
|
|
|
+ return bit;
|
|
|
+}
|
|
|
+
|
|
|
+#include <asm-generic/bitops/sched.h>
|
|
|
+#include <asm-generic/bitops/hweight.h>
|
|
|
+#include <asm-generic/bitops/lock.h>
|
|
|
+
|
|
|
+#include <asm-generic/bitops/le.h>
|
|
|
+
|
|
|
+#include <asm-generic/bitops/ext2-atomic-setbit.h>
|
|
|
+
|
|
|
+#endif /* __KERNEL__ */
|
|
|
+
|
|
|
+#endif /* _ASM_BITOPS_H */
|