<div dir="ltr"><div dir="ltr"><br></div><br><div class="gmail_quote"><div dir="ltr" class="gmail_attr">Le jeu. 29 août 2019 à 12:28, <<a href="mailto:commits@source.squeak.org">commits@source.squeak.org</a>> a écrit :<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">Nicolas Cellier uploaded a new version of Kernel to project The Trunk:<br>
<a href="http://source.squeak.org/trunk/Kernel-nice.1262.mcz" rel="noreferrer" target="_blank">http://source.squeak.org/trunk/Kernel-nice.1262.mcz</a><br>
<br>
==================== Summary ====================<br>
<br>
Name: Kernel-nice.1262<br>
Author: nice<br>
Time: 29 August 2019, 12:28:31.665288 pm<br>
UUID: 3e167d35-e96e-7649-8324-28d9cb255a65<br>
Ancestors: Kernel-nice.1261<br>
<br>
Accelerate LargeInteger asFloat in 64bits images.<br>
<br>
=============== Diff against Kernel-nice.1261 ===============<br>
<br>
Item was added:<br>
+ ----- Method: Integer>>digitsAsFloat (in category 'private') -----<br>
+ digitsAsFloat<br>
+       "private - naive conversion method.<br>
+       This method should be used if and only if<br>
+       Float precision + 8 >= self highBit.<br>
+       This way, all floating point operations will be exact, but eventually the<br>
+       last one, giving a guaranty that result will be the nearest Float."<br>
+       | result n |<br>
+       result := (self digitAt: (n := self digitLength)) asFloat.<br>
+       [(n := n - 1) > 0]<br>
+               whileTrue: [result := 256.0 * result + (self digitAt: n) asFloat].<br>
+       ^ result!<br>
<br></blockquote><div>Some additional notes:</div><div><br></div><div>I've also tried a simple split of the 53 (Float precision) + 7 (excess bits) significand into two 30 bits SmallInteger:</div><div><br></div><div>digitsAsFloat2<br></div><div>^(self bitShift: -30) asFloat * 1073741824.0<br>  + (self bitAnd: 16r3FFFFFFF) asFloat</div><div><br></div><div>Unfortunately, this is slower than looping on the byte-digits like proposed above:</div><div><br></div><div>self assert: 923456789012345678 highBit = 60.<br>[923456789012345678 digitsAsFloat] bench. '7,290,000 per second. 137 nanoseconds per run.'<br>[923456789012345678 digitsAsFloat2] bench. '3,430,000 per second. 291 nanoseconds per run.'</div><div><br></div><div>I did not use latest VM when trying this (Macos-metal pb <a href="https://github.com/OpenSmalltalk/opensmalltalk-vm/issues/397">https://github.com/OpenSmalltalk/opensmalltalk-vm/issues/397</a>)</div><div><br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
Item was changed:<br>
  ----- Method: LargePositiveInteger>>asFloat (in category 'converting') -----<br>
  asFloat<br>
        "Answer a Float that best approximates the value of the receiver.<br>
        This algorithm is optimized to process only the significant digits of a LargeInteger.<br>
        And it does honour IEEE 754 round to nearest even mode in case of excess precision (see details below)."<br>
<br>
        "How numbers are rounded in IEEE 754 default rounding mode:<br>
        A shift is applied so that the highest 53 bits are placed before the floating point to form a mantissa.<br>
        The trailing bits form the fraction part placed after the floating point.<br>
        This fractional number must be rounded to the nearest integer.<br>
        If fraction part is 2r0.1, exactly between two consecutive integers, there is a tie.<br>
        The nearest even integer is chosen in this case.<br>
        Examples (First 52bits of mantissa are omitted for brevity):<br>
        2r0.00001 is rounded downward to 2r0<br>
        2r1.00001 is rounded downward to 2r1<br>
        2r0.1 is a tie and rounded to 2r0 (nearest even)<br>
        2r1.1 is a tie and rounded to 2r10 (nearest even)<br>
        2r0.10001 is rounded upward to 2r1<br>
        2r1.10001 is rounded upward to 2r10<br>
        Thus, if the next bit after floating point is 0, the mantissa is left unchanged.<br>
        If next bit after floating point is 1, an odd mantissa is always rounded upper.<br>
        An even mantissa is rounded upper only if the fraction part is not a tie."<br>
<br>
        "Algorihm details:<br>
        The floating point hardware can perform the rounding correctly with several excess bits as long as there is a single inexact operation.<br>
-       This can be obtained by splitting the mantissa plus excess bits in two part with less bits than Float precision.<br>
        Note 1: the inexact flag in floating point hardware must not be trusted because in some cases the operations would be exact but would not take into account some bits that were truncated before the Floating point operations.<br>
        Note 2: the floating point hardware is presumed configured in default rounding mode."<br>
<br>
+       | mantissa shift excess |<br>
-       | mantissa shift excess result n |<br>
<br>
        "Check how many bits excess the maximum precision of a Float mantissa."<br>
        excess := self highBitOfMagnitude - Float precision.<br>
        excess > 7<br>
                ifTrue:<br>
                        ["Remove the excess bits but seven."<br>
                        mantissa := self bitShiftMagnitude: 7 - excess.<br>
                        shift := excess - 7.<br>
                        "An even mantissa with a single excess bit immediately following would be truncated.<br>
                        But this would not be correct if above shift has truncated some extra bits.<br>
                        Check this case, and round excess bits upper manually."<br>
                        ((mantissa digitAt: 1) = 2r01000000 and: [self anyBitOfMagnitudeFrom: 1 to: shift])<br>
                                ifTrue: [mantissa := mantissa + 1]]<br>
                ifFalse:<br>
                        [mantissa := self.<br>
                        shift := 0].<br>
<br>
+       "We can use naive digit by digit conversion because there will be a single inexact round off at last iteration.<br>
+       But the nice thing is that Float precision + 7 excess bits = 60 which fit in a SmallInteger in Spur64.<br>
+       So the best to do is to delegate this final operation"<br>
+       ^mantissa digitsAsFloat timesTwoPower: shift.!<br>
-       "There will be a single inexact round off at last iteration"<br>
-       result := (mantissa digitAt: (n := mantissa digitLength)) asFloat.<br>
-       [(n := n - 1) > 0] whileTrue: [<br>
-               result := 256.0 * result + (mantissa digitAt: n) asFloat].<br>
-       ^result timesTwoPower: shift.!<br>
<br>
Item was added:<br>
+ ----- Method: SmallInteger>>digitsAsFloat (in category 'private') -----<br>
+ digitsAsFloat<br>
+       "private - let the primitive take care to answer the nearest float"<br>
+       <primitive: 40><br>
+       ^super digitsAsFloat!<br>
<br>
<br>
</blockquote></div></div>