fpu_arith.h 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. /* $NetBSD: fpu_arith.h,v 1.4 2005/12/24 20:07:28 perry Exp $ */
  2. /* $FreeBSD$ */
  3. /*-
  4. * SPDX-License-Identifier: BSD-3-Clause
  5. *
  6. * Copyright (c) 1992, 1993
  7. * The Regents of the University of California. All rights reserved.
  8. *
  9. * This software was developed by the Computer Systems Engineering group
  10. * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
  11. * contributed to Berkeley.
  12. *
  13. * All advertising materials mentioning features or use of this software
  14. * must display the following acknowledgement:
  15. * This product includes software developed by the University of
  16. * California, Lawrence Berkeley Laboratory.
  17. *
  18. * Redistribution and use in source and binary forms, with or without
  19. * modification, are permitted provided that the following conditions
  20. * are met:
  21. * 1. Redistributions of source code must retain the above copyright
  22. * notice, this list of conditions and the following disclaimer.
  23. * 2. Redistributions in binary form must reproduce the above copyright
  24. * notice, this list of conditions and the following disclaimer in the
  25. * documentation and/or other materials provided with the distribution.
  26. * 3. Neither the name of the University nor the names of its contributors
  27. * may be used to endorse or promote products derived from this software
  28. * without specific prior written permission.
  29. *
  30. * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  31. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  32. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  33. * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  34. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  35. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  36. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  37. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  38. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  39. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  40. * SUCH DAMAGE.
  41. *
  42. * @(#)fpu_arith.h 8.1 (Berkeley) 6/11/93
  43. */
  44. /*
  45. * Extended-precision arithmetic.
  46. *
  47. * We hold the notion of a `carry register', which may or may not be a
  48. * machine carry bit or register. On the SPARC, it is just the machine's
  49. * carry bit.
  50. *
  51. * In the worst case, you can compute the carry from x+y as
  52. * (unsigned)(x + y) < (unsigned)x
  53. * and from x+y+c as
  54. * ((unsigned)(x + y + c) <= (unsigned)x && (y|c) != 0)
  55. * for example.
  56. */
  57. #ifndef FPE_USE_ASM
  58. /* set up for extended-precision arithemtic */
  59. #define FPU_DECL_CARRY quad_t fpu_carry, fpu_tmp;
  60. /*
  61. * We have three kinds of add:
  62. * add with carry: r = x + y + c
  63. * add (ignoring current carry) and set carry: c'r = x + y + 0
  64. * add with carry and set carry: c'r = x + y + c
  65. * The macros use `C' for `use carry' and `S' for `set carry'.
  66. * Note that the state of the carry is undefined after ADDC and SUBC,
  67. * so if all you have for these is `add with carry and set carry',
  68. * that is OK.
  69. *
  70. * The same goes for subtract, except that we compute x - y - c.
  71. *
  72. * Finally, we have a way to get the carry into a `regular' variable,
  73. * or set it from a value. SET_CARRY turns 0 into no-carry, nonzero
  74. * into carry; GET_CARRY sets its argument to 0 or 1.
  75. */
  76. #define FPU_ADDC(r, x, y) \
  77. (r) = (x) + (y) + (!!fpu_carry)
  78. #define FPU_ADDS(r, x, y) \
  79. { \
  80. fpu_tmp = (quad_t)(x) + (quad_t)(y); \
  81. (r) = (u_int)fpu_tmp; \
  82. fpu_carry = ((fpu_tmp & 0xffffffff00000000LL) != 0); \
  83. }
  84. #define FPU_ADDCS(r, x, y) \
  85. { \
  86. fpu_tmp = (quad_t)(x) + (quad_t)(y) + (!!fpu_carry); \
  87. (r) = (u_int)fpu_tmp; \
  88. fpu_carry = ((fpu_tmp & 0xffffffff00000000LL) != 0); \
  89. }
  90. #define FPU_SUBC(r, x, y) \
  91. (r) = (x) - (y) - (!!fpu_carry)
  92. #define FPU_SUBS(r, x, y) \
  93. { \
  94. fpu_tmp = (quad_t)(x) - (quad_t)(y); \
  95. (r) = (u_int)fpu_tmp; \
  96. fpu_carry = ((fpu_tmp & 0xffffffff00000000LL) != 0); \
  97. }
  98. #define FPU_SUBCS(r, x, y) \
  99. { \
  100. fpu_tmp = (quad_t)(x) - (quad_t)(y) - (!!fpu_carry); \
  101. (r) = (u_int)fpu_tmp; \
  102. fpu_carry = ((fpu_tmp & 0xffffffff00000000LL) != 0); \
  103. }
  104. #define FPU_GET_CARRY(r) (r) = (!!fpu_carry)
  105. #define FPU_SET_CARRY(v) fpu_carry = ((v) != 0)
  106. #else
  107. /* set up for extended-precision arithemtic */
  108. #define FPU_DECL_CARRY
  109. /*
  110. * We have three kinds of add:
  111. * add with carry: r = x + y + c
  112. * add (ignoring current carry) and set carry: c'r = x + y + 0
  113. * add with carry and set carry: c'r = x + y + c
  114. * The macros use `C' for `use carry' and `S' for `set carry'.
  115. * Note that the state of the carry is undefined after ADDC and SUBC,
  116. * so if all you have for these is `add with carry and set carry',
  117. * that is OK.
  118. *
  119. * The same goes for subtract, except that we compute x - y - c.
  120. *
  121. * Finally, we have a way to get the carry into a `regular' variable,
  122. * or set it from a value. SET_CARRY turns 0 into no-carry, nonzero
  123. * into carry; GET_CARRY sets its argument to 0 or 1.
  124. */
  125. #define FPU_ADDC(r, x, y) \
  126. __asm volatile("adde %0,%1,%2" : "=r"(r) : "r"(x), "r"(y))
  127. #define FPU_ADDS(r, x, y) \
  128. __asm volatile("addc %0,%1,%2" : "=r"(r) : "r"(x), "r"(y))
  129. #define FPU_ADDCS(r, x, y) \
  130. __asm volatile("adde %0,%1,%2" : "=r"(r) : "r"(x), "r"(y))
  131. #define FPU_SUBC(r, x, y) \
  132. __asm volatile("subfe %0,%2,%1" : "=r"(r) : "r"(x), "r"(y))
  133. #define FPU_SUBS(r, x, y) \
  134. __asm volatile("subfc %0,%2,%1" : "=r"(r) : "r"(x), "r"(y))
  135. #define FPU_SUBCS(r, x, y) \
  136. __asm volatile("subfe %0,%2,%1" : "=r"(r) : "r"(x), "r"(y))
  137. #define FPU_GET_CARRY(r) __asm volatile("li %0,0; addie %0,%0,0" : "=r"(r))
  138. /* This one needs to destroy a temp register. */
  139. #define FPU_SET_CARRY(v) do { int __tmp; \
  140. __asm volatile("addic %0,%0,-1" : "r"(__tmp) : "r"(v)); \
  141. } while (0)
  142. #define FPU_SHL1_BY_ADD /* shift left 1 faster by ADDC than (a<<1)|(b>>31) */
  143. #endif