1 #ifndef __ASM_ARM_DIV64
 
   2 #define __ASM_ARM_DIV64
 
   4 #include <asm/system.h>
 
   5 #include <linux/types.h>
 
   8  * The semantics of do_div() are:
 
  10  * uint32_t do_div(uint64_t *n, uint32_t base)
 
  12  *      uint32_t remainder = *n % base;
 
  17  * In other words, a 64-bit dividend with a 32-bit divisor producing
 
  18  * a 64-bit result and a 32-bit remainder.  To accomplish this optimally
 
  19  * we call a special __do_div64 helper with completely non standard
 
  20  * calling convention for arguments and results (beware).
 
  31 #define __do_div_asm(n, base)                                   \
 
  33         register unsigned int __base      asm("r4") = base;     \
 
  34         register unsigned long long __n   asm("r0") = n;        \
 
  35         register unsigned long long __res asm("r2");            \
 
  36         register unsigned int __rem       asm(__xh);            \
 
  37         asm(    __asmeq("%0", __xh)                             \
 
  42                 : "=r" (__rem), "=r" (__res)                    \
 
  43                 : "r" (__n), "r" (__base)                       \
 
  44                 : "ip", "lr", "cc");                            \
 
  52  * gcc versions earlier than 4.0 are simply too problematic for the
 
  53  * optimized implementation below. First there is gcc PR 15089 that
 
  54  * tend to trig on more complex constructs, spurious .global __udivsi3
 
  55  * are inserted even if none of those symbols are referenced in the
 
  56  * generated code, and those gcc versions are not able to do constant
 
  57  * propagation on long long values anyway.
 
  59 #define do_div(n, base) __do_div_asm(n, base)
 
  66  * If the divisor happens to be constant, we determine the appropriate
 
  67  * inverse at compile time to turn the division into a few inline
 
  68  * multiplications instead which is much faster. And yet only if compiling
 
  69  * for ARMv4 or higher (we need umull/umlal) and if the gcc version is
 
  70  * sufficiently recent to perform proper long long constant propagation.
 
  71  * (It is unfortunate that gcc doesn't perform all this internally.)
 
  73 #define do_div(n, base)                                                 \
 
  75         unsigned int __r, __b = (base);                                 \
 
  76         if (!__builtin_constant_p(__b) || __b == 0 ||                   \
 
  77             (__LINUX_ARM_ARCH__ < 4 && (__b & (__b - 1)) != 0)) {       \
 
  78                 /* non-constant divisor (or zero): slow path */         \
 
  79                 __r = __do_div_asm(n, __b);                             \
 
  80         } else if ((__b & (__b - 1)) == 0) {                            \
 
  81                 /* Trivial: __b is constant and a power of 2 */         \
 
  82                 /* gcc does the right thing with this code.  */         \
 
  87                 /* Multiply by inverse of __b: n/b = n*(p/b)/p       */ \
 
  88                 /* We rely on the fact that most of this code gets   */ \
 
  89                 /* optimized away at compile time due to constant    */ \
 
  90                 /* propagation and only a couple inline assembly     */ \
 
  91                 /* instructions should remain. Better avoid any      */ \
 
  92                 /* code construct that might prevent that.           */ \
 
  93                 unsigned long long __res, __x, __t, __m, __n = n;       \
 
  94                 unsigned int __c, __p, __z = 0;                         \
 
  95                 /* preserve low part of n for reminder computation */   \
 
  97                 /* determine number of bits to represent __b */         \
 
  98                 __p = 1 << __div64_fls(__b);                            \
 
  99                 /* compute __m = ((__p << 64) + __b - 1) / __b */       \
 
 100                 __m = (~0ULL / __b) * __p;                              \
 
 101                 __m += (((~0ULL % __b + 1) * __p) + __b - 1) / __b;     \
 
 102                 /* compute __res = __m*(~0ULL/__b*__b-1)/(__p << 64) */ \
 
 103                 __x = ~0ULL / __b * __b - 1;                            \
 
 104                 __res = (__m & 0xffffffff) * (__x & 0xffffffff);        \
 
 106                 __res += (__m & 0xffffffff) * (__x >> 32);              \
 
 108                 __res += (__x & 0xffffffff) * (__m >> 32);              \
 
 109                 __t = (__res < __t) ? (1ULL << 32) : 0;                 \
 
 110                 __res = (__res >> 32) + __t;                            \
 
 111                 __res += (__m >> 32) * (__x >> 32);                     \
 
 113                 /* Now sanitize and optimize what we've got. */         \
 
 114                 if (~0ULL % (__b / (__b & -__b)) == 0) {                \
 
 115                         /* those cases can be simplified with: */       \
 
 116                         __n /= (__b & -__b);                            \
 
 117                         __m = ~0ULL / (__b / (__b & -__b));             \
 
 120                 } else if (__res != __x / __b) {                        \
 
 121                         /* We can't get away without a correction    */ \
 
 122                         /* to compensate for bit truncation errors.  */ \
 
 123                         /* To avoid it we'd need an additional bit   */ \
 
 124                         /* to represent __m which would overflow it. */ \
 
 125                         /* Instead we do m=p/b and n/b=(n*m+m)/p.    */ \
 
 127                         /* Compute __m = (__p << 64) / __b */           \
 
 128                         __m = (~0ULL / __b) * __p;                      \
 
 129                         __m += ((~0ULL % __b + 1) * __p) / __b;         \
 
 131                         /* Reduce __m/__p, and try to clear bit 31   */ \
 
 132                         /* of __m when possible otherwise that'll    */ \
 
 133                         /* need extra overflow handling later.       */ \
 
 134                         unsigned int __bits = -(__m & -__m);            \
 
 135                         __bits |= __m >> 32;                            \
 
 136                         __bits = (~__bits) << 1;                        \
 
 137                         /* If __bits == 0 then setting bit 31 is     */ \
 
 138                         /* unavoidable.  Simply apply the maximum    */ \
 
 139                         /* possible reduction in that case.          */ \
 
 140                         /* Otherwise the MSB of __bits indicates the */ \
 
 141                         /* best reduction we should apply.           */ \
 
 143                                 __p /= (__m & -__m);                    \
 
 144                                 __m /= (__m & -__m);                    \
 
 146                                 __p >>= __div64_fls(__bits);            \
 
 147                                 __m >>= __div64_fls(__bits);            \
 
 149                         /* No correction needed. */                     \
 
 152                 /* Now we have a combination of 2 conditions:        */ \
 
 153                 /* 1) whether or not we need a correction (__c), and */ \
 
 154                 /* 2) whether or not there might be an overflow in   */ \
 
 155                 /*    the cross product (__m & ((1<<63) | (1<<31)))  */ \
 
 156                 /* Select the best insn combination to perform the   */ \
 
 157                 /* actual __m * __n / (__p << 64) operation.         */ \
 
 159                         asm (   "umull  %Q0, %R0, %1, %Q2\n\t"          \
 
 162                                 : "r" (__m), "r" (__n)                  \
 
 164                 } else if (!(__m & ((1ULL << 63) | (1ULL << 31)))) {    \
 
 166                         asm (   "umlal  %Q0, %R0, %Q1, %Q2\n\t"         \
 
 169                                 : "r" (__m), "r" (__n)                  \
 
 172                         asm (   "umull  %Q0, %R0, %Q1, %Q2\n\t"         \
 
 174                                 "adcs   %R0, %R0, %R1\n\t"              \
 
 177                                 : "r" (__m), "r" (__n), "r" (__z)       \
 
 180                 if (!(__m & ((1ULL << 63) | (1ULL << 31)))) {           \
 
 181                         asm (   "umlal  %R0, %Q0, %R1, %Q2\n\t"         \
 
 182                                 "umlal  %R0, %Q0, %Q1, %R2\n\t"         \
 
 184                                 "umlal  %Q0, %R0, %R1, %R2"             \
 
 186                                 : "r" (__m), "r" (__n)                  \
 
 189                         asm (   "umlal  %R0, %Q0, %R2, %Q3\n\t"         \
 
 190                                 "umlal  %R0, %1, %Q2, %R3\n\t"          \
 
 192                                 "adds   %Q0, %1, %Q0\n\t"               \
 
 193                                 "adc    %R0, %R0, #0\n\t"               \
 
 194                                 "umlal  %Q0, %R0, %R2, %R3"             \
 
 195                                 : "+r" (__res), "+r" (__z)              \
 
 196                                 : "r" (__m), "r" (__n)                  \
 
 200                 /* The reminder can be computed with 32-bit regs     */ \
 
 201                 /* only, and gcc is good at that.                    */ \
 
 203                         unsigned int __res0 = __res;                    \
 
 204                         unsigned int __b0 = __b;                        \
 
 205                         __r -= __res0 * __b0;                           \
 
 207                 /* BUG_ON(__r >= __b || __res * __b + __r != n); */     \
 
 213 /* our own fls implementation to make sure constant propagation is fine */
 
 214 #define __div64_fls(bits)                                               \
 
 216         unsigned int __left = (bits), __nr = 0;                         \
 
 217         if (__left & 0xffff0000) __nr += 16, __left >>= 16;             \
 
 218         if (__left & 0x0000ff00) __nr +=  8, __left >>=  8;             \
 
 219         if (__left & 0x000000f0) __nr +=  4, __left >>=  4;             \
 
 220         if (__left & 0x0000000c) __nr +=  2, __left >>=  2;             \
 
 221         if (__left & 0x00000002) __nr +=  1;                            \