165 строки
4.0 KiB

  1. //
  2. // Lol Engine
  3. //
  4. // Copyright: (c) 2010-2011 Sam Hocevar <sam@hocevar.net>
  5. // This program is free software; you can redistribute it and/or
  6. // modify it under the terms of the Do What The Fuck You Want To
  7. // Public License, Version 2, as published by Sam Hocevar. See
  8. // http://sam.zoy.org/projects/COPYING.WTFPL for more details.
  9. //
  10. #if defined HAVE_CONFIG_H
  11. # include "config.h"
  12. #endif
  13. #include <cstring>
  14. #include <cstdio>
  15. #include "core.h"
  16. using namespace std;
  17. namespace lol
  18. {
  19. real::real(float f)
  20. {
  21. union { float f; uint32_t x; } u = { f };
  22. uint32_t sign = u.x & 0x80000000u;
  23. int e = ((u.x >> 23) & 0xff) + (1 << 30) - (1 << 7);
  24. m_signexp = sign | e;
  25. m_mantissa[0] = u.x >> 7;
  26. m_mantissa[1] = u.x << 9;
  27. memset(m_mantissa + 2, 0, sizeof(m_mantissa) - sizeof(m_mantissa[0]));
  28. }
  29. real::operator float() const
  30. {
  31. union { float f; uint32_t x; } u;
  32. u.x = m_mantissa[0] << 7;
  33. u.x |= m_mantissa[1] >> 9;
  34. u.x |= ((m_signexp & 0x7fffffffu) - (1 << 30) + (1 << 7)) << 23;
  35. u.x |= m_signexp & 0x80000000u;
  36. return u.f;
  37. }
  38. real real::operator -()
  39. {
  40. m_signexp ^= 0x80000000u;
  41. return *this;
  42. }
  43. real real::operator +(real const &x) const
  44. {
  45. if ((m_signexp << 1) < (x.m_signexp << 1))
  46. return x + *this;
  47. /* For now, assume both numbers are positive. */
  48. real ret;
  49. int e1 = (m_signexp & 0x7fffffffu) - (1 << 30) + 1;
  50. int e2 = (x.m_signexp & 0x7fffffffu) - (1 << 30) + 1;
  51. int bigoff = (e1 - e2) / (sizeof(uint16_t) * 8);
  52. int off = e1 - e2 - bigoff * (sizeof(uint16_t) * 8);
  53. ret.m_signexp = m_signexp;
  54. uint32_t carry = 0;
  55. for (int i = 0; i < BIGITS; i++)
  56. {
  57. carry = m_mantissa[BIGITS - 1 - i];
  58. if (BIGITS - 1 - i - bigoff >= 0)
  59. carry += x.m_mantissa[BIGITS - 1 - i - bigoff] >> off;
  60. else if (BIGITS - 1 - i - bigoff == -1)
  61. carry += 0x0001u >> off;
  62. if (BIGITS - 1 - i - bigoff - 1 >= 0)
  63. carry += (x.m_mantissa[BIGITS - 1 - i - bigoff - 1] << (16 - off)) & 0xffffu;
  64. else if (BIGITS - 1 - i - bigoff - 1 == -1)
  65. carry += 0x0001u << (16 - off);
  66. ret.m_mantissa[BIGITS - 1 - i] = carry;
  67. carry >>= 16;
  68. }
  69. /* Renormalise in case we overflowed the mantissa */
  70. if (carry)
  71. {
  72. carry--;
  73. for (int i = 0; i < BIGITS; i++)
  74. {
  75. uint16_t tmp = ret.m_mantissa[i];
  76. ret.m_mantissa[i] = (carry << 15) | (tmp >> 1);
  77. carry = tmp & 0x0001u;
  78. }
  79. ret.m_signexp++;
  80. }
  81. return ret;
  82. }
  83. real real::operator *(real const &x) const
  84. {
  85. real ret;
  86. ret.m_signexp = (m_signexp ^ x.m_signexp) & 0x80000000u;
  87. int e = (m_signexp & 0x7fffffffu) - (1 << 30) + 1
  88. + (x.m_signexp & 0x7fffffffu) - (1 << 30) + 1;
  89. /* Accumulate low order product; no need to store it, we just
  90. * want the carry value */
  91. uint32_t carry = 0;
  92. for (int i = 0; i < BIGITS; i++)
  93. {
  94. for (int j = 0; j < i + 1; j++)
  95. carry += m_mantissa[BIGITS - 1 - j]
  96. * x.m_mantissa[BIGITS - 1 + j - i];
  97. carry >>= 16;
  98. }
  99. for (int i = 0; i < BIGITS; i++)
  100. {
  101. for (int j = i + 1; j < BIGITS; j++)
  102. carry += m_mantissa[BIGITS - 1 - j]
  103. * x.m_mantissa[j - 1 - i];
  104. carry += m_mantissa[BIGITS - 1 - i];
  105. carry += x.m_mantissa[BIGITS - 1 - i];
  106. ret.m_mantissa[BIGITS - 1 - i] = carry & 0xffffu;
  107. carry >>= 16;
  108. }
  109. /* Renormalise in case we overflowed the mantissa */
  110. if (carry)
  111. {
  112. carry--;
  113. for (int i = 0; i < BIGITS; i++)
  114. {
  115. uint16_t tmp = ret.m_mantissa[i];
  116. ret.m_mantissa[i] = (carry << 15) | (tmp >> 1);
  117. carry = tmp & 0x0001u;
  118. }
  119. e++;
  120. }
  121. ret.m_signexp |= e + (1 << 30) - 1;
  122. return ret;
  123. }
  124. void real::print() const
  125. {
  126. printf("%x %08x ", m_signexp >> 31, (m_signexp << 1) >> 1);
  127. for (int i = 0; i < BIGITS; i++)
  128. printf("%04x ", m_mantissa[i]);
  129. printf("\n");
  130. }
  131. } /* namespace lol */