@@ -803,6 +803,69 @@ def testHypot(self):
803
803
scale = FLOAT_MIN / 2.0 ** exp
804
804
self .assertEqual (math .hypot (4 * scale , 3 * scale ), 5 * scale )
805
805
806
+ @requires_IEEE_754
807
+ @unittest .skipIf (HAVE_DOUBLE_ROUNDING ,
808
+ "hypot() loses accuracy on machines with double rounding" )
809
+ def testHypotAccuracy (self ):
810
+ # Verify improved accuracy in cases that were known to be inaccurate.
811
+ #
812
+ # The new algorithm's accuracy depends on IEEE 754 arithmetic
813
+ # guarantees, on having the usual ROUND HALF EVEN rounding mode, on
814
+ # the system not having double rounding due to extended precision,
815
+ # and on the compiler maintaining the specified order of operations.
816
+ #
817
+ # This test is known to succeed on most of our builds. If it fails
818
+ # some build, we either need to add another skipIf if the cause is
819
+ # identifiable; otherwise, we can remove this test entirely.
820
+
821
+ hypot = math .hypot
822
+ Decimal = decimal .Decimal
823
+ high_precision = decimal .Context (prec = 500 )
824
+
825
+ for hx , hy in [
826
+ # Cases with a 1 ulp error in Python 3.7 compiled with Clang
827
+ ('0x1.10e89518dca48p+29' , '0x1.1970f7565b7efp+30' ),
828
+ ('0x1.10106eb4b44a2p+29' , '0x1.ef0596cdc97f8p+29' ),
829
+ ('0x1.459c058e20bb7p+30' , '0x1.993ca009b9178p+29' ),
830
+ ('0x1.378371ae67c0cp+30' , '0x1.fbe6619854b4cp+29' ),
831
+ ('0x1.f4cd0574fb97ap+29' , '0x1.50fe31669340ep+30' ),
832
+ ('0x1.494b2cdd3d446p+29' , '0x1.212a5367b4c7cp+29' ),
833
+ ('0x1.f84e649f1e46dp+29' , '0x1.1fa56bef8eec4p+30' ),
834
+ ('0x1.2e817edd3d6fap+30' , '0x1.eb0814f1e9602p+29' ),
835
+ ('0x1.0d3a6e3d04245p+29' , '0x1.32a62fea52352p+30' ),
836
+ ('0x1.888e19611bfc5p+29' , '0x1.52b8e70b24353p+29' ),
837
+
838
+ # Cases with 2 ulp error in Python 3.8
839
+ ('0x1.538816d48a13fp+29' , '0x1.7967c5ca43e16p+29' ),
840
+ ('0x1.57b47b7234530p+29' , '0x1.74e2c7040e772p+29' ),
841
+ ('0x1.821b685e9b168p+30' , '0x1.677dc1c1e3dc6p+29' ),
842
+ ('0x1.9e8247f67097bp+29' , '0x1.24bd2dc4f4baep+29' ),
843
+ ('0x1.b73b59e0cb5f9p+29' , '0x1.da899ab784a97p+28' ),
844
+ ('0x1.94a8d2842a7cfp+30' , '0x1.326a51d4d8d8ap+30' ),
845
+ ('0x1.e930b9cd99035p+29' , '0x1.5a1030e18dff9p+30' ),
846
+ ('0x1.1592bbb0e4690p+29' , '0x1.a9c337b33fb9ap+29' ),
847
+ ('0x1.1243a50751fd4p+29' , '0x1.a5a10175622d9p+29' ),
848
+ ('0x1.57a8596e74722p+30' , '0x1.42d1af9d04da9p+30' ),
849
+
850
+ # Cases with 1 ulp error in version fff3c28052e6b0
851
+ ('0x1.ee7dbd9565899p+29' , '0x1.7ab4d6fc6e4b4p+29' ),
852
+ ('0x1.5c6bfbec5c4dcp+30' , '0x1.02511184b4970p+30' ),
853
+ ('0x1.59dcebba995cap+30' , '0x1.50ca7e7c38854p+29' ),
854
+ ('0x1.768cdd94cf5aap+29' , '0x1.9cfdc5571d38ep+29' ),
855
+ ('0x1.dcf137d60262ep+29' , '0x1.1101621990b3ep+30' ),
856
+ ('0x1.3a2d006e288b0p+30' , '0x1.e9a240914326cp+29' ),
857
+ ('0x1.62a32f7f53c61p+29' , '0x1.47eb6cd72684fp+29' ),
858
+ ('0x1.d3bcb60748ef2p+29' , '0x1.3f13c4056312cp+30' ),
859
+ ('0x1.282bdb82f17f3p+30' , '0x1.640ba4c4eed3ap+30' ),
860
+ ('0x1.89d8c423ea0c6p+29' , '0x1.d35dcfe902bc3p+29' ),
861
+ ]:
862
+ x = float .fromhex (hx )
863
+ y = float .fromhex (hy )
864
+ with self .subTest (hx = hx , hy = hy , x = x , y = y ):
865
+ with decimal .localcontext (high_precision ):
866
+ z = float ((Decimal (x )** 2 + Decimal (y )** 2 ).sqrt ())
867
+ self .assertEqual (hypot (x , y ), z )
868
+
806
869
def testDist (self ):
807
870
from decimal import Decimal as D
808
871
from fractions import Fraction as F
0 commit comments