diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..4aa9c95 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +SOURCES/numpy-1.14.2.tar.gz +SOURCES/numpy-html-1.13.0.zip diff --git a/.numpy.metadata b/.numpy.metadata new file mode 100644 index 0000000..2aa818e --- /dev/null +++ b/.numpy.metadata @@ -0,0 +1,2 @@ +c00e70468703830a26ee9173ba1cf4aedf08718f SOURCES/numpy-1.14.2.tar.gz +b23d66880bba5f56baa81ce02eb5a55de046c0a7 SOURCES/numpy-html-1.13.0.zip diff --git a/SOURCES/numpy-1.14.2-CVE-2019-6446.patch b/SOURCES/numpy-1.14.2-CVE-2019-6446.patch new file mode 100644 index 0000000..95ee13c --- /dev/null +++ b/SOURCES/numpy-1.14.2-CVE-2019-6446.patch @@ -0,0 +1,166 @@ +From 0fcfa065d900040c80628b31b8b6ea606c131086 Mon Sep 17 00:00:00 2001 +From: Paul Ivanov +Date: Wed, 30 Jan 2019 14:22:44 -0800 +Subject: [PATCH] BUG: load fails when using pickle without allow_pickle=True + +a partial mitigation of #12759. + +see also https://nvd.nist.gov/vuln/detail/CVE-2019-6446 +--- + numpy/core/tests/test_regression.py | 2 +- + numpy/lib/format.py | 8 ++++++-- + numpy/lib/npyio.py | 17 ++++++++++++----- + numpy/lib/tests/test_format.py | 15 +++++++++------ + numpy/lib/tests/test_io.py | 2 +- + 5 files changed, 29 insertions(+), 15 deletions(-) + +diff --git a/numpy/core/tests/test_regression.py b/numpy/core/tests/test_regression.py +index a3b0114..2be6bf3 100644 +--- a/numpy/core/tests/test_regression.py ++++ b/numpy/core/tests/test_regression.py +@@ -96,7 +96,7 @@ class TestRegression(object): + ca = np.char.array(np.arange(1000, 1010), itemsize=4) + ca.dump(f) + f.seek(0) +- ca = np.load(f) ++ ca = np.load(f, allow_pickle=True) + f.close() + + def test_noncontiguous_fill(self): +diff --git a/numpy/lib/format.py b/numpy/lib/format.py +index 363bb21..b91142c 100644 +--- a/numpy/lib/format.py ++++ b/numpy/lib/format.py +@@ -602,7 +602,7 @@ def write_array(fp, array, version=None, allow_pickle=True, pickle_kwargs=None): + fp.write(chunk.tobytes('C')) + + +-def read_array(fp, allow_pickle=True, pickle_kwargs=None): ++def read_array(fp, allow_pickle=False, pickle_kwargs=None): + """ + Read an array from an NPY file. + +@@ -612,7 +612,11 @@ def read_array(fp, allow_pickle=True, pickle_kwargs=None): + If this is not a real file object, then this may take extra memory + and time. + allow_pickle : bool, optional +- Whether to allow reading pickled data. Default: True ++ Whether to allow writing pickled data. Default: False ++ ++ .. versionchanged:: 1.14.2 ++ Made default False in response to CVE-2019-6446. ++ + pickle_kwargs : dict + Additional keyword arguments to pass to pickle.load. These are only + useful when loading object arrays saved on Python 2 when using +diff --git a/numpy/lib/npyio.py b/numpy/lib/npyio.py +index 76b135c..c6522f5 100644 +--- a/numpy/lib/npyio.py ++++ b/numpy/lib/npyio.py +@@ -130,7 +130,11 @@ class NpzFile(object): + An object on which attribute can be performed as an alternative + to getitem access on the `NpzFile` instance itself. + allow_pickle : bool, optional +- Allow loading pickled data. Default: True ++ Allow loading pickled data. Default: False ++ ++ .. versionchanged:: 1.14.2 ++ Made default False in response to CVE-2019-6446. ++ + pickle_kwargs : dict, optional + Additional keyword arguments to pass on to pickle.load. + These are only useful when loading object arrays saved on +@@ -166,7 +170,7 @@ class NpzFile(object): + + """ + +- def __init__(self, fid, own_fid=False, allow_pickle=True, ++ def __init__(self, fid, own_fid=False, allow_pickle=False, + pickle_kwargs=None): + # Import is postponed to here since zipfile depends on gzip, an + # optional component of the so-called standard library. +@@ -265,7 +269,7 @@ class NpzFile(object): + return self.files.__contains__(key) + + +-def load(file, mmap_mode=None, allow_pickle=True, fix_imports=True, ++def load(file, mmap_mode=None, allow_pickle=False, fix_imports=True, + encoding='ASCII'): + """ + Load arrays or pickled objects from ``.npy``, ``.npz`` or pickled files. +@@ -287,8 +291,11 @@ def load(file, mmap_mode=None, allow_pickle=True, fix_imports=True, + Allow loading pickled object arrays stored in npy files. Reasons for + disallowing pickles include security, as loading pickled data can + execute arbitrary code. If pickles are disallowed, loading object +- arrays will fail. +- Default: True ++ arrays will fail. Default: False ++ ++ .. versionchanged:: 1.14.2 ++ Made default False in response to CVE-2019-6446. ++ + fix_imports : bool, optional + Only useful when loading Python 2 generated pickled files on Python 3, + which includes npy/npz files containing object arrays. If `fix_imports` +diff --git a/numpy/lib/tests/test_format.py b/numpy/lib/tests/test_format.py +index 2d2b4ce..04e090c 100644 +--- a/numpy/lib/tests/test_format.py ++++ b/numpy/lib/tests/test_format.py +@@ -426,7 +426,7 @@ def roundtrip(arr): + f = BytesIO() + format.write_array(f, arr) + f2 = BytesIO(f.getvalue()) +- arr2 = format.read_array(f2) ++ arr2 = format.read_array(f2, allow_pickle=True) + return arr2 + + +@@ -553,7 +553,7 @@ def test_pickle_python2_python3(): + path = os.path.join(data_dir, fname) + + for encoding in ['bytes', 'latin1']: +- data_f = np.load(path, encoding=encoding) ++ data_f = np.load(path, allow_pickle=True, encoding=encoding) + if fname.endswith('.npz'): + data = data_f['x'] + data_f.close() +@@ -575,16 +575,19 @@ def test_pickle_python2_python3(): + if sys.version_info[0] >= 3: + if fname.startswith('py2'): + if fname.endswith('.npz'): +- data = np.load(path) ++ data = np.load(path, allow_pickle=True) + assert_raises(UnicodeError, data.__getitem__, 'x') + data.close() +- data = np.load(path, fix_imports=False, encoding='latin1') ++ data = np.load(path, allow_pickle=True, fix_imports=False, ++ encoding='latin1') + assert_raises(ImportError, data.__getitem__, 'x') + data.close() + else: +- assert_raises(UnicodeError, np.load, path) ++ assert_raises(UnicodeError, np.load, path, ++ allow_pickle=True) + assert_raises(ImportError, np.load, path, +- encoding='latin1', fix_imports=False) ++ allow_pickle=True, fix_imports=False, ++ encoding='latin1') + + + def test_pickle_disallow(): +diff --git a/numpy/lib/tests/test_io.py b/numpy/lib/tests/test_io.py +index 2daa015..bde2567 100644 +--- a/numpy/lib/tests/test_io.py ++++ b/numpy/lib/tests/test_io.py +@@ -87,7 +87,7 @@ class RoundtripTest(object): + + """ + save_kwds = kwargs.get('save_kwds', {}) +- load_kwds = kwargs.get('load_kwds', {}) ++ load_kwds = kwargs.get('load_kwds', {"allow_pickle": True}) + file_on_disk = kwargs.get('file_on_disk', False) + + if file_on_disk: +-- +2.21.0 + diff --git a/SOURCES/numpy-1.14.2-float128.patch b/SOURCES/numpy-1.14.2-float128.patch new file mode 100644 index 0000000..26635d7 --- /dev/null +++ b/SOURCES/numpy-1.14.2-float128.patch @@ -0,0 +1,2867 @@ +From 85ed88218164ceecb360bd9b8539e55076df9d19 Mon Sep 17 00:00:00 2001 +From: =?UTF-8?q?Nikola=20Forr=C3=B3?= +Date: Thu, 30 May 2019 15:52:23 +0200 +Subject: [PATCH] Fix broken float128 on all arches except x86_64 + +Backport of the following upstream commits: + +3d21112d6 MAINT: Add bitmask helper functions +4d7c529d7 MAINT: Reorganize Dragon4 code to clarify float formats +727756e63 MAINT: replace static BigInts in dragon4 by dummy manager +888d37086 MAINT: Add comments to long_double detection code +cdc6b6847 BUG: Implement float128 dragon4 for IBM double-double (ppc64) +718d9e866 ENH: Add support for the 64-bit RISC-V architecture +b35dfc2d5 ENH: Add AARCH32 support. +3973b2508 Adjust endianness header file to accommodate AARCHXX changes. +f07359b22 BLD: Modify cpu detection and function to get build working on aarch64 (#11568) +dd949d5c6 BUG: Fix printing of longdouble on ppc64le. +641058fb7 MAINT: remove darwin hardcoded LDOUBLE detection +ec843c700 BUG: Fix undefined functions on big-endian systems. +00bc5bc88 BUG: Fix test sensitive to platform byte order. +--- + numpy/core/include/numpy/npy_cpu.h | 30 +- + numpy/core/include/numpy/npy_endian.h | 44 +- + numpy/core/setup.py | 13 +- + numpy/core/setup_common.py | 18 +- + numpy/core/src/multiarray/dragon4.c | 1613 ++++++++++++------- + numpy/core/src/multiarray/dragon4.h | 66 +- + numpy/core/src/multiarray/scalartypes.c.src | 5 +- + numpy/core/src/npymath/ieee754.c.src | 3 +- + numpy/core/src/npymath/npy_math_private.h | 9 +- + numpy/core/src/private/npy_config.h | 3 +- + numpy/core/src/private/npy_fpmath.h | 41 +- + numpy/core/tests/test_scalarprint.py | 63 +- + numpy/core/tests/test_umath.py | 12 +- + numpy/ma/tests/test_core.py | 11 +- + 14 files changed, 1277 insertions(+), 654 deletions(-) + +diff --git a/numpy/core/include/numpy/npy_cpu.h b/numpy/core/include/numpy/npy_cpu.h +index 84653ea..1109426 100644 +--- a/numpy/core/include/numpy/npy_cpu.h ++++ b/numpy/core/include/numpy/npy_cpu.h +@@ -17,6 +17,7 @@ + * NPY_CPU_SH_BE + * NPY_CPU_ARCEL + * NPY_CPU_ARCEB ++ * NPY_CPU_RISCV64 + */ + #ifndef _NPY_CPUARCH_H_ + #define _NPY_CPUARCH_H_ +@@ -60,10 +61,27 @@ + #define NPY_CPU_HPPA + #elif defined(__alpha__) + #define NPY_CPU_ALPHA +-#elif defined(__arm__) && defined(__ARMEL__) +- #define NPY_CPU_ARMEL +-#elif defined(__arm__) && defined(__ARMEB__) +- #define NPY_CPU_ARMEB ++#elif defined(__arm__) || defined(__aarch64__) ++ #if defined(__ARMEB__) || defined(__AARCH64EB__) ++ #if defined(__ARM_32BIT_STATE) ++ #define NPY_CPU_ARMEB_AARCH32 ++ #elif defined(__ARM_64BIT_STATE) ++ #define NPY_CPU_ARMEB_AARCH64 ++ #else ++ #define NPY_CPU_ARMEB ++ #endif ++ #elif defined(__ARMEL__) || defined(__AARCH64EL__) ++ #if defined(__ARM_32BIT_STATE) ++ #define NPY_CPU_ARMEL_AARCH32 ++ #elif defined(__ARM_64BIT_STATE) ++ #define NPY_CPU_ARMEL_AARCH64 ++ #else ++ #define NPY_CPU_ARMEL ++ #endif ++ #else ++ # error Unknown ARM CPU, please report this to numpy maintainers with \ ++ information about your platform (OS, CPU and compiler) ++ #endif + #elif defined(__sh__) && defined(__LITTLE_ENDIAN__) + #define NPY_CPU_SH_LE + #elif defined(__sh__) && defined(__BIG_ENDIAN__) +@@ -74,14 +92,14 @@ + #define NPY_CPU_MIPSEB + #elif defined(__or1k__) + #define NPY_CPU_OR1K +-#elif defined(__aarch64__) +- #define NPY_CPU_AARCH64 + #elif defined(__mc68000__) + #define NPY_CPU_M68K + #elif defined(__arc__) && defined(__LITTLE_ENDIAN__) + #define NPY_CPU_ARCEL + #elif defined(__arc__) && defined(__BIG_ENDIAN__) + #define NPY_CPU_ARCEB ++#elif defined(__riscv) && defined(__riscv_xlen) && __riscv_xlen == 64 ++ #define NPY_CPU_RISCV64 + #else + #error Unknown CPU, please report this to numpy maintainers with \ + information about your platform (OS, CPU and compiler) +diff --git a/numpy/core/include/numpy/npy_endian.h b/numpy/core/include/numpy/npy_endian.h +index 1a42121..44cdffd 100644 +--- a/numpy/core/include/numpy/npy_endian.h ++++ b/numpy/core/include/numpy/npy_endian.h +@@ -37,27 +37,31 @@ + #define NPY_LITTLE_ENDIAN 1234 + #define NPY_BIG_ENDIAN 4321 + +- #if defined(NPY_CPU_X86) \ +- || defined(NPY_CPU_AMD64) \ +- || defined(NPY_CPU_IA64) \ +- || defined(NPY_CPU_ALPHA) \ +- || defined(NPY_CPU_ARMEL) \ +- || defined(NPY_CPU_AARCH64) \ +- || defined(NPY_CPU_SH_LE) \ +- || defined(NPY_CPU_MIPSEL) \ +- || defined(NPY_CPU_PPC64LE) \ +- || defined(NPY_CPU_ARCEL) ++ #if defined(NPY_CPU_X86) \ ++ || defined(NPY_CPU_AMD64) \ ++ || defined(NPY_CPU_IA64) \ ++ || defined(NPY_CPU_ALPHA) \ ++ || defined(NPY_CPU_ARMEL) \ ++ || defined(NPY_CPU_ARMEL_AARCH32) \ ++ || defined(NPY_CPU_ARMEL_AARCH64) \ ++ || defined(NPY_CPU_SH_LE) \ ++ || defined(NPY_CPU_MIPSEL) \ ++ || defined(NPY_CPU_PPC64LE) \ ++ || defined(NPY_CPU_ARCEL) \ ++ || defined(NPY_CPU_RISCV64) + #define NPY_BYTE_ORDER NPY_LITTLE_ENDIAN +- #elif defined(NPY_CPU_PPC) \ +- || defined(NPY_CPU_SPARC) \ +- || defined(NPY_CPU_S390) \ +- || defined(NPY_CPU_HPPA) \ +- || defined(NPY_CPU_PPC64) \ +- || defined(NPY_CPU_ARMEB) \ +- || defined(NPY_CPU_SH_BE) \ +- || defined(NPY_CPU_MIPSEB) \ +- || defined(NPY_CPU_OR1K) \ +- || defined(NPY_CPU_M68K) \ ++ #elif defined(NPY_CPU_PPC) \ ++ || defined(NPY_CPU_SPARC) \ ++ || defined(NPY_CPU_S390) \ ++ || defined(NPY_CPU_HPPA) \ ++ || defined(NPY_CPU_PPC64) \ ++ || defined(NPY_CPU_ARMEB) \ ++ || defined(NPY_CPU_ARMEB_AARCH32) \ ++ || defined(NPY_CPU_ARMEB_AARCH64) \ ++ || defined(NPY_CPU_SH_BE) \ ++ || defined(NPY_CPU_MIPSEB) \ ++ || defined(NPY_CPU_OR1K) \ ++ || defined(NPY_CPU_M68K) \ + || defined(NPY_CPU_ARCEB) + #define NPY_BYTE_ORDER NPY_BIG_ENDIAN + #else +diff --git a/numpy/core/setup.py b/numpy/core/setup.py +index 371df5b..cc1bb5b 100644 +--- a/numpy/core/setup.py ++++ b/numpy/core/setup.py +@@ -452,17 +452,8 @@ def configuration(parent_package='',top_path=None): + moredefs.append(('NPY_RELAXED_STRIDES_DEBUG', 1)) + + # Get long double representation +- if sys.platform != 'darwin': +- rep = check_long_double_representation(config_cmd) +- if rep in ['INTEL_EXTENDED_12_BYTES_LE', +- 'INTEL_EXTENDED_16_BYTES_LE', +- 'MOTOROLA_EXTENDED_12_BYTES_BE', +- 'IEEE_QUAD_LE', 'IEEE_QUAD_BE', +- 'IEEE_DOUBLE_LE', 'IEEE_DOUBLE_BE', +- 'DOUBLE_DOUBLE_BE', 'DOUBLE_DOUBLE_LE']: +- moredefs.append(('HAVE_LDOUBLE_%s' % rep, 1)) +- else: +- raise ValueError("Unrecognized long double format: %s" % rep) ++ rep = check_long_double_representation(config_cmd) ++ moredefs.append(('HAVE_LDOUBLE_%s' % rep, 1)) + + # Py3K check + if sys.version_info[0] == 3: +diff --git a/numpy/core/setup_common.py b/numpy/core/setup_common.py +index bd093c5..343318c 100644 +--- a/numpy/core/setup_common.py ++++ b/numpy/core/setup_common.py +@@ -333,9 +333,9 @@ _MOTOROLA_EXTENDED_12B = ['300', '031', '000', '000', '353', '171', + _IEEE_QUAD_PREC_BE = ['300', '031', '326', '363', '105', '100', '000', '000', + '000', '000', '000', '000', '000', '000', '000', '000'] + _IEEE_QUAD_PREC_LE = _IEEE_QUAD_PREC_BE[::-1] +-_DOUBLE_DOUBLE_BE = (['301', '235', '157', '064', '124', '000', '000', '000'] + ++_IBM_DOUBLE_DOUBLE_BE = (['301', '235', '157', '064', '124', '000', '000', '000'] + + ['000'] * 8) +-_DOUBLE_DOUBLE_LE = (['000', '000', '000', '124', '064', '157', '235', '301'] + ++_IBM_DOUBLE_DOUBLE_LE = (['000', '000', '000', '124', '064', '157', '235', '301'] + + ['000'] * 8) + + def long_double_representation(lines): +@@ -362,11 +362,16 @@ def long_double_representation(lines): + # the long double + if read[-8:] == _AFTER_SEQ: + saw = copy.copy(read) ++ # if the content was 12 bytes, we only have 32 - 8 - 12 = 12 ++ # "before" bytes. In other words the first 4 "before" bytes went ++ # past the sliding window. + if read[:12] == _BEFORE_SEQ[4:]: + if read[12:-8] == _INTEL_EXTENDED_12B: + return 'INTEL_EXTENDED_12_BYTES_LE' + if read[12:-8] == _MOTOROLA_EXTENDED_12B: + return 'MOTOROLA_EXTENDED_12_BYTES_BE' ++ # if the content was 16 bytes, we are left with 32-8-16 = 16 ++ # "before" bytes, so 8 went past the sliding window. + elif read[:8] == _BEFORE_SEQ[8:]: + if read[8:-8] == _INTEL_EXTENDED_16B: + return 'INTEL_EXTENDED_16_BYTES_LE' +@@ -374,10 +379,11 @@ def long_double_representation(lines): + return 'IEEE_QUAD_BE' + elif read[8:-8] == _IEEE_QUAD_PREC_LE: + return 'IEEE_QUAD_LE' +- elif read[8:-8] == _DOUBLE_DOUBLE_BE: +- return 'DOUBLE_DOUBLE_BE' +- elif read[8:-8] == _DOUBLE_DOUBLE_LE: +- return 'DOUBLE_DOUBLE_LE' ++ elif read[8:-8] == _IBM_DOUBLE_DOUBLE_LE: ++ return 'IBM_DOUBLE_DOUBLE_LE' ++ elif read[8:-8] == _IBM_DOUBLE_DOUBLE_BE: ++ return 'IBM_DOUBLE_DOUBLE_BE' ++ # if the content was 8 bytes, left with 32-8-8 = 16 bytes + elif read[:16] == _BEFORE_SEQ: + if read[16:-8] == _IEEE_DOUBLE_LE: + return 'IEEE_DOUBLE_LE' +diff --git a/numpy/core/src/multiarray/dragon4.c b/numpy/core/src/multiarray/dragon4.c +index e005234..e3da79f 100644 +--- a/numpy/core/src/multiarray/dragon4.c ++++ b/numpy/core/src/multiarray/dragon4.c +@@ -42,6 +42,18 @@ + #define DEBUG_ASSERT(stmnt) do {} while(0) + #endif + ++static inline npy_uint64 ++bitmask_u64(npy_uint32 n) ++{ ++ return ~(~((npy_uint64)0) << n); ++} ++ ++static inline npy_uint32 ++bitmask_u32(npy_uint32 n) ++{ ++ return ~(~((npy_uint32)0) << n); ++} ++ + /* + * Get the log base 2 of a 32-bit unsigned integer. + * http://graphics.stanford.edu/~seander/bithacks.html#IntegerLogLookup +@@ -102,6 +114,17 @@ LogBase2_64(npy_uint64 val) + return LogBase2_32((npy_uint32)val); + } + ++#if defined(HAVE_LDOUBLE_IEEE_QUAD_LE) || defined(HAVE_LDOUBLE_IEEE_QUAD_BE) ++static npy_uint32 ++LogBase2_128(npy_uint64 hi, npy_uint64 lo) ++{ ++ if (hi) { ++ return 64 + LogBase2_64(hi); ++ } ++ ++ return LogBase2_64(lo); ++} ++#endif /* HAVE_LDOUBLE_IEEE_QUAD_LE */ + + /* + * Maximum number of 32 bit blocks needed in high precision arithmetic to print +@@ -122,6 +145,45 @@ typedef struct BigInt { + npy_uint32 blocks[c_BigInt_MaxBlocks]; + } BigInt; + ++/* ++ * Dummy implementation of a memory manager for BigInts. Currently, only ++ * supports a single call to Dragon4, but that is OK because Dragon4 ++ * does not release the GIL. ++ * ++ * We try to raise an error anyway if dragon4 re-enters, and this code serves ++ * as a placeholder if we want to make it re-entrant in the future. ++ * ++ * Each call to dragon4 uses 7 BigInts. ++ */ ++#define BIGINT_DRAGON4_GROUPSIZE 7 ++typedef struct { ++ BigInt bigints[BIGINT_DRAGON4_GROUPSIZE]; ++ char repr[16384]; ++} Dragon4_Scratch; ++ ++static int _bigint_static_in_use = 0; ++static Dragon4_Scratch _bigint_static; ++ ++static Dragon4_Scratch* ++get_dragon4_bigint_scratch(void) { ++ /* this test+set is not threadsafe, but no matter because we have GIL */ ++ if (_bigint_static_in_use) { ++ PyErr_SetString(PyExc_RuntimeError, ++ "numpy float printing code is not re-entrant. " ++ "Ping the devs to fix it."); ++ return NULL; ++ } ++ _bigint_static_in_use = 1; ++ ++ /* in this dummy implementation we only return the static allocation */ ++ return &_bigint_static; ++} ++ ++static void ++free_dragon4_bigint_scratch(Dragon4_Scratch *mem){ ++ _bigint_static_in_use = 0; ++} ++ + /* Copy integer */ + static void + BigInt_Copy(BigInt *dst, const BigInt *src) +@@ -139,32 +201,86 @@ BigInt_Copy(BigInt *dst, const BigInt *src) + static void + BigInt_Set_uint64(BigInt *i, npy_uint64 val) + { +- if (val > 0xFFFFFFFF) { +- i->blocks[0] = val & 0xFFFFFFFF; +- i->blocks[1] = (val >> 32) & 0xFFFFFFFF; ++ if (val > bitmask_u64(32)) { ++ i->blocks[0] = val & bitmask_u64(32); ++ i->blocks[1] = (val >> 32) & bitmask_u64(32); + i->length = 2; + } + else if (val != 0) { +- i->blocks[0] = val & 0xFFFFFFFF; ++ i->blocks[0] = val & bitmask_u64(32); ++ i->length = 1; ++ } ++ else { ++ i->length = 0; ++ } ++} ++ ++#if (defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE) || \ ++ defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE) || \ ++ defined(HAVE_LDOUBLE_IEEE_QUAD_LE) || \ ++ defined(HAVE_LDOUBLE_IEEE_QUAD_BE)) ++static void ++BigInt_Set_2x_uint64(BigInt *i, npy_uint64 hi, npy_uint64 lo) ++{ ++ if (hi > bitmask_u64(32)) { ++ i->length = 4; ++ } ++ else if (hi != 0) { ++ i->length = 3; ++ } ++ else if (lo > bitmask_u64(32)) { ++ i->length = 2; ++ } ++ else if (lo != 0) { + i->length = 1; + } + else { + i->length = 0; + } ++ ++ switch (i->length) { ++ case 4: ++ i->blocks[3] = (hi >> 32) & bitmask_u64(32); ++ case 3: ++ i->blocks[2] = hi & bitmask_u64(32); ++ case 2: ++ i->blocks[1] = (lo >> 32) & bitmask_u64(32); ++ case 1: ++ i->blocks[0] = lo & bitmask_u64(32); ++ } + } ++#endif /* DOUBLE_DOUBLE and QUAD */ + + static void + BigInt_Set_uint32(BigInt *i, npy_uint32 val) + { + if (val != 0) { + i->blocks[0] = val; +- i->length = (val != 0); ++ i->length = 1; + } + else { + i->length = 0; + } + } + ++/* ++ * Returns 1 if the value is zero ++ */ ++static int ++BigInt_IsZero(const BigInt *i) ++{ ++ return i->length == 0; ++} ++ ++/* ++ * Returns 1 if the value is even ++ */ ++static int ++BigInt_IsEven(const BigInt *i) ++{ ++ return (i->length == 0) || ( (i->blocks[0] % 2) == 0); ++} ++ + /* + * Returns 0 if (lhs = rhs), negative if (lhs < rhs), positive if (lhs > rhs) + */ +@@ -228,7 +344,7 @@ BigInt_Add(BigInt *result, const BigInt *lhs, const BigInt *rhs) + npy_uint64 sum = carry + (npy_uint64)(*largeCur) + + (npy_uint64)(*smallCur); + carry = sum >> 32; +- *resultCur = sum & 0xFFFFFFFF; ++ *resultCur = sum & bitmask_u64(32); + ++largeCur; + ++smallCur; + ++resultCur; +@@ -238,7 +354,7 @@ BigInt_Add(BigInt *result, const BigInt *lhs, const BigInt *rhs) + while (largeCur != largeEnd) { + npy_uint64 sum = carry + (npy_uint64)(*largeCur); + carry = sum >> 32; +- (*resultCur) = sum & 0xFFFFFFFF; ++ (*resultCur) = sum & bitmask_u64(32); + ++largeCur; + ++resultCur; + } +@@ -307,13 +423,13 @@ BigInt_Multiply(BigInt *result, const BigInt *lhs, const BigInt *rhs) + npy_uint64 product = (*resultCur) + + (*largeCur)*(npy_uint64)multiplier + carry; + carry = product >> 32; +- *resultCur = product & 0xFFFFFFFF; ++ *resultCur = product & bitmask_u64(32); + ++largeCur; + ++resultCur; + } while(largeCur != large->blocks + large->length); + + DEBUG_ASSERT(resultCur < result->blocks + maxResultLen); +- *resultCur = (npy_uint32)(carry & 0xFFFFFFFF); ++ *resultCur = (npy_uint32)(carry & bitmask_u64(32)); + } + } + +@@ -337,7 +453,7 @@ BigInt_Multiply_int(BigInt *result, const BigInt *lhs, npy_uint32 rhs) + const npy_uint32 *pLhsEnd = lhs->blocks + lhs->length; + for ( ; pLhsCur != pLhsEnd; ++pLhsCur, ++resultCur) { + npy_uint64 product = (npy_uint64)(*pLhsCur) * rhs + carry; +- *resultCur = (npy_uint32)(product & 0xFFFFFFFF); ++ *resultCur = (npy_uint32)(product & bitmask_u64(32)); + carry = product >> 32; + } + +@@ -414,7 +530,7 @@ BigInt_Multiply10(BigInt *result) + npy_uint32 *end = result->blocks + result->length; + for ( ; cur != end; ++cur) { + npy_uint64 product = (npy_uint64)(*cur) * 10ull + carry; +- (*cur) = (npy_uint32)(product & 0xFFFFFFFF); ++ (*cur) = (npy_uint32)(product & bitmask_u64(32)); + carry = product >> 32; + } + +@@ -637,13 +753,11 @@ static BigInt g_PowerOf10_Big[] = + + /* result = 10^exponent */ + static void +-BigInt_Pow10(BigInt *result, npy_uint32 exponent) ++BigInt_Pow10(BigInt *result, npy_uint32 exponent, BigInt *temp) + { +- /* create two temporary values to reduce large integer copy operations */ +- BigInt temp1; +- BigInt temp2; +- BigInt *curTemp = &temp1; +- BigInt *pNextTemp = &temp2; ++ /* use two temporary values to reduce large integer copy operations */ ++ BigInt *curTemp = result; ++ BigInt *pNextTemp = temp; + npy_uint32 smallExponent; + npy_uint32 tableIdx = 0; + +@@ -654,7 +768,7 @@ BigInt_Pow10(BigInt *result, npy_uint32 exponent) + * initialize the result by looking up a 32-bit power of 10 corresponding to + * the first 3 bits + */ +- smallExponent = exponent & 0x7; ++ smallExponent = exponent & bitmask_u32(3); + BigInt_Set_uint32(curTemp, g_PowerOf10_U32[smallExponent]); + + /* remove the low bits that we used for the 32-bit lookup table */ +@@ -681,19 +795,17 @@ BigInt_Pow10(BigInt *result, npy_uint32 exponent) + } + + /* output the result */ +- BigInt_Copy(result, curTemp); ++ if (curTemp != result) { ++ BigInt_Copy(result, curTemp); ++ } + } + +-/* result = in * 10^exponent */ ++/* in = in * 10^exponent */ + static void +-BigInt_MultiplyPow10(BigInt *result, const BigInt *in, npy_uint32 exponent) ++BigInt_MultiplyPow10(BigInt *in, npy_uint32 exponent, BigInt *temp) + { +- +- /* create two temporary values to reduce large integer copy operations */ +- BigInt temp1; +- BigInt temp2; +- BigInt *curTemp = &temp1; +- BigInt *pNextTemp = &temp2; ++ /* use two temporary values to reduce large integer copy operations */ ++ BigInt *curTemp, *pNextTemp; + npy_uint32 smallExponent; + npy_uint32 tableIdx = 0; + +@@ -704,12 +816,15 @@ BigInt_MultiplyPow10(BigInt *result, const BigInt *in, npy_uint32 exponent) + * initialize the result by looking up a 32-bit power of 10 corresponding to + * the first 3 bits + */ +- smallExponent = exponent & 0x7; ++ smallExponent = exponent & bitmask_u32(3); + if (smallExponent != 0) { +- BigInt_Multiply_int(curTemp, in, g_PowerOf10_U32[smallExponent]); ++ BigInt_Multiply_int(temp, in, g_PowerOf10_U32[smallExponent]); ++ curTemp = temp; ++ pNextTemp = in; + } + else { +- BigInt_Copy(curTemp, in); ++ curTemp = in; ++ pNextTemp = temp; + } + + /* remove the low bits that we used for the 32-bit lookup table */ +@@ -724,7 +839,7 @@ BigInt_MultiplyPow10(BigInt *result, const BigInt *in, npy_uint32 exponent) + /* multiply into the next temporary */ + BigInt_Multiply(pNextTemp, curTemp, &g_PowerOf10_Big[tableIdx]); + +- // swap to the next temporary ++ /* swap to the next temporary */ + pSwap = curTemp; + curTemp = pNextTemp; + pNextTemp = pSwap; +@@ -736,7 +851,9 @@ BigInt_MultiplyPow10(BigInt *result, const BigInt *in, npy_uint32 exponent) + } + + /* output the result */ +- BigInt_Copy(result, curTemp); ++ if (curTemp != in){ ++ BigInt_Copy(in, curTemp); ++ } + } + + /* result = 2^exponent */ +@@ -788,7 +905,7 @@ BigInt_DivideWithRemainder_MaxQuotient9(BigInt *dividend, const BigInt *divisor) + */ + DEBUG_ASSERT(!divisor->length == 0 && + divisor->blocks[divisor->length-1] >= 8 && +- divisor->blocks[divisor->length-1] < 0xFFFFFFFF && ++ divisor->blocks[divisor->length-1] < bitmask_u64(32) && + dividend->length <= divisor->length); + + /* +@@ -825,10 +942,10 @@ BigInt_DivideWithRemainder_MaxQuotient9(BigInt *dividend, const BigInt *divisor) + carry = product >> 32; + + difference = (npy_uint64)*dividendCur +- - (product & 0xFFFFFFFF) - borrow; ++ - (product & bitmask_u64(32)) - borrow; + borrow = (difference >> 32) & 1; + +- *dividendCur = difference & 0xFFFFFFFF; ++ *dividendCur = difference & bitmask_u64(32); + + ++divisorCur; + ++dividendCur; +@@ -860,7 +977,7 @@ BigInt_DivideWithRemainder_MaxQuotient9(BigInt *dividend, const BigInt *divisor) + - (npy_uint64)*divisorCur - borrow; + borrow = (difference >> 32) & 1; + +- *dividendCur = difference & 0xFFFFFFFF; ++ *dividendCur = difference & bitmask_u64(32); + + ++divisorCur; + ++dividendCur; +@@ -993,12 +1110,20 @@ BigInt_ShiftLeft(BigInt *result, npy_uint32 shift) + * There is some more documentation of these changes on Ryan Juckett's website + * at http://www.ryanjuckett.com/programming/printing-floating-point-numbers/ + * +- * Ryan Juckett's implementation did not implement "IEEE unbiased rounding", +- * except in the last digit. This has been added back, following the Burger & +- * Dybvig code, using the isEven variable. ++ * This code also has a few implementation differences from Ryan Juckett's ++ * version: ++ * 1. fixed overflow problems when mantissa was 64 bits (in float128 types), ++ * by replacing multiplication by 2 or 4 by BigInt_ShiftLeft calls. ++ * 2. Increased c_BigInt_MaxBlocks, for 128-bit floats ++ * 3. Added more entries to the g_PowerOf10_Big table, for 128-bit floats. ++ * 4. Added unbiased rounding calculation with isEven. Ryan Juckett's ++ * implementation did not implement "IEEE unbiased rounding", except in the ++ * last digit. This has been added back, following the Burger & Dybvig ++ * code, using the isEven variable. + * + * Arguments: +- * * mantissa - value significand ++ * * bigints - memory to store all bigints needed (7) for dragon4 computation. ++ * The first BigInt should be filled in with the mantissa. + * * exponent - value exponent in base 2 + * * mantissaBit - index of the highest set mantissa bit + * * hasUnequalMargins - is the high margin twice as large as the low margin +@@ -1007,9 +1132,11 @@ BigInt_ShiftLeft(BigInt *result, npy_uint32 shift) + * * pOutBuffer - buffer to output into + * * bufferSize - maximum characters that can be printed to pOutBuffer + * * pOutExponent - the base 10 exponent of the first digit ++ * ++ * Returns the number of digits written to the output buffer. + */ + static npy_uint32 +-Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, ++Dragon4(BigInt *bigints, const npy_int32 exponent, + const npy_uint32 mantissaBit, const npy_bool hasUnequalMargins, + const DigitMode digitMode, const CutoffMode cutoffMode, + npy_int32 cutoffNumber, char *pOutBuffer, +@@ -1025,21 +1152,24 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * Here, marginLow and marginHigh represent 1/2 of the distance to the next + * floating point value above/below the mantissa. + * +- * scaledMarginHigh is a pointer so that it can point to scaledMarginLow in +- * the case they must be equal to each other, otherwise it will point to +- * optionalMarginHigh. ++ * scaledMarginHigh will point to scaledMarginLow in the case they must be ++ * equal to each other, otherwise it will point to optionalMarginHigh. + */ +- BigInt scale; +- BigInt scaledValue; +- BigInt scaledMarginLow; ++ BigInt *mantissa = &bigints[0]; /* the only initialized bigint */ ++ BigInt *scale = &bigints[1]; ++ BigInt *scaledValue = &bigints[2]; ++ BigInt *scaledMarginLow = &bigints[3]; + BigInt *scaledMarginHigh; +- BigInt optionalMarginHigh; ++ BigInt *optionalMarginHigh = &bigints[4]; ++ ++ BigInt *temp1 = &bigints[5]; ++ BigInt *temp2 = &bigints[6]; + + const npy_float64 log10_2 = 0.30102999566398119521373889472449; + npy_int32 digitExponent, cutoffExponent, hiBlock; + npy_uint32 outputDigit; /* current digit being output */ + npy_uint32 outputLen; +- npy_bool isEven = (mantissa % 2) == 0; ++ npy_bool isEven = BigInt_IsEven(mantissa); + npy_int32 cmp; + + /* values used to determine how to round */ +@@ -1048,12 +1178,14 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + DEBUG_ASSERT(bufferSize > 0); + + /* if the mantissa is zero, the value is zero regardless of the exponent */ +- if (mantissa == 0) { ++ if (BigInt_IsZero(mantissa)) { + *curDigit = '0'; + *pOutExponent = 0; + return 1; + } + ++ BigInt_Copy(scaledValue, mantissa); ++ + if (hasUnequalMargins) { + /* if we have no fractional component */ + if (exponent > 0) { +@@ -1067,17 +1199,13 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + */ + + /* scaledValue = 2 * 2 * mantissa*2^exponent */ +- BigInt_Set_uint64(&scaledValue, mantissa); +- BigInt_ShiftLeft(&scaledValue, exponent + 2); +- ++ BigInt_ShiftLeft(scaledValue, exponent + 2); + /* scale = 2 * 2 * 1 */ +- BigInt_Set_uint32(&scale, 4); +- ++ BigInt_Set_uint32(scale, 4); + /* scaledMarginLow = 2 * 2^(exponent-1) */ +- BigInt_Pow2(&scaledMarginLow, exponent); +- ++ BigInt_Pow2(scaledMarginLow, exponent); + /* scaledMarginHigh = 2 * 2 * 2^(exponent-1) */ +- BigInt_Pow2(&optionalMarginHigh, exponent + 1); ++ BigInt_Pow2(optionalMarginHigh, exponent + 1); + } + /* else we have a fractional exponent */ + else { +@@ -1087,34 +1215,27 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + */ + + /* scaledValue = 2 * 2 * mantissa */ +- BigInt_Set_uint64(&scaledValue, mantissa); +- BigInt_ShiftLeft(&scaledValue, 2); +- ++ BigInt_ShiftLeft(scaledValue, 2); + /* scale = 2 * 2 * 2^(-exponent) */ +- BigInt_Pow2(&scale, -exponent + 2); +- ++ BigInt_Pow2(scale, -exponent + 2); + /* scaledMarginLow = 2 * 2^(-1) */ +- BigInt_Set_uint32(&scaledMarginLow, 1); +- ++ BigInt_Set_uint32(scaledMarginLow, 1); + /* scaledMarginHigh = 2 * 2 * 2^(-1) */ +- BigInt_Set_uint32(&optionalMarginHigh, 2); ++ BigInt_Set_uint32(optionalMarginHigh, 2); + } + + /* the high and low margins are different */ +- scaledMarginHigh = &optionalMarginHigh; ++ scaledMarginHigh = optionalMarginHigh; + } + else { + /* if we have no fractional component */ + if (exponent > 0) { + /* scaledValue = 2 * mantissa*2^exponent */ +- BigInt_Set_uint64(&scaledValue, mantissa); +- BigInt_ShiftLeft(&scaledValue, exponent + 1); +- ++ BigInt_ShiftLeft(scaledValue, exponent + 1); + /* scale = 2 * 1 */ +- BigInt_Set_uint32(&scale, 2); +- ++ BigInt_Set_uint32(scale, 2); + /* scaledMarginLow = 2 * 2^(exponent-1) */ +- BigInt_Pow2(&scaledMarginLow, exponent); ++ BigInt_Pow2(scaledMarginLow, exponent); + } + /* else we have a fractional exponent */ + else { +@@ -1124,18 +1245,15 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + */ + + /* scaledValue = 2 * mantissa */ +- BigInt_Set_uint64(&scaledValue, mantissa); +- BigInt_ShiftLeft(&scaledValue, 1); +- ++ BigInt_ShiftLeft(scaledValue, 1); + /* scale = 2 * 2^(-exponent) */ +- BigInt_Pow2(&scale, -exponent + 1); +- ++ BigInt_Pow2(scale, -exponent + 1); + /* scaledMarginLow = 2 * 2^(-1) */ +- BigInt_Set_uint32(&scaledMarginLow, 1); ++ BigInt_Set_uint32(scaledMarginLow, 1); + } + + /* the high and low margins are equal */ +- scaledMarginHigh = &scaledMarginLow; ++ scaledMarginHigh = scaledMarginLow; + } + + /* +@@ -1158,6 +1276,9 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * <= log10(v) + log10(2) + * floor(log10(v)) < ceil((mantissaBit + exponent) * log10(2)) + * <= floor(log10(v)) + 1 ++ * ++ * Warning: This calculation assumes npy_float64 is an IEEE-binary64 ++ * float. This line may need to be updated if this is not the case. + */ + digitExponent = (npy_int32)( + ceil((npy_float64)((npy_int32)mantissaBit + exponent) * log10_2 - 0.69)); +@@ -1179,31 +1300,29 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + /* Divide value by 10^digitExponent. */ + if (digitExponent > 0) { + /* A positive exponent creates a division so we multiply the scale. */ +- BigInt temp; +- BigInt_MultiplyPow10(&temp, &scale, digitExponent); +- BigInt_Copy(&scale, &temp); ++ BigInt_MultiplyPow10(scale, digitExponent, temp1); + } + else if (digitExponent < 0) { + /* + * A negative exponent creates a multiplication so we multiply up the + * scaledValue, scaledMarginLow and scaledMarginHigh. + */ +- BigInt pow10, temp; +- BigInt_Pow10(&pow10, -digitExponent); ++ BigInt *temp=temp1, *pow10=temp2; ++ BigInt_Pow10(pow10, -digitExponent, temp); + +- BigInt_Multiply(&temp, &scaledValue, &pow10); +- BigInt_Copy(&scaledValue, &temp); ++ BigInt_Multiply(temp, scaledValue, pow10); ++ BigInt_Copy(scaledValue, temp); + +- BigInt_Multiply(&temp, &scaledMarginLow, &pow10); +- BigInt_Copy(&scaledMarginLow, &temp); ++ BigInt_Multiply(temp, scaledMarginLow, pow10); ++ BigInt_Copy(scaledMarginLow, temp); + +- if (scaledMarginHigh != &scaledMarginLow) { +- BigInt_Multiply2(scaledMarginHigh, &scaledMarginLow); ++ if (scaledMarginHigh != scaledMarginLow) { ++ BigInt_Multiply2(scaledMarginHigh, scaledMarginLow); + } + } + + /* If (value >= 1), our estimate for digitExponent was too low */ +- if (BigInt_Compare(&scaledValue, &scale) >= 0) { ++ if (BigInt_Compare(scaledValue, scale) >= 0) { + /* + * The exponent estimate was incorrect. + * Increment the exponent and don't perform the premultiply needed +@@ -1217,10 +1336,10 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * Multiply larger by the output base to prepare for the first loop + * iteration. + */ +- BigInt_Multiply10(&scaledValue); +- BigInt_Multiply10(&scaledMarginLow); +- if (scaledMarginHigh != &scaledMarginLow) { +- BigInt_Multiply2(scaledMarginHigh, &scaledMarginLow); ++ BigInt_Multiply10(scaledValue); ++ BigInt_Multiply10(scaledMarginLow); ++ if (scaledMarginHigh != scaledMarginLow) { ++ BigInt_Multiply2(scaledMarginHigh, scaledMarginLow); + } + } + +@@ -1261,8 +1380,8 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * to be less than or equal to 429496729 which is the highest number that + * can be multiplied by 10 without overflowing to a new block. + */ +- DEBUG_ASSERT(scale.length > 0); +- hiBlock = scale.blocks[scale.length - 1]; ++ DEBUG_ASSERT(scale->length > 0); ++ hiBlock = scale->blocks[scale->length - 1]; + if (hiBlock < 8 || hiBlock > 429496729) { + npy_uint32 hiBlockLog2, shift; + +@@ -1280,11 +1399,11 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + DEBUG_ASSERT(hiBlockLog2 < 3 || hiBlockLog2 > 27); + shift = (32 + 27 - hiBlockLog2) % 32; + +- BigInt_ShiftLeft(&scale, shift); +- BigInt_ShiftLeft(&scaledValue, shift); +- BigInt_ShiftLeft(&scaledMarginLow, shift); +- if (scaledMarginHigh != &scaledMarginLow) { +- BigInt_Multiply2(scaledMarginHigh, &scaledMarginLow); ++ BigInt_ShiftLeft(scale, shift); ++ BigInt_ShiftLeft(scaledValue, shift); ++ BigInt_ShiftLeft(scaledMarginLow, shift); ++ if (scaledMarginHigh != scaledMarginLow) { ++ BigInt_Multiply2(scaledMarginHigh, scaledMarginLow); + } + } + +@@ -1296,25 +1415,25 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * terminate early. + */ + for (;;) { +- BigInt scaledValueHigh; ++ BigInt *scaledValueHigh = temp1; + + digitExponent = digitExponent-1; + + /* divide out the scale to extract the digit */ + outputDigit = +- BigInt_DivideWithRemainder_MaxQuotient9(&scaledValue, &scale); ++ BigInt_DivideWithRemainder_MaxQuotient9(scaledValue, scale); + DEBUG_ASSERT(outputDigit < 10); + + /* update the high end of the value */ +- BigInt_Add(&scaledValueHigh, &scaledValue, scaledMarginHigh); ++ BigInt_Add(scaledValueHigh, scaledValue, scaledMarginHigh); + + /* + * stop looping if we are far enough away from our neighboring + * values or if we have reached the cutoff digit + */ +- cmp = BigInt_Compare(&scaledValue, &scaledMarginLow); ++ cmp = BigInt_Compare(scaledValue, scaledMarginLow); + low = isEven ? (cmp <= 0) : (cmp < 0); +- cmp = BigInt_Compare(&scaledValueHigh, &scale); ++ cmp = BigInt_Compare(scaledValueHigh, scale); + high = isEven ? (cmp >= 0) : (cmp > 0); + if (low | high | (digitExponent == cutoffExponent)) + break; +@@ -1324,10 +1443,10 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + ++curDigit; + + /* multiply larger by the output base */ +- BigInt_Multiply10(&scaledValue); +- BigInt_Multiply10(&scaledMarginLow); +- if (scaledMarginHigh != &scaledMarginLow) { +- BigInt_Multiply2(scaledMarginHigh, &scaledMarginLow); ++ BigInt_Multiply10(scaledValue); ++ BigInt_Multiply10(scaledMarginLow); ++ if (scaledMarginHigh != scaledMarginLow) { ++ BigInt_Multiply2(scaledMarginHigh, scaledMarginLow); + } + } + } +@@ -1345,10 +1464,11 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + + /* divide out the scale to extract the digit */ + outputDigit = +- BigInt_DivideWithRemainder_MaxQuotient9(&scaledValue, &scale); ++ BigInt_DivideWithRemainder_MaxQuotient9(scaledValue, scale); + DEBUG_ASSERT(outputDigit < 10); + +- if ((scaledValue.length == 0) | (digitExponent == cutoffExponent)) { ++ if ((scaledValue->length == 0) | ++ (digitExponent == cutoffExponent)) { + break; + } + +@@ -1357,7 +1477,7 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + ++curDigit; + + /* multiply larger by the output base */ +- BigInt_Multiply10(&scaledValue); ++ BigInt_Multiply10(scaledValue); + } + } + +@@ -1375,8 +1495,8 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + * compare( scale * value, scale * 0.5 ) + * compare( 2 * scale * value, scale ) + */ +- BigInt_Multiply2_inplace(&scaledValue); +- compare = BigInt_Compare(&scaledValue, &scale); ++ BigInt_Multiply2_inplace(scaledValue); ++ compare = BigInt_Compare(scaledValue, scale); + roundDown = compare < 0; + + /* +@@ -1431,134 +1551,53 @@ Dragon4(const npy_uint64 mantissa, const npy_int32 exponent, + + + /* +- * Helper union to decompose a 16-bit IEEE float. +- * sign: 1 bit +- * exponent: 5 bits +- * mantissa: 10 bits +- */ +-typedef union FloatUnion16 +-{ +- npy_uint16 integer; +-} FloatUnion16; +- +-npy_bool IsNegative_F16(FloatUnion16 *v) { return (v->integer >> 15) != 0; } +-npy_uint32 GetExponent_F16(FloatUnion16 *v) { return (v->integer >> 10) & 0x1F;} +-npy_uint32 GetMantissa_F16(FloatUnion16 *v) { return v->integer & 0x3FF; } +- +- +-/* +- * Helper union to decompose a 32-bit IEEE float. +- * sign: 1 bit +- * exponent: 8 bits +- * mantissa: 23 bits +- */ +-typedef union FloatUnion32 +-{ +- npy_float32 floatingPoint; +- npy_uint32 integer; +-} FloatUnion32; +- +-npy_bool IsNegative_F32(FloatUnion32 *v) { return (v->integer >> 31) != 0; } +-npy_uint32 GetExponent_F32(FloatUnion32 *v) { return (v->integer >> 23) & 0xFF;} +-npy_uint32 GetMantissa_F32(FloatUnion32 *v) { return v->integer & 0x7FFFFF; } +- +-/* +- * Helper union to decompose a 64-bit IEEE float. +- * sign: 1 bit +- * exponent: 11 bits +- * mantissa: 52 bits +- */ +-typedef union FloatUnion64 +-{ +- npy_float64 floatingPoint; +- npy_uint64 integer; +-} FloatUnion64; +-npy_bool IsNegative_F64(FloatUnion64 *v) { return (v->integer >> 63) != 0; } +-npy_uint32 GetExponent_F64(FloatUnion64 *v) { return (v->integer >> 52) & 0x7FF; } +-npy_uint64 GetMantissa_F64(FloatUnion64 *v) { return v->integer & 0xFFFFFFFFFFFFFull; } +- +-/* +- * Helper unions and datatype to decompose a 80-bit IEEE float +- * sign: 1 bit, second u64 +- * exponent: 15 bits, second u64 +- * intbit 1 bit, first u64 +- * mantissa: 63 bits, first u64 +- */ +- +-/* +- * Since systems have different types of long doubles, and may not necessarily +- * have a 128-byte format we can use to pass values around, here we create +- * our own 128-bit storage type for convenience. +- */ +-typedef struct FloatVal128 { +- npy_uint64 integer[2]; +-} FloatVal128; +-npy_bool IsNegative_F128(FloatVal128 *v) { +- return ((v->integer[1] >> 15) & 0x1) != 0; +-} +-npy_uint32 GetExponent_F128(FloatVal128 *v) { return v->integer[1] & 0x7FFF; } +-npy_uint64 GetMantissa_F128(FloatVal128 *v) { +- return v->integer[0] & 0x7FFFFFFFFFFFFFFFull; +-} +- +-/* +- * then for each different definition of long double, we create a union to +- * unpack the float data safely. We can then copy these integers to a +- * FloatVal128. ++ * The FormatPositional and FormatScientific functions have been more ++ * significantly rewritten relative to Ryan Juckett's code. ++ * ++ * The binary16 and the various 128-bit float functions are new, and adapted ++ * from the 64 bit version. The python interface functions are new. + */ +-#ifdef NPY_FLOAT128 +-typedef union FloatUnion128 +-{ +- npy_float128 floatingPoint; +- struct { +- npy_uint64 a; +- npy_uint16 b; +- } integer; +-} FloatUnion128; +-#endif +- +-#ifdef NPY_FLOAT96 +-typedef union FloatUnion96 +-{ +- npy_float96 floatingPoint; +- struct { +- npy_uint64 a; +- npy_uint32 b; +- } integer; +-} FloatUnion96; +-#endif +- +-#ifdef NPY_FLOAT80 +-typedef union FloatUnion80 +-{ +- npy_float80 floatingPoint; +- struct { +- npy_uint64 a; +- npy_uint16 b; +- } integer; +-} FloatUnion80; +-#endif + + +-/* +- * The main changes above this point, relative to Ryan Juckett's code, are: +- * 1. fixed overflow problems when mantissa was 64 bits (in float128 types), +- * by replacing multiplication by 2 or 4 by BigInt_ShiftLeft calls. +- * 2. Increased c_BigInt_MaxBlocks +- * 3. Added more entries to the g_PowerOf10_Big table +- * 4. Added unbiased rounding calculation with isEven ++/* Options struct for easy passing of Dragon4 options. + * +- * Below this point, the FormatPositional and FormatScientific functions have +- * been more significantly rewritten. The Dragon4_PrintFloat16 and +- * Dragon4_PrintFloat128 functions are new, and were adapted from the 64 and 32 +- * bit versions. The python interfacing functions (in the header) are new. ++ * scientific - boolean controlling whether scientific notation is used ++ * digit_mode - whether to use unique or fixed fracional output ++ * cutoff_mode - whether 'precision' refers to toal digits, or digits past ++ * the decimal point. ++ * precision - When negative, prints as many digits as needed for a unique ++ * number. When positive specifies the maximum number of ++ * significant digits to print. ++ * sign - whether to always show sign ++ * trim_mode - how to treat trailing 0s and '.'. See TrimMode comments. ++ * digits_left - pad characters to left of decimal point. -1 for no padding ++ * digits_right - pad characters to right of decimal point. -1 for no padding. ++ * Padding adds whitespace until there are the specified ++ * number characters to sides of decimal point. Applies after ++ * trim_mode characters were removed. If digits_right is ++ * positive and the decimal point was trimmed, decimal point ++ * will be replaced by a whitespace character. ++ * exp_digits - Only affects scientific output. If positive, pads the ++ * exponent with 0s until there are this many digits. If ++ * negative, only use sufficient digits. + */ +- ++typedef struct Dragon4_Options { ++ npy_bool scientific; ++ DigitMode digit_mode; ++ CutoffMode cutoff_mode; ++ npy_int32 precision; ++ npy_bool sign; ++ TrimMode trim_mode; ++ npy_int32 digits_left; ++ npy_int32 digits_right; ++ npy_int32 exp_digits; ++} Dragon4_Options; + + /* + * Outputs the positive number with positional notation: ddddd.dddd + * The output is always NUL terminated and the output length (not including the + * NUL) is returned. ++ * + * Arguments: + * buffer - buffer to output into + * bufferSize - maximum characters that can be printed to buffer +@@ -1567,20 +1606,11 @@ typedef union FloatUnion80 + * signbit - value of the sign position. Should be '+', '-' or '' + * mantissaBit - index of the highest set mantissa bit + * hasUnequalMargins - is the high margin twice as large as the low margin +- * precision - Negative prints as many digits as are needed for a unique +- * number. Positive specifies the maximum number of significant +- * digits to print past the decimal point. +- * trim_mode - how to treat trailing 0s and '.'. See TrimMode comments. +- * digits_left - pad characters to left of decimal point. -1 for no padding +- * digits_right - pad characters to right of decimal point. -1 for no padding +- * padding adds whitespace until there are the specified +- * number characters to sides of decimal point. Applies after +- * trim_mode characters were removed. If digits_right is +- * positive and the decimal point was trimmed, decimal point +- * will be replaced by a whitespace character. ++ * ++ * See Dragon4_Options for description of remaining arguments. + */ + static npy_uint32 +-FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, ++FormatPositional(char *buffer, npy_uint32 bufferSize, BigInt *mantissa, + npy_int32 exponent, char signbit, npy_uint32 mantissaBit, + npy_bool hasUnequalMargins, DigitMode digit_mode, + CutoffMode cutoff_mode, npy_int32 precision, +@@ -1707,7 +1737,7 @@ FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + + /* always add decimal point, except for DprZeros mode */ + if (trim_mode != TrimMode_DptZeros && numFractionDigits == 0 && +- pos < maxPrintLen){ ++ pos < maxPrintLen) { + buffer[pos++] = '.'; + } + +@@ -1745,7 +1775,7 @@ FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + * when rounding, we may still end up with trailing zeros. Remove them + * depending on trim settings. + */ +- if (precision >= 0 && trim_mode != TrimMode_None && numFractionDigits > 0){ ++ if (precision >= 0 && trim_mode != TrimMode_None && numFractionDigits > 0) { + while (buffer[pos-1] == '0') { + pos--; + numFractionDigits--; +@@ -1779,7 +1809,7 @@ FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + npy_int32 shift = digits_left - (numWholeDigits + has_sign); + npy_int32 count = pos; + +- if (count + shift > maxPrintLen){ ++ if (count + shift > maxPrintLen) { + count = maxPrintLen - shift; + } + +@@ -1803,6 +1833,7 @@ FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + * Outputs the positive number with scientific notation: d.dddde[sign]ddd + * The output is always NUL terminated and the output length (not including the + * NUL) is returned. ++ * + * Arguments: + * buffer - buffer to output into + * bufferSize - maximum characters that can be printed to buffer +@@ -1811,15 +1842,11 @@ FormatPositional(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + * signbit - value of the sign position. Should be '+', '-' or '' + * mantissaBit - index of the highest set mantissa bit + * hasUnequalMargins - is the high margin twice as large as the low margin +- * precision - Negative prints as many digits as are needed for a unique +- * number. Positive specifies the maximum number of significant +- * digits to print past the decimal point. +- * trim_mode - how to treat trailing 0s and '.'. See TrimMode comments. +- * digits_left - pad characters to left of decimal point. -1 for no padding +- * exp_digits - pads exponent with zeros until it has this many digits ++ * ++ * See Dragon4_Options for description of remaining arguments. + */ + static npy_uint32 +-FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, ++FormatScientific (char *buffer, npy_uint32 bufferSize, BigInt *mantissa, + npy_int32 exponent, char signbit, npy_uint32 mantissaBit, + npy_bool hasUnequalMargins, DigitMode digit_mode, + npy_int32 precision, TrimMode trim_mode, +@@ -1844,7 +1871,7 @@ FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + leftchars = 1 + (signbit == '-' || signbit == '+'); + if (digits_left > leftchars) { + int i; +- for (i = 0; i < digits_left - leftchars && bufferSize > 1; i++){ ++ for (i = 0; i < digits_left - leftchars && bufferSize > 1; i++) { + *pCurOut = ' '; + pCurOut++; + --bufferSize; +@@ -1892,7 +1919,7 @@ FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + + /* always add decimal point, except for DprZeros mode */ + if (trim_mode != TrimMode_DptZeros && numFractionDigits == 0 && +- bufferSize > 1){ ++ bufferSize > 1) { + *pCurOut = '.'; + ++pCurOut; + --bufferSize; +@@ -1931,7 +1958,7 @@ FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + * when rounding, we may still end up with trailing zeros. Remove them + * depending on trim settings. + */ +- if (precision >= 0 && trim_mode != TrimMode_None && numFractionDigits > 0){ ++ if (precision >= 0 && trim_mode != TrimMode_None && numFractionDigits > 0) { + --pCurOut; + while (*pCurOut == '0') { + --pCurOut; +@@ -1972,7 +1999,7 @@ FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + DEBUG_ASSERT(printExponent < 100000); + + /* get exp digits */ +- for (i = 0; i < 5; i++){ ++ for (i = 0; i < 5; i++) { + digits[i] = printExponent % 10; + printExponent /= 10; + } +@@ -1981,7 +2008,7 @@ FormatScientific (char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + } + exp_size = i; + /* write remaining digits to tmp buf */ +- for (i = exp_size; i > 0; i--){ ++ for (i = exp_size; i > 0; i--) { + exponentBuffer[2 + (exp_size-i)] = (char)('0' + digits[i-1]); + } + +@@ -2057,12 +2084,12 @@ PrintInfNan(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + + /* only print sign for inf values (though nan can have a sign set) */ + if (signbit == '+') { +- if (pos < maxPrintLen-1){ ++ if (pos < maxPrintLen-1) { + buffer[pos++] = '+'; + } + } + else if (signbit == '-') { +- if (pos < maxPrintLen-1){ ++ if (pos < maxPrintLen-1) { + buffer[pos++] = '-'; + } + } +@@ -2080,7 +2107,9 @@ PrintInfNan(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + buffer[pos + printLen] = '\0'; + + /* +- * // XXX: Should we change this for numpy? ++ * For numpy we ignore unusual mantissa values for nan, but keep this ++ * code in case we change our mind later. ++ * + * // append HEX value + * if (maxPrintLen > 3) { + * printLen += PrintHex(buffer+3, bufferSize-3, mantissa, +@@ -2093,34 +2122,63 @@ PrintInfNan(char *buffer, npy_uint32 bufferSize, npy_uint64 mantissa, + } + + /* +- * These functions print a floating-point number as a decimal string. +- * The output string is always NUL terminated and the string length (not +- * including the NUL) is returned. ++ * The functions below format a floating-point numbers stored in particular ++ * formats, as a decimal string. The output string is always NUL terminated ++ * and the string length (not including the NUL) is returned. ++ * ++ * For 16, 32 and 64 bit floats we assume they are the IEEE 754 type. ++ * For 128 bit floats we account for different definitions. + * + * Arguments are: + * buffer - buffer to output into + * bufferSize - maximum characters that can be printed to buffer +- * value - value significand +- * scientific - boolean controlling whether scientific notation is used +- * precision - If positive, specifies the number of decimals to show after +- * decimal point. If negative, sufficient digits to uniquely +- * specify the float will be output. +- * trim_mode - how to treat trailing zeros and decimal point. See TrimMode. +- * digits_right - pad the result with '' on the right past the decimal point +- * digits_left - pad the result with '' on the right past the decimal point +- * exp_digits - Only affects scientific output. If positive, pads the +- * exponent with 0s until there are this many digits. If +- * negative, only use sufficient digits. ++ * value - value to print ++ * opt - Dragon4 options, see above ++ */ ++ ++/* ++ * Helper function that takes Dragon4 parameters and options and ++ * calls Dragon4. + */ + static npy_uint32 +-Dragon4_PrintFloat16(char *buffer, npy_uint32 bufferSize, npy_uint16 value, +- npy_bool scientific, DigitMode digit_mode, +- CutoffMode cutoff_mode, npy_int32 precision, +- npy_bool sign, TrimMode trim_mode, npy_int32 digits_left, +- npy_int32 digits_right, npy_int32 exp_digits) ++Format_floatbits(char *buffer, npy_uint32 bufferSize, BigInt *mantissa, ++ npy_int32 exponent, char signbit, npy_uint32 mantissaBit, ++ npy_bool hasUnequalMargins, Dragon4_Options *opt) + { +- FloatUnion16 floatUnion; +- npy_uint32 floatExponent, floatMantissa; ++ /* format the value */ ++ if (opt->scientific) { ++ return FormatScientific(buffer, bufferSize, mantissa, exponent, ++ signbit, mantissaBit, hasUnequalMargins, ++ opt->digit_mode, opt->precision, ++ opt->trim_mode, opt->digits_left, ++ opt->exp_digits); ++ } ++ else { ++ return FormatPositional(buffer, bufferSize, mantissa, exponent, ++ signbit, mantissaBit, hasUnequalMargins, ++ opt->digit_mode, opt->cutoff_mode, ++ opt->precision, opt->trim_mode, ++ opt->digits_left, opt->digits_right); ++ } ++} ++ ++/* ++ * IEEE binary16 floating-point format ++ * ++ * sign: 1 bit ++ * exponent: 5 bits ++ * mantissa: 10 bits ++ */ ++static npy_uint32 ++Dragon4_PrintFloat_IEEE_binary16( ++ Dragon4_Scratch *scratch, npy_half *value, Dragon4_Options *opt) ++{ ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ npy_uint16 val = *value; ++ npy_uint32 floatExponent, floatMantissa, floatSign; + + npy_uint32 mantissa; + npy_int32 exponent; +@@ -2138,20 +2196,20 @@ Dragon4_PrintFloat16(char *buffer, npy_uint32 bufferSize, npy_uint16 value, + } + + /* deconstruct the floating point value */ +- floatUnion.integer = value; +- floatExponent = GetExponent_F16(&floatUnion); +- floatMantissa = GetMantissa_F16(&floatUnion); ++ floatMantissa = val & bitmask_u32(10); ++ floatExponent = (val >> 10) & bitmask_u32(5); ++ floatSign = val >> 15; + + /* output the sign */ +- if (IsNegative_F16(&floatUnion)) { ++ if (floatSign != 0) { + signbit = '-'; + } +- else if (sign) { ++ else if (opt->sign) { + signbit = '+'; + } + + /* if this is a special value */ +- if (floatExponent == 0x1F) { ++ if (floatExponent == bitmask_u32(5)) { + return PrintInfNan(buffer, bufferSize, floatMantissa, 3, signbit); + } + /* else this is a number */ +@@ -2195,29 +2253,33 @@ Dragon4_PrintFloat16(char *buffer, npy_uint32 bufferSize, npy_uint16 value, + hasUnequalMargins = NPY_FALSE; + } + +- /* format the value */ +- if (scientific) { +- return FormatScientific(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- precision, trim_mode, digits_left, exp_digits); +- } +- else { +- return FormatPositional(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- cutoff_mode, precision, trim_mode, +- digits_left, digits_right); +- } ++ BigInt_Set_uint32(&bigints[0], mantissa); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent, ++ signbit, mantissaBit, hasUnequalMargins, opt); + } + ++/* ++ * IEEE binary32 floating-point format ++ * ++ * sign: 1 bit ++ * exponent: 8 bits ++ * mantissa: 23 bits ++ */ + static npy_uint32 +-Dragon4_PrintFloat32(char *buffer, npy_uint32 bufferSize, npy_float32 value, +- npy_bool scientific, DigitMode digit_mode, +- CutoffMode cutoff_mode, npy_int32 precision, +- npy_bool sign, TrimMode trim_mode, npy_int32 digits_left, +- npy_int32 digits_right, npy_int32 exp_digits) ++Dragon4_PrintFloat_IEEE_binary32( ++ Dragon4_Scratch *scratch, npy_float32 *value, ++ Dragon4_Options *opt) + { +- FloatUnion32 floatUnion; +- npy_uint32 floatExponent, floatMantissa; ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ union ++ { ++ npy_float32 floatingPoint; ++ npy_uint32 integer; ++ } floatUnion; ++ npy_uint32 floatExponent, floatMantissa, floatSign; + + npy_uint32 mantissa; + npy_int32 exponent; +@@ -2235,20 +2297,21 @@ Dragon4_PrintFloat32(char *buffer, npy_uint32 bufferSize, npy_float32 value, + } + + /* deconstruct the floating point value */ +- floatUnion.floatingPoint = value; +- floatExponent = GetExponent_F32(&floatUnion); +- floatMantissa = GetMantissa_F32(&floatUnion); ++ floatUnion.floatingPoint = *value; ++ floatMantissa = floatUnion.integer & bitmask_u32(23); ++ floatExponent = (floatUnion.integer >> 23) & bitmask_u32(8); ++ floatSign = floatUnion.integer >> 31; + + /* output the sign */ +- if (IsNegative_F32(&floatUnion)) { ++ if (floatSign != 0) { + signbit = '-'; + } +- else if (sign) { ++ else if (opt->sign) { + signbit = '+'; + } + + /* if this is a special value */ +- if (floatExponent == 0xFF) { ++ if (floatExponent == bitmask_u32(8)) { + return PrintInfNan(buffer, bufferSize, floatMantissa, 6, signbit); + } + /* else this is a number */ +@@ -2292,29 +2355,32 @@ Dragon4_PrintFloat32(char *buffer, npy_uint32 bufferSize, npy_float32 value, + hasUnequalMargins = NPY_FALSE; + } + +- /* format the value */ +- if (scientific) { +- return FormatScientific(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- precision, trim_mode, digits_left, exp_digits); +- } +- else { +- return FormatPositional(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- cutoff_mode, precision, trim_mode, +- digits_left, digits_right); +- } ++ BigInt_Set_uint32(&bigints[0], mantissa); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent, ++ signbit, mantissaBit, hasUnequalMargins, opt); + } + ++/* ++ * IEEE binary64 floating-point format ++ * ++ * sign: 1 bit ++ * exponent: 11 bits ++ * mantissa: 52 bits ++ */ + static npy_uint32 +-Dragon4_PrintFloat64(char *buffer, npy_uint32 bufferSize, npy_float64 value, +- npy_bool scientific, DigitMode digit_mode, +- CutoffMode cutoff_mode, npy_int32 precision, +- npy_bool sign, TrimMode trim_mode, npy_int32 digits_left, +- npy_int32 digits_right, npy_int32 exp_digits) ++Dragon4_PrintFloat_IEEE_binary64( ++ Dragon4_Scratch *scratch, npy_float64 *value, Dragon4_Options *opt) + { +- FloatUnion64 floatUnion; +- npy_uint32 floatExponent; ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ union ++ { ++ npy_float64 floatingPoint; ++ npy_uint64 integer; ++ } floatUnion; ++ npy_uint32 floatExponent, floatSign; + npy_uint64 floatMantissa; + + npy_uint64 mantissa; +@@ -2333,20 +2399,21 @@ Dragon4_PrintFloat64(char *buffer, npy_uint32 bufferSize, npy_float64 value, + } + + /* deconstruct the floating point value */ +- floatUnion.floatingPoint = value; +- floatExponent = GetExponent_F64(&floatUnion); +- floatMantissa = GetMantissa_F64(&floatUnion); ++ floatUnion.floatingPoint = *value; ++ floatMantissa = floatUnion.integer & bitmask_u64(52); ++ floatExponent = (floatUnion.integer >> 52) & bitmask_u32(11); ++ floatSign = floatUnion.integer >> 63; + + /* output the sign */ +- if (IsNegative_F64(&floatUnion)) { ++ if (floatSign != 0) { + signbit = '-'; + } +- else if (sign) { ++ else if (opt->sign) { + signbit = '+'; + } + + /* if this is a special value */ +- if (floatExponent == 0x7FF) { ++ if (floatExponent == bitmask_u32(11)) { + return PrintInfNan(buffer, bufferSize, floatMantissa, 13, signbit); + } + /* else this is a number */ +@@ -2390,28 +2457,48 @@ Dragon4_PrintFloat64(char *buffer, npy_uint32 bufferSize, npy_float64 value, + hasUnequalMargins = NPY_FALSE; + } + +- /* format the value */ +- if (scientific) { +- return FormatScientific(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- precision, trim_mode, digits_left, exp_digits); +- } +- else { +- return FormatPositional(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- cutoff_mode, precision, trim_mode, +- digits_left, digits_right); +- } ++ BigInt_Set_uint64(&bigints[0], mantissa); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent, ++ signbit, mantissaBit, hasUnequalMargins, opt); + } + ++ ++/* ++ * Since systems have different types of long doubles, and may not necessarily ++ * have a 128-byte format we can use to pass values around, here we create ++ * our own 128-bit storage type for convenience. ++ */ ++typedef struct FloatVal128 { ++ npy_uint64 hi, lo; ++} FloatVal128; ++ ++#if defined(HAVE_LDOUBLE_INTEL_EXTENDED_10_BYTES_LE) || \ ++ defined(HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE) || \ ++ defined(HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE) || \ ++ defined(HAVE_LDOUBLE_MOTOROLA_EXTENDED_12_BYTES_BE) ++/* ++ * Intel's 80-bit IEEE extended precision floating-point format ++ * ++ * "long doubles" with this format are stored as 96 or 128 bits, but ++ * are equivalent to the 80 bit type with some zero padding on the high bits. ++ * This method expects the user to pass in the value using a 128-bit ++ * FloatVal128, so can support 80, 96, or 128 bit storage formats, ++ * and is endian-independent. ++ * ++ * sign: 1 bit, second u64 ++ * exponent: 15 bits, second u64 ++ * intbit 1 bit, first u64 ++ * mantissa: 63 bits, first u64 ++ */ + static npy_uint32 +-Dragon4_PrintFloat128(char *buffer, npy_uint32 bufferSize, FloatVal128 value, +- npy_bool scientific, DigitMode digit_mode, +- CutoffMode cutoff_mode, npy_int32 precision, +- npy_bool sign, TrimMode trim_mode, npy_int32 digits_left, +- npy_int32 digits_right, npy_int32 exp_digits) ++Dragon4_PrintFloat_Intel_extended( ++ Dragon4_Scratch *scratch, FloatVal128 value, Dragon4_Options *opt) + { +- npy_uint32 floatExponent; ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ npy_uint32 floatExponent, floatSign; + npy_uint64 floatMantissa; + + npy_uint64 mantissa; +@@ -2429,20 +2516,27 @@ Dragon4_PrintFloat128(char *buffer, npy_uint32 bufferSize, FloatVal128 value, + return 0; + } + +- /* deconstruct the floating point value */ +- floatExponent = GetExponent_F128(&value); +- floatMantissa = GetMantissa_F128(&value); ++ /* deconstruct the floating point value (we ignore the intbit) */ ++ floatMantissa = value.lo & bitmask_u64(63); ++ floatExponent = value.hi & bitmask_u32(15); ++ floatSign = (value.hi >> 15) & 0x1; + + /* output the sign */ +- if (IsNegative_F128(&value)) { ++ if (floatSign != 0) { + signbit = '-'; + } +- else if (sign) { ++ else if (opt->sign) { + signbit = '+'; + } + + /* if this is a special value */ +- if (floatExponent == 0x7FFF) { ++ if (floatExponent == bitmask_u32(15)) { ++ /* ++ * Note: Technically there are other special extended values defined if ++ * the intbit is 0, like Pseudo-Infinity, Pseudo-Nan, Quiet-NaN. We ++ * ignore all of these since they are not generated on modern ++ * processors. We treat Quiet-Nan as simply Nan. ++ */ + return PrintInfNan(buffer, bufferSize, floatMantissa, 16, signbit); + } + /* else this is a number */ +@@ -2486,247 +2580,668 @@ Dragon4_PrintFloat128(char *buffer, npy_uint32 bufferSize, FloatVal128 value, + hasUnequalMargins = NPY_FALSE; + } + +- /* format the value */ +- if (scientific) { +- return FormatScientific(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- precision, trim_mode, digits_left, exp_digits); ++ BigInt_Set_uint64(&bigints[0], mantissa); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent, ++ signbit, mantissaBit, hasUnequalMargins, opt); ++ ++} ++ ++#endif /* INTEL_EXTENDED group */ ++ ++ ++#ifdef HAVE_LDOUBLE_INTEL_EXTENDED_10_BYTES_LE ++/* ++ * Intel's 80-bit IEEE extended precision format, 80-bit storage ++ * ++ * Note: It is not clear if a long double with 10-byte storage exists on any ++ * system. But numpy defines NPY_FLOAT80, so if we come across it, assume it is ++ * an Intel extended format. ++ */ ++static npy_uint32 ++Dragon4_PrintFloat_Intel_extended80( ++ Dragon4_Scratch *scratch, npy_float80 *value, Dragon4_Options *opt) ++{ ++ FloatVal128 val128; ++ union { ++ npy_float80 floatingPoint; ++ struct { ++ npy_uint64 a; ++ npy_uint16 b; ++ } integer; ++ } buf80; ++ ++ buf80.floatingPoint = *value; ++ /* Intel is little-endian */ ++ val128.lo = buf80.integer.a; ++ val128.hi = buf80.integer.b; ++ ++ return Dragon4_PrintFloat_Intel_extended(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_INTEL_EXTENDED_10_BYTES_LE */ ++ ++#ifdef HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE ++/* Intel's 80-bit IEEE extended precision format, 96-bit storage */ ++static npy_uint32 ++Dragon4_PrintFloat_Intel_extended96( ++ Dragon4_Scratch *scratch, npy_float96 *value, Dragon4_Options *opt) ++{ ++ FloatVal128 val128; ++ union { ++ npy_float96 floatingPoint; ++ struct { ++ npy_uint64 a; ++ npy_uint32 b; ++ } integer; ++ } buf96; ++ ++ buf96.floatingPoint = *value; ++ /* Intel is little-endian */ ++ val128.lo = buf96.integer.a; ++ val128.hi = buf96.integer.b; ++ ++ return Dragon4_PrintFloat_Intel_extended(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE */ ++ ++#ifdef HAVE_LDOUBLE_MOTOROLA_EXTENDED_12_BYTES_BE ++/* Motorola Big-endian equivalent of the Intel-extended 96 fp format */ ++static npy_uint32 ++Dragon4_PrintFloat_Motorola_extended96( ++ Dragon4_Scratch *scratch, npy_float96 *value, Dragon4_Options *opt) ++{ ++ FloatVal128 val128; ++ union { ++ npy_float96 floatingPoint; ++ struct { ++ npy_uint64 a; ++ npy_uint32 b; ++ } integer; ++ } buf96; ++ ++ buf96.floatingPoint = *value; ++ /* Motorola is big-endian */ ++ val128.lo = buf96.integer.b; ++ val128.hi = buf96.integer.a >> 16; ++ /* once again we assume the int has same endianness as the float */ ++ ++ return Dragon4_PrintFloat_Intel_extended(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_MOTOROLA_EXTENDED_12_BYTES_BE */ ++ ++ ++#ifdef NPY_FLOAT128 ++ ++typedef union FloatUnion128 ++{ ++ npy_float128 floatingPoint; ++ struct { ++ npy_uint64 a; ++ npy_uint64 b; ++ } integer; ++} FloatUnion128; ++ ++#ifdef HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE ++/* Intel's 80-bit IEEE extended precision format, 128-bit storage */ ++static npy_uint32 ++Dragon4_PrintFloat_Intel_extended128( ++ Dragon4_Scratch *scratch, npy_float128 *value, Dragon4_Options *opt) ++{ ++ FloatVal128 val128; ++ FloatUnion128 buf128; ++ ++ buf128.floatingPoint = *value; ++ /* Intel is little-endian */ ++ val128.lo = buf128.integer.a; ++ val128.hi = buf128.integer.b; ++ ++ return Dragon4_PrintFloat_Intel_extended(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE */ ++ ++#if defined(HAVE_LDOUBLE_IEEE_QUAD_LE) || defined(HAVE_LDOUBLE_IEEE_QUAD_BE) ++/* ++ * IEEE binary128 floating-point format ++ * ++ * sign: 1 bit ++ * exponent: 15 bits ++ * mantissa: 112 bits ++ * ++ * Currently binary128 format exists on only a few CPUs, such as on the POWER9 ++ * arch or aarch64. Because of this, this code has not been extensively tested. ++ * I am not sure if the arch also supports uint128, and C does not seem to ++ * support int128 literals. So we use uint64 to do manipulation. ++ */ ++static npy_uint32 ++Dragon4_PrintFloat_IEEE_binary128( ++ Dragon4_Scratch *scratch, FloatVal128 val128, Dragon4_Options *opt) ++{ ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ npy_uint32 floatExponent, floatSign; ++ ++ npy_uint64 mantissa_hi, mantissa_lo; ++ npy_int32 exponent; ++ npy_uint32 mantissaBit; ++ npy_bool hasUnequalMargins; ++ char signbit = '\0'; ++ ++ if (bufferSize == 0) { ++ return 0; ++ } ++ ++ if (bufferSize == 1) { ++ buffer[0] = '\0'; ++ return 0; ++ } ++ ++ mantissa_hi = val128.hi & bitmask_u64(48); ++ mantissa_lo = val128.lo; ++ floatExponent = (val128.hi >> 48) & bitmask_u32(15); ++ floatSign = val128.hi >> 63; ++ ++ /* output the sign */ ++ if (floatSign != 0) { ++ signbit = '-'; ++ } ++ else if (opt->sign) { ++ signbit = '+'; ++ } ++ ++ /* if this is a special value */ ++ if (floatExponent == bitmask_u32(15)) { ++ npy_uint64 mantissa_zero = mantissa_hi == 0 && mantissa_lo == 0; ++ return PrintInfNan(buffer, bufferSize, !mantissa_zero, 16, signbit); ++ } ++ /* else this is a number */ ++ ++ /* factor the value into its parts */ ++ if (floatExponent != 0) { ++ /* ++ * normal ++ * The floating point equation is: ++ * value = (1 + mantissa/2^112) * 2 ^ (exponent-16383) ++ * We convert the integer equation by factoring a 2^112 out of the ++ * exponent ++ * value = (1 + mantissa/2^112) * 2^112 * 2 ^ (exponent-16383-112) ++ * value = (2^112 + mantissa) * 2 ^ (exponent-16383-112) ++ * Because of the implied 1 in front of the mantissa we have 112 bits of ++ * precision. ++ * m = (2^112 + mantissa) ++ * e = (exponent-16383+1-112) ++ * ++ * Adding 2^112 to the mantissa is the same as adding 2^48 to the hi ++ * 64 bit part. ++ */ ++ mantissa_hi = (1ull << 48) | mantissa_hi; ++ /* mantissa_lo is unchanged */ ++ exponent = floatExponent - 16383 - 112; ++ mantissaBit = 112; ++ hasUnequalMargins = (floatExponent != 1) && (mantissa_hi == 0 && ++ mantissa_lo == 0); + } + else { +- return FormatPositional(buffer, bufferSize, mantissa, exponent, signbit, +- mantissaBit, hasUnequalMargins, digit_mode, +- cutoff_mode, precision, trim_mode, +- digits_left, digits_right); ++ /* ++ * subnormal ++ * The floating point equation is: ++ * value = (mantissa/2^112) * 2 ^ (1-16383) ++ * We convert the integer equation by factoring a 2^112 out of the ++ * exponent ++ * value = (mantissa/2^112) * 2^112 * 2 ^ (1-16383-112) ++ * value = mantissa * 2 ^ (1-16383-112) ++ * We have up to 112 bits of precision. ++ * m = (mantissa) ++ * e = (1-16383-112) ++ */ ++ exponent = 1 - 16383 - 112; ++ mantissaBit = LogBase2_128(mantissa_hi, mantissa_lo); ++ hasUnequalMargins = NPY_FALSE; + } ++ ++ BigInt_Set_2x_uint64(&bigints[0], mantissa_hi, mantissa_lo); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent, ++ signbit, mantissaBit, hasUnequalMargins, opt); + } + +-PyObject * +-Dragon4_Positional_AnySize(void *val, size_t size, DigitMode digit_mode, +- CutoffMode cutoff_mode, int precision, int sign, +- TrimMode trim, int pad_left, int pad_right) ++#if defined(HAVE_LDOUBLE_IEEE_QUAD_LE) ++static npy_uint32 ++Dragon4_PrintFloat_IEEE_binary128_le( ++ Dragon4_Scratch *scratch, npy_float128 *value, Dragon4_Options *opt) + { +- /* +- * Use a very large buffer in case anyone tries to output a large numberG. +- * 16384 should be enough to uniquely print any float128, which goes up +- * to about 10^4932 */ +- static char repr[16384]; + FloatVal128 val128; +-#ifdef NPY_FLOAT80 +- FloatUnion80 buf80;; +-#endif +-#ifdef NPY_FLOAT96 +- FloatUnion96 buf96; +-#endif +-#ifdef NPY_FLOAT128 + FloatUnion128 buf128; +-#endif + +- switch (size) { +- case 2: +- Dragon4_PrintFloat16(repr, sizeof(repr), *(npy_float16*)val, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +- case 4: +- Dragon4_PrintFloat32(repr, sizeof(repr), *(npy_float32*)val, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +- case 8: +- Dragon4_PrintFloat64(repr, sizeof(repr), *(npy_float64*)val, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +-#ifdef NPY_FLOAT80 +- case 10: +- buf80.floatingPoint = *(npy_float80*)val; +- val128.integer[0] = buf80.integer.a; +- val128.integer[1] = buf80.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +-#endif +-#ifdef NPY_FLOAT96 +- case 12: +- buf96.floatingPoint = *(npy_float96*)val; +- val128.integer[0] = buf96.integer.a; +- val128.integer[1] = buf96.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +-#endif +-#ifdef NPY_FLOAT128 +- case 16: +- buf128.floatingPoint = *(npy_float128*)val; +- val128.integer[0] = buf128.integer.a; +- val128.integer[1] = buf128.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 0, digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right, -1); +- break; +-#endif +- default: +- PyErr_Format(PyExc_ValueError, "unexpected itemsize %zu", size); +- return NULL; ++ buf128.floatingPoint = *value; ++ val128.lo = buf128.integer.a; ++ val128.hi = buf128.integer.b; ++ ++ return Dragon4_PrintFloat_IEEE_binary128(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_IEEE_QUAD_LE */ ++ ++#if defined(HAVE_LDOUBLE_IEEE_QUAD_BE) ++/* ++ * This function is untested, very few, if any, architectures implement ++ * big endian IEEE binary128 floating point. ++ */ ++static npy_uint32 ++Dragon4_PrintFloat_IEEE_binary128_be( ++ Dragon4_Scratch *scratch, npy_float128 *value, Dragon4_Options *opt) ++{ ++ FloatVal128 val128; ++ FloatUnion128 buf128; ++ ++ buf128.floatingPoint = *value; ++ val128.lo = buf128.integer.b; ++ val128.hi = buf128.integer.a; ++ ++ return Dragon4_PrintFloat_IEEE_binary128(scratch, val128, opt); ++} ++#endif /* HAVE_LDOUBLE_IEEE_QUAD_BE */ ++ ++#endif /* HAVE_LDOUBLE_IEEE_QUAD_LE | HAVE_LDOUBLE_IEEE_BE*/ ++ ++#if (defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE) || \ ++ defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE)) ++/* ++ * IBM extended precision 128-bit floating-point format, aka IBM double-double ++ * ++ * IBM's double-double type is a pair of IEEE binary64 values, which you add ++ * together to get a total value. The exponents are arranged so that the lower ++ * double is about 2^52 times smaller than the high one, and the nearest ++ * float64 value is simply the upper double, in which case the pair is ++ * considered "normalized" (not to confuse with "normal" and "subnormal" ++ * binary64 values). We assume normalized values. You can see the glibc's ++ * printf on ppc does so too by constructing un-normalized values to get ++ * strange behavior from the OS printf: ++ * ++ * >>> from numpy.core._multiarray_tests import format_float_OSprintf_g ++ * >>> x = np.array([0.3,0.3], dtype='f8').view('f16')[0] ++ * >>> format_float_OSprintf_g(x, 2) ++ * 0.30 ++ * >>> format_float_OSprintf_g(2*x, 2) ++ * 1.20 ++ * ++ * If we don't assume normalization, x should really print as 0.6. ++ * ++ * For normalized values gcc assumes that the total mantissa is no ++ * more than 106 bits (53+53), so we can drop bits from the second double which ++ * would be pushed past 106 when left-shifting by its exponent, as happens ++ * sometimes. (There has been debate about this, see ++ * https://gcc.gnu.org/bugzilla/show_bug.cgi?format=multiple&id=70117, ++ * https://sourceware.org/bugzilla/show_bug.cgi?id=22752 ) ++ * ++ * Note: This function is for the IBM-double-double which is a pair of IEEE ++ * binary64 floats, like on ppc64 systems. This is *not* the hexadecimal ++ * IBM-double-double type, which is a pair of IBM hexadecimal64 floats. ++ * ++ * See also: ++ * https://gcc.gnu.org/wiki/Ieee128PowerPCA ++ * https://www.ibm.com/support/knowledgecenter/en/ssw_aix_71/com.ibm.aix.genprogc/128bit_long_double_floating-point_datatype.htm ++ */ ++static npy_uint32 ++Dragon4_PrintFloat_IBM_double_double( ++ Dragon4_Scratch *scratch, npy_float128 *value, Dragon4_Options *opt) ++{ ++ char *buffer = scratch->repr; ++ npy_uint32 bufferSize = sizeof(scratch->repr); ++ BigInt *bigints = scratch->bigints; ++ ++ FloatVal128 val128; ++ FloatUnion128 buf128; ++ ++ npy_uint32 floatExponent1, floatExponent2; ++ npy_uint64 floatMantissa1, floatMantissa2; ++ npy_uint32 floatSign1, floatSign2; ++ ++ npy_uint64 mantissa1, mantissa2; ++ npy_int32 exponent1, exponent2; ++ int shift; ++ npy_uint32 mantissaBit; ++ npy_bool hasUnequalMargins; ++ char signbit = '\0'; ++ ++ if (bufferSize == 0) { ++ return 0; ++ } ++ ++ if (bufferSize == 1) { ++ buffer[0] = '\0'; ++ return 0; ++ } ++ ++ /* The high part always comes before the low part, regardless of the ++ * endianness of the system. */ ++ buf128.floatingPoint = *value; ++ val128.hi = buf128.integer.a; ++ val128.lo = buf128.integer.b; ++ ++ /* deconstruct the floating point values */ ++ floatMantissa1 = val128.hi & bitmask_u64(52); ++ floatExponent1 = (val128.hi >> 52) & bitmask_u32(11); ++ floatSign1 = (val128.hi >> 63) != 0; ++ ++ floatMantissa2 = val128.lo & bitmask_u64(52); ++ floatExponent2 = (val128.lo >> 52) & bitmask_u32(11); ++ floatSign2 = (val128.lo >> 63) != 0; ++ ++ /* output the sign using 1st float's sign */ ++ if (floatSign1) { ++ signbit = '-'; ++ } ++ else if (opt->sign) { ++ signbit = '+'; ++ } ++ ++ /* we only need to look at the first float for inf/nan */ ++ if (floatExponent1 == bitmask_u32(11)) { ++ return PrintInfNan(buffer, bufferSize, floatMantissa1, 13, signbit); ++ } ++ ++ /* else this is a number */ ++ ++ /* Factor the 1st value into its parts, see binary64 for comments. */ ++ if (floatExponent1 == 0) { ++ /* ++ * If the first number is a subnormal value, the 2nd has to be 0 for ++ * the float128 to be normalized, so we can ignore it. In this case ++ * the float128 only has the precision of a single binary64 value. ++ */ ++ mantissa1 = floatMantissa1; ++ exponent1 = 1 - 1023 - 52; ++ mantissaBit = LogBase2_64(mantissa1); ++ hasUnequalMargins = NPY_FALSE; ++ ++ BigInt_Set_uint64(&bigints[0], mantissa1); ++ } ++ else { ++ mantissa1 = (1ull << 52) | floatMantissa1; ++ exponent1 = floatExponent1 - 1023 - 52; ++ mantissaBit = 52 + 53; ++ ++ /* ++ * Computing hasUnequalMargins and mantissaBit: ++ * This is a little trickier than for IEEE formats. ++ * ++ * When both doubles are "normal" it is clearer since we can think of ++ * it as an IEEE type with a 106 bit mantissa. This value can never ++ * have "unequal" margins because of the implied 1 bit in the 2nd ++ * value. (unequal margins only happen when the mantissa has a value ++ * like "10000000000...", all zeros except the implied bit at the ++ * start, since the next lowest number has a different exponent). ++ * mantissaBits will always be 52+53 in this case. ++ * ++ * If the 1st number is a very small normal, and the 2nd is subnormal ++ * and not 2^52 times smaller, the number behaves like a subnormal ++ * overall, where the upper number just adds some bits on the left. ++ * Like usual subnormals, it has "equal" margins. The slightly tricky ++ * thing is that the number of mantissaBits varies. It will be 52 ++ * (from lower double) plus a variable number depending on the upper ++ * number's exponent. We recompute the number of bits in the shift ++ * calculation below, because the shift will be equal to the number of ++ * lost bits. ++ * ++ * We can get unequal margins only if the first value has all-0 ++ * mantissa (except implied bit), and the second value is exactly 0. As ++ * a special exception the smallest normal value (smallest exponent, 0 ++ * mantissa) should have equal margins, since it is "next to" a ++ * subnormal value. ++ */ ++ ++ /* factor the 2nd value into its parts */ ++ if (floatExponent2 != 0) { ++ mantissa2 = (1ull << 52) | floatMantissa2; ++ exponent2 = floatExponent2 - 1023 - 52; ++ hasUnequalMargins = NPY_FALSE; ++ } ++ else { ++ /* shift exp by one so that leading mantissa bit is still bit 53 */ ++ mantissa2 = floatMantissa2 << 1; ++ exponent2 = - 1023 - 52; ++ hasUnequalMargins = (floatExponent1 != 1) && (floatMantissa1 == 0) ++ && (floatMantissa2 == 0); ++ } ++ ++ /* ++ * The 2nd val's exponent might not be exactly 52 smaller than the 1st, ++ * it can vary a little bit. So do some shifting of the low mantissa, ++ * so that the total mantissa is equivalent to bits 53 to 0 of the ++ * first double immediately followed by bits 53 to 0 of the second. ++ */ ++ shift = exponent1 - exponent2 - 53; ++ if (shift > 0) { ++ /* shift more than 64 is undefined behavior */ ++ mantissa2 = shift < 64 ? mantissa2 >> shift : 0; ++ } ++ else if (shift < 0) { ++ /* ++ * This only happens if the 2nd value is subnormal. ++ * We expect that shift > -64, but check it anyway ++ */ ++ mantissa2 = -shift < 64 ? mantissa2 << -shift : 0; ++ } ++ ++ /* ++ * If the low double is a different sign from the high double, ++ * rearrange so that the total mantissa is the sum of the two ++ * mantissas, instead of a subtraction. ++ * hi - lo -> (hi-1) + (1-lo), where lo < 1 ++ */ ++ if (floatSign1 != floatSign2 && mantissa2 != 0) { ++ mantissa1--; ++ mantissa2 = (1ull << 53) - mantissa2; ++ } ++ ++ /* ++ * Compute the number of bits if we are in the subnormal range. ++ * The value "shift" happens to be exactly the number of lost bits. ++ * Also, shift the bits so that the least significant bit is at ++ * bit position 0, like a typical subnormal. After this exponent1 ++ * should always be 2^-1022 ++ */ ++ if (shift < 0) { ++ mantissa2 = (mantissa2 >> -shift) | (mantissa1 << (53 + shift)); ++ mantissa1 = mantissa1 >> -shift; ++ mantissaBit = mantissaBit -(-shift); ++ exponent1 -= shift; ++ DEBUG_ASSERT(exponent1 == -1022); ++ } ++ ++ /* ++ * set up the BigInt mantissa, by shifting the parts as needed ++ * We can use | instead of + since the mantissas should not overlap ++ */ ++ BigInt_Set_2x_uint64(&bigints[0], mantissa1 >> 11, ++ (mantissa1 << 53) | (mantissa2)); ++ exponent1 = exponent1 - 53; + } + +- return PyUString_FromString(repr); ++ return Format_floatbits(buffer, bufferSize, bigints, exponent1, ++ signbit, mantissaBit, hasUnequalMargins, opt); + } + ++#endif /* HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE | HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE */ ++ ++#endif /* NPY_FLOAT128 */ ++ ++ ++/* ++ * Here we define two Dragon4 entry functions for each type. One of them ++ * accepts the args in a Dragon4_Options struct for convenience, the ++ * other enumerates only the necessary parameters. ++ * ++ * Use a very large string buffer in case anyone tries to output a large number. ++ * 16384 should be enough to exactly print the integer part of any float128, ++ * which goes up to about 10^4932. The Dragon4_scratch struct provides a string ++ * buffer of this size. ++ */ ++#define make_dragon4_typefuncs_inner(Type, npy_type, format) \ ++\ ++PyObject *\ ++Dragon4_Positional_##Type##_opt(npy_type *val, Dragon4_Options *opt)\ ++{\ ++ PyObject *ret;\ ++ Dragon4_Scratch *scratch = get_dragon4_bigint_scratch();\ ++ if (scratch == NULL) {\ ++ return NULL;\ ++ }\ ++ if (Dragon4_PrintFloat_##format(scratch, val, opt) < 0) {\ ++ free_dragon4_bigint_scratch(scratch);\ ++ return NULL;\ ++ }\ ++ ret = PyUString_FromString(scratch->repr);\ ++ free_dragon4_bigint_scratch(scratch);\ ++ return ret;\ ++}\ ++\ ++PyObject *\ ++Dragon4_Positional_##Type(npy_type *val, DigitMode digit_mode,\ ++ CutoffMode cutoff_mode, int precision,\ ++ int sign, TrimMode trim, int pad_left, int pad_right)\ ++{\ ++ Dragon4_Options opt;\ ++ \ ++ opt.scientific = 0;\ ++ opt.digit_mode = digit_mode;\ ++ opt.cutoff_mode = cutoff_mode;\ ++ opt.precision = precision;\ ++ opt.sign = sign;\ ++ opt.trim_mode = trim;\ ++ opt.digits_left = pad_left;\ ++ opt.digits_right = pad_right;\ ++ opt.exp_digits = -1;\ ++\ ++ return Dragon4_Positional_##Type##_opt(val, &opt);\ ++}\ ++\ ++PyObject *\ ++Dragon4_Scientific_##Type##_opt(npy_type *val, Dragon4_Options *opt)\ ++{\ ++ PyObject *ret;\ ++ Dragon4_Scratch *scratch = get_dragon4_bigint_scratch();\ ++ if (scratch == NULL) {\ ++ return NULL;\ ++ }\ ++ if (Dragon4_PrintFloat_##format(scratch, val, opt) < 0) {\ ++ free_dragon4_bigint_scratch(scratch);\ ++ return NULL;\ ++ }\ ++ ret = PyUString_FromString(scratch->repr);\ ++ free_dragon4_bigint_scratch(scratch);\ ++ return ret;\ ++}\ ++PyObject *\ ++Dragon4_Scientific_##Type(npy_type *val, DigitMode digit_mode, int precision,\ ++ int sign, TrimMode trim, int pad_left, int exp_digits)\ ++{\ ++ Dragon4_Options opt;\ ++\ ++ opt.scientific = 1;\ ++ opt.digit_mode = digit_mode;\ ++ opt.cutoff_mode = CutoffMode_TotalLength;\ ++ opt.precision = precision;\ ++ opt.sign = sign;\ ++ opt.trim_mode = trim;\ ++ opt.digits_left = pad_left;\ ++ opt.digits_right = -1;\ ++ opt.exp_digits = exp_digits;\ ++\ ++ return Dragon4_Scientific_##Type##_opt(val, &opt);\ ++} ++ ++#define make_dragon4_typefuncs(Type, npy_type, format) \ ++ make_dragon4_typefuncs_inner(Type, npy_type, format) ++ ++make_dragon4_typefuncs(Half, npy_half, NPY_HALF_BINFMT_NAME) ++make_dragon4_typefuncs(Float, npy_float, NPY_FLOAT_BINFMT_NAME) ++make_dragon4_typefuncs(Double, npy_double, NPY_DOUBLE_BINFMT_NAME) ++make_dragon4_typefuncs(LongDouble, npy_longdouble, NPY_LONGDOUBLE_BINFMT_NAME) ++ ++#undef make_dragon4_typefuncs ++#undef make_dragon4_typefuncs_inner ++ + PyObject * + Dragon4_Positional(PyObject *obj, DigitMode digit_mode, CutoffMode cutoff_mode, + int precision, int sign, TrimMode trim, int pad_left, + int pad_right) + { +- double val; ++ npy_double val; ++ Dragon4_Options opt; ++ ++ opt.scientific = 0; ++ opt.digit_mode = digit_mode; ++ opt.cutoff_mode = cutoff_mode; ++ opt.precision = precision; ++ opt.sign = sign; ++ opt.trim_mode = trim; ++ opt.digits_left = pad_left; ++ opt.digits_right = pad_right; ++ opt.exp_digits = -1; + + if (PyArray_IsScalar(obj, Half)) { + npy_half x = ((PyHalfScalarObject *)obj)->obval; +- return Dragon4_Positional_AnySize(&x, sizeof(npy_half), +- digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right); ++ return Dragon4_Positional_Half_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, Float)) { + npy_float x = ((PyFloatScalarObject *)obj)->obval; +- return Dragon4_Positional_AnySize(&x, sizeof(npy_float), +- digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right); ++ return Dragon4_Positional_Float_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, Double)) { + npy_double x = ((PyDoubleScalarObject *)obj)->obval; +- return Dragon4_Positional_AnySize(&x, sizeof(npy_double), +- digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right); ++ return Dragon4_Positional_Double_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, LongDouble)) { + npy_longdouble x = ((PyLongDoubleScalarObject *)obj)->obval; +- return Dragon4_Positional_AnySize(&x, sizeof(npy_longdouble), +- digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right); ++ return Dragon4_Positional_LongDouble_opt(&x, &opt); + } + + val = PyFloat_AsDouble(obj); + if (PyErr_Occurred()) { + return NULL; + } +- return Dragon4_Positional_AnySize(&val, sizeof(double), +- digit_mode, cutoff_mode, precision, +- sign, trim, pad_left, pad_right); +-} +- +-PyObject * +-Dragon4_Scientific_AnySize(void *val, size_t size, DigitMode digit_mode, +- int precision, int sign, TrimMode trim, +- int pad_left, int exp_digits) +-{ +- /* use a very large buffer in case anyone tries to output a large precision */ +- static char repr[4096]; +- FloatVal128 val128; +-#ifdef NPY_FLOAT80 +- FloatUnion80 buf80;; +-#endif +-#ifdef NPY_FLOAT96 +- FloatUnion96 buf96; +-#endif +-#ifdef NPY_FLOAT128 +- FloatUnion128 buf128; +-#endif +- +- /* dummy, is ignored in scientific mode */ +- CutoffMode cutoff_mode = CutoffMode_TotalLength; +- +- switch (size) { +- case 2: +- Dragon4_PrintFloat16(repr, sizeof(repr), *(npy_float16*)val, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +- case 4: +- Dragon4_PrintFloat32(repr, sizeof(repr), *(npy_float32*)val, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +- case 8: +- Dragon4_PrintFloat64(repr, sizeof(repr), *(npy_float64*)val, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +-#ifdef NPY_FLOAT80 +- case 10: +- buf80.floatingPoint = *(npy_float80*)val; +- val128.integer[0] = buf80.integer.a; +- val128.integer[1] = buf80.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +-#endif +-#ifdef NPY_FLOAT96 +- case 12: +- buf96.floatingPoint = *(npy_float96*)val; +- val128.integer[0] = buf96.integer.a; +- val128.integer[1] = buf96.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +-#endif +-#ifdef NPY_FLOAT128 +- case 16: +- buf128.floatingPoint = *(npy_float128*)val; +- val128.integer[0] = buf128.integer.a; +- val128.integer[1] = buf128.integer.b; +- Dragon4_PrintFloat128(repr, sizeof(repr), val128, +- 1, digit_mode, cutoff_mode, precision, sign, +- trim, pad_left, -1, exp_digits); +- break; +-#endif +- default: +- PyErr_Format(PyExc_ValueError, "unexpected itemsize %zu", size); +- return NULL; +- } +- +- return PyUString_FromString(repr); ++ return Dragon4_Positional_Double_opt(&val, &opt); + } + + PyObject * + Dragon4_Scientific(PyObject *obj, DigitMode digit_mode, int precision, + int sign, TrimMode trim, int pad_left, int exp_digits) + { +- double val; ++ npy_double val; ++ Dragon4_Options opt; ++ ++ opt.scientific = 1; ++ opt.digit_mode = digit_mode; ++ opt.cutoff_mode = CutoffMode_TotalLength; ++ opt.precision = precision; ++ opt.sign = sign; ++ opt.trim_mode = trim; ++ opt.digits_left = pad_left; ++ opt.digits_right = -1; ++ opt.exp_digits = exp_digits; + + if (PyArray_IsScalar(obj, Half)) { + npy_half x = ((PyHalfScalarObject *)obj)->obval; +- return Dragon4_Scientific_AnySize(&x, sizeof(npy_half), +- digit_mode, precision, +- sign, trim, pad_left, exp_digits); ++ return Dragon4_Scientific_Half_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, Float)) { + npy_float x = ((PyFloatScalarObject *)obj)->obval; +- return Dragon4_Scientific_AnySize(&x, sizeof(npy_float), +- digit_mode, precision, +- sign, trim, pad_left, exp_digits); ++ return Dragon4_Scientific_Float_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, Double)) { + npy_double x = ((PyDoubleScalarObject *)obj)->obval; +- return Dragon4_Scientific_AnySize(&x, sizeof(npy_double), +- digit_mode, precision, +- sign, trim, pad_left, exp_digits); ++ return Dragon4_Scientific_Double_opt(&x, &opt); + } + else if (PyArray_IsScalar(obj, LongDouble)) { + npy_longdouble x = ((PyLongDoubleScalarObject *)obj)->obval; +- return Dragon4_Scientific_AnySize(&x, sizeof(npy_longdouble), +- digit_mode, precision, +- sign, trim, pad_left, exp_digits); ++ return Dragon4_Scientific_LongDouble_opt(&x, &opt); + } + + val = PyFloat_AsDouble(obj); + if (PyErr_Occurred()) { + return NULL; + } +- return Dragon4_Scientific_AnySize(&val, sizeof(double), +- digit_mode, precision, +- sign, trim, pad_left, exp_digits); ++ return Dragon4_Scientific_Double_opt(&val, &opt); + } ++ ++#undef DEBUG_ASSERT +diff --git a/numpy/core/src/multiarray/dragon4.h b/numpy/core/src/multiarray/dragon4.h +index 5559c51..2b8b4ce 100644 +--- a/numpy/core/src/multiarray/dragon4.h ++++ b/numpy/core/src/multiarray/dragon4.h +@@ -40,6 +40,48 @@ + #include "npy_pycompat.h" + #include "numpy/arrayscalars.h" + ++/* Half binary format */ ++#define NPY_HALF_BINFMT_NAME IEEE_binary16 ++ ++/* Float binary format */ ++#if NPY_BITSOF_FLOAT == 32 ++ #define NPY_FLOAT_BINFMT_NAME IEEE_binary32 ++#elif NPY_BITSOF_FLOAT == 64 ++ #define NPY_FLOAT_BINFMT_NAME IEEE_binary64 ++#else ++ #error No float representation defined ++#endif ++ ++/* Double binary format */ ++#if NPY_BITSOF_DOUBLE == 32 ++ #define NPY_DOUBLE_BINFMT_NAME IEEE_binary32 ++#elif NPY_BITSOF_DOUBLE == 64 ++ #define NPY_DOUBLE_BINFMT_NAME IEEE_binary64 ++#else ++ #error No double representation defined ++#endif ++ ++/* LongDouble binary format */ ++#if defined(HAVE_LDOUBLE_IEEE_QUAD_BE) ++ #define NPY_LONGDOUBLE_BINFMT_NAME IEEE_binary128_be ++#elif defined(HAVE_LDOUBLE_IEEE_QUAD_LE) ++ #define NPY_LONGDOUBLE_BINFMT_NAME IEEE_binary128_le ++#elif (defined(HAVE_LDOUBLE_IEEE_DOUBLE_LE) || \ ++ defined(HAVE_LDOUBLE_IEEE_DOUBLE_BE)) ++ #define NPY_LONGDOUBLE_BINFMT_NAME IEEE_binary64 ++#elif defined(HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE) ++ #define NPY_LONGDOUBLE_BINFMT_NAME Intel_extended96 ++#elif defined(HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE) ++ #define NPY_LONGDOUBLE_BINFMT_NAME Intel_extended128 ++#elif defined(HAVE_LDOUBLE_MOTOROLA_EXTENDED_12_BYTES_BE) ++ #define NPY_LONGDOUBLE_BINFMT_NAME Motorola_extended96 ++#elif (defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE) || \ ++ defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE)) ++ #define NPY_LONGDOUBLE_BINFMT_NAME IBM_double_double ++#else ++ #error No long double representation defined ++#endif ++ + typedef enum DigitMode + { + /* Round digits to print shortest uniquely identifiable number. */ +@@ -64,15 +106,23 @@ typedef enum TrimMode + TrimMode_DptZeros, /* trim trailing zeros & trailing decimal point */ + } TrimMode; + +-PyObject * +-Dragon4_Positional_AnySize(void *val, size_t size, DigitMode digit_mode, +- CutoffMode cutoff_mode, int precision, int sign, +- TrimMode trim, int pad_left, int pad_right); ++#define make_dragon4_typedecl(Type, npy_type) \ ++ PyObject *\ ++ Dragon4_Positional_##Type(npy_type *val, DigitMode digit_mode,\ ++ CutoffMode cutoff_mode, int precision,\ ++ int sign, TrimMode trim, int pad_left,\ ++ int pad_right);\ ++ PyObject *\ ++ Dragon4_Scientific_##Type(npy_type *val, DigitMode digit_mode,\ ++ int precision, int sign, TrimMode trim,\ ++ int pad_left, int exp_digits); + +-PyObject * +-Dragon4_Scientific_AnySize(void *val, size_t size, DigitMode digit_mode, +- int precision, int sign, TrimMode trim, +- int pad_left, int pad_right); ++make_dragon4_typedecl(Half, npy_half) ++make_dragon4_typedecl(Float, npy_float) ++make_dragon4_typedecl(Double, npy_double) ++make_dragon4_typedecl(LongDouble, npy_longdouble) ++ ++#undef make_dragon4_typedecl + + PyObject * + Dragon4_Positional(PyObject *obj, DigitMode digit_mode, CutoffMode cutoff_mode, +diff --git a/numpy/core/src/multiarray/scalartypes.c.src b/numpy/core/src/multiarray/scalartypes.c.src +index 9fc6e17..be2516d 100644 +--- a/numpy/core/src/multiarray/scalartypes.c.src ++++ b/numpy/core/src/multiarray/scalartypes.c.src +@@ -432,6 +432,7 @@ gentype_format(PyObject *self, PyObject *args) + + /**begin repeat + * #name = half, float, double, longdouble# ++ * #Name = Half, Float, Double, LongDouble# + * #NAME = HALF, FLOAT, DOUBLE, LONGDOUBLE# + * #type = npy_half, npy_float, npy_double, npy_longdouble# + * #suff = h, f, d, l# +@@ -443,12 +444,12 @@ format_@name@(@type@ val, npy_bool scientific, + int pad_left, int pad_right, int exp_digits) + { + if (scientific) { +- return Dragon4_Scientific_AnySize(&val, sizeof(@type@), ++ return Dragon4_Scientific_@Name@(&val, + DigitMode_Unique, precision, + sign, trim, pad_left, exp_digits); + } + else { +- return Dragon4_Positional_AnySize(&val, sizeof(@type@), ++ return Dragon4_Positional_@Name@(&val, + DigitMode_Unique, CutoffMode_TotalLength, precision, + sign, trim, pad_left, pad_right); + } +diff --git a/numpy/core/src/npymath/ieee754.c.src b/numpy/core/src/npymath/ieee754.c.src +index 0370ea6..170471f 100644 +--- a/numpy/core/src/npymath/ieee754.c.src ++++ b/numpy/core/src/npymath/ieee754.c.src +@@ -183,6 +183,7 @@ static npy_longdouble _nextl(npy_longdouble x, int p) + { + npy_int64 hx,ihx,ilx; + npy_uint64 lx; ++ npy_longdouble u; + + GET_LDOUBLE_WORDS64(hx, lx, x); + ihx = hx & 0x7fffffffffffffffLL; /* |hx| */ +@@ -193,7 +194,6 @@ static npy_longdouble _nextl(npy_longdouble x, int p) + return x; /* signal the nan */ + } + if(ihx == 0 && ilx == 0) { /* x == 0 */ +- npy_longdouble u; + SET_LDOUBLE_WORDS64(x, p, 0ULL);/* return +-minsubnormal */ + u = x * x; + if (u == x) { +@@ -203,7 +203,6 @@ static npy_longdouble _nextl(npy_longdouble x, int p) + } + } + +- npy_longdouble u; + if(p < 0) { /* p < 0, x -= ulp */ + if((hx==0xffefffffffffffffLL)&&(lx==0xfc8ffffffffffffeLL)) + return x+x; /* overflow, return -inf */ +diff --git a/numpy/core/src/npymath/npy_math_private.h b/numpy/core/src/npymath/npy_math_private.h +index d75b9e9..e4a919d 100644 +--- a/numpy/core/src/npymath/npy_math_private.h ++++ b/numpy/core/src/npymath/npy_math_private.h +@@ -287,8 +287,7 @@ do { \ + typedef npy_uint32 ldouble_man_t; + typedef npy_uint32 ldouble_exp_t; + typedef npy_uint32 ldouble_sign_t; +-#elif defined(HAVE_LDOUBLE_IEEE_DOUBLE_16_BYTES_BE) || \ +- defined(HAVE_LDOUBLE_IEEE_DOUBLE_BE) ++#elif defined(HAVE_LDOUBLE_IEEE_DOUBLE_BE) + /* 64 bits IEEE double precision aligned on 16 bytes: used by ppc arch on + * Mac OS X */ + +@@ -435,8 +434,8 @@ do { \ + typedef npy_uint32 ldouble_sign_t; + #endif + +-#if !defined(HAVE_LDOUBLE_DOUBLE_DOUBLE_BE) && \ +- !defined(HAVE_LDOUBLE_DOUBLE_DOUBLE_LE) ++#if !defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE) && \ ++ !defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE) + /* Get the sign bit of x. x should be of type IEEEl2bitsrep */ + #define GET_LDOUBLE_SIGN(x) \ + (((x).a[LDBL_SIGN_INDEX] & LDBL_SIGN_MASK) >> LDBL_SIGN_SHIFT) +@@ -477,7 +476,7 @@ do { \ + ((x).a[LDBL_MANH_INDEX] & ~LDBL_MANH_MASK) | \ + (((IEEEl2bitsrep_part)(v) << LDBL_MANH_SHIFT) & LDBL_MANH_MASK)) + +-#endif /* #ifndef HAVE_LDOUBLE_DOUBLE_DOUBLE_BE */ ++#endif /* !HAVE_LDOUBLE_DOUBLE_DOUBLE_* */ + + /* + * Those unions are used to convert a pointer of npy_cdouble to native C99 +diff --git a/numpy/core/src/private/npy_config.h b/numpy/core/src/private/npy_config.h +index 107b3cb..8143e77 100644 +--- a/numpy/core/src/private/npy_config.h ++++ b/numpy/core/src/private/npy_config.h +@@ -15,7 +15,8 @@ + * amd64 is not harmed much by the bloat as the system provides 16 byte + * alignment by default. + */ +-#if (defined NPY_CPU_X86 || defined _WIN32) ++#if (defined NPY_CPU_X86 || defined _WIN32 || defined NPY_CPU_ARMEL_AARCH32 ||\ ++ defined NPY_CPU_ARMEB_AARCH32) + #define NPY_MAX_COPY_ALIGNMENT 8 + #else + #define NPY_MAX_COPY_ALIGNMENT 16 +diff --git a/numpy/core/src/private/npy_fpmath.h b/numpy/core/src/private/npy_fpmath.h +index 86b9cf3..dbb3fb2 100644 +--- a/numpy/core/src/private/npy_fpmath.h ++++ b/numpy/core/src/private/npy_fpmath.h +@@ -7,45 +7,24 @@ + #include "numpy/npy_cpu.h" + #include "numpy/npy_common.h" + +-#ifdef NPY_OS_DARWIN +- /* This hardcoded logic is fragile, but universal builds makes it +- * difficult to detect arch-specific features */ +- +- /* MAC OS X < 10.4 and gcc < 4 does not support proper long double, and +- * is the same as double on those platforms */ +- #if NPY_BITSOF_LONGDOUBLE == NPY_BITSOF_DOUBLE +- /* This assumes that FPU and ALU have the same endianness */ +- #if NPY_BYTE_ORDER == NPY_LITTLE_ENDIAN +- #define HAVE_LDOUBLE_IEEE_DOUBLE_LE +- #elif NPY_BYTE_ORDER == NPY_BIG_ENDIAN +- #define HAVE_LDOUBLE_IEEE_DOUBLE_BE +- #else +- #error Endianness undefined ? +- #endif +- #else +- #if defined(NPY_CPU_X86) +- #define HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE +- #elif defined(NPY_CPU_AMD64) +- #define HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE +- #elif defined(NPY_CPU_PPC) || defined(NPY_CPU_PPC64) +- #define HAVE_LDOUBLE_IEEE_DOUBLE_16_BYTES_BE +- #elif defined(NPY_CPU_PPC64LE) +- #define HAVE_LDOUBLE_IEEE_DOUBLE_16_BYTES_LE +- #endif +- #endif +-#endif +- + #if !(defined(HAVE_LDOUBLE_IEEE_QUAD_BE) || \ + defined(HAVE_LDOUBLE_IEEE_QUAD_LE) || \ + defined(HAVE_LDOUBLE_IEEE_DOUBLE_LE) || \ + defined(HAVE_LDOUBLE_IEEE_DOUBLE_BE) || \ +- defined(HAVE_LDOUBLE_IEEE_DOUBLE_16_BYTES_BE) || \ + defined(HAVE_LDOUBLE_INTEL_EXTENDED_16_BYTES_LE) || \ + defined(HAVE_LDOUBLE_INTEL_EXTENDED_12_BYTES_LE) || \ + defined(HAVE_LDOUBLE_MOTOROLA_EXTENDED_12_BYTES_BE) || \ +- defined(HAVE_LDOUBLE_DOUBLE_DOUBLE_BE) || \ +- defined(HAVE_LDOUBLE_DOUBLE_DOUBLE_LE)) ++ defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE) || \ ++ defined(HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE)) + #error No long double representation defined + #endif + ++/* for back-compat, also keep old name for double-double */ ++#ifdef HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_LE ++ #define HAVE_LDOUBLE_DOUBLE_DOUBLE_LE ++#endif ++#ifdef HAVE_LDOUBLE_IBM_DOUBLE_DOUBLE_BE ++ #define HAVE_LDOUBLE_DOUBLE_DOUBLE_BE ++#endif ++ + #endif +diff --git a/numpy/core/tests/test_scalarprint.py b/numpy/core/tests/test_scalarprint.py +index d57f1a8..e04de61 100644 +--- a/numpy/core/tests/test_scalarprint.py ++++ b/numpy/core/tests/test_scalarprint.py +@@ -5,7 +5,8 @@ + from __future__ import division, absolute_import, print_function + + import numpy as np +-from numpy.testing import assert_, assert_equal, run_module_suite ++from numpy.testing import assert_, assert_equal, run_module_suite, dec ++import platform + + + class TestRealScalars(object): +@@ -201,6 +202,66 @@ class TestRealScalars(object): + "1.2" if tp != np.float16 else "1.2002") + assert_equal(fpos(tp('1.'), trim='-'), "1") + ++ @dec.skipif(not platform.machine().startswith("ppc64"), ++ "only applies to ppc float128 values") ++ def test_ppc64_ibm_double_double128(self): ++ # check that the precision decreases once we get into the subnormal ++ # range. Unlike float64, this starts around 1e-292 instead of 1e-308, ++ # which happens when the first double is normal and the second is ++ # subnormal. ++ x = np.float128('2.123123123123123123123123123123123e-286') ++ got = [str(x/np.float128('2e' + str(i))) for i in range(0,40)] ++ expected = [ ++ "1.06156156156156156156156156156157e-286", ++ "1.06156156156156156156156156156158e-287", ++ "1.06156156156156156156156156156159e-288", ++ "1.0615615615615615615615615615616e-289", ++ "1.06156156156156156156156156156157e-290", ++ "1.06156156156156156156156156156156e-291", ++ "1.0615615615615615615615615615616e-292", ++ "1.0615615615615615615615615615615e-293", ++ "1.061561561561561561561561561562e-294", ++ "1.06156156156156156156156156155e-295", ++ "1.0615615615615615615615615616e-296", ++ "1.06156156156156156156156156e-297", ++ "1.06156156156156156156156157e-298", ++ "1.0615615615615615615615616e-299", ++ "1.06156156156156156156156e-300", ++ "1.06156156156156156156155e-301", ++ "1.0615615615615615615616e-302", ++ "1.061561561561561561562e-303", ++ "1.06156156156156156156e-304", ++ "1.0615615615615615618e-305", ++ "1.06156156156156156e-306", ++ "1.06156156156156157e-307", ++ "1.0615615615615616e-308", ++ "1.06156156156156e-309", ++ "1.06156156156157e-310", ++ "1.0615615615616e-311", ++ "1.06156156156e-312", ++ "1.06156156154e-313", ++ "1.0615615616e-314", ++ "1.06156156e-315", ++ "1.06156155e-316", ++ "1.061562e-317", ++ "1.06156e-318", ++ "1.06155e-319", ++ "1.0617e-320", ++ "1.06e-321", ++ "1.04e-322", ++ "1e-323", ++ "0.0", ++ "0.0"] ++ assert_equal(got, expected) ++ ++ # Note: we follow glibc behavior, but it (or gcc) might not be right. ++ # In particular we can get two values that print the same but are not ++ # equal: ++ a = np.float128('2')/np.float128('3') ++ b = np.float128(str(a)) ++ assert_equal(str(a), str(b)) ++ assert_(a != b) ++ + def float32_roundtrip(self): + # gh-9360 + x = np.float32(1024 - 2**-14) +diff --git a/numpy/core/tests/test_umath.py b/numpy/core/tests/test_umath.py +index bebeddc..09d1bbe 100644 +--- a/numpy/core/tests/test_umath.py ++++ b/numpy/core/tests/test_umath.py +@@ -2515,8 +2515,10 @@ def test_nextafter(): + def test_nextafterf(): + return _test_nextafter(np.float32) + +-@dec.knownfailureif(sys.platform == 'win32', +- "Long double support buggy on win32, ticket 1664.") ++@dec.skipif(np.finfo(np.double) == np.finfo(np.longdouble), ++ "long double is same as double") ++@dec.knownfailureif(platform.machine().startswith("ppc64"), ++ "IBM double double") + def test_nextafterl(): + return _test_nextafter(np.longdouble) + +@@ -2544,8 +2546,10 @@ def test_spacing(): + def test_spacingf(): + return _test_spacing(np.float32) + +-@dec.knownfailureif(sys.platform == 'win32', +- "Long double support buggy on win32, ticket 1664.") ++@dec.skipif(np.finfo(np.double) == np.finfo(np.longdouble), ++ "long double is same as double") ++@dec.knownfailureif(platform.machine().startswith("ppc64"), ++ "IBM double double") + def test_spacingl(): + return _test_spacing(np.longdouble) + +diff --git a/numpy/ma/tests/test_core.py b/numpy/ma/tests/test_core.py +index 708c12e..1b47021 100644 +--- a/numpy/ma/tests/test_core.py ++++ b/numpy/ma/tests/test_core.py +@@ -3278,18 +3278,13 @@ class TestMaskedArrayMethods(object): + assert_equal(test.mask, mask_first.mask) + + # Test sort on dtype with subarray (gh-8069) ++ # Just check that the sort does not error, structured array subarrays ++ # are treated as byte strings and that leads to differing behavior ++ # depending on endianess and `endwith`. + dt = np.dtype([('v', int, 2)]) + a = a.view(dt) +- mask_last = mask_last.view(dt) +- mask_first = mask_first.view(dt) +- + test = sort(a) +- assert_equal(test, mask_last) +- assert_equal(test.mask, mask_last.mask) +- + test = sort(a, endwith=False) +- assert_equal(test, mask_first) +- assert_equal(test.mask, mask_first.mask) + + def test_argsort(self): + # Test argsort +-- +2.21.0 + diff --git a/SPECS/numpy.spec b/SPECS/numpy.spec new file mode 100644 index 0000000..8e972b8 --- /dev/null +++ b/SPECS/numpy.spec @@ -0,0 +1,858 @@ +%if 0%{?fedora} || 0%{?rhel} > 7 +%bcond_without python3 +%else +%bcond_with python3 +%{!?python2_sitearch: %global python2_sitearch %(%{__python2} -c "from distutils.sysconfig import get_python_lib; print get_python_lib(1)")} +%endif + +#uncomment next line for a release candidate or a beta +#%%global relc rc1 + +%global modname numpy + +Name: numpy +Version: 1.14.2 +Release: 16%{?dist} +Epoch: 1 +Summary: A fast multidimensional array facility for Python + +Group: Development/Languages +# Everything is BSD except for class SafeEval in numpy/lib/utils.py which is Python +License: BSD and Python +URL: http://www.numpy.org/ +Source0: https://github.com/%{name}/%{name}/releases/download/v%{version}/%{name}-%{version}.tar.gz +Source1: https://docs.scipy.org/doc/numpy/numpy-html-1.13.0.zip +Patch0: numpy-1.14.2-float128.patch +Patch1: numpy-1.14.2-CVE-2019-6446.patch + +BuildRequires: python2-devel lapack-devel python2-setuptools gcc-gfortran python2-nose +BuildRequires: /usr/bin/sed +BuildRequires: python2-Cython +%ifarch %{openblas_arches} +BuildRequires: openblas-devel +%else +BuildRequires: atlas-devel +%endif + + +%description +NumPy is a general-purpose array-processing package designed to +efficiently manipulate large multi-dimensional arrays of arbitrary +records without sacrificing too much speed for small multi-dimensional +arrays. NumPy is built on the Numeric code base and adds features +introduced by numarray as well as an extended C-API and the ability to +create arrays of arbitrary type. + +There are also basic facilities for discrete fourier transform, +basic linear algebra and random number generation. Also included in +this package is a version of f2py that works properly with NumPy. + + +%package -n python2-numpy +Summary: A fast multidimensional array facility for Python +Requires: python2-nose +%{?python_provide:%python_provide python2-%{modname}} +Obsoletes: numpy < 1:1.10.1-3 +%description -n python2-numpy +NumPy is a general-purpose array-processing package designed to +efficiently manipulate large multi-dimensional arrays of arbitrary +records without sacrificing too much speed for small multi-dimensional +arrays. NumPy is built on the Numeric code base and adds features +introduced by numarray as well as an extended C-API and the ability to +create arrays of arbitrary type. + +There are also basic facilities for discrete fourier transform, +basic linear algebra and random number generation. Also included in +this package is a version of f2py that works properly with NumPy. + + +%package -n python2-numpy-f2py +Summary: f2py for numpy +Group: Development/Libraries +Requires: python2-%{name} = %{epoch}:%{version}-%{release} +Requires: python2-devel +Obsoletes: numpy-f2py < 1:1.10.1-3 +%{?python_provide:%python_provide python2-numpy-f2py} + + +%description -n python2-numpy-f2py +This package includes a version of f2py that works properly with NumPy. + + +%package -n python2-numpy-doc +Summary: Documentation for numpy +Requires: python2-%{name} = %{epoch}:%{version}-%{release} +BuildArch: noarch + +%description -n python2-numpy-doc +This package provides the complete documentation for NumPy. + +%if %{with python3} +%package -n python3-numpy +Summary: A fast multidimensional array facility for Python + +Group: Development/Languages +License: BSD +%{?python_provide:%python_provide python3-numpy} +BuildRequires: python3-devel +BuildRequires: python3-setuptools +BuildRequires: python3-nose + +%description -n python3-numpy +NumPy is a general-purpose array-processing package designed to +efficiently manipulate large multi-dimensional arrays of arbitrary +records without sacrificing too much speed for small multi-dimensional +arrays. NumPy is built on the Numeric code base and adds features +introduced by numarray as well as an extended C-API and the ability to +create arrays of arbitrary type. + +There are also basic facilities for discrete fourier transform, +basic linear algebra and random number generation. Also included in +this package is a version of f2py that works properly with NumPy. + +%package -n python3-numpy-f2py +Summary: f2py for numpy +Group: Development/Libraries +Requires: python3-numpy = %{epoch}:%{version}-%{release} +Requires: python3-devel +Provides: python3-f2py = %{version}-%{release} +Obsoletes: python3-f2py <= 2.45.241_1927 +%{?python_provide:%python_provide python3-numpy-f2py} + +%description -n python3-numpy-f2py +This package includes a version of f2py that works properly with NumPy. + +%package -n python3-numpy-doc +Summary: Documentation for numpy +Requires: python3-numpy = %{epoch}:%{version}-%{release} +BuildArch: noarch + +%description -n python3-numpy-doc +This package provides the complete documentation for NumPy. + +%endif # with python3 + +%prep +%setup -q -n %{name}-%{version}%{?relc} +#%setup -q -n numpy-cc2b04 +%patch0 -p1 +%patch1 -p1 + +# workaround for rhbz#849713 +# http://mail.scipy.org/pipermail/numpy-discussion/2012-July/063530.html +rm numpy/distutils/command/__init__.py && touch numpy/distutils/command/__init__.py + +%ifarch %{openblas_arches} +# Use openblas pthreads as recommended by upstream (see comment in site.cfg.example) +cat >> site.cfg <= 21 || 0%{?rhel} > 7 +cat >> site.cfg < /dev/null + +# The custom install script gets the Python version from the executable name, +# e.g. "python3" -> "3", but when built by "platform-python" it guesses the +# version as "rm-python". Renaming the file here is the easiest correction. +mv f2pyrm-python f2py3 + +popd &> /dev/null + +popd + +%endif # with python3 + +#%%{__python2} setup.py install -O1 --skip-build --root %%{buildroot} +# skip-build currently broken, this works around it for now +%ifarch %{openblas_arches} +env OPENBLAS=%{_libdir} \ +%else +env ATLAS=%{_libdir} \ +%endif + FFTW=%{_libdir} BLAS=%{_libdir} \ + LAPACK=%{_libdir} CFLAGS="%{optflags}" \ + %{__python2} setup.py install --root %{buildroot} + +pushd %{buildroot}%{_bindir} &> /dev/null +# symlink for anyone who was using f2py.numpy +mv f2py{2,-%{python2_version}} +ln -s f2py-%{python2_version} f2py-2 +ln -s f2py-%{python2_version} f2py2 +popd &> /dev/null +#install -D -p -m 0644 docs/f2py/f2py.1 %{buildroot}%{_mandir}/man1/f2py.1 + +#symlink for includes, BZ 185079 +mkdir -p %{buildroot}%{_includedir} +ln -s %{python2_sitearch}/%{name}/core/include/numpy/ %{buildroot}%{_includedir}/numpy + + + +%check +# Having LDFLAGS set in the environment (for Flatpak builds, in particular), breaks +# f2py Makefiles which use that variable like: 'LDFLAGS = -s -shared' +unset LDFLAGS +pushd doc &> /dev/null +PATH="%{buildroot}%{_bindir}:${PATH}" \ +PYTHONPATH="%{buildroot}%{python2_sitearch}" \ +%{__python2} -c "import pkg_resources, numpy, sys ; sys.exit(0 if numpy.test(verbose=2).wasSuccessful() else 1)" \ +%ifarch s390 s390x +|| : +%endif +# don't remove this comment +popd &> /dev/null + +%if %{with python3} +pushd doc &> /dev/null +PATH="%{buildroot}%{_bindir}:${PATH}" \ +PYTHONPATH="%{buildroot}%{python3_sitearch}" \ +%{__python3} -c "import pkg_resources, numpy, sys ; sys.exit(0 if numpy.test(verbose=2).wasSuccessful() else 1)" \ +%ifarch s390 s390x +|| : +%endif +# don't remove this comment +popd &> /dev/null + +%endif # with python3 + + +%files -n python2-numpy +%license LICENSE.txt +%doc THANKS.txt site.cfg.example +%dir %{python2_sitearch}/%{name} +%{python2_sitearch}/%{name}/*.py* +%{python2_sitearch}/%{name}/core +%{python2_sitearch}/%{name}/distutils +%{python2_sitearch}/%{name}/doc +%{python2_sitearch}/%{name}/fft +%{python2_sitearch}/%{name}/lib +%{python2_sitearch}/%{name}/linalg +%{python2_sitearch}/%{name}/ma +%{python2_sitearch}/%{name}/random +%{python2_sitearch}/%{name}/testing +%{python2_sitearch}/%{name}/tests +%{python2_sitearch}/%{name}/compat +%{python2_sitearch}/%{name}/matrixlib +%{python2_sitearch}/%{name}/polynomial +%{python2_sitearch}/%{name}-*.egg-info +%{_includedir}/numpy +%exclude %{python2_sitearch}/%{name}/LICENSE.txt + +%files -n python2-numpy-f2py +%doc docs/f2py/*.html +#%{_mandir}/man*/* +%{_bindir}/f2py2 +%{_bindir}/f2py-2 +%{_bindir}/f2py-%{python2_version} +%{python2_sitearch}/%{name}/f2py + +%files -n python2-numpy-doc +%doc docs/* + +%if %{with python3} +%files -n python3-numpy +%license LICENSE.txt +%doc THANKS.txt site.cfg.example +%{python3_sitearch}/%{name}/__pycache__ +%dir %{python3_sitearch}/%{name} +%{python3_sitearch}/%{name}/*.py* +%{python3_sitearch}/%{name}/core +%{python3_sitearch}/%{name}/distutils +%{python3_sitearch}/%{name}/doc +%{python3_sitearch}/%{name}/fft +%{python3_sitearch}/%{name}/lib +%{python3_sitearch}/%{name}/linalg +%{python3_sitearch}/%{name}/ma +%{python3_sitearch}/%{name}/random +%{python3_sitearch}/%{name}/testing +%{python3_sitearch}/%{name}/tests +%{python3_sitearch}/%{name}/compat +%{python3_sitearch}/%{name}/matrixlib +%{python3_sitearch}/%{name}/polynomial +%{python3_sitearch}/%{name}-*.egg-info +%exclude %{python3_sitearch}/%{name}/LICENSE.txt + +%files -n python3-numpy-f2py +%{_bindir}/f2py3 +%{python3_sitearch}/%{name}/f2py + +%files -n python3-numpy-doc +%doc docs/* + +%endif # with python3 + + +%changelog +* Fri Jan 08 2021 Nikola Forró - 1:1.14.2-16 +- Fix include path +- Related: rhbz#1907601 + +* Wed Dec 16 2020 Nikola Forró - 1:1.14.2-15 +- Fix %check +- Related: rhbz#1907601 + +* Tue Dec 15 2020 Nikola Forró - 1:1.14.2-14 +- Use macros rather than hardcoded paths +- Resolves: rhbz#1907601 + +* Wed Jun 05 2019 Nikola Forró - 1:1.14.2-13 +- Fix CVE-2019-6446 +- Resolves: rhbz#1668829 + +* Thu May 30 2019 Charalampos Stratakis - 1.14.2-12 +- Set proper build flags for https://fedoraproject.org/wiki/Changes/Python_Extension_Flags +- Resolves: rhbz#1715036 + +* Thu May 30 2019 Nikola Forró - 1.14.2-11 +- Fix broken float128 on all arches except x86_64 +- Resolves: rhbz#1688709 + +* Thu Apr 25 2019 Tomas Orsava - 1.14.2-10 +- Bumping due to problems with modular RPM upgrade path +- Resolves: rhbz#1695587 + +* Tue Oct 09 2018 Lumír Balhar - 1:1.14.2-9 +- Remove unversioned provides +- Resolves: rhbz#1628242 + +* Tue Oct 02 2018 Lumír Balhar - 1:1.14.2-8 +- Fix unversioned requires/buildrequires +- Resolves: rhbz#1628242 + +* Tue Aug 14 2018 Lumír Balhar - 1:1.14.2-7 +- Bring symlink f2py2 back for symlink modules +- Resolves: rhbz#1615727 + +* Wed Aug 08 2018 Lumír Balhar - 1:1.14.2-6 +- Remove unversioned binaries from python2 subpackage +- Resolves: rhbz#1613343 + +* Tue Jul 31 2018 Lumír Balhar - 1:1.14.2-5 +- Switch python3 coditions to bcond + +* Mon Jun 25 2018 Tomas Orsava - 1:1.14.2-4 +- Use python2 macros instead of unversioned python macros + +* Sat Apr 28 2018 Tomas Orsava - 1:1.14.2-3 +- Change the shebang of f2py to the versioned /usr/bin/python2 + +* Fri Apr 27 2018 Tomas Orsava - 1:1.14.2-2 +- Fix incorrect Python version guess when building on Platform-Python + +* Mon Mar 12 2018 Gwyn Ciesla - 1:1.14.2-1 +- 1.14.2 + +* Wed Feb 21 2018 Gwyn Ciesla - 1:1.14.1-1 +- 1.14.1 + +* Thu Feb 08 2018 Fedora Release Engineering - 1:1.14.0-0.rc1.1 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_28_Mass_Rebuild + +* Wed Dec 13 2017 Gwyn Ciesla - 1:1.14.0-0.rc1 +- 1.14.0 rc1 + +* Mon Dec 11 2017 Iryna Shcherbina - 1:1.13.3-5 +- Fix ambiguous Python 2 dependency declarations + (See https://fedoraproject.org/wiki/FinalizingFedoraSwitchtoPython3) + +* Thu Nov 16 2017 Gwyn Ciesla - 1:1.13.3-4 +- Split out doc subpackage. + +* Mon Nov 06 2017 Merlin Mathesius - 1:1.13.3-3 +- Cleanup spec file conditionals + +* Tue Oct 31 2017 Christian Dersch - 1:1.13.3-2 +- set proper environment variables for openblas + +* Wed Oct 04 2017 Gwyn Ciesla - 1:1.13.3-1 +- 1.13.3 + +* Thu Sep 28 2017 Gwyn Ciesla - 1:1.13.2-1 +- 1.13.2 + +* Tue Aug 08 2017 Gwyn Ciesla - 1:1.13.1-4 +- Use openblas where available, BZ 1472318. + +* Thu Aug 03 2017 Fedora Release Engineering - 1:1.13.1-3 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_27_Binutils_Mass_Rebuild + +* Thu Jul 27 2017 Fedora Release Engineering - 1:1.13.1-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_27_Mass_Rebuild + +* Fri Jul 07 2017 Gwyn Ciesla - 1:1.13.1-1 +- 1.13.1 final + +* Fri Jun 09 2017 Gwyn Ciesla - 1:1.13.0-1 +- 1.13.0 final + +* Fri May 19 2017 Gwyn Ciesla - 1:1.13.0-0.rc2 +- 1.13.0 rc2 + +* Thu May 11 2017 Gwyn Ciesla - 1:1.13.0-0.rc1 +- 1.13.0 rc1 + +* Wed Mar 29 2017 Gwyn Ciesla - 1:1.12.1-1 +- 1.12.1 + +* Tue Jan 31 2017 Simone Caronni - 1:1.12.0-1 +- Update to 1.12.0, build with gcc 7.0. + +* Mon Dec 12 2016 Charalampos Stratakis - 1:1.11.2-2 +- Rebuild for Python 3.6 + +* Mon Oct 3 2016 Orion Poplawski - 1:1.11.2-1 +- Update to 1.11.2 final + +* Thu Sep 15 2016 Jon Ciesla - 1:1.11.2-0.rc1 +- Update to 1.11.2rc1, BZ 1340440. + +* Tue Jul 19 2016 Fedora Release Engineering - 1:1.11.1-2 +- https://fedoraproject.org/wiki/Changes/Automatic_Provides_for_Python_RPM_Packages + +* Tue Jun 28 2016 Orion Poplawski - 1:1.11.1-1 +- Update to 1.11.1 final + +* Tue Jun 07 2016 Jon Ciesla - 1:1.11.1-0.rc1 +- Update to 1.11.1rc1, BZ 1340440. + +* Mon Mar 28 2016 Orion Poplawski - 1:1.11.0-4 +- Update to 1.11.0 final + +* Wed Mar 23 2016 Orion Poplawski - 1:1.11.0-3.rc2 +- Update to 1.11.0rc2 + +* Sun Mar 6 2016 Peter Robinson 1:1.11.0-2.b3 +- Bump Release. 1b2 is higher than 0b3 + +* Wed Feb 10 2016 Jon Ciesla - 1:1.11.0-0.b3 +- Update to 1.11.0b2, BZ 1306249. + +* Thu Feb 04 2016 Fedora Release Engineering - 1:1.11.0-1b2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_24_Mass_Rebuild + +* Sun Jan 31 2016 Jon Ciesla - 1:1.11.0-0.b2 +- Update to 1.11.0b2, BZ 1303387. + +* Tue Jan 26 2016 Jon Ciesla - 1:1.11.0-020161016.cc2b04git +- Update to git snapshot (due to build issue) after 1.11.0b1, BZ 1301943. + +* Thu Jan 07 2016 Jon Ciesla - 1:1.10.4-1 +- Update to 1.10.4, BZ 1296509. + +* Tue Dec 15 2015 Jon Ciesla - 1:1.10.2-1 +- Update to 1.10.2, BZ 1291674. + +* Tue Dec 08 2015 Jon Ciesla - 1:1.10.2-0.2.rc2 +- Update to 1.10.2rc1, BZ 1289550. + +* Fri Nov 13 2015 Orion Poplawski - 1:1.10.2-0.1.rc1 +- Update to 1.10.2rc1 +- Drop opt-flags patch applied upstream + +* Fri Nov 13 2015 Kalev Lember - 1:1.10.1-6 +- Add provides to satisfy numpy%%{_isa} requires in other packages + +* Thu Nov 12 2015 Orion Poplawski - 1:1.10.1-5 +- Re-add provides f2py + +* Thu Nov 12 2015 Kalev Lember - 1:1.10.1-4 +- Fix obsoletes / provides for numpy -> python2-numpy rename + +* Wed Oct 14 2015 Thomas Spura - 1:1.10.1-3 +- Remove fortran flags or arm would build with -march=x86-64 + +* Wed Oct 14 2015 Thomas Spura - 1:1.10.1-2 +- Provide python2-* packages +- Run tests with verbose=2 + +* Tue Oct 13 2015 Jon Ciesla - 1:1.10.1-1 +- Update to 1.10.1, BZ 1271022. + +* Tue Oct 13 2015 Robert Kuska - 1:1.10.0-2 +- Rebuilt for Python3.5 rebuild + +* Tue Oct 06 2015 Jon Ciesla - 1:1.10.0-1 +- Update to 1.10.0 final. + +* Wed Sep 02 2015 Jon Ciesla - 1:1.10.0-0.b1 +- Update to 1.10.0b1, BZ 1252641. + +* Thu Aug 13 2015 Orion Poplawski - 1:1.9.2-3 +- Add python2-numpy provides (bug #1249423) +- Spec cleanup + +* Wed Jun 17 2015 Fedora Release Engineering - 1:1.9.2-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_23_Mass_Rebuild + +* Sun Mar 1 2015 Orion Poplawski - 1:1.9.2-1 +- Update to 1.9.2 + +* Tue Jan 6 2015 Orion Poplawski - 1:1.9.1-2 +- Add upstream patch to fix xerbla linkage (bug #1172834) + +* Tue Nov 04 2014 Jon Ciesla - 1:1.9.1-1 +- Update to 1.9.1, BZ 1160273. + +* Sun Sep 7 2014 Orion Poplawski - 1:1.9.0-1 +- Update to 1.9.0 + +* Wed Aug 27 2014 Orion Poplawski - 1:1.9.0-0.1.rc1 +- Update to 1.9.0rc1 + +* Sun Aug 17 2014 Fedora Release Engineering - 1:1.8.2-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_21_22_Mass_Rebuild + +* Sun Aug 10 2014 Orion Poplawski - 1:1.8.2-1 +- Update to 1.8.2 + +* Sat Jun 07 2014 Fedora Release Engineering - 1:1.8.1-4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_21_Mass_Rebuild + +* Fri May 9 2014 Orion Poplawski - 1:1.8.1-3 +- Rebuild for Python 3.4 + +* Wed May 07 2014 Jaromir Capik - 1:1.8.1-2 +- Fixing FTBFS on ppc64le (#1078354) + +* Tue Mar 25 2014 Orion Poplawski - 1:1.8.1-1 +- Update to 1.8.1 + +* Tue Mar 4 2014 Orion Poplawski - 1:1.8.0-5 +- Fix __pycache__ ownership (bug #1072467) + +* Mon Feb 10 2014 Thomas Spura - 1:1.8.0-4 +- Fix CVE-2014-1858, CVE-2014-1859: #1062009, #1062359 + +* Mon Nov 25 2013 Orion Poplawski - 1:1.8.0-3 +- Ship doc module (bug #1034357) + +* Wed Nov 6 2013 Orion Poplawski - 1:1.8.0-2 +- Move f2py documentation to f2py package (bug #1027394) + +* Wed Oct 30 2013 Orion Poplawski - 1:1.8.0-1 +- Update to 1.8.0 final + +* Mon Oct 14 2013 Orion Poplawski - 1:1.8.0-0.7.rc2 +- Update to 1.8.0rc2 +- Create clean site.cfg +- Use serial atlas + +* Mon Sep 23 2013 Orion Poplawski - 1:1.8.0-0.6.b2 +- Add [atlas] to site.cfg for new atlas library names + +* Sun Sep 22 2013 Orion Poplawski - 1:1.8.0-0.5.b2 +- Update site.cfg for new atlas library names + +* Sat Sep 21 2013 David Tardon - 1:1.8.0-0.4.b2 +- rebuild for atlas 3.10 + +* Tue Sep 10 2013 Jon Ciesla - 1:1.8.0-0.3.b2 +- Fix libdir path in site.cfg, BZ 1006242. + +* Sun Sep 8 2013 Orion Poplawski - 1:1.8.0-0.2.b2 +- Update to 1.8.0b2 + +* Wed Sep 4 2013 Orion Poplawski - 1:1.8.0-0.1.b1 +- Update to 1.8.0b1 +- Drop f2py patch applied upstream + +* Tue Aug 27 2013 Jon Ciesla - 1:1.7.1-5 +- URL Fix, BZ 1001337 + +* Sat Aug 03 2013 Fedora Release Engineering - 1:1.7.1-4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_20_Mass_Rebuild + +* Tue Jul 30 2013 Tomas Tomecek - 1:1.7.1-3 +- Fix rpmlint warnings +- Update License +- Apply patch: change shebang of f2py to use binary directly + +* Sun Jun 2 2013 Orion Poplawski - 1:1.7.1-2 +- Specfile cleanup (bug #969854) + +* Wed Apr 10 2013 Orion Poplawski - 1:1.7.1-1 +- Update to 1.7.1 + +* Sat Feb 9 2013 Orion Poplawski - 1:1.7.0-1 +- Update to 1.7.0 final + +* Sun Dec 30 2012 Orion Poplawski - 1:1.7.0-0.5.rc1 +- Update to 1.7.0rc1 + +* Thu Sep 20 2012 Orion Poplawski - 1:1.7.0-0.4.b2 +- Update to 1.7.0b2 +- Drop patches applied upstream + +* Wed Aug 22 2012 Orion Poplawski - 1:1.7.0-0.3.b1 +- Add patch from github pull 371 to fix python 3.3 pickle issue +- Remove cython .c source regeneration - fails now + +* Wed Aug 22 2012 Orion Poplawski - 1:1.7.0-0.2.b1 +- add workaround for rhbz#849713 (fixes FTBFS) + +* Tue Aug 21 2012 Orion Poplawski - 1:1.7.0-0.1.b1 +- Update to 1.7.0b1 +- Rebase python 3.3 patchs to current git master +- Drop patches applied upstream + +* Sun Aug 5 2012 David Malcolm - 1:1.6.2-5 +- rework patches for 3.3 to more directly reflect upstream's commits +- re-enable test suite on python 3 +- forcibly regenerate Cython .c source to avoid import issues on Python 3.3 + +* Sun Aug 5 2012 Thomas Spura - 1:1.6.2-4 +- rebuild for https://fedoraproject.org/wiki/Features/Python_3.3 +- needs unicode patch + +* Fri Aug 3 2012 David Malcolm - 1:1.6.2-3 +- remove rhel logic from with_python3 conditional + +* Fri Jul 20 2012 Fedora Release Engineering - 1:1.6.2-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_18_Mass_Rebuild + +* Sun May 20 2012 Orion Poplawski - 1:1.6.2-1 +- Update to 1.6.2 final + +* Sat May 12 2012 Orion Poplawski - 1:1.6.2rc1-0.1 +- Update to 1.6.2rc1 + +* Fri Jan 13 2012 Fedora Release Engineering - 1:1.6.1-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_17_Mass_Rebuild + +* Mon Nov 7 2011 Orion Poplawski - 1:1.6.1-1 +- Update to 1.6.1 + +* Fri Jun 17 2011 Jon Ciesla - 1:1.6.0-2 +- Bump and rebuild for BZ 712251. + +* Mon May 16 2011 Orion Poplawski - 1:1.6.0-1 +- Update to 1.6.0 final + +* Mon Apr 4 2011 Orion Poplawski - 1:1.6.0-0.2.b2 +- Update to 1.6.0b2 +- Drop import patch fixed upstream + +* Thu Mar 31 2011 Orion Poplawski - 1:1.6.0-0.1.b1 +- Update to 1.6.0b1 +- Build python3 module with python3 +- Add patch from upstream to fix build time import error + +* Wed Mar 30 2011 Orion Poplawski - 1:1.5.1-1 +- Update to 1.5.1 final + +* Tue Feb 08 2011 Fedora Release Engineering - 1:1.5.1-0.4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_15_Mass_Rebuild + +* Thu Jan 13 2011 Dan Horák - 1:1.5.1-0.3 +- fix the AttributeError during tests +- fix build on s390(x) + +* Wed Dec 29 2010 David Malcolm - 1:1.5.1-0.2 +- rebuild for newer python3 + +* Wed Oct 27 2010 Thomas Spura - 1:1.5.1-0.1 +- update to 1.5.1rc1 +- add python3 subpackage +- some spec-cleanups + +* Thu Jul 22 2010 David Malcolm - 1:1.4.1-6 +- actually add the patch this time + +* Thu Jul 22 2010 David Malcolm - 1:1.4.1-5 +- fix segfault within %%check on 2.7 (patch 2) + +* Wed Jul 21 2010 David Malcolm - 1:1.4.1-4 +- Rebuilt for https://fedoraproject.org/wiki/Features/Python_2.7/MassRebuild + +* Sun Jul 18 2010 Dan Horák 1.4.1-3 +- ignore the "Ticket #1299 second test" failure on s390(x) + +* Thu Jun 24 2010 Jef Spaleta 1.4.1-2 +- source commit fix + +* Thu Jun 24 2010 Jef Spaleta 1.4.1-1 +- New upstream release. Include backported doublefree patch + +* Mon Apr 26 2010 Jon Ciesla 1.3.0-8 +- Moved distutils back to the main package, BZ 572820. + +* Thu Apr 08 2010 Jon Ciesla 1.3.0-7 +- Reverted to 1.3.0 after upstream pulled 1.4.0, BZ 579065. + +* Tue Mar 02 2010 Jon Ciesla 1.4.0-5 +- Linking /usr/include/numpy to .h files, BZ 185079. + +* Tue Feb 16 2010 Jon Ciesla 1.4.0-4 +- Re-enabling atlas BR, dropping lapack Requires. + +* Wed Feb 10 2010 Jon Ciesla 1.4.0-3 +- Since the previous didn't work, Requiring lapack. + +* Tue Feb 09 2010 Jon Ciesla 1.4.0-2 +- Temporarily dropping atlas BR to work around 562577. + +* Fri Jan 22 2010 Jon Ciesla 1.4.0-1 +- 1.4.0. +- Dropped ARM patch, ARM support added upstream. + +* Tue Nov 17 2009 Jitesh Shah - 1.3.0-6.fa1 +- Add ARM support + +* Sat Jul 25 2009 Fedora Release Engineering - 1.3.0-6 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_12_Mass_Rebuild + +* Thu Jun 11 2009 Jon Ciesla 1.3.0-5 +- Fixed atlas BR, BZ 505376. + +* Fri Apr 17 2009 Jon Ciesla 1.3.0-4 +- EVR bump for pygame chainbuild. + +* Fri Apr 17 2009 Jon Ciesla 1.3.0-3 +- Moved linalg, fft back to main package. + +* Tue Apr 14 2009 Jon Ciesla 1.3.0-2 +- Split out f2py into subpackage, thanks Peter Robinson pbrobinson@gmail.com. + +* Tue Apr 07 2009 Jon Ciesla 1.3.0-1 +- Update to latest upstream. +- Fixed Source0 URL. + +* Thu Apr 02 2009 Jon Ciesla 1.3.0-0.rc1 +- Update to latest upstream. + +* Thu Mar 05 2009 Jon Ciesla 1.2.1-3 +- Require python-devel, BZ 488464. + +* Wed Feb 25 2009 Fedora Release Engineering - 1.2.1-2 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_11_Mass_Rebuild + +* Fri Dec 19 2008 Jon Ciesla 1.2.1-1 +- Update to 1.2.1. + +* Sat Nov 29 2008 Ignacio Vazquez-Abrams - 1.2.0-2 +- Rebuild for Python 2.6 + +* Tue Oct 07 2008 Jon Ciesla 1.2.0-1 +- New upstream release, added python-nose BR. BZ 465999. +- Using atlas blas, not blas-devel. BZ 461472. + +* Wed Aug 06 2008 Jon Ciesla 1.1.1-1 +- New upstream release + +* Thu May 29 2008 Jarod Wilson 1.1.0-1 +- New upstream release + +* Tue May 06 2008 Jarod Wilson 1.0.4-1 +- New upstream release + +* Mon Feb 11 2008 Jarod Wilson 1.0.3.1-2 +- Add python egg to %%files on f9+ + +* Wed Aug 22 2007 Jarod Wilson 1.0.3.1-1 +- New upstream release + +* Wed Jun 06 2007 Jarod Wilson 1.0.3-1 +- New upstream release + +* Mon May 14 2007 Jarod Wilson 1.0.2-2 +- Drop BR: atlas-devel, since it just provides binary-compat + blas and lapack libs. Atlas can still be optionally used + at runtime. (Note: this is all per the atlas maintainer). + +* Mon May 14 2007 Jarod Wilson 1.0.2-1 +- New upstream release + +* Tue Apr 17 2007 Jarod Wilson 1.0.1-4 +- Update gfortran patch to recognize latest gfortran f95 support +- Resolves rhbz#236444 + +* Fri Feb 23 2007 Jarod Wilson 1.0.1-3 +- Fix up cpuinfo bug (#229753). Upstream bug/change: + http://projects.scipy.org/scipy/scipy/ticket/349 + +* Thu Jan 04 2007 Jarod Wilson 1.0.1-2 +- Per discussion w/Jose Matos, Obsolete/Provide f2py, as the + stand-alone one is no longer supported/maintained upstream + +* Wed Dec 13 2006 Jarod Wilson 1.0.1-1 +- New upstream release + +* Tue Dec 12 2006 Jarod Wilson 1.0-2 +- Rebuild for python 2.5 + +* Wed Oct 25 2006 Jarod Wilson 1.0-1 +- New upstream release + +* Wed Sep 06 2006 Jarod Wilson 0.9.8-1 +- New upstream release + +* Wed Apr 26 2006 Ignacio Vazquez-Abrams 0.9.6-1 +- Upstream update + +* Thu Feb 16 2006 Ignacio Vazquez-Abrams 0.9.5-1 +- Upstream update + +* Mon Feb 13 2006 Ignacio Vazquez-Abrams 0.9.4-2 +- Rebuild for Fedora Extras 5 + +* Thu Feb 2 2006 Ignacio Vazquez-Abrams 0.9.4-1 +- Initial RPM release +- Added gfortran patch from Neal Becker