blob: 9a673824bda3d0ad25ee8b2903f1cf0f9c7486e6 [file] [log] [blame]
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001
2import unittest, struct
Christian Heimes284d9272007-12-10 22:28:56 +00003import os
Michael W. Hudsonba283e22005-05-27 15:23:20 +00004from test import test_support
Christian Heimes6f341092008-04-18 23:13:07 +00005import math
6from math import isinf, isnan
7import operator
Michael W. Hudsonba283e22005-05-27 15:23:20 +00008
Christian Heimes6f341092008-04-18 23:13:07 +00009INF = float("inf")
10NAN = float("nan")
Christian Heimes0a8143f2007-12-18 23:22:54 +000011
Michael W. Hudsonba283e22005-05-27 15:23:20 +000012class FormatFunctionsTestCase(unittest.TestCase):
13
14 def setUp(self):
15 self.save_formats = {'double':float.__getformat__('double'),
16 'float':float.__getformat__('float')}
17
18 def tearDown(self):
19 float.__setformat__('double', self.save_formats['double'])
20 float.__setformat__('float', self.save_formats['float'])
21
22 def test_getformat(self):
23 self.assert_(float.__getformat__('double') in
24 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
25 self.assert_(float.__getformat__('float') in
26 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
27 self.assertRaises(ValueError, float.__getformat__, 'chicken')
28 self.assertRaises(TypeError, float.__getformat__, 1)
29
30 def test_setformat(self):
31 for t in 'double', 'float':
32 float.__setformat__(t, 'unknown')
33 if self.save_formats[t] == 'IEEE, big-endian':
34 self.assertRaises(ValueError, float.__setformat__,
35 t, 'IEEE, little-endian')
36 elif self.save_formats[t] == 'IEEE, little-endian':
37 self.assertRaises(ValueError, float.__setformat__,
38 t, 'IEEE, big-endian')
39 else:
40 self.assertRaises(ValueError, float.__setformat__,
41 t, 'IEEE, big-endian')
42 self.assertRaises(ValueError, float.__setformat__,
43 t, 'IEEE, little-endian')
44 self.assertRaises(ValueError, float.__setformat__,
45 t, 'chicken')
46 self.assertRaises(ValueError, float.__setformat__,
47 'chicken', 'unknown')
48
49BE_DOUBLE_INF = '\x7f\xf0\x00\x00\x00\x00\x00\x00'
50LE_DOUBLE_INF = ''.join(reversed(BE_DOUBLE_INF))
51BE_DOUBLE_NAN = '\x7f\xf8\x00\x00\x00\x00\x00\x00'
52LE_DOUBLE_NAN = ''.join(reversed(BE_DOUBLE_NAN))
53
54BE_FLOAT_INF = '\x7f\x80\x00\x00'
55LE_FLOAT_INF = ''.join(reversed(BE_FLOAT_INF))
56BE_FLOAT_NAN = '\x7f\xc0\x00\x00'
57LE_FLOAT_NAN = ''.join(reversed(BE_FLOAT_NAN))
58
59# on non-IEEE platforms, attempting to unpack a bit pattern
60# representing an infinity or a NaN should raise an exception.
61
62class UnknownFormatTestCase(unittest.TestCase):
63 def setUp(self):
64 self.save_formats = {'double':float.__getformat__('double'),
65 'float':float.__getformat__('float')}
66 float.__setformat__('double', 'unknown')
67 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +000068
Michael W. Hudsonba283e22005-05-27 15:23:20 +000069 def tearDown(self):
70 float.__setformat__('double', self.save_formats['double'])
71 float.__setformat__('float', self.save_formats['float'])
72
73 def test_double_specials_dont_unpack(self):
74 for fmt, data in [('>d', BE_DOUBLE_INF),
75 ('>d', BE_DOUBLE_NAN),
76 ('<d', LE_DOUBLE_INF),
77 ('<d', LE_DOUBLE_NAN)]:
78 self.assertRaises(ValueError, struct.unpack, fmt, data)
79
80 def test_float_specials_dont_unpack(self):
81 for fmt, data in [('>f', BE_FLOAT_INF),
82 ('>f', BE_FLOAT_NAN),
83 ('<f', LE_FLOAT_INF),
84 ('<f', LE_FLOAT_NAN)]:
85 self.assertRaises(ValueError, struct.unpack, fmt, data)
86
87
88# on an IEEE platform, all we guarantee is that bit patterns
89# representing infinities or NaNs do not raise an exception; all else
90# is accident (today).
Alex Martellid8672aa2007-08-22 21:14:17 +000091# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +000092
93class IEEEFormatTestCase(unittest.TestCase):
94 if float.__getformat__("double").startswith("IEEE"):
95 def test_double_specials_do_unpack(self):
96 for fmt, data in [('>d', BE_DOUBLE_INF),
97 ('>d', BE_DOUBLE_NAN),
98 ('<d', LE_DOUBLE_INF),
99 ('<d', LE_DOUBLE_NAN)]:
100 struct.unpack(fmt, data)
101
102 if float.__getformat__("float").startswith("IEEE"):
103 def test_float_specials_do_unpack(self):
104 for fmt, data in [('>f', BE_FLOAT_INF),
105 ('>f', BE_FLOAT_NAN),
106 ('<f', LE_FLOAT_INF),
107 ('<f', LE_FLOAT_NAN)]:
108 struct.unpack(fmt, data)
109
Alex Martellid8672aa2007-08-22 21:14:17 +0000110 if float.__getformat__("double").startswith("IEEE"):
111 def test_negative_zero(self):
112 import math
113 def pos_pos():
114 return 0.0, math.atan2(0.0, -1)
115 def pos_neg():
116 return 0.0, math.atan2(-0.0, -1)
117 def neg_pos():
118 return -0.0, math.atan2(0.0, -1)
119 def neg_neg():
120 return -0.0, math.atan2(-0.0, -1)
121 self.assertEquals(pos_pos(), neg_pos())
122 self.assertEquals(pos_neg(), neg_neg())
123
Guido van Rossum3b835492008-01-05 00:59:59 +0000124 if float.__getformat__("double").startswith("IEEE"):
125 def test_underflow_sign(self):
126 import math
127 # check that -1e-1000 gives -0.0, not 0.0
128 self.assertEquals(math.atan2(-1e-1000, -1), math.atan2(-0.0, -1))
129 self.assertEquals(math.atan2(float('-1e-1000'), -1),
130 math.atan2(-0.0, -1))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000131
Christian Heimes284d9272007-12-10 22:28:56 +0000132class ReprTestCase(unittest.TestCase):
133 def test_repr(self):
134 floats_file = open(os.path.join(os.path.split(__file__)[0],
135 'floating_points.txt'))
136 for line in floats_file:
137 line = line.strip()
138 if not line or line.startswith('#'):
139 continue
140 v = eval(line)
141 self.assertEqual(v, eval(repr(v)))
142 floats_file.close()
143
Christian Heimes0a8143f2007-12-18 23:22:54 +0000144# Beginning with Python 2.6 float has cross platform compatible
145# ways to create and representate inf and nan
146class InfNanTest(unittest.TestCase):
147 def test_inf_from_str(self):
148 self.assert_(isinf(float("inf")))
149 self.assert_(isinf(float("+inf")))
150 self.assert_(isinf(float("-inf")))
151
152 self.assertEqual(repr(float("inf")), "inf")
153 self.assertEqual(repr(float("+inf")), "inf")
154 self.assertEqual(repr(float("-inf")), "-inf")
155
156 self.assertEqual(repr(float("INF")), "inf")
157 self.assertEqual(repr(float("+Inf")), "inf")
158 self.assertEqual(repr(float("-iNF")), "-inf")
159
160 self.assertEqual(str(float("inf")), "inf")
161 self.assertEqual(str(float("+inf")), "inf")
162 self.assertEqual(str(float("-inf")), "-inf")
163
164 self.assertRaises(ValueError, float, "info")
165 self.assertRaises(ValueError, float, "+info")
166 self.assertRaises(ValueError, float, "-info")
167 self.assertRaises(ValueError, float, "in")
168 self.assertRaises(ValueError, float, "+in")
169 self.assertRaises(ValueError, float, "-in")
170
171 def test_inf_as_str(self):
172 self.assertEqual(repr(1e300 * 1e300), "inf")
173 self.assertEqual(repr(-1e300 * 1e300), "-inf")
174
175 self.assertEqual(str(1e300 * 1e300), "inf")
176 self.assertEqual(str(-1e300 * 1e300), "-inf")
177
178 def test_nan_from_str(self):
179 self.assert_(isnan(float("nan")))
180 self.assert_(isnan(float("+nan")))
181 self.assert_(isnan(float("-nan")))
182
183 self.assertEqual(repr(float("nan")), "nan")
184 self.assertEqual(repr(float("+nan")), "nan")
185 self.assertEqual(repr(float("-nan")), "nan")
186
187 self.assertEqual(repr(float("NAN")), "nan")
188 self.assertEqual(repr(float("+NAn")), "nan")
189 self.assertEqual(repr(float("-NaN")), "nan")
190
191 self.assertEqual(str(float("nan")), "nan")
192 self.assertEqual(str(float("+nan")), "nan")
193 self.assertEqual(str(float("-nan")), "nan")
194
195 self.assertRaises(ValueError, float, "nana")
196 self.assertRaises(ValueError, float, "+nana")
197 self.assertRaises(ValueError, float, "-nana")
198 self.assertRaises(ValueError, float, "na")
199 self.assertRaises(ValueError, float, "+na")
200 self.assertRaises(ValueError, float, "-na")
201
202 def test_nan_as_str(self):
203 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
204 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
205
206 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
207 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes284d9272007-12-10 22:28:56 +0000208
Christian Heimes6f341092008-04-18 23:13:07 +0000209 def notest_float_nan(self):
210 self.assert_(NAN.is_nan())
211 self.failIf(INF.is_nan())
212 self.failIf((0.).is_nan())
213
214 def notest_float_inf(self):
215 self.assert_(INF.is_inf())
216 self.failIf(NAN.is_inf())
217 self.failIf((0.).is_inf())
218
219
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000220def test_main():
221 test_support.run_unittest(
222 FormatFunctionsTestCase,
223 UnknownFormatTestCase,
Christian Heimes284d9272007-12-10 22:28:56 +0000224 IEEEFormatTestCase,
Christian Heimes0a8143f2007-12-18 23:22:54 +0000225 ReprTestCase,
226 InfNanTest,
Christian Heimesf15c66e2007-12-11 00:54:34 +0000227 )
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000228
229if __name__ == '__main__':
230 test_main()