forked from mdickinson/bigfloat
-
Notifications
You must be signed in to change notification settings - Fork 0
/
setup.py
279 lines (223 loc) · 9.59 KB
/
setup.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
# -*- coding: UTF-8
# Copyright 2009--2019 Mark Dickinson.
#
# This file is part of the bigfloat package.
#
# The bigfloat package is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or (at your
# option) any later version.
#
# The bigfloat package is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
# for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with the bigfloat package. If not, see <http://www.gnu.org/licenses/>.
import os
from setuptools import setup, Extension
DESCRIPTION = """\
Arbitrary-precision correctly-rounded floating-point arithmetic, via MPFR.\
"""
LONG_DESCRIPTION = """\
The ``bigfloat`` package is a Python package providing arbitrary-precision
correctly-rounded binary floating-point arithmetic. It is implemented as a
`Cython <http://cython.org>`_ wrapper around the `GNU MPFR library
<http://www.mpfr.org>`_. A couple of lines of Python code should give the
idea::
>>> from bigfloat import *
>>> with precision(200) + RoundTowardZero:
... print(sqrt(2))
...
1.4142135623730950488016887242096980785696718753769480731766796
>>> with quadruple_precision:
... const_pi()
...
BigFloat.exact('3.14159265358979323846264338327950280', precision=113)
Features
--------
- Supports Python 2 (version 2.7) and Python 3 (version 3.5 or later).
- Exactly reproducible correctly-rounded results across platforms;
precisely-defined semantics compatible with the IEEE 754-2008 standard.
- Support for mixed-type operations with Python integers and floats.
- Support for emulating IEEE 754 arithmetic in any of the IEEE binary
interchange formats described in IEEE 754-2008. Infinities, NaNs,
signed zeros, and subnormals are all supported.
- Easy control of rounding modes and precisions via ``Context`` objects
and Python's ``with`` statement.
Documentation
-------------
Full `package documentation <http://bigfloat.readthedocs.org>`_ is hosted at
Read the Docs. Read on for a quick tour.
A quick tour
------------
The ``bigfloat`` package is small and simple to use. Here's a quick
tour of some of its features.
For demonstration purposes, start with::
>>> from bigfloat import *
Note that this import shadows some builtin Python functions, namely ``abs``,
``max``, ``min``, ``pow``, ``round`` and (on Python 2 only) ``cmp``. In normal
usage you'll probably only want to import the classes and functions that you
actually need.
The main class is the ``BigFloat`` class::
>>> BigFloat(1) # can be constructed from an integer, float or string
BigFloat.exact('1.0000000000000000', precision=53)
>>> BigFloat('3.14159') ** 2 / 6.0 # can combine with ints and floats
BigFloat.exact('1.6449312880166664', precision=53)
>>> BigFloat('0.1', precision(200)) # high-precision value from string
BigFloat.exact('0.1000000000000000000000000000000000000000000000000000
0000000002', precision=200)
Newly-created ``BigFloat`` instances refer to the current *context* to
determine what precision and rounding modes to use. This current
context is represented by a ``Context`` instance, and can be retrieved
by calling ``getcontext``::
>>> getcontext()
Context(precision=53, emax=1073741823, emin=-1073741823,
subnormalize=False, rounding=ROUND_TIES_TO_EVEN)
The ``precision(200)`` argument passed to the ``BigFloat`` constructor
above is also an example of a ``Context``::
>>> precision(200)
Context(precision=200)
The context used for a calculation can be set using the ``setcontext``
function, but a better way to make a temporary change to the context
is to use Python's ``with`` statement::
>>> with precision(1000):
... print sqrt(2)
...
1.41421356237309504880168872420969807856967187537694807317667973
7990732478462107038850387534327641572735013846230912297024924836
0558507372126441214970999358314132226659275055927557999505011527
8206057147010955997160597027453459686201472851741864088919860955
232923048430871432145083976260362799525140798964
Here, ``sqrt`` is one of a number of mathematical functions that the
``bigfloat`` package exports. As you can see, these functions operate on
integers and floats as well as ``BigFloat`` instances, but always
return a ``BigFloat`` instance.
Rounding modes can be controlled similarly. Here are upper and lower
bounds for π, accurate to 53 significant bits::
>>> with RoundTowardPositive:
... const_pi()
...
BigFloat.exact('3.1415926535897936', precision=53)
>>> with RoundTowardNegative:
... const_pi()
...
BigFloat.exact('3.1415926535897931', precision=53)
And as you'd expect, ``with`` statements like those above can be
nested. ``Context`` objects can also be combined using addition::
>>> with RoundTowardPositive + precision(24):
... BigFloat(1) / 3
...
BigFloat.exact('0.333333343', precision=24)
Various ``Context`` objects corresponding to IEEE 754 interchange
formats are predefined::
>>> quadruple_precision
Context(precision=113, emax=16384, emin=-16493, subnormalize=True)
>>> half_precision
Context(precision=11, emax=16, emin=-23, subnormalize=True)
>>> with half_precision:
log(2)
...
BigFloat.exact('0.69336', precision=11)
Installation
------------
The ``bigfloat`` package is `available on the Python package index
<https://pypi.python.org/pypi/bigfloat>`_, and can be installed in the usual
way using ``easy_install`` or ``pip``. Alternatively, the development sources
may be downloaded from the project's `homepage
<https:/github.com/mdickinson/bigfloat>`_ on GitHub.
For more comprehensive installation instructions, please see the `full
documentation <http://bigfloat.readthedocs.org/en/latest/#installation>`_.
Feedback
--------
Feedback is welcome! Please use the `GitHub issue tracker
<https://github.com/mdickinson/bigfloat/issues>`_ to report issues.
Alternatively, you can contact Mark Dickinson directly at [email protected]
with suggestions, complaints, bug reports, etc.
License
-------
The bigfloat package is copyright (C) 2009--2019 Mark Dickinson
The bigfloat package is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
The bigfloat package is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
for more details.
You should have received a copy of the GNU Lesser General Public License
along with the bigfloat package. If not, see <http://www.gnu.org/licenses/>.
Links
-----
- `Documentation at Read the Docs <http://bigfloat.readthedocs.org>`_
- `Python package index <https://pypi.python.org/pypi/bigfloat>`_
- `Project homepage at GitHub <https://github.com/mdickinson/bigfloat>`_
- `Issue tracker <https://github.com/mdickinson/bigfloat/issues>`_
"""
# During package development we want to use Cython, but the distributed
# egg shouldn't use Cython. We use the presence of PKG-INFO to determine
# which of these is true.
if os.path.exists("PKG-INFO"):
USE_CYTHON = False
else:
try:
from Cython.Build import cythonize
USE_CYTHON = True
except:
print("Failed to import Cython, using pre-generated .c version")
USE_CYTHON = False
if USE_CYTHON:
try:
extensions = cythonize(
Extension("mpfr", ["mpfr.pyx"], libraries=["mpfr", "gmp"],)
)
except:
print("Failed Cythonize, fallback to pre-generated .c version")
USE_CYTHON = False
if not USE_CYTHON:
extensions = [
Extension("mpfr", ["mpfr.c"], libraries=["mpfr", "gmp"],),
]
CLASSIFIERS = """\
Development Status :: 4 - Beta
Intended Audience :: Developers
Intended Audience :: Education
Intended Audience :: Science/Research
License :: OSI Approved :: GNU Lesser General Public License v3 (LGPLv3)
Operating System :: OS Independent
Programming Language :: Python :: 2
Programming Language :: Python :: 2.7
Programming Language :: Python :: 3
Programming Language :: Python :: 3.5
Programming Language :: Python :: 3.6
Programming Language :: Python :: 3.7
Programming Language :: Python :: 3.8
Programming Language :: Python :: Implementation :: CPython
Topic :: Scientific/Engineering :: Mathematics
""".splitlines()
def get_version_info():
"""Extract version information as a dictionary from version.py."""
version_info = {}
with open(os.path.join("bigfloat", "version.py"), "r") as f:
version_code = compile(f.read(), "version.py", "exec")
exec(version_code, version_info)
return version_info
version_info = get_version_info()
setup(
name="bigfloat",
version=version_info["release"],
description=DESCRIPTION,
long_description=LONG_DESCRIPTION,
long_description_content_type="text/x-rst",
install_requires=["six"],
author="Mark Dickinson",
author_email="[email protected]",
url="http://github.com/mdickinson/bigfloat",
classifiers=CLASSIFIERS,
platforms=["Linux", "OS X"],
license="GNU Library or Lesser General Public License (LGPL)",
ext_modules=extensions,
packages=["bigfloat", "bigfloat.test"],
package_data={"bigfloat.test": ["test_data/*.bft"]},
)