blob: ddf93bcf8773f269923951d61e27e9139b029479 [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, "-.")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000046 self.assertRaises(ValueError, float, b"-")
47 self.assertRaises(TypeError, float, {})
48 # Lone surrogate
49 self.assertRaises(UnicodeEncodeError, float, '\uD8F0')
Mark Dickinsonc2d86892010-02-12 21:18:34 +000050 # check that we don't accept alternate exponent markers
51 self.assertRaises(ValueError, float, "-1.7d29")
52 self.assertRaises(ValueError, float, "3D-14")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000053 self.assertEqual(float(" \u0663.\u0661\u0664 "), 3.14)
54 self.assertEqual(float("\N{EM SPACE}3.14\N{EN SPACE}"), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000055 # extra long strings should not be a problem
56 float(b'.' + b'1'*1000)
57 float('.' + '1'*1000)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000058
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000059 def test_error_message(self):
60 testlist = ('\xbd', '123\xbd', ' 123 456 ')
61 for s in testlist:
62 try:
63 float(s)
64 except ValueError as e:
65 self.assertIn(s.strip(), e.args[0])
66 else:
67 self.fail("Expected int(%r) to raise a ValueError", s)
68
69
Benjamin Petersonee8712c2008-05-20 21:35:26 +000070 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +000071 def test_float_with_comma(self):
72 # set locale to something that doesn't use '.' for the decimal point
73 # float must not accept the locale specific decimal point but
74 # it still has to accept the normal python syntac
75 import locale
76 if not locale.localeconv()['decimal_point'] == ',':
77 return
78
79 self.assertEqual(float(" 3.14 "), 3.14)
80 self.assertEqual(float("+3.14 "), 3.14)
81 self.assertEqual(float("-3.14 "), -3.14)
82 self.assertEqual(float(".14 "), .14)
83 self.assertEqual(float("3. "), 3.0)
84 self.assertEqual(float("3.e3 "), 3000.0)
85 self.assertEqual(float("3.2e3 "), 3200.0)
86 self.assertEqual(float("2.5e-1 "), 0.25)
87 self.assertEqual(float("5e-1"), 0.5)
88 self.assertRaises(ValueError, float, " 3,14 ")
89 self.assertRaises(ValueError, float, " +3,14 ")
90 self.assertRaises(ValueError, float, " -3,14 ")
91 self.assertRaises(ValueError, float, " 0x3.1 ")
92 self.assertRaises(ValueError, float, " -0x3.p-1 ")
93 self.assertRaises(ValueError, float, " +0x3.p-1 ")
94 self.assertEqual(float(" 25.e-1 "), 2.5)
Benjamin Petersond43029b2008-09-06 23:33:21 +000095 self.assertEqual(support.fcmp(float(" .25e-1 "), .025), 0)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000096
97 def test_floatconversion(self):
98 # Make sure that calls to __float__() work properly
99 class Foo0:
100 def __float__(self):
101 return 42.
102
103 class Foo1(object):
104 def __float__(self):
105 return 42.
106
107 class Foo2(float):
108 def __float__(self):
109 return 42.
110
111 class Foo3(float):
112 def __new__(cls, value=0.):
113 return float.__new__(cls, 2*value)
114
115 def __float__(self):
116 return self
117
118 class Foo4(float):
119 def __float__(self):
120 return 42
121
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000122 # Issue 5759: __float__ not called on str subclasses (though it is on
123 # unicode subclasses).
124 class FooStr(str):
125 def __float__(self):
126 return float(str(self)) + 1
127
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000128 self.assertAlmostEqual(float(Foo0()), 42.)
129 self.assertAlmostEqual(float(Foo1()), 42.)
130 self.assertAlmostEqual(float(Foo2()), 42.)
131 self.assertAlmostEqual(float(Foo3(21)), 42.)
132 self.assertRaises(TypeError, float, Foo4(42))
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000133 self.assertAlmostEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000134
135 def test_floatasratio(self):
136 for f, ratio in [
137 (0.875, (7, 8)),
138 (-0.875, (-7, 8)),
139 (0.0, (0, 1)),
140 (11.5, (23, 2)),
141 ]:
142 self.assertEqual(f.as_integer_ratio(), ratio)
143
144 for i in range(10000):
145 f = random.random()
146 f *= 10 ** random.randint(-100, 100)
147 n, d = f.as_integer_ratio()
148 self.assertEqual(float(n).__truediv__(d), f)
149
150 R = fractions.Fraction
151 self.assertEqual(R(0, 1),
152 R(*float(0.0).as_integer_ratio()))
153 self.assertEqual(R(5, 2),
154 R(*float(2.5).as_integer_ratio()))
155 self.assertEqual(R(1, 2),
156 R(*float(0.5).as_integer_ratio()))
157 self.assertEqual(R(4728779608739021, 2251799813685248),
158 R(*float(2.1).as_integer_ratio()))
159 self.assertEqual(R(-4728779608739021, 2251799813685248),
160 R(*float(-2.1).as_integer_ratio()))
161 self.assertEqual(R(-2100, 1),
162 R(*float(-2100.0).as_integer_ratio()))
163
164 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
165 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
166 self.assertRaises(ValueError, float('nan').as_integer_ratio)
167
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000168 def test_float_containment(self):
169 floats = (INF, -INF, 0.0, 1.0, NAN)
170 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000171 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000172 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000173 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000174 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000175 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000176 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000177
178 for f in floats:
179 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000180 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
181 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
182 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
183 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000184 "{%r : None}" % (f, f))
185
186 # identical containers
187 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000188 self.assertTrue(l == l, "[%r] not equal to itself" % f)
189 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
190 self.assertTrue(s == s, "{%r} not equal to itself" % f)
191 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000192
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000193 def assertEqualAndEqualSign(self, a, b):
194 # fail unless a == b and a and b have the same sign bit;
195 # the only difference from assertEqual is that this test
196 # distingishes -0.0 and 0.0.
197 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
198
199 @requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000200 def test_float_mod(self):
201 # Check behaviour of % operator for IEEE 754 special cases.
202 # In particular, check signs of zeros.
203 mod = operator.mod
204
205 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
206 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
207 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
208 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
209 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
210 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
211
212 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
213 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
214 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
215 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
216 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
217 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
218
219 @requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000220 def test_float_pow(self):
221 # test builtin pow and ** operator for IEEE 754 special cases.
222 # Special cases taken from section F.9.4.4 of the C99 specification
223
224 for pow_op in pow, operator.pow:
225 # x**NAN is NAN for any x except 1
226 self.assertTrue(isnan(pow_op(-INF, NAN)))
227 self.assertTrue(isnan(pow_op(-2.0, NAN)))
228 self.assertTrue(isnan(pow_op(-1.0, NAN)))
229 self.assertTrue(isnan(pow_op(-0.5, NAN)))
230 self.assertTrue(isnan(pow_op(-0.0, NAN)))
231 self.assertTrue(isnan(pow_op(0.0, NAN)))
232 self.assertTrue(isnan(pow_op(0.5, NAN)))
233 self.assertTrue(isnan(pow_op(2.0, NAN)))
234 self.assertTrue(isnan(pow_op(INF, NAN)))
235 self.assertTrue(isnan(pow_op(NAN, NAN)))
236
237 # NAN**y is NAN for any y except +-0
238 self.assertTrue(isnan(pow_op(NAN, -INF)))
239 self.assertTrue(isnan(pow_op(NAN, -2.0)))
240 self.assertTrue(isnan(pow_op(NAN, -1.0)))
241 self.assertTrue(isnan(pow_op(NAN, -0.5)))
242 self.assertTrue(isnan(pow_op(NAN, 0.5)))
243 self.assertTrue(isnan(pow_op(NAN, 1.0)))
244 self.assertTrue(isnan(pow_op(NAN, 2.0)))
245 self.assertTrue(isnan(pow_op(NAN, INF)))
246
247 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
248 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
249 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
250
251 # (+-0)**y raises ZeroDivisionError for y finite and negative
252 # but not an odd integer
253 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
254 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
255 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
256 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
257
258 # (+-0)**y is +-0 for y a positive odd integer
259 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
260 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
261
262 # (+-0)**y is 0 for y finite and positive but not an odd integer
263 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
264 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
265 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
266 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
267
268 # (-1)**+-inf is 1
269 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
270 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
271
272 # 1**y is 1 for any y, even if y is an infinity or nan
273 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
274 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
275 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
276 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
277 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
278 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
279 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
280 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
281 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
282 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
283 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
284
285 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
286 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
287 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
288 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
289 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
290 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
291 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
292 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
293 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
294 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
295 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
296 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
297 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
298 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
299 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
300 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
301 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
302 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
303 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
304 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
305 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
306 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
307 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
308
309 # x**y defers to complex pow for finite negative x and
310 # non-integral y.
311 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
312 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
313 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
314 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
315 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
316 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
317
318 # x**-INF is INF for abs(x) < 1
319 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
320 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
321 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
322 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
323
324 # x**-INF is 0 for abs(x) > 1
325 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
326 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
327 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
328 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
329
330 # x**INF is 0 for abs(x) < 1
331 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
332 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
333 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
334 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
335
336 # x**INF is INF for abs(x) > 1
337 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
338 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
339 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
340 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
341
342 # (-INF)**y is -0.0 for y a negative odd integer
343 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
344
345 # (-INF)**y is 0.0 for y negative but not an odd integer
346 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
347 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
348
349 # (-INF)**y is -INF for y a positive odd integer
350 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
351
352 # (-INF)**y is INF for y positive but not an odd integer
353 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
354 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
355
356 # INF**y is INF for y positive
357 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
358 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
359 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
360
361 # INF**y is 0.0 for y negative
362 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
363 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
364 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
365
366 # basic checks not covered by the special cases above
367 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
368 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
369 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
370 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
371 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
372 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
373 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
374 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
375 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
376 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
377 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
378 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
379 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
380 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
381 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
382 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
383 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
384 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
385
386 # 1 ** large and -1 ** large; some libms apparently
387 # have problems with these
388 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
389 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
390 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
391 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
392
393 # check sign for results that underflow to 0
394 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
395 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
396 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
397 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
398 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
399 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
400 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
401 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
402 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
403 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
404 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
405 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
406
407 # check we don't raise an exception for subnormal results,
408 # and validate signs. Tests currently disabled, since
409 # they fail on systems where a subnormal result from pow
410 # is flushed to zero (e.g. Debian/ia64.)
411 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
412 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
413 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
414 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
415 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
416 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
417 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
418 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000419
420
Benjamin Petersone401c682010-07-02 23:25:44 +0000421@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000422class FormatFunctionsTestCase(unittest.TestCase):
423
424 def setUp(self):
425 self.save_formats = {'double':float.__getformat__('double'),
426 'float':float.__getformat__('float')}
427
428 def tearDown(self):
429 float.__setformat__('double', self.save_formats['double'])
430 float.__setformat__('float', self.save_formats['float'])
431
432 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000433 self.assertIn(float.__getformat__('double'),
434 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
435 self.assertIn(float.__getformat__('float'),
436 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000437 self.assertRaises(ValueError, float.__getformat__, 'chicken')
438 self.assertRaises(TypeError, float.__getformat__, 1)
439
440 def test_setformat(self):
441 for t in 'double', 'float':
442 float.__setformat__(t, 'unknown')
443 if self.save_formats[t] == 'IEEE, big-endian':
444 self.assertRaises(ValueError, float.__setformat__,
445 t, 'IEEE, little-endian')
446 elif self.save_formats[t] == 'IEEE, little-endian':
447 self.assertRaises(ValueError, float.__setformat__,
448 t, 'IEEE, big-endian')
449 else:
450 self.assertRaises(ValueError, float.__setformat__,
451 t, 'IEEE, big-endian')
452 self.assertRaises(ValueError, float.__setformat__,
453 t, 'IEEE, little-endian')
454 self.assertRaises(ValueError, float.__setformat__,
455 t, 'chicken')
456 self.assertRaises(ValueError, float.__setformat__,
457 'chicken', 'unknown')
458
Guido van Rossum2be161d2007-05-15 20:43:51 +0000459BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000460LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000461BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000462LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000463
Guido van Rossum2be161d2007-05-15 20:43:51 +0000464BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000465LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000466BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000467LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000468
469# on non-IEEE platforms, attempting to unpack a bit pattern
470# representing an infinity or a NaN should raise an exception.
471
Benjamin Petersone401c682010-07-02 23:25:44 +0000472@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000473class UnknownFormatTestCase(unittest.TestCase):
474 def setUp(self):
475 self.save_formats = {'double':float.__getformat__('double'),
476 'float':float.__getformat__('float')}
477 float.__setformat__('double', 'unknown')
478 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000479
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000480 def tearDown(self):
481 float.__setformat__('double', self.save_formats['double'])
482 float.__setformat__('float', self.save_formats['float'])
483
484 def test_double_specials_dont_unpack(self):
485 for fmt, data in [('>d', BE_DOUBLE_INF),
486 ('>d', BE_DOUBLE_NAN),
487 ('<d', LE_DOUBLE_INF),
488 ('<d', LE_DOUBLE_NAN)]:
489 self.assertRaises(ValueError, struct.unpack, fmt, data)
490
491 def test_float_specials_dont_unpack(self):
492 for fmt, data in [('>f', BE_FLOAT_INF),
493 ('>f', BE_FLOAT_NAN),
494 ('<f', LE_FLOAT_INF),
495 ('<f', LE_FLOAT_NAN)]:
496 self.assertRaises(ValueError, struct.unpack, fmt, data)
497
498
499# on an IEEE platform, all we guarantee is that bit patterns
500# representing infinities or NaNs do not raise an exception; all else
501# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000502# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000503
504class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000505
Benjamin Petersone401c682010-07-02 23:25:44 +0000506 @requires_IEEE_754
507 def test_double_specials_do_unpack(self):
508 for fmt, data in [('>d', BE_DOUBLE_INF),
509 ('>d', BE_DOUBLE_NAN),
510 ('<d', LE_DOUBLE_INF),
511 ('<d', LE_DOUBLE_NAN)]:
512 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000513
Benjamin Petersone401c682010-07-02 23:25:44 +0000514 @requires_IEEE_754
515 def test_float_specials_do_unpack(self):
516 for fmt, data in [('>f', BE_FLOAT_INF),
517 ('>f', BE_FLOAT_NAN),
518 ('<f', LE_FLOAT_INF),
519 ('<f', LE_FLOAT_NAN)]:
520 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000521
Eric Smith8c663262007-08-25 02:26:07 +0000522class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000523
Eric Smith11fe3e02007-08-31 01:33:06 +0000524 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000525 # these should be rewritten to use both format(x, spec) and
526 # x.__format__(spec)
527
528 self.assertEqual(format(0.0, 'f'), '0.000000')
529
530 # the default is 'g', except for empty format spec
531 self.assertEqual(format(0.0, ''), '0.0')
532 self.assertEqual(format(0.01, ''), '0.01')
533 self.assertEqual(format(0.01, 'g'), '0.01')
534
Eric Smith63376222009-05-05 14:04:18 +0000535 # empty presentation type should format in the same way as str
536 # (issue 5920)
537 x = 100/7.
538 self.assertEqual(format(x, ''), str(x))
539 self.assertEqual(format(x, '-'), str(x))
540 self.assertEqual(format(x, '>'), str(x))
541 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000542
543 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000544
545 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000546
547 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
548 self.assertEqual(format(-1.0, ' f'), '-1.000000')
549 self.assertEqual(format( 1.0, '+f'), '+1.000000')
550 self.assertEqual(format(-1.0, '+f'), '-1.000000')
551
552 # % formatting
553 self.assertEqual(format(-1.0, '%'), '-100.000000%')
554
555 # conversion to string should fail
556 self.assertRaises(ValueError, format, 3.0, "s")
557
Eric Smith7b69c6c2008-01-27 21:07:59 +0000558 # other format specifiers shouldn't work on floats,
559 # in particular int specifiers
560 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
561 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
562 if not format_spec in 'eEfFgGn%':
563 self.assertRaises(ValueError, format, 0.0, format_spec)
564 self.assertRaises(ValueError, format, 1.0, format_spec)
565 self.assertRaises(ValueError, format, -1.0, format_spec)
566 self.assertRaises(ValueError, format, 1e100, format_spec)
567 self.assertRaises(ValueError, format, -1e100, format_spec)
568 self.assertRaises(ValueError, format, 1e-100, format_spec)
569 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000570
Eric Smith741191f2009-05-06 13:08:15 +0000571 # issue 3382
572 self.assertEqual(format(NAN, 'f'), 'nan')
573 self.assertEqual(format(NAN, 'F'), 'NAN')
574 self.assertEqual(format(INF, 'f'), 'inf')
575 self.assertEqual(format(INF, 'F'), 'INF')
576
Benjamin Petersone401c682010-07-02 23:25:44 +0000577 @requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000578 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000579 with open(format_testfile) as testfile:
580 for line in testfile:
581 if line.startswith('--'):
582 continue
583 line = line.strip()
584 if not line:
585 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000586
Brian Curtin076623b2010-10-31 00:03:45 +0000587 lhs, rhs = map(str.strip, line.split('->'))
588 fmt, arg = lhs.split()
589 self.assertEqual(fmt % float(arg), rhs)
590 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000591
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000592 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000593 self.assertEqual(format(123.456, '.4'), '123.5')
594 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
595 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000596
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000597class ReprTestCase(unittest.TestCase):
598 def test_repr(self):
599 floats_file = open(os.path.join(os.path.split(__file__)[0],
600 'floating_points.txt'))
601 for line in floats_file:
602 line = line.strip()
603 if not line or line.startswith('#'):
604 continue
605 v = eval(line)
606 self.assertEqual(v, eval(repr(v)))
607 floats_file.close()
608
Eric Smith0923d1d2009-04-16 20:16:10 +0000609 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
610 "applies only when using short float repr style")
611 def test_short_repr(self):
612 # test short float repr introduced in Python 3.1. One aspect
613 # of this repr is that we get some degree of str -> float ->
614 # str roundtripping. In particular, for any numeric string
615 # containing 15 or fewer significant digits, those exact same
616 # digits (modulo trailing zeros) should appear in the output.
617 # No more repr(0.03) -> "0.029999999999999999"!
618
619 test_strings = [
620 # output always includes *either* a decimal point and at
621 # least one digit after that point, or an exponent.
622 '0.0',
623 '1.0',
624 '0.01',
625 '0.02',
626 '0.03',
627 '0.04',
628 '0.05',
629 '1.23456789',
630 '10.0',
631 '100.0',
632 # values >= 1e16 get an exponent...
633 '1000000000000000.0',
634 '9999999999999990.0',
635 '1e+16',
636 '1e+17',
637 # ... and so do values < 1e-4
638 '0.001',
639 '0.001001',
640 '0.00010000000000001',
641 '0.0001',
642 '9.999999999999e-05',
643 '1e-05',
644 # values designed to provoke failure if the FPU rounding
645 # precision isn't set correctly
646 '8.72293771110361e+25',
647 '7.47005307342313e+26',
648 '2.86438000439698e+28',
649 '8.89142905246179e+28',
650 '3.08578087079232e+35',
651 ]
652
653 for s in test_strings:
654 negs = '-'+s
655 self.assertEqual(s, repr(float(s)))
656 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000657 # Since Python 3.2, repr and str are identical
658 self.assertEqual(repr(float(s)), str(float(s)))
659 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000660
661@requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000662class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000663
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000664 def test_inf_nan(self):
665 self.assertRaises(OverflowError, round, INF)
666 self.assertRaises(OverflowError, round, -INF)
667 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000668 self.assertRaises(TypeError, round, INF, 0.0)
669 self.assertRaises(TypeError, round, -INF, 1.0)
670 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
671 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000672
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000673 def test_large_n(self):
674 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
675 self.assertEqual(round(123.456, n), 123.456)
676 self.assertEqual(round(-123.456, n), -123.456)
677 self.assertEqual(round(1e300, n), 1e300)
678 self.assertEqual(round(1e-320, n), 1e-320)
679 self.assertEqual(round(1e150, 300), 1e150)
680 self.assertEqual(round(1e300, 307), 1e300)
681 self.assertEqual(round(-3.1415, 308), -3.1415)
682 self.assertEqual(round(1e150, 309), 1e150)
683 self.assertEqual(round(1.4e-315, 315), 1e-315)
684
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000685 def test_small_n(self):
686 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
687 self.assertEqual(round(123.456, n), 0.0)
688 self.assertEqual(round(-123.456, n), -0.0)
689 self.assertEqual(round(1e300, n), 0.0)
690 self.assertEqual(round(1e-320, n), 0.0)
691
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000692 def test_overflow(self):
693 self.assertRaises(OverflowError, round, 1.6e308, -308)
694 self.assertRaises(OverflowError, round, -1.7e308, -308)
695
696 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
697 "applies only when using short float repr style")
698 def test_previous_round_bugs(self):
699 # particular cases that have occurred in bug reports
700 self.assertEqual(round(562949953421312.5, 1),
701 562949953421312.5)
702 self.assertEqual(round(56294995342131.5, 3),
703 56294995342131.5)
704 # round-half-even
705 self.assertEqual(round(25.0, -1), 20.0)
706 self.assertEqual(round(35.0, -1), 40.0)
707 self.assertEqual(round(45.0, -1), 40.0)
708 self.assertEqual(round(55.0, -1), 60.0)
709 self.assertEqual(round(65.0, -1), 60.0)
710 self.assertEqual(round(75.0, -1), 80.0)
711 self.assertEqual(round(85.0, -1), 80.0)
712 self.assertEqual(round(95.0, -1), 100.0)
713
714 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
715 "applies only when using short float repr style")
716 def test_matches_float_format(self):
717 # round should give the same results as float formatting
718 for i in range(500):
719 x = i/1000.
720 self.assertEqual(float(format(x, '.0f')), round(x, 0))
721 self.assertEqual(float(format(x, '.1f')), round(x, 1))
722 self.assertEqual(float(format(x, '.2f')), round(x, 2))
723 self.assertEqual(float(format(x, '.3f')), round(x, 3))
724
725 for i in range(5, 5000, 10):
726 x = i/1000.
727 self.assertEqual(float(format(x, '.0f')), round(x, 0))
728 self.assertEqual(float(format(x, '.1f')), round(x, 1))
729 self.assertEqual(float(format(x, '.2f')), round(x, 2))
730 self.assertEqual(float(format(x, '.3f')), round(x, 3))
731
732 for i in range(500):
733 x = random.random()
734 self.assertEqual(float(format(x, '.0f')), round(x, 0))
735 self.assertEqual(float(format(x, '.1f')), round(x, 1))
736 self.assertEqual(float(format(x, '.2f')), round(x, 2))
737 self.assertEqual(float(format(x, '.3f')), round(x, 3))
738
Eric Smith8a10ecc2009-12-02 17:58:24 +0000739 def test_format_specials(self):
740 # Test formatting of nans and infs.
741
742 def test(fmt, value, expected):
743 # Test with both % and format().
744 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000745 fmt = fmt[1:] # strip off the %
746 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000747
748 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
749 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
750 pfmt = '%+' + fmt[1:]
751 sfmt = '% ' + fmt[1:]
752 test(fmt, INF, 'inf')
753 test(fmt, -INF, '-inf')
754 test(fmt, NAN, 'nan')
755 test(fmt, -NAN, 'nan')
756 # When asking for a sign, it's always provided. nans are
757 # always positive.
758 test(pfmt, INF, '+inf')
759 test(pfmt, -INF, '-inf')
760 test(pfmt, NAN, '+nan')
761 test(pfmt, -NAN, '+nan')
762 # When using ' ' for a sign code, only infs can be negative.
763 # Others have a space.
764 test(sfmt, INF, ' inf')
765 test(sfmt, -INF, '-inf')
766 test(sfmt, NAN, ' nan')
767 test(sfmt, -NAN, ' nan')
768
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000769
Christian Heimes99170a52007-12-19 02:07:34 +0000770# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000771# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000772class InfNanTest(unittest.TestCase):
773 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000774 self.assertTrue(isinf(float("inf")))
775 self.assertTrue(isinf(float("+inf")))
776 self.assertTrue(isinf(float("-inf")))
777 self.assertTrue(isinf(float("infinity")))
778 self.assertTrue(isinf(float("+infinity")))
779 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000780
781 self.assertEqual(repr(float("inf")), "inf")
782 self.assertEqual(repr(float("+inf")), "inf")
783 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000784 self.assertEqual(repr(float("infinity")), "inf")
785 self.assertEqual(repr(float("+infinity")), "inf")
786 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000787
788 self.assertEqual(repr(float("INF")), "inf")
789 self.assertEqual(repr(float("+Inf")), "inf")
790 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000791 self.assertEqual(repr(float("Infinity")), "inf")
792 self.assertEqual(repr(float("+iNfInItY")), "inf")
793 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000794
795 self.assertEqual(str(float("inf")), "inf")
796 self.assertEqual(str(float("+inf")), "inf")
797 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000798 self.assertEqual(str(float("infinity")), "inf")
799 self.assertEqual(str(float("+infinity")), "inf")
800 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000801
802 self.assertRaises(ValueError, float, "info")
803 self.assertRaises(ValueError, float, "+info")
804 self.assertRaises(ValueError, float, "-info")
805 self.assertRaises(ValueError, float, "in")
806 self.assertRaises(ValueError, float, "+in")
807 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000808 self.assertRaises(ValueError, float, "infinit")
809 self.assertRaises(ValueError, float, "+Infin")
810 self.assertRaises(ValueError, float, "-INFI")
811 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000812
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000813 self.assertRaises(ValueError, float, "++Inf")
814 self.assertRaises(ValueError, float, "-+inf")
815 self.assertRaises(ValueError, float, "+-infinity")
816 self.assertRaises(ValueError, float, "--Infinity")
817
Christian Heimes99170a52007-12-19 02:07:34 +0000818 def test_inf_as_str(self):
819 self.assertEqual(repr(1e300 * 1e300), "inf")
820 self.assertEqual(repr(-1e300 * 1e300), "-inf")
821
822 self.assertEqual(str(1e300 * 1e300), "inf")
823 self.assertEqual(str(-1e300 * 1e300), "-inf")
824
825 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000826 self.assertTrue(isnan(float("nan")))
827 self.assertTrue(isnan(float("+nan")))
828 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000829
830 self.assertEqual(repr(float("nan")), "nan")
831 self.assertEqual(repr(float("+nan")), "nan")
832 self.assertEqual(repr(float("-nan")), "nan")
833
834 self.assertEqual(repr(float("NAN")), "nan")
835 self.assertEqual(repr(float("+NAn")), "nan")
836 self.assertEqual(repr(float("-NaN")), "nan")
837
838 self.assertEqual(str(float("nan")), "nan")
839 self.assertEqual(str(float("+nan")), "nan")
840 self.assertEqual(str(float("-nan")), "nan")
841
842 self.assertRaises(ValueError, float, "nana")
843 self.assertRaises(ValueError, float, "+nana")
844 self.assertRaises(ValueError, float, "-nana")
845 self.assertRaises(ValueError, float, "na")
846 self.assertRaises(ValueError, float, "+na")
847 self.assertRaises(ValueError, float, "-na")
848
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000849 self.assertRaises(ValueError, float, "++nan")
850 self.assertRaises(ValueError, float, "-+NAN")
851 self.assertRaises(ValueError, float, "+-NaN")
852 self.assertRaises(ValueError, float, "--nAn")
853
Christian Heimes99170a52007-12-19 02:07:34 +0000854 def test_nan_as_str(self):
855 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
856 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
857
858 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
859 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000860
Christian Heimes53876d92008-04-19 00:31:39 +0000861 def notest_float_nan(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000862 self.assertTrue(NAN.is_nan())
863 self.assertFalse(INF.is_nan())
864 self.assertFalse((0.).is_nan())
Christian Heimes53876d92008-04-19 00:31:39 +0000865
866 def notest_float_inf(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000867 self.assertTrue(INF.is_inf())
868 self.assertFalse(NAN.is_inf())
869 self.assertFalse((0.).is_inf())
Christian Heimes53876d92008-04-19 00:31:39 +0000870
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000871fromHex = float.fromhex
872toHex = float.hex
873class HexFloatTestCase(unittest.TestCase):
874 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
875 MIN = fromHex('0x1p-1022') # min normal
876 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
877 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
878
879 def identical(self, x, y):
880 # check that floats x and y are identical, or that both
881 # are NaNs
882 if isnan(x) or isnan(y):
883 if isnan(x) == isnan(y):
884 return
885 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
886 return
887 self.fail('%r not identical to %r' % (x, y))
888
889 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +0000890 self.identical(self.MIN, ldexp(1.0, -1022))
891 self.identical(self.TINY, ldexp(1.0, -1074))
892 self.identical(self.EPS, ldexp(1.0, -52))
893 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000894
895 def test_invalid_inputs(self):
896 invalid_inputs = [
897 'infi', # misspelt infinities and nans
898 '-Infinit',
899 '++inf',
900 '-+Inf',
901 '--nan',
902 '+-NaN',
903 'snan',
904 'NaNs',
905 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000906 'an',
907 'nf',
908 'nfinity',
909 'inity',
910 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000911 '0xnan',
912 '',
913 ' ',
914 'x1.0p0',
915 '0xX1.0p0',
916 '+ 0x1.0p0', # internal whitespace
917 '- 0x1.0p0',
918 '0 x1.0p0',
919 '0x 1.0p0',
920 '0x1 2.0p0',
921 '+0x1 .0p0',
922 '0x1. 0p0',
923 '-0x1.0 1p0',
924 '-0x1.0 p0',
925 '+0x1.0p +0',
926 '0x1.0p -0',
927 '0x1.0p 0',
928 '+0x1.0p+ 0',
929 '-0x1.0p- 0',
930 '++0x1.0p-0', # double signs
931 '--0x1.0p0',
932 '+-0x1.0p+0',
933 '-+0x1.0p0',
934 '0x1.0p++0',
935 '+0x1.0p+-0',
936 '-0x1.0p-+0',
937 '0x1.0p--0',
938 '0x1.0.p0',
939 '0x.p0', # no hex digits before or after point
940 '0x1,p0', # wrong decimal point character
941 '0x1pa',
942 '0x1p\uff10', # fullwidth Unicode digits
943 '\uff10x1p0',
944 '0x\uff11p0',
945 '0x1.\uff10p0',
946 '0x1p0 \n 0x2p0',
947 '0x1p0\0 0x1p0', # embedded null byte is not end of string
948 ]
949 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +0000950 try:
951 result = fromHex(x)
952 except ValueError:
953 pass
954 else:
955 self.fail('Expected float.fromhex(%r) to raise ValueError; '
956 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000957
958
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000959 def test_whitespace(self):
960 value_pairs = [
961 ('inf', INF),
962 ('-Infinity', -INF),
963 ('nan', NAN),
964 ('1.0', 1.0),
965 ('-0x.2', -0.125),
966 ('-0.0', -0.0)
967 ]
968 whitespace = [
969 '',
970 ' ',
971 '\t',
972 '\n',
973 '\n \t',
974 '\f',
975 '\v',
976 '\r'
977 ]
978 for inp, expected in value_pairs:
979 for lead in whitespace:
980 for trail in whitespace:
981 got = fromHex(lead + inp + trail)
982 self.identical(got, expected)
983
984
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000985 def test_from_hex(self):
986 MIN = self.MIN;
987 MAX = self.MAX;
988 TINY = self.TINY;
989 EPS = self.EPS;
990
991 # two spellings of infinity, with optional signs; case-insensitive
992 self.identical(fromHex('inf'), INF)
993 self.identical(fromHex('+Inf'), INF)
994 self.identical(fromHex('-INF'), -INF)
995 self.identical(fromHex('iNf'), INF)
996 self.identical(fromHex('Infinity'), INF)
997 self.identical(fromHex('+INFINITY'), INF)
998 self.identical(fromHex('-infinity'), -INF)
999 self.identical(fromHex('-iNFiNitY'), -INF)
1000
1001 # nans with optional sign; case insensitive
1002 self.identical(fromHex('nan'), NAN)
1003 self.identical(fromHex('+NaN'), NAN)
1004 self.identical(fromHex('-NaN'), NAN)
1005 self.identical(fromHex('-nAN'), NAN)
1006
1007 # variations in input format
1008 self.identical(fromHex('1'), 1.0)
1009 self.identical(fromHex('+1'), 1.0)
1010 self.identical(fromHex('1.'), 1.0)
1011 self.identical(fromHex('1.0'), 1.0)
1012 self.identical(fromHex('1.0p0'), 1.0)
1013 self.identical(fromHex('01'), 1.0)
1014 self.identical(fromHex('01.'), 1.0)
1015 self.identical(fromHex('0x1'), 1.0)
1016 self.identical(fromHex('0x1.'), 1.0)
1017 self.identical(fromHex('0x1.0'), 1.0)
1018 self.identical(fromHex('+0x1.0'), 1.0)
1019 self.identical(fromHex('0x1p0'), 1.0)
1020 self.identical(fromHex('0X1p0'), 1.0)
1021 self.identical(fromHex('0X1P0'), 1.0)
1022 self.identical(fromHex('0x1P0'), 1.0)
1023 self.identical(fromHex('0x1.p0'), 1.0)
1024 self.identical(fromHex('0x1.0p0'), 1.0)
1025 self.identical(fromHex('0x.1p4'), 1.0)
1026 self.identical(fromHex('0x.1p04'), 1.0)
1027 self.identical(fromHex('0x.1p004'), 1.0)
1028 self.identical(fromHex('0x1p+0'), 1.0)
1029 self.identical(fromHex('0x1P-0'), 1.0)
1030 self.identical(fromHex('+0x1p0'), 1.0)
1031 self.identical(fromHex('0x01p0'), 1.0)
1032 self.identical(fromHex('0x1p00'), 1.0)
1033 self.identical(fromHex(' 0x1p0 '), 1.0)
1034 self.identical(fromHex('\n 0x1p0'), 1.0)
1035 self.identical(fromHex('0x1p0 \t'), 1.0)
1036 self.identical(fromHex('0xap0'), 10.0)
1037 self.identical(fromHex('0xAp0'), 10.0)
1038 self.identical(fromHex('0xaP0'), 10.0)
1039 self.identical(fromHex('0xAP0'), 10.0)
1040 self.identical(fromHex('0xbep0'), 190.0)
1041 self.identical(fromHex('0xBep0'), 190.0)
1042 self.identical(fromHex('0xbEp0'), 190.0)
1043 self.identical(fromHex('0XBE0P-4'), 190.0)
1044 self.identical(fromHex('0xBEp0'), 190.0)
1045 self.identical(fromHex('0xB.Ep4'), 190.0)
1046 self.identical(fromHex('0x.BEp8'), 190.0)
1047 self.identical(fromHex('0x.0BEp12'), 190.0)
1048
1049 # moving the point around
1050 pi = fromHex('0x1.921fb54442d18p1')
1051 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1052 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1053 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1054 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1055 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1056 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1057 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1058 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1059 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1060 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1061 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1062 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1063 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1064 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1065 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1066 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1067 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1068 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1069 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1070 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1071 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1072 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1073 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1074 # ...
1075 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1076 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1077 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1078 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1079 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1080 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1081 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1082 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1083 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1084
1085
1086 # results that should overflow...
1087 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1088 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1089 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1090 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1091 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1092 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1093 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1094 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1095 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1096 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1097 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1098 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1099 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1100 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1101 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1102 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1103 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1104 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1105 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1106
1107 # ...and those that round to +-max float
1108 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1109 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1110 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1111
1112 # zeros
1113 self.identical(fromHex('0x0p0'), 0.0)
1114 self.identical(fromHex('0x0p1000'), 0.0)
1115 self.identical(fromHex('-0x0p1023'), -0.0)
1116 self.identical(fromHex('0X0p1024'), 0.0)
1117 self.identical(fromHex('-0x0p1025'), -0.0)
1118 self.identical(fromHex('0X0p2000'), 0.0)
1119 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1120 self.identical(fromHex('-0X0p-0'), -0.0)
1121 self.identical(fromHex('-0X0p-1000'), -0.0)
1122 self.identical(fromHex('0x0p-1023'), 0.0)
1123 self.identical(fromHex('-0X0p-1024'), -0.0)
1124 self.identical(fromHex('-0x0p-1025'), -0.0)
1125 self.identical(fromHex('-0x0p-1072'), -0.0)
1126 self.identical(fromHex('0X0p-1073'), 0.0)
1127 self.identical(fromHex('-0x0p-1074'), -0.0)
1128 self.identical(fromHex('0x0p-1075'), 0.0)
1129 self.identical(fromHex('0X0p-1076'), 0.0)
1130 self.identical(fromHex('-0X0p-2000'), -0.0)
1131 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1132
1133 # values that should underflow to 0
1134 self.identical(fromHex('0X1p-1075'), 0.0)
1135 self.identical(fromHex('-0X1p-1075'), -0.0)
1136 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1137 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1138 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1139 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1140
1141 # check round-half-even is working correctly near 0 ...
1142 self.identical(fromHex('0x1p-1076'), 0.0)
1143 self.identical(fromHex('0X2p-1076'), 0.0)
1144 self.identical(fromHex('0X3p-1076'), TINY)
1145 self.identical(fromHex('0x4p-1076'), TINY)
1146 self.identical(fromHex('0X5p-1076'), TINY)
1147 self.identical(fromHex('0X6p-1076'), 2*TINY)
1148 self.identical(fromHex('0x7p-1076'), 2*TINY)
1149 self.identical(fromHex('0X8p-1076'), 2*TINY)
1150 self.identical(fromHex('0X9p-1076'), 2*TINY)
1151 self.identical(fromHex('0xap-1076'), 2*TINY)
1152 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1153 self.identical(fromHex('0xcp-1076'), 3*TINY)
1154 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1155 self.identical(fromHex('0Xep-1076'), 4*TINY)
1156 self.identical(fromHex('0xfp-1076'), 4*TINY)
1157 self.identical(fromHex('0x10p-1076'), 4*TINY)
1158 self.identical(fromHex('-0x1p-1076'), -0.0)
1159 self.identical(fromHex('-0X2p-1076'), -0.0)
1160 self.identical(fromHex('-0x3p-1076'), -TINY)
1161 self.identical(fromHex('-0X4p-1076'), -TINY)
1162 self.identical(fromHex('-0x5p-1076'), -TINY)
1163 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1164 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1165 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1166 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1167 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1168 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1169 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1170 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1171 self.identical(fromHex('-0xep-1076'), -4*TINY)
1172 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1173 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1174
1175 # ... and near MIN ...
1176 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1177 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1178 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1179 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1180 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1181 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1182 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1183 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1184 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1185 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1186 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1187 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1188 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1189 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1190 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1191 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1192 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1193 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1194 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1195 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1196 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1197 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1198 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1199 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1200 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1201 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1202 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1203 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1204 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1205 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1206 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1207 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1208 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1209 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1210
1211 # ... and near 1.0.
1212 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1213 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1214 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1215 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1216 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1217 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1218 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1219 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1220 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1221 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1222 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1223 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1224 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1225 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1226 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1227 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1228 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1229 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1230 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1231 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1232 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1233 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1234 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1235 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1236 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1237 1.0)
1238 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1239 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1240 1+EPS)
1241 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1242 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1243 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1244 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1245 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1246 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1247 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1248 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1249 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1250 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1251 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1252 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1253 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1254 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1255 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1256 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1257 1.0+EPS)
1258 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1259 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1260 1.0+2*EPS)
1261 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1262 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1263 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1264 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1265 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1266 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1267 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1268 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1269
1270 def test_roundtrip(self):
1271 def roundtrip(x):
1272 return fromHex(toHex(x))
1273
1274 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1275 self.identical(x, roundtrip(x))
1276 self.identical(-x, roundtrip(-x))
1277
1278 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1279 import random
1280 for i in range(10000):
1281 e = random.randrange(-1200, 1200)
1282 m = random.random()
1283 s = random.choice([1.0, -1.0])
1284 try:
1285 x = s*ldexp(m, e)
1286 except OverflowError:
1287 pass
1288 else:
1289 self.identical(x, fromHex(toHex(x)))
1290
Christian Heimes53876d92008-04-19 00:31:39 +00001291
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001292def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001293 support.run_unittest(
Amaury Forgeot d'Arc7e958d12008-09-06 21:03:22 +00001294 GeneralFloatCases,
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001295 FormatFunctionsTestCase,
1296 UnknownFormatTestCase,
Eric Smith8c663262007-08-25 02:26:07 +00001297 IEEEFormatTestCase,
Christian Heimes827b35c2007-12-10 22:19:17 +00001298 FormatTestCase,
Christian Heimes99170a52007-12-19 02:07:34 +00001299 ReprTestCase,
Mark Dickinsone6a076d2009-04-18 11:48:33 +00001300 RoundTestCase,
Christian Heimes99170a52007-12-19 02:07:34 +00001301 InfNanTest,
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001302 HexFloatTestCase,
Christian Heimesb76922a2007-12-11 01:06:40 +00001303 )
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001304
1305if __name__ == '__main__':
1306 test_main()