blob: 0072133aae3bdfaea57333fa52c44710df79a37c [file] [log] [blame]
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001
2import unittest, struct
Christian Heimes827b35c2007-12-10 22:19:17 +00003import os
Eric Smith0923d1d2009-04-16 20:16:10 +00004import sys
Benjamin Petersonee8712c2008-05-20 21:35:26 +00005from test import support
Christian Heimes53876d92008-04-19 00:31:39 +00006import math
Mark Dickinson65fe25e2008-07-16 11:30:51 +00007from math import isinf, isnan, copysign, ldexp
Christian Heimes53876d92008-04-19 00:31:39 +00008import operator
Amaury Forgeot d'Arc7e958d12008-09-06 21:03:22 +00009import random, fractions
Michael W. Hudsonba283e22005-05-27 15:23:20 +000010
Christian Heimes53876d92008-04-19 00:31:39 +000011INF = float("inf")
12NAN = float("nan")
Christian Heimes99170a52007-12-19 02:07:34 +000013
Benjamin Petersone401c682010-07-02 23:25:44 +000014have_getformat = hasattr(float, "__getformat__")
15requires_getformat = unittest.skipUnless(have_getformat,
16 "requires __getformat__")
17requires_setformat = unittest.skipUnless(hasattr(float, "__setformat__"),
18 "requires __setformat__")
Mark Dickinson9ab44b52009-12-30 16:22:49 +000019# decorator for skipping tests on non-IEEE 754 platforms
Benjamin Petersone401c682010-07-02 23:25:44 +000020requires_IEEE_754 = unittest.skipUnless(have_getformat and
Mark Dickinson9ab44b52009-12-30 16:22:49 +000021 float.__getformat__("double").startswith("IEEE"),
22 "test requires IEEE 754 doubles")
23
Eric Smith0923d1d2009-04-16 20:16:10 +000024#locate file with float format test values
25test_dir = os.path.dirname(__file__) or os.curdir
26format_testfile = os.path.join(test_dir, 'formatfloat_testcases.txt')
27
Christian Heimes81ee3ef2008-05-04 22:42:01 +000028class GeneralFloatCases(unittest.TestCase):
29
30 def test_float(self):
31 self.assertEqual(float(3.14), 3.14)
32 self.assertEqual(float(314), 314.0)
33 self.assertEqual(float(" 3.14 "), 3.14)
Amaury Forgeot d'Arc7e958d12008-09-06 21:03:22 +000034 self.assertEqual(float(b" 3.14 "), 3.14)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000035 self.assertRaises(ValueError, float, " 0x3.1 ")
36 self.assertRaises(ValueError, float, " -0x3.p-1 ")
37 self.assertRaises(ValueError, float, " +0x3.p-1 ")
38 self.assertRaises(ValueError, float, "++3.14")
39 self.assertRaises(ValueError, float, "+-3.14")
40 self.assertRaises(ValueError, float, "-+3.14")
41 self.assertRaises(ValueError, float, "--3.14")
Eric Smith0923d1d2009-04-16 20:16:10 +000042 self.assertRaises(ValueError, float, ".nan")
43 self.assertRaises(ValueError, float, "+.inf")
44 self.assertRaises(ValueError, float, ".")
45 self.assertRaises(ValueError, float, "-.")
Mark Dickinsonc2d86892010-02-12 21:18:34 +000046 # check that we don't accept alternate exponent markers
47 self.assertRaises(ValueError, float, "-1.7d29")
48 self.assertRaises(ValueError, float, "3D-14")
Amaury Forgeot d'Arc7e958d12008-09-06 21:03:22 +000049 self.assertEqual(float(b" \u0663.\u0661\u0664 ".decode('raw-unicode-escape')), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000050 # extra long strings should not be a problem
51 float(b'.' + b'1'*1000)
52 float('.' + '1'*1000)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000053
Benjamin Petersonee8712c2008-05-20 21:35:26 +000054 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +000055 def test_float_with_comma(self):
56 # set locale to something that doesn't use '.' for the decimal point
57 # float must not accept the locale specific decimal point but
58 # it still has to accept the normal python syntac
59 import locale
60 if not locale.localeconv()['decimal_point'] == ',':
61 return
62
63 self.assertEqual(float(" 3.14 "), 3.14)
64 self.assertEqual(float("+3.14 "), 3.14)
65 self.assertEqual(float("-3.14 "), -3.14)
66 self.assertEqual(float(".14 "), .14)
67 self.assertEqual(float("3. "), 3.0)
68 self.assertEqual(float("3.e3 "), 3000.0)
69 self.assertEqual(float("3.2e3 "), 3200.0)
70 self.assertEqual(float("2.5e-1 "), 0.25)
71 self.assertEqual(float("5e-1"), 0.5)
72 self.assertRaises(ValueError, float, " 3,14 ")
73 self.assertRaises(ValueError, float, " +3,14 ")
74 self.assertRaises(ValueError, float, " -3,14 ")
75 self.assertRaises(ValueError, float, " 0x3.1 ")
76 self.assertRaises(ValueError, float, " -0x3.p-1 ")
77 self.assertRaises(ValueError, float, " +0x3.p-1 ")
78 self.assertEqual(float(" 25.e-1 "), 2.5)
Benjamin Petersond43029b2008-09-06 23:33:21 +000079 self.assertEqual(support.fcmp(float(" .25e-1 "), .025), 0)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000080
81 def test_floatconversion(self):
82 # Make sure that calls to __float__() work properly
83 class Foo0:
84 def __float__(self):
85 return 42.
86
87 class Foo1(object):
88 def __float__(self):
89 return 42.
90
91 class Foo2(float):
92 def __float__(self):
93 return 42.
94
95 class Foo3(float):
96 def __new__(cls, value=0.):
97 return float.__new__(cls, 2*value)
98
99 def __float__(self):
100 return self
101
102 class Foo4(float):
103 def __float__(self):
104 return 42
105
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000106 # Issue 5759: __float__ not called on str subclasses (though it is on
107 # unicode subclasses).
108 class FooStr(str):
109 def __float__(self):
110 return float(str(self)) + 1
111
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000112 self.assertAlmostEqual(float(Foo0()), 42.)
113 self.assertAlmostEqual(float(Foo1()), 42.)
114 self.assertAlmostEqual(float(Foo2()), 42.)
115 self.assertAlmostEqual(float(Foo3(21)), 42.)
116 self.assertRaises(TypeError, float, Foo4(42))
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000117 self.assertAlmostEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000118
119 def test_floatasratio(self):
120 for f, ratio in [
121 (0.875, (7, 8)),
122 (-0.875, (-7, 8)),
123 (0.0, (0, 1)),
124 (11.5, (23, 2)),
125 ]:
126 self.assertEqual(f.as_integer_ratio(), ratio)
127
128 for i in range(10000):
129 f = random.random()
130 f *= 10 ** random.randint(-100, 100)
131 n, d = f.as_integer_ratio()
132 self.assertEqual(float(n).__truediv__(d), f)
133
134 R = fractions.Fraction
135 self.assertEqual(R(0, 1),
136 R(*float(0.0).as_integer_ratio()))
137 self.assertEqual(R(5, 2),
138 R(*float(2.5).as_integer_ratio()))
139 self.assertEqual(R(1, 2),
140 R(*float(0.5).as_integer_ratio()))
141 self.assertEqual(R(4728779608739021, 2251799813685248),
142 R(*float(2.1).as_integer_ratio()))
143 self.assertEqual(R(-4728779608739021, 2251799813685248),
144 R(*float(-2.1).as_integer_ratio()))
145 self.assertEqual(R(-2100, 1),
146 R(*float(-2100.0).as_integer_ratio()))
147
148 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
149 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
150 self.assertRaises(ValueError, float('nan').as_integer_ratio)
151
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000152 def test_float_containment(self):
153 floats = (INF, -INF, 0.0, 1.0, NAN)
154 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000155 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000156 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000157 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000158 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000159 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000160 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000161
162 for f in floats:
163 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000164 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
165 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
166 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
167 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000168 "{%r : None}" % (f, f))
169
170 # identical containers
171 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000172 self.assertTrue(l == l, "[%r] not equal to itself" % f)
173 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
174 self.assertTrue(s == s, "{%r} not equal to itself" % f)
175 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000176
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000177 def assertEqualAndEqualSign(self, a, b):
178 # fail unless a == b and a and b have the same sign bit;
179 # the only difference from assertEqual is that this test
180 # distingishes -0.0 and 0.0.
181 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
182
183 @requires_IEEE_754
184 def test_float_pow(self):
185 # test builtin pow and ** operator for IEEE 754 special cases.
186 # Special cases taken from section F.9.4.4 of the C99 specification
187
188 for pow_op in pow, operator.pow:
189 # x**NAN is NAN for any x except 1
190 self.assertTrue(isnan(pow_op(-INF, NAN)))
191 self.assertTrue(isnan(pow_op(-2.0, NAN)))
192 self.assertTrue(isnan(pow_op(-1.0, NAN)))
193 self.assertTrue(isnan(pow_op(-0.5, NAN)))
194 self.assertTrue(isnan(pow_op(-0.0, NAN)))
195 self.assertTrue(isnan(pow_op(0.0, NAN)))
196 self.assertTrue(isnan(pow_op(0.5, NAN)))
197 self.assertTrue(isnan(pow_op(2.0, NAN)))
198 self.assertTrue(isnan(pow_op(INF, NAN)))
199 self.assertTrue(isnan(pow_op(NAN, NAN)))
200
201 # NAN**y is NAN for any y except +-0
202 self.assertTrue(isnan(pow_op(NAN, -INF)))
203 self.assertTrue(isnan(pow_op(NAN, -2.0)))
204 self.assertTrue(isnan(pow_op(NAN, -1.0)))
205 self.assertTrue(isnan(pow_op(NAN, -0.5)))
206 self.assertTrue(isnan(pow_op(NAN, 0.5)))
207 self.assertTrue(isnan(pow_op(NAN, 1.0)))
208 self.assertTrue(isnan(pow_op(NAN, 2.0)))
209 self.assertTrue(isnan(pow_op(NAN, INF)))
210
211 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
212 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
213 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
214
215 # (+-0)**y raises ZeroDivisionError for y finite and negative
216 # but not an odd integer
217 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
218 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
219 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
220 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
221
222 # (+-0)**y is +-0 for y a positive odd integer
223 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
224 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
225
226 # (+-0)**y is 0 for y finite and positive but not an odd integer
227 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
228 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
229 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
230 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
231
232 # (-1)**+-inf is 1
233 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
234 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
235
236 # 1**y is 1 for any y, even if y is an infinity or nan
237 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
238 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
239 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
240 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
241 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
242 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
243 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
244 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
245 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
246 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
247 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
248
249 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
250 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
251 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
252 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
253 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
254 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
255 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
256 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
257 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
258 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
259 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
260 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
261 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
262 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
263 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
264 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
265 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
266 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
267 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
268 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
269 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
270 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
271 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
272
273 # x**y defers to complex pow for finite negative x and
274 # non-integral y.
275 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
276 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
277 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
278 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
279 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
280 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
281
282 # x**-INF is INF for abs(x) < 1
283 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
284 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
285 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
286 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
287
288 # x**-INF is 0 for abs(x) > 1
289 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
290 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
291 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
292 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
293
294 # x**INF is 0 for abs(x) < 1
295 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
296 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
297 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
298 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
299
300 # x**INF is INF for abs(x) > 1
301 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
302 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
303 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
304 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
305
306 # (-INF)**y is -0.0 for y a negative odd integer
307 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
308
309 # (-INF)**y is 0.0 for y negative but not an odd integer
310 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
311 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
312
313 # (-INF)**y is -INF for y a positive odd integer
314 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
315
316 # (-INF)**y is INF for y positive but not an odd integer
317 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
318 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
319
320 # INF**y is INF for y positive
321 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
322 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
323 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
324
325 # INF**y is 0.0 for y negative
326 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
327 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
328 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
329
330 # basic checks not covered by the special cases above
331 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
332 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
333 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
334 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
335 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
336 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
337 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
338 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
339 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
340 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
341 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
342 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
343 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
344 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
345 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
346 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
347 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
348 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
349
350 # 1 ** large and -1 ** large; some libms apparently
351 # have problems with these
352 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
353 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
354 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
355 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
356
357 # check sign for results that underflow to 0
358 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
359 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
360 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
361 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
362 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
363 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
364 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
365 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
366 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
367 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
368 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
369 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
370
371 # check we don't raise an exception for subnormal results,
372 # and validate signs. Tests currently disabled, since
373 # they fail on systems where a subnormal result from pow
374 # is flushed to zero (e.g. Debian/ia64.)
375 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
376 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
377 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
378 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
379 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
380 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
381 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
382 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000383
384
Benjamin Petersone401c682010-07-02 23:25:44 +0000385@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000386class FormatFunctionsTestCase(unittest.TestCase):
387
388 def setUp(self):
389 self.save_formats = {'double':float.__getformat__('double'),
390 'float':float.__getformat__('float')}
391
392 def tearDown(self):
393 float.__setformat__('double', self.save_formats['double'])
394 float.__setformat__('float', self.save_formats['float'])
395
396 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000397 self.assertIn(float.__getformat__('double'),
398 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
399 self.assertIn(float.__getformat__('float'),
400 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000401 self.assertRaises(ValueError, float.__getformat__, 'chicken')
402 self.assertRaises(TypeError, float.__getformat__, 1)
403
404 def test_setformat(self):
405 for t in 'double', 'float':
406 float.__setformat__(t, 'unknown')
407 if self.save_formats[t] == 'IEEE, big-endian':
408 self.assertRaises(ValueError, float.__setformat__,
409 t, 'IEEE, little-endian')
410 elif self.save_formats[t] == 'IEEE, little-endian':
411 self.assertRaises(ValueError, float.__setformat__,
412 t, 'IEEE, big-endian')
413 else:
414 self.assertRaises(ValueError, float.__setformat__,
415 t, 'IEEE, big-endian')
416 self.assertRaises(ValueError, float.__setformat__,
417 t, 'IEEE, little-endian')
418 self.assertRaises(ValueError, float.__setformat__,
419 t, 'chicken')
420 self.assertRaises(ValueError, float.__setformat__,
421 'chicken', 'unknown')
422
Guido van Rossum2be161d2007-05-15 20:43:51 +0000423BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000424LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000425BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000426LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000427
Guido van Rossum2be161d2007-05-15 20:43:51 +0000428BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000429LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000430BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000431LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000432
433# on non-IEEE platforms, attempting to unpack a bit pattern
434# representing an infinity or a NaN should raise an exception.
435
Benjamin Petersone401c682010-07-02 23:25:44 +0000436@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000437class UnknownFormatTestCase(unittest.TestCase):
438 def setUp(self):
439 self.save_formats = {'double':float.__getformat__('double'),
440 'float':float.__getformat__('float')}
441 float.__setformat__('double', 'unknown')
442 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000443
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000444 def tearDown(self):
445 float.__setformat__('double', self.save_formats['double'])
446 float.__setformat__('float', self.save_formats['float'])
447
448 def test_double_specials_dont_unpack(self):
449 for fmt, data in [('>d', BE_DOUBLE_INF),
450 ('>d', BE_DOUBLE_NAN),
451 ('<d', LE_DOUBLE_INF),
452 ('<d', LE_DOUBLE_NAN)]:
453 self.assertRaises(ValueError, struct.unpack, fmt, data)
454
455 def test_float_specials_dont_unpack(self):
456 for fmt, data in [('>f', BE_FLOAT_INF),
457 ('>f', BE_FLOAT_NAN),
458 ('<f', LE_FLOAT_INF),
459 ('<f', LE_FLOAT_NAN)]:
460 self.assertRaises(ValueError, struct.unpack, fmt, data)
461
462
463# on an IEEE platform, all we guarantee is that bit patterns
464# representing infinities or NaNs do not raise an exception; all else
465# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000466# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000467
468class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000469
Benjamin Petersone401c682010-07-02 23:25:44 +0000470 @requires_IEEE_754
471 def test_double_specials_do_unpack(self):
472 for fmt, data in [('>d', BE_DOUBLE_INF),
473 ('>d', BE_DOUBLE_NAN),
474 ('<d', LE_DOUBLE_INF),
475 ('<d', LE_DOUBLE_NAN)]:
476 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000477
Benjamin Petersone401c682010-07-02 23:25:44 +0000478 @requires_IEEE_754
479 def test_float_specials_do_unpack(self):
480 for fmt, data in [('>f', BE_FLOAT_INF),
481 ('>f', BE_FLOAT_NAN),
482 ('<f', LE_FLOAT_INF),
483 ('<f', LE_FLOAT_NAN)]:
484 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000485
Eric Smith8c663262007-08-25 02:26:07 +0000486class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000487
Eric Smith11fe3e02007-08-31 01:33:06 +0000488 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000489 # these should be rewritten to use both format(x, spec) and
490 # x.__format__(spec)
491
492 self.assertEqual(format(0.0, 'f'), '0.000000')
493
494 # the default is 'g', except for empty format spec
495 self.assertEqual(format(0.0, ''), '0.0')
496 self.assertEqual(format(0.01, ''), '0.01')
497 self.assertEqual(format(0.01, 'g'), '0.01')
498
Eric Smith63376222009-05-05 14:04:18 +0000499 # empty presentation type should format in the same way as str
500 # (issue 5920)
501 x = 100/7.
502 self.assertEqual(format(x, ''), str(x))
503 self.assertEqual(format(x, '-'), str(x))
504 self.assertEqual(format(x, '>'), str(x))
505 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000506
507 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000508
509 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000510
511 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
512 self.assertEqual(format(-1.0, ' f'), '-1.000000')
513 self.assertEqual(format( 1.0, '+f'), '+1.000000')
514 self.assertEqual(format(-1.0, '+f'), '-1.000000')
515
516 # % formatting
517 self.assertEqual(format(-1.0, '%'), '-100.000000%')
518
519 # conversion to string should fail
520 self.assertRaises(ValueError, format, 3.0, "s")
521
Eric Smith7b69c6c2008-01-27 21:07:59 +0000522 # other format specifiers shouldn't work on floats,
523 # in particular int specifiers
524 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
525 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
526 if not format_spec in 'eEfFgGn%':
527 self.assertRaises(ValueError, format, 0.0, format_spec)
528 self.assertRaises(ValueError, format, 1.0, format_spec)
529 self.assertRaises(ValueError, format, -1.0, format_spec)
530 self.assertRaises(ValueError, format, 1e100, format_spec)
531 self.assertRaises(ValueError, format, -1e100, format_spec)
532 self.assertRaises(ValueError, format, 1e-100, format_spec)
533 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000534
Eric Smith741191f2009-05-06 13:08:15 +0000535 # issue 3382
536 self.assertEqual(format(NAN, 'f'), 'nan')
537 self.assertEqual(format(NAN, 'F'), 'NAN')
538 self.assertEqual(format(INF, 'f'), 'inf')
539 self.assertEqual(format(INF, 'F'), 'INF')
540
Benjamin Petersone401c682010-07-02 23:25:44 +0000541 @requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000542 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000543 with open(format_testfile) as testfile:
544 for line in testfile:
545 if line.startswith('--'):
546 continue
547 line = line.strip()
548 if not line:
549 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000550
Brian Curtin076623b2010-10-31 00:03:45 +0000551 lhs, rhs = map(str.strip, line.split('->'))
552 fmt, arg = lhs.split()
553 self.assertEqual(fmt % float(arg), rhs)
554 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000555
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000556 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000557 self.assertEqual(format(123.456, '.4'), '123.5')
558 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
559 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000560
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000561class ReprTestCase(unittest.TestCase):
562 def test_repr(self):
563 floats_file = open(os.path.join(os.path.split(__file__)[0],
564 'floating_points.txt'))
565 for line in floats_file:
566 line = line.strip()
567 if not line or line.startswith('#'):
568 continue
569 v = eval(line)
570 self.assertEqual(v, eval(repr(v)))
571 floats_file.close()
572
Eric Smith0923d1d2009-04-16 20:16:10 +0000573 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
574 "applies only when using short float repr style")
575 def test_short_repr(self):
576 # test short float repr introduced in Python 3.1. One aspect
577 # of this repr is that we get some degree of str -> float ->
578 # str roundtripping. In particular, for any numeric string
579 # containing 15 or fewer significant digits, those exact same
580 # digits (modulo trailing zeros) should appear in the output.
581 # No more repr(0.03) -> "0.029999999999999999"!
582
583 test_strings = [
584 # output always includes *either* a decimal point and at
585 # least one digit after that point, or an exponent.
586 '0.0',
587 '1.0',
588 '0.01',
589 '0.02',
590 '0.03',
591 '0.04',
592 '0.05',
593 '1.23456789',
594 '10.0',
595 '100.0',
596 # values >= 1e16 get an exponent...
597 '1000000000000000.0',
598 '9999999999999990.0',
599 '1e+16',
600 '1e+17',
601 # ... and so do values < 1e-4
602 '0.001',
603 '0.001001',
604 '0.00010000000000001',
605 '0.0001',
606 '9.999999999999e-05',
607 '1e-05',
608 # values designed to provoke failure if the FPU rounding
609 # precision isn't set correctly
610 '8.72293771110361e+25',
611 '7.47005307342313e+26',
612 '2.86438000439698e+28',
613 '8.89142905246179e+28',
614 '3.08578087079232e+35',
615 ]
616
617 for s in test_strings:
618 negs = '-'+s
619 self.assertEqual(s, repr(float(s)))
620 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000621 # Since Python 3.2, repr and str are identical
622 self.assertEqual(repr(float(s)), str(float(s)))
623 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000624
625@requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000626class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000627
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000628 def test_inf_nan(self):
629 self.assertRaises(OverflowError, round, INF)
630 self.assertRaises(OverflowError, round, -INF)
631 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000632 self.assertRaises(TypeError, round, INF, 0.0)
633 self.assertRaises(TypeError, round, -INF, 1.0)
634 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
635 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000636
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000637 def test_large_n(self):
638 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
639 self.assertEqual(round(123.456, n), 123.456)
640 self.assertEqual(round(-123.456, n), -123.456)
641 self.assertEqual(round(1e300, n), 1e300)
642 self.assertEqual(round(1e-320, n), 1e-320)
643 self.assertEqual(round(1e150, 300), 1e150)
644 self.assertEqual(round(1e300, 307), 1e300)
645 self.assertEqual(round(-3.1415, 308), -3.1415)
646 self.assertEqual(round(1e150, 309), 1e150)
647 self.assertEqual(round(1.4e-315, 315), 1e-315)
648
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000649 def test_small_n(self):
650 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
651 self.assertEqual(round(123.456, n), 0.0)
652 self.assertEqual(round(-123.456, n), -0.0)
653 self.assertEqual(round(1e300, n), 0.0)
654 self.assertEqual(round(1e-320, n), 0.0)
655
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000656 def test_overflow(self):
657 self.assertRaises(OverflowError, round, 1.6e308, -308)
658 self.assertRaises(OverflowError, round, -1.7e308, -308)
659
660 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
661 "applies only when using short float repr style")
662 def test_previous_round_bugs(self):
663 # particular cases that have occurred in bug reports
664 self.assertEqual(round(562949953421312.5, 1),
665 562949953421312.5)
666 self.assertEqual(round(56294995342131.5, 3),
667 56294995342131.5)
668 # round-half-even
669 self.assertEqual(round(25.0, -1), 20.0)
670 self.assertEqual(round(35.0, -1), 40.0)
671 self.assertEqual(round(45.0, -1), 40.0)
672 self.assertEqual(round(55.0, -1), 60.0)
673 self.assertEqual(round(65.0, -1), 60.0)
674 self.assertEqual(round(75.0, -1), 80.0)
675 self.assertEqual(round(85.0, -1), 80.0)
676 self.assertEqual(round(95.0, -1), 100.0)
677
678 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
679 "applies only when using short float repr style")
680 def test_matches_float_format(self):
681 # round should give the same results as float formatting
682 for i in range(500):
683 x = i/1000.
684 self.assertEqual(float(format(x, '.0f')), round(x, 0))
685 self.assertEqual(float(format(x, '.1f')), round(x, 1))
686 self.assertEqual(float(format(x, '.2f')), round(x, 2))
687 self.assertEqual(float(format(x, '.3f')), round(x, 3))
688
689 for i in range(5, 5000, 10):
690 x = i/1000.
691 self.assertEqual(float(format(x, '.0f')), round(x, 0))
692 self.assertEqual(float(format(x, '.1f')), round(x, 1))
693 self.assertEqual(float(format(x, '.2f')), round(x, 2))
694 self.assertEqual(float(format(x, '.3f')), round(x, 3))
695
696 for i in range(500):
697 x = random.random()
698 self.assertEqual(float(format(x, '.0f')), round(x, 0))
699 self.assertEqual(float(format(x, '.1f')), round(x, 1))
700 self.assertEqual(float(format(x, '.2f')), round(x, 2))
701 self.assertEqual(float(format(x, '.3f')), round(x, 3))
702
Eric Smith8a10ecc2009-12-02 17:58:24 +0000703 def test_format_specials(self):
704 # Test formatting of nans and infs.
705
706 def test(fmt, value, expected):
707 # Test with both % and format().
708 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000709 fmt = fmt[1:] # strip off the %
710 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000711
712 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
713 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
714 pfmt = '%+' + fmt[1:]
715 sfmt = '% ' + fmt[1:]
716 test(fmt, INF, 'inf')
717 test(fmt, -INF, '-inf')
718 test(fmt, NAN, 'nan')
719 test(fmt, -NAN, 'nan')
720 # When asking for a sign, it's always provided. nans are
721 # always positive.
722 test(pfmt, INF, '+inf')
723 test(pfmt, -INF, '-inf')
724 test(pfmt, NAN, '+nan')
725 test(pfmt, -NAN, '+nan')
726 # When using ' ' for a sign code, only infs can be negative.
727 # Others have a space.
728 test(sfmt, INF, ' inf')
729 test(sfmt, -INF, '-inf')
730 test(sfmt, NAN, ' nan')
731 test(sfmt, -NAN, ' nan')
732
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000733
Christian Heimes99170a52007-12-19 02:07:34 +0000734# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000735# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000736class InfNanTest(unittest.TestCase):
737 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000738 self.assertTrue(isinf(float("inf")))
739 self.assertTrue(isinf(float("+inf")))
740 self.assertTrue(isinf(float("-inf")))
741 self.assertTrue(isinf(float("infinity")))
742 self.assertTrue(isinf(float("+infinity")))
743 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000744
745 self.assertEqual(repr(float("inf")), "inf")
746 self.assertEqual(repr(float("+inf")), "inf")
747 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000748 self.assertEqual(repr(float("infinity")), "inf")
749 self.assertEqual(repr(float("+infinity")), "inf")
750 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000751
752 self.assertEqual(repr(float("INF")), "inf")
753 self.assertEqual(repr(float("+Inf")), "inf")
754 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000755 self.assertEqual(repr(float("Infinity")), "inf")
756 self.assertEqual(repr(float("+iNfInItY")), "inf")
757 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000758
759 self.assertEqual(str(float("inf")), "inf")
760 self.assertEqual(str(float("+inf")), "inf")
761 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000762 self.assertEqual(str(float("infinity")), "inf")
763 self.assertEqual(str(float("+infinity")), "inf")
764 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000765
766 self.assertRaises(ValueError, float, "info")
767 self.assertRaises(ValueError, float, "+info")
768 self.assertRaises(ValueError, float, "-info")
769 self.assertRaises(ValueError, float, "in")
770 self.assertRaises(ValueError, float, "+in")
771 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000772 self.assertRaises(ValueError, float, "infinit")
773 self.assertRaises(ValueError, float, "+Infin")
774 self.assertRaises(ValueError, float, "-INFI")
775 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000776
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000777 self.assertRaises(ValueError, float, "++Inf")
778 self.assertRaises(ValueError, float, "-+inf")
779 self.assertRaises(ValueError, float, "+-infinity")
780 self.assertRaises(ValueError, float, "--Infinity")
781
Christian Heimes99170a52007-12-19 02:07:34 +0000782 def test_inf_as_str(self):
783 self.assertEqual(repr(1e300 * 1e300), "inf")
784 self.assertEqual(repr(-1e300 * 1e300), "-inf")
785
786 self.assertEqual(str(1e300 * 1e300), "inf")
787 self.assertEqual(str(-1e300 * 1e300), "-inf")
788
789 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000790 self.assertTrue(isnan(float("nan")))
791 self.assertTrue(isnan(float("+nan")))
792 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000793
794 self.assertEqual(repr(float("nan")), "nan")
795 self.assertEqual(repr(float("+nan")), "nan")
796 self.assertEqual(repr(float("-nan")), "nan")
797
798 self.assertEqual(repr(float("NAN")), "nan")
799 self.assertEqual(repr(float("+NAn")), "nan")
800 self.assertEqual(repr(float("-NaN")), "nan")
801
802 self.assertEqual(str(float("nan")), "nan")
803 self.assertEqual(str(float("+nan")), "nan")
804 self.assertEqual(str(float("-nan")), "nan")
805
806 self.assertRaises(ValueError, float, "nana")
807 self.assertRaises(ValueError, float, "+nana")
808 self.assertRaises(ValueError, float, "-nana")
809 self.assertRaises(ValueError, float, "na")
810 self.assertRaises(ValueError, float, "+na")
811 self.assertRaises(ValueError, float, "-na")
812
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000813 self.assertRaises(ValueError, float, "++nan")
814 self.assertRaises(ValueError, float, "-+NAN")
815 self.assertRaises(ValueError, float, "+-NaN")
816 self.assertRaises(ValueError, float, "--nAn")
817
Christian Heimes99170a52007-12-19 02:07:34 +0000818 def test_nan_as_str(self):
819 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
820 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
821
822 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
823 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000824
Christian Heimes53876d92008-04-19 00:31:39 +0000825 def notest_float_nan(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000826 self.assertTrue(NAN.is_nan())
827 self.assertFalse(INF.is_nan())
828 self.assertFalse((0.).is_nan())
Christian Heimes53876d92008-04-19 00:31:39 +0000829
830 def notest_float_inf(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000831 self.assertTrue(INF.is_inf())
832 self.assertFalse(NAN.is_inf())
833 self.assertFalse((0.).is_inf())
Christian Heimes53876d92008-04-19 00:31:39 +0000834
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000835fromHex = float.fromhex
836toHex = float.hex
837class HexFloatTestCase(unittest.TestCase):
838 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
839 MIN = fromHex('0x1p-1022') # min normal
840 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
841 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
842
843 def identical(self, x, y):
844 # check that floats x and y are identical, or that both
845 # are NaNs
846 if isnan(x) or isnan(y):
847 if isnan(x) == isnan(y):
848 return
849 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
850 return
851 self.fail('%r not identical to %r' % (x, y))
852
853 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +0000854 self.identical(self.MIN, ldexp(1.0, -1022))
855 self.identical(self.TINY, ldexp(1.0, -1074))
856 self.identical(self.EPS, ldexp(1.0, -52))
857 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000858
859 def test_invalid_inputs(self):
860 invalid_inputs = [
861 'infi', # misspelt infinities and nans
862 '-Infinit',
863 '++inf',
864 '-+Inf',
865 '--nan',
866 '+-NaN',
867 'snan',
868 'NaNs',
869 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000870 'an',
871 'nf',
872 'nfinity',
873 'inity',
874 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000875 '0xnan',
876 '',
877 ' ',
878 'x1.0p0',
879 '0xX1.0p0',
880 '+ 0x1.0p0', # internal whitespace
881 '- 0x1.0p0',
882 '0 x1.0p0',
883 '0x 1.0p0',
884 '0x1 2.0p0',
885 '+0x1 .0p0',
886 '0x1. 0p0',
887 '-0x1.0 1p0',
888 '-0x1.0 p0',
889 '+0x1.0p +0',
890 '0x1.0p -0',
891 '0x1.0p 0',
892 '+0x1.0p+ 0',
893 '-0x1.0p- 0',
894 '++0x1.0p-0', # double signs
895 '--0x1.0p0',
896 '+-0x1.0p+0',
897 '-+0x1.0p0',
898 '0x1.0p++0',
899 '+0x1.0p+-0',
900 '-0x1.0p-+0',
901 '0x1.0p--0',
902 '0x1.0.p0',
903 '0x.p0', # no hex digits before or after point
904 '0x1,p0', # wrong decimal point character
905 '0x1pa',
906 '0x1p\uff10', # fullwidth Unicode digits
907 '\uff10x1p0',
908 '0x\uff11p0',
909 '0x1.\uff10p0',
910 '0x1p0 \n 0x2p0',
911 '0x1p0\0 0x1p0', # embedded null byte is not end of string
912 ]
913 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +0000914 try:
915 result = fromHex(x)
916 except ValueError:
917 pass
918 else:
919 self.fail('Expected float.fromhex(%r) to raise ValueError; '
920 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000921
922
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000923 def test_whitespace(self):
924 value_pairs = [
925 ('inf', INF),
926 ('-Infinity', -INF),
927 ('nan', NAN),
928 ('1.0', 1.0),
929 ('-0x.2', -0.125),
930 ('-0.0', -0.0)
931 ]
932 whitespace = [
933 '',
934 ' ',
935 '\t',
936 '\n',
937 '\n \t',
938 '\f',
939 '\v',
940 '\r'
941 ]
942 for inp, expected in value_pairs:
943 for lead in whitespace:
944 for trail in whitespace:
945 got = fromHex(lead + inp + trail)
946 self.identical(got, expected)
947
948
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000949 def test_from_hex(self):
950 MIN = self.MIN;
951 MAX = self.MAX;
952 TINY = self.TINY;
953 EPS = self.EPS;
954
955 # two spellings of infinity, with optional signs; case-insensitive
956 self.identical(fromHex('inf'), INF)
957 self.identical(fromHex('+Inf'), INF)
958 self.identical(fromHex('-INF'), -INF)
959 self.identical(fromHex('iNf'), INF)
960 self.identical(fromHex('Infinity'), INF)
961 self.identical(fromHex('+INFINITY'), INF)
962 self.identical(fromHex('-infinity'), -INF)
963 self.identical(fromHex('-iNFiNitY'), -INF)
964
965 # nans with optional sign; case insensitive
966 self.identical(fromHex('nan'), NAN)
967 self.identical(fromHex('+NaN'), NAN)
968 self.identical(fromHex('-NaN'), NAN)
969 self.identical(fromHex('-nAN'), NAN)
970
971 # variations in input format
972 self.identical(fromHex('1'), 1.0)
973 self.identical(fromHex('+1'), 1.0)
974 self.identical(fromHex('1.'), 1.0)
975 self.identical(fromHex('1.0'), 1.0)
976 self.identical(fromHex('1.0p0'), 1.0)
977 self.identical(fromHex('01'), 1.0)
978 self.identical(fromHex('01.'), 1.0)
979 self.identical(fromHex('0x1'), 1.0)
980 self.identical(fromHex('0x1.'), 1.0)
981 self.identical(fromHex('0x1.0'), 1.0)
982 self.identical(fromHex('+0x1.0'), 1.0)
983 self.identical(fromHex('0x1p0'), 1.0)
984 self.identical(fromHex('0X1p0'), 1.0)
985 self.identical(fromHex('0X1P0'), 1.0)
986 self.identical(fromHex('0x1P0'), 1.0)
987 self.identical(fromHex('0x1.p0'), 1.0)
988 self.identical(fromHex('0x1.0p0'), 1.0)
989 self.identical(fromHex('0x.1p4'), 1.0)
990 self.identical(fromHex('0x.1p04'), 1.0)
991 self.identical(fromHex('0x.1p004'), 1.0)
992 self.identical(fromHex('0x1p+0'), 1.0)
993 self.identical(fromHex('0x1P-0'), 1.0)
994 self.identical(fromHex('+0x1p0'), 1.0)
995 self.identical(fromHex('0x01p0'), 1.0)
996 self.identical(fromHex('0x1p00'), 1.0)
997 self.identical(fromHex(' 0x1p0 '), 1.0)
998 self.identical(fromHex('\n 0x1p0'), 1.0)
999 self.identical(fromHex('0x1p0 \t'), 1.0)
1000 self.identical(fromHex('0xap0'), 10.0)
1001 self.identical(fromHex('0xAp0'), 10.0)
1002 self.identical(fromHex('0xaP0'), 10.0)
1003 self.identical(fromHex('0xAP0'), 10.0)
1004 self.identical(fromHex('0xbep0'), 190.0)
1005 self.identical(fromHex('0xBep0'), 190.0)
1006 self.identical(fromHex('0xbEp0'), 190.0)
1007 self.identical(fromHex('0XBE0P-4'), 190.0)
1008 self.identical(fromHex('0xBEp0'), 190.0)
1009 self.identical(fromHex('0xB.Ep4'), 190.0)
1010 self.identical(fromHex('0x.BEp8'), 190.0)
1011 self.identical(fromHex('0x.0BEp12'), 190.0)
1012
1013 # moving the point around
1014 pi = fromHex('0x1.921fb54442d18p1')
1015 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1016 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1017 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1018 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1019 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1020 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1021 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1022 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1023 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1024 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1025 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1026 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1027 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1028 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1029 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1030 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1031 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1032 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1033 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1034 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1035 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1036 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1037 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1038 # ...
1039 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1040 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1041 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1042 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1043 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1044 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1045 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1046 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1047 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1048
1049
1050 # results that should overflow...
1051 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1052 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1053 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1054 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1055 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1056 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1057 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1058 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1059 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1060 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1061 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1062 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1063 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1064 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1065 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1066 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1067 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1068 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1069 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1070
1071 # ...and those that round to +-max float
1072 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1073 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1074 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1075
1076 # zeros
1077 self.identical(fromHex('0x0p0'), 0.0)
1078 self.identical(fromHex('0x0p1000'), 0.0)
1079 self.identical(fromHex('-0x0p1023'), -0.0)
1080 self.identical(fromHex('0X0p1024'), 0.0)
1081 self.identical(fromHex('-0x0p1025'), -0.0)
1082 self.identical(fromHex('0X0p2000'), 0.0)
1083 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1084 self.identical(fromHex('-0X0p-0'), -0.0)
1085 self.identical(fromHex('-0X0p-1000'), -0.0)
1086 self.identical(fromHex('0x0p-1023'), 0.0)
1087 self.identical(fromHex('-0X0p-1024'), -0.0)
1088 self.identical(fromHex('-0x0p-1025'), -0.0)
1089 self.identical(fromHex('-0x0p-1072'), -0.0)
1090 self.identical(fromHex('0X0p-1073'), 0.0)
1091 self.identical(fromHex('-0x0p-1074'), -0.0)
1092 self.identical(fromHex('0x0p-1075'), 0.0)
1093 self.identical(fromHex('0X0p-1076'), 0.0)
1094 self.identical(fromHex('-0X0p-2000'), -0.0)
1095 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1096
1097 # values that should underflow to 0
1098 self.identical(fromHex('0X1p-1075'), 0.0)
1099 self.identical(fromHex('-0X1p-1075'), -0.0)
1100 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1101 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1102 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1103 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1104
1105 # check round-half-even is working correctly near 0 ...
1106 self.identical(fromHex('0x1p-1076'), 0.0)
1107 self.identical(fromHex('0X2p-1076'), 0.0)
1108 self.identical(fromHex('0X3p-1076'), TINY)
1109 self.identical(fromHex('0x4p-1076'), TINY)
1110 self.identical(fromHex('0X5p-1076'), TINY)
1111 self.identical(fromHex('0X6p-1076'), 2*TINY)
1112 self.identical(fromHex('0x7p-1076'), 2*TINY)
1113 self.identical(fromHex('0X8p-1076'), 2*TINY)
1114 self.identical(fromHex('0X9p-1076'), 2*TINY)
1115 self.identical(fromHex('0xap-1076'), 2*TINY)
1116 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1117 self.identical(fromHex('0xcp-1076'), 3*TINY)
1118 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1119 self.identical(fromHex('0Xep-1076'), 4*TINY)
1120 self.identical(fromHex('0xfp-1076'), 4*TINY)
1121 self.identical(fromHex('0x10p-1076'), 4*TINY)
1122 self.identical(fromHex('-0x1p-1076'), -0.0)
1123 self.identical(fromHex('-0X2p-1076'), -0.0)
1124 self.identical(fromHex('-0x3p-1076'), -TINY)
1125 self.identical(fromHex('-0X4p-1076'), -TINY)
1126 self.identical(fromHex('-0x5p-1076'), -TINY)
1127 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1128 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1129 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1130 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1131 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1132 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1133 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1134 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1135 self.identical(fromHex('-0xep-1076'), -4*TINY)
1136 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1137 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1138
1139 # ... and near MIN ...
1140 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1141 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1142 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1143 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1144 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1145 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1146 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1147 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1148 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1149 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1150 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1151 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1152 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1153 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1154 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1155 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1156 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1157 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1158 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1159 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1160 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1161 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1162 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1163 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1164 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1165 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1166 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1167 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1168 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1169 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1170 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1171 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1172 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1173 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1174
1175 # ... and near 1.0.
1176 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1177 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1178 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1179 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1180 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1181 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1182 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1183 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1184 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1185 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1186 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1187 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1188 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1189 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1190 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1191 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1192 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1193 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1194 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1195 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1196 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1197 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1198 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1199 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1200 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1201 1.0)
1202 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1203 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1204 1+EPS)
1205 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1206 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1207 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1208 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1209 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1210 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1211 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1212 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1213 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1214 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1215 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1216 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1217 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1218 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1219 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1220 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1221 1.0+EPS)
1222 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1223 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1224 1.0+2*EPS)
1225 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1226 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1227 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1228 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1229 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1230 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1231 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1232 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1233
1234 def test_roundtrip(self):
1235 def roundtrip(x):
1236 return fromHex(toHex(x))
1237
1238 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1239 self.identical(x, roundtrip(x))
1240 self.identical(-x, roundtrip(-x))
1241
1242 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1243 import random
1244 for i in range(10000):
1245 e = random.randrange(-1200, 1200)
1246 m = random.random()
1247 s = random.choice([1.0, -1.0])
1248 try:
1249 x = s*ldexp(m, e)
1250 except OverflowError:
1251 pass
1252 else:
1253 self.identical(x, fromHex(toHex(x)))
1254
Christian Heimes53876d92008-04-19 00:31:39 +00001255
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001256def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001257 support.run_unittest(
Amaury Forgeot d'Arc7e958d12008-09-06 21:03:22 +00001258 GeneralFloatCases,
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001259 FormatFunctionsTestCase,
1260 UnknownFormatTestCase,
Eric Smith8c663262007-08-25 02:26:07 +00001261 IEEEFormatTestCase,
Christian Heimes827b35c2007-12-10 22:19:17 +00001262 FormatTestCase,
Christian Heimes99170a52007-12-19 02:07:34 +00001263 ReprTestCase,
Mark Dickinsone6a076d2009-04-18 11:48:33 +00001264 RoundTestCase,
Christian Heimes99170a52007-12-19 02:07:34 +00001265 InfNanTest,
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001266 HexFloatTestCase,
Christian Heimesb76922a2007-12-11 01:06:40 +00001267 )
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001268
1269if __name__ == '__main__':
1270 test_main()