internal_windows.odin 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. package runtime
  2. @(link_name="__umodti3")
  3. umodti3 :: proc "c" (a, b: u128) -> u128 {
  4. r: u128 = ---;
  5. _ = udivmod128(a, b, &r);
  6. return r;
  7. }
  8. @(link_name="__udivmodti4")
  9. udivmodti4 :: proc "c" (a, b: u128, rem: ^u128) -> u128 {
  10. return udivmod128(a, b, rem);
  11. }
  12. @(link_name="__udivti3")
  13. udivti3 :: proc "c" (a, b: u128) -> u128 {
  14. return udivmodti4(a, b, nil);
  15. }
  16. @(link_name="__modti3")
  17. modti3 :: proc "c" (a, b: i128) -> i128 {
  18. s_a := a >> (128 - 1);
  19. s_b := b >> (128 - 1);
  20. an := (a ~ s_a) - s_a;
  21. bn := (b ~ s_b) - s_b;
  22. r: u128 = ---;
  23. _ = udivmod128(transmute(u128)an, transmute(u128)bn, &r);
  24. return (transmute(i128)r ~ s_a) - s_a;
  25. }
  26. @(link_name="__divmodti4")
  27. divmodti4 :: proc "c" (a, b: i128, rem: ^i128) -> i128 {
  28. u := udivmod128(transmute(u128)a, transmute(u128)b, cast(^u128)rem);
  29. return transmute(i128)u;
  30. }
  31. @(link_name="__divti3")
  32. divti3 :: proc "c" (a, b: i128) -> i128 {
  33. u := udivmodti4(transmute(u128)a, transmute(u128)b, nil);
  34. return transmute(i128)u;
  35. }
  36. @(link_name="__fixdfti")
  37. fixdfti :: proc(a: u64) -> i128 {
  38. significandBits :: 52;
  39. typeWidth :: (size_of(u64)*8);
  40. exponentBits :: (typeWidth - significandBits - 1);
  41. maxExponent :: ((1 << exponentBits) - 1);
  42. exponentBias :: (maxExponent >> 1);
  43. implicitBit :: (u64(1) << significandBits);
  44. significandMask :: (implicitBit - 1);
  45. signBit :: (u64(1) << (significandBits + exponentBits));
  46. absMask :: (signBit - 1);
  47. exponentMask :: (absMask ~ significandMask);
  48. // Break a into sign, exponent, significand
  49. aRep := a;
  50. aAbs := aRep & absMask;
  51. sign := i128(-1 if aRep & signBit != 0 else 1);
  52. exponent := u64((aAbs >> significandBits) - exponentBias);
  53. significand := u64((aAbs & significandMask) | implicitBit);
  54. // If exponent is negative, the result is zero.
  55. if exponent < 0 {
  56. return 0;
  57. }
  58. // If the value is too large for the integer type, saturate.
  59. if exponent >= size_of(i128) * 8 {
  60. return max(i128) if sign == 1 else min(i128);
  61. }
  62. // If 0 <= exponent < significandBits, right shift to get the result.
  63. // Otherwise, shift left.
  64. if exponent < significandBits {
  65. return sign * i128(significand >> (significandBits - exponent));
  66. } else {
  67. return sign * (i128(significand) << (exponent - significandBits));
  68. }
  69. }
  70. @(default_calling_convention = "none")
  71. foreign {
  72. @(link_name="llvm.ctlz.i128") _clz_i128 :: proc(x: i128, is_zero_undef := false) -> i128 ---
  73. }
  74. @(link_name="__floattidf")
  75. floattidf :: proc(a: i128) -> f64 {
  76. DBL_MANT_DIG :: 53;
  77. if a == 0 {
  78. return 0.0;
  79. }
  80. a := a;
  81. N :: size_of(i128) * 8;
  82. s := a >> (N-1);
  83. a = (a ~ s) - s;
  84. sd: = N - _clz_i128(a); // number of significant digits
  85. e := u32(sd - 1); // exponent
  86. if sd > DBL_MANT_DIG {
  87. switch sd {
  88. case DBL_MANT_DIG + 1:
  89. a <<= 1;
  90. case DBL_MANT_DIG + 2:
  91. // okay
  92. case:
  93. a = i128(u128(a) >> u128(sd - (DBL_MANT_DIG+2))) |
  94. i128(u128(a) & (~u128(0) >> u128(N + DBL_MANT_DIG+2 - sd)) != 0);
  95. };
  96. a |= i128((a & 4) != 0);
  97. a += 1;
  98. a >>= 2;
  99. if a & (1 << DBL_MANT_DIG) != 0 {
  100. a >>= 1;
  101. e += 1;
  102. }
  103. } else {
  104. a <<= u128(DBL_MANT_DIG - sd);
  105. }
  106. fb: [2]u32;
  107. fb[1] = (u32(s) & 0x80000000) | // sign
  108. ((e + 1023) << 20) | // exponent
  109. ((u32(a) >> 32) & 0x000FFFFF); // mantissa-high
  110. fb[1] = u32(a); // mantissa-low
  111. return transmute(f64)fb;
  112. }