All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apfloat.internal.DoubleApfloatImpl Maven / Gradle / Ivy

The newest version!
/*
 * MIT License
 *
 * Copyright (c) 2002-2023 Mikko Tommila
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
package org.apfloat.internal;

import java.io.ObjectInputStream;
import java.io.PushbackReader;
import java.io.Writer;
import java.io.StringWriter;
import java.io.IOException;

import org.apfloat.Apfloat;
import org.apfloat.ApfloatContext;
import org.apfloat.ApfloatRuntimeException;
import org.apfloat.InfiniteExpansionException;
import org.apfloat.OverflowException;
import org.apfloat.spi.ApfloatImpl;
import org.apfloat.spi.DataStorageBuilder;
import org.apfloat.spi.DataStorage;
import org.apfloat.spi.ArrayAccess;
import org.apfloat.spi.AdditionBuilder;
import org.apfloat.spi.AdditionStrategy;
import org.apfloat.spi.ConvolutionBuilder;
import org.apfloat.spi.ConvolutionStrategy;
import org.apfloat.spi.Util;
import static org.apfloat.spi.RadixConstants.*;
import static org.apfloat.internal.DoubleRadixConstants.*;

/**
 * Immutable apfloat implementation class for the
 * double data element type.

* * The associated {@link DataStorage} is assumed to be immutable also. * This way performance can be improved by sharing the data storage between * different ApfloatImpl's and by only varying the * ApfloatImpl specific fields, like sign, precision and exponent.

* * This implementation doesn't necessarily store any extra digits for added * precision, so the last digit of any operation may be inaccurate. * * @version 1.11.0 * @author Mikko Tommila */ public class DoubleApfloatImpl extends DoubleBaseMath implements ApfloatImpl { // Implementation notes: // - The dataStorage must never contain leading zeros or trailing zeros // - If precision is reduced then the dataStorage can contain trailing zeros (physically in the middle) // - The dataStorage should not be unnecessarily subsequenced if precision is reduced e.g. to allow autoconvolution // - Precision is in digits but exponent is in base units private DoubleApfloatImpl(int sign, long precision, long exponent, DataStorage dataStorage, int radix) { super(radix); assert (sign == 0 || sign == -1 || sign == 1); assert (precision > 0); assert (sign != 0 || precision == Apfloat.INFINITE); assert (sign != 0 || exponent == 0); assert (sign != 0 || dataStorage == null); assert (sign == 0 || dataStorage != null); assert (exponent <= MAX_EXPONENT[radix] && exponent >= -MAX_EXPONENT[radix]); assert (dataStorage == null || dataStorage.isReadOnly()); this.sign = sign; this.precision = precision; this.exponent = exponent; this.dataStorage = dataStorage; this.radix = radix; } /** * Create a new DoubleApfloatImpl instance from a String. * * @param value The string to be parsed to a number. * @param precision The precision of the number (in digits of the radix). * @param radix The radix in which the number is created. * @param isInteger Specifies if the number to be parsed from the string is to be treated as an integer or not. * * @exception NumberFormatException If the number is not valid. */ public DoubleApfloatImpl(String value, long precision, int radix, boolean isInteger) throws NumberFormatException, ApfloatRuntimeException { super(checkRadix(radix)); assert (precision == Apfloat.DEFAULT || precision > 0); this.radix = radix; // Default sign if not specified this.sign = 1; int startIndex = -1, pointIndex = -1, expIndex = -1, leadingZeros = 0, trailingZeros = 0, digitSize = 0; // Scan through the string looking for various things for (int i = 0; i < value.length(); i++) { char c = value.charAt(i); int digit = Character.digit(c, radix); // Note that checking for a valid digit takes place before checking for e or E in the string if (digit == -1) { if (i == 0 && (c == '-' || c == '+')) { // Get sign this.sign = (c == '-' ? -1 : 1); } else if (!isInteger && c == '.' && pointIndex == -1) { // Mark decimal point location pointIndex = digitSize; } else if (!isInteger && (c == 'e' || c == 'E') && expIndex == -1) { // Mark index after which the exponent is specified expIndex = i; break; } else { throw new NumberFormatException("Invalid character: " + c + " at position " + i); } } else { if (leadingZeros == digitSize && digit == 0) { // Increase number of leading zeros leadingZeros++; } else if (startIndex == -1) { // Mark index where the significant digits start startIndex = i; } // Increase number of digits digitSize++; if (digit == 0) { // Increase number of trailing zeros trailingZeros++; } else { // Reset number of trailing zeros trailingZeros = 0; } } } // Check if no digits were specified if (digitSize == 0) { throw new NumberFormatException("No digits"); } // Check if this number is zero if (startIndex == -1) { this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // Default precision is number of significant digits, if not specified if (precision == Apfloat.DEFAULT) { assert (!isInteger); precision = digitSize - leadingZeros; } this.precision = precision; // Size of integer part int integerSize = (pointIndex >= 0 ? pointIndex : digitSize) - leadingZeros; // Read exponent as specified in string if (expIndex >= 0) { // Thanks to Charles Oliver Nutter for finding this bug String expString = value.substring(expIndex + 1); if (expString.startsWith("+")) { expString = expString.substring(1); } try { this.exponent = Long.parseLong(expString); } catch (NumberFormatException nfe) { throw new NumberFormatException("Invalid exponent: " + expString); } } else { this.exponent = 0; } // Do not allow the exponent to be too close to the limits (MIN_VALUE, MAX_VALUE), leave some slack int slack = BASE_DIGITS[radix]; // Check for overflow in exponent, roughly if (integerSize >= -slack && this.exponent >= Long.MAX_VALUE - integerSize - slack) { throw new NumberFormatException("Exponent overflow"); } else if (integerSize <= slack && this.exponent <= Long.MIN_VALUE - integerSize + slack) { // Underflow this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // Adjust exponent by decimal point location this.exponent += integerSize; // Exponent rounded towards positive infinity to base unit long baseExp = (this.exponent + (this.exponent > 0 ? BASE_DIGITS[radix] - 1 : 0)) / BASE_DIGITS[radix]; // Check for overflow in exponent as represented in base units if (baseExp > MAX_EXPONENT[this.radix]) { throw new OverflowException("Overflow"); } else if (baseExp < -MAX_EXPONENT[this.radix]) { // Underflow this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // Leading zeros in first base unit int digitsInBase = (int) (baseExp * BASE_DIGITS[radix] - this.exponent); // The stored exponent is really the one per base unit this.exponent = baseExp; // Remove leading and trailing zeros from size digitSize -= leadingZeros + trailingZeros; // Limit number of significant digits by specified precision digitSize = (int) Math.min(digitSize, precision); // Needed storage size in doubles int size = (int) getBasePrecision(digitSize, BASE_DIGITS[radix] - digitsInBase); this.dataStorage = createDataStorage(size); this.dataStorage.setSize(size); // Base unit that is constructed and stored to an element of the data storage double word = 0; DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.WRITE, 0, size); // Set the data for (int i = startIndex; digitSize > 0; i++) { char c = value.charAt(i); if (c == '.') { continue; } int digit = Character.digit(c, radix); word *= (double) radix; word += (double) digit; if (digitSize == 1) { // Last digit while (digitsInBase < BASE_DIGITS[radix] - 1) { // Fill last word with trailing zeros word *= (double) radix; digitsInBase++; } } if (++digitsInBase == BASE_DIGITS[radix]) { // Word is full, write word digitsInBase = 0; iterator.setDouble(word); iterator.next(); word = 0; } digitSize--; } assert (!iterator.hasNext()); this.dataStorage.setReadOnly(); } /** * Create a new DoubleApfloatImpl instance from a long. * * @param value The value of the number. * @param precision The precision of the number (in digits of the radix). * @param radix The radix in which the number is created. * * @exception NumberFormatException If the number is not valid. */ public DoubleApfloatImpl(long value, long precision, int radix) throws NumberFormatException, ApfloatRuntimeException { super(checkRadix(radix)); assert (precision > 0); this.radix = radix; // Faster to set now than calculate later this.isOne = (value == 1 ? 1 : 0); if (value > 0) { this.sign = 1; value = -value; // Calculate here as negative to handle 0x8000000000000000 } else if (value < 0) { this.sign = -1; } else { this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } this.precision = precision; int size; double[] data = new double[MAX_LONG_SIZE]; long longBase = (long) BASE[radix]; if (-longBase < value) { size = 1; // Nonzero data[MAX_LONG_SIZE - 1] = (double) -value; } else { for (size = 0; value != 0; size++) { long newValue = value / longBase; data[MAX_LONG_SIZE - 1 - size] = (double) (newValue * longBase - value); // Negated here value = newValue; } } this.exponent = size; this.initialDigits = getDigits(data[MAX_LONG_SIZE - size]); // Check if precision in doubles is less than size; truncate size if so long basePrecision = getBasePrecision(precision, this.initialDigits); if (basePrecision < size) { size = (int) basePrecision; } // Remove trailing zeros from data while (data[MAX_LONG_SIZE - 1 - (int) this.exponent + size] == 0) { size--; } this.dataStorage = createDataStorage(size); this.dataStorage.setSize(size); try (ArrayAccess arrayAccess = this.dataStorage.getArray(DataStorage.WRITE, 0, size)) { System.arraycopy(data, MAX_LONG_SIZE - (int) this.exponent, arrayAccess.getData(), arrayAccess.getOffset(), size); } this.dataStorage.setReadOnly(); } /** * Create a new DoubleApfloatImpl instance from a double. * * @param value The value of the number. * @param precision The precision of the number (in digits of the radix). * @param radix The radix in which the number is created. * * @exception NumberFormatException If the number is not valid. */ public DoubleApfloatImpl(double value, long precision, int radix) throws NumberFormatException, ApfloatRuntimeException { super(checkRadix(radix)); if (Double.isInfinite(value) || Double.isNaN(value)) { throw new NumberFormatException(value + " is not a valid number"); } this.radix = radix; if (value > 0) { this.sign = 1; } else if (value < 0) { this.sign = -1; value = -value; } else { this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } this.precision = precision; int size; double[] data = new double[MAX_DOUBLE_SIZE]; double doubleBase = (double) BASE[radix]; this.exponent = (long) Math.floor(Math.log(value) / Math.log(doubleBase)); // Avoid overflow in intermediate value if (this.exponent > 0) { value *= Math.pow(doubleBase, (double) -this.exponent); } else if (this.exponent < 0) { value *= Math.pow(doubleBase, (double) (-this.exponent - MAX_DOUBLE_SIZE)); value *= Math.pow(doubleBase, (double) MAX_DOUBLE_SIZE); } this.exponent++; if (value < 1.0) { // Round-off error in case the input was very close but just under the base, e.g. 9.999999999999996E-10 value = 1.0; } for (size = 0; size < MAX_DOUBLE_SIZE && value > 0.0; size++) { double tmp = Math.floor(value); assert (tmp <= doubleBase); if (tmp == doubleBase) { // Round-off error e.g. in case of the number being exactly 1/radix tmp -= 1.0; } data[size] = (double) tmp; value -= tmp; value *= doubleBase; } this.initialDigits = getDigits(data[0]); // Check if precision in doubles is less than size; truncate size if so long basePrecision = getBasePrecision(precision, this.initialDigits); if (basePrecision < size) { size = (int) basePrecision; } // Remove trailing zeros from data while (data[size - 1] == 0) { size--; } this.dataStorage = createDataStorage(size); this.dataStorage.setSize(size); try (ArrayAccess arrayAccess = this.dataStorage.getArray(DataStorage.WRITE, 0, size)) { System.arraycopy(data, 0, arrayAccess.getData(), arrayAccess.getOffset(), size); } this.dataStorage.setReadOnly(); } private static long readExponent(PushbackReader in) throws IOException, NumberFormatException { StringBuilder buffer = new StringBuilder(20); int input; for (long i = 0; (input = in.read()) != -1; i++) { char c = (char) input; int digit = Character.digit(c, 10); // Exponent is always in base 10 if (i == 0 && c == '-' || digit != -1) { buffer.append(c); } else { // Stop at first invalid character and put it back in.unread(input); break; } } return Long.parseLong(buffer.toString()); } /** * Create a new DoubleApfloatImpl instance reading from a stream.

* * Implementation note: this constructor calls the in stream's * single-character read() method. If the underlying stream doesn't * explicitly implement this method in some efficient way, but simply inherits it * from the Reader base class, performance will suffer as the default * Reader method creates a new char[1] on every call to * read(). * * @param in The stream to read from. * @param precision The precision of the number (in digits of the radix). * @param radix The radix in which the number is created. * @param isInteger Specifies if the number to be parsed from the stream is to be treated as an integer or not. * * @exception IOException If an I/O error occurs accessing the stream. * @exception NumberFormatException If the number is not valid. */ public DoubleApfloatImpl(PushbackReader in, long precision, int radix, boolean isInteger) throws IOException, NumberFormatException, ApfloatRuntimeException { super(checkRadix(radix)); assert (precision == Apfloat.DEFAULT || precision > 0); this.radix = radix; // Default sign if not specified this.sign = 1; // Allocate a reasonable memory block, since we don't know how much data to expect long initialSize = getBlockSize(), previousAllocatedSize = 0, allocatedSize = initialSize; this.dataStorage = createDataStorage(initialSize); this.dataStorage.setSize(initialSize); // Base unit that is constructed and stored to an element of the data storage double word = 0; // Number of digits stored in word int digitsInBase = 0; DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.WRITE, previousAllocatedSize, allocatedSize); int input; long actualSize = 0, startIndex = -1, pointIndex = -1, leadingZeros = 0, trailingZeros = 0, digitSize = 0; // Scan through the string looking for various things for (long i = 0; (input = in.read()) != -1; i++) { char c = (char) input; int digit = Character.digit(c, radix); // Note that checking for a valid digit takes place before checking for e or E in the string if (digit == -1) { if (i == 0 && (c == '-' || c == '+')) { // Get sign this.sign = (c == '-' ? -1 : 1); } else if (!isInteger && c == '.' && pointIndex == -1) { // Mark decimal point location pointIndex = digitSize; } else if (!isInteger && digitSize > 0 && (c == 'e' || c == 'E')) { // Read the exponent and stop this.exponent = readExponent(in); break; } else { // Stop at first invalid character and put it back in.unread(input); break; } } else { if (leadingZeros == digitSize && digit == 0) { // Increase number of leading zeros leadingZeros++; } else { if (startIndex == -1) { // Mark index where the significant digits start startIndex = i; } // Set the data word *= (double) radix; word += (double) digit; // Reallocate storage if needed; done here to prepare storing last (partial) word if (actualSize == allocatedSize) { if (actualSize == initialSize) { // Initial memory block size exceeded; prepare to allocate anything DataStorage dataStorage = createDataStorage(Long.MAX_VALUE / Double.BYTES); dataStorage.copyFrom(this.dataStorage, actualSize); this.dataStorage = dataStorage; } previousAllocatedSize = allocatedSize; allocatedSize += getBlockSize(); this.dataStorage.setSize(allocatedSize); iterator.close(); iterator = this.dataStorage.iterator(DataStorage.WRITE, previousAllocatedSize, allocatedSize); } if (++digitsInBase == BASE_DIGITS[radix]) { // Word is full, write word digitsInBase = 0; iterator.setDouble(word); iterator.next(); word = 0; actualSize++; } } // Increase number of digits digitSize++; if (digit == 0) { // Increase number of trailing zeros trailingZeros++; } else { // Reset number of trailing zeros trailingZeros = 0; } } } // Check if no digits were specified if (digitSize == 0) { throw new NumberFormatException("No digits"); } // Check if this number is zero if (startIndex == -1) { this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // Handle last word if (digitsInBase > 0 && word != 0) { // Last digit while (digitsInBase < BASE_DIGITS[radix]) { // Fill last word with trailing zeros word *= (double) radix; digitsInBase++; } // Write word iterator.setDouble(word); actualSize++; } iterator.close(); // Default precision is number of significant digits, if not specified if (precision == Apfloat.DEFAULT) { assert (!isInteger); precision = digitSize - leadingZeros; } this.precision = precision; // Size of integer part long integerSize = (pointIndex >= 0 ? pointIndex : digitSize) - leadingZeros; // Do not allow the exponent to be too close to the limits (MIN_VALUE, MAX_VALUE), leave some slack int slack = BASE_DIGITS[radix]; // Check for overflow in exponent, roughly if (integerSize >= -slack && this.exponent >= Long.MAX_VALUE - integerSize - slack) { throw new NumberFormatException("Exponent overflow"); } else if (integerSize <= slack && this.exponent <= Long.MIN_VALUE - integerSize + slack) { // Underflow this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // Adjust exponent by decimal point location this.exponent += integerSize; // Exponent rounded towards negative infinity to base unit long baseExp = (this.exponent - (this.exponent < 0 ? BASE_DIGITS[radix] - 1 : 0)) / BASE_DIGITS[radix]; // Check for overflow in exponent as represented in base units if (baseExp > MAX_EXPONENT[this.radix]) { throw new OverflowException("Overflow"); } else if (baseExp < -MAX_EXPONENT[this.radix]) { // Underflow this.sign = 0; this.precision = Apfloat.INFINITE; this.exponent = 0; this.dataStorage = null; return; } // How much the data needs to be shifted int bias = (int) (this.exponent - baseExp * BASE_DIGITS[radix]); // The stored exponent is really the one per base unit this.exponent = baseExp; // Remove leading and trailing zeros from size digitSize -= leadingZeros + trailingZeros; // Limit number of significant digits by specified precision digitSize = Math.min(digitSize, precision); // Needed storage size in doubles actualSize = (digitSize + BASE_DIGITS[radix] - 1) / BASE_DIGITS[radix]; // Truncate allocated space to actually used amount this.dataStorage.setSize(actualSize); this.dataStorage.setReadOnly(); if (bias != 0) { // Shift by bias long factor = 1; for (int i = 0; i < bias; i++) { factor *= radix; } DoubleApfloatImpl tmp = (DoubleApfloatImpl) multiply(new DoubleApfloatImpl(factor, Apfloat.INFINITE, radix)); this.exponent = tmp.exponent; this.dataStorage = tmp.dataStorage; this.initialDigits = UNDEFINED; // Needs to be reset } } // Returns number of trailing zeros before specified index private static long getTrailingZeros(DataStorage dataStorage, long index) throws ApfloatRuntimeException { long count = 0; try (DataStorage.Iterator iterator = dataStorage.iterator(DataStorage.READ, index, 0)) { while (iterator.hasNext()) { if (iterator.getDouble() != 0) { break; } iterator.next(); count++; } } return count; } // Returns number of leading zeros starting from specified index private static long getLeadingZeros(DataStorage dataStorage, long index) throws ApfloatRuntimeException { long count = 0; try (DataStorage.Iterator iterator = dataStorage.iterator(DataStorage.READ, index, dataStorage.getSize())) { while (iterator.hasNext()) { if (iterator.getDouble() != 0) { break; } iterator.next(); count++; } } return count; } @Override public ApfloatImpl addOrSubtract(ApfloatImpl x, boolean subtract) throws ApfloatRuntimeException { if (!(x instanceof DoubleApfloatImpl)) { throw new ImplementationMismatchException("Wrong operand type: " + x.getClass().getName()); } DoubleApfloatImpl that = (DoubleApfloatImpl) x; if (this.radix != that.radix) { throw new RadixMismatchException("Cannot use numbers with different radixes: " + this.radix + " and " + that.radix); } assert (this.sign != 0); assert (that.sign != 0); int realThatSign = (subtract ? -that.sign : that.sign); boolean reallySubtract = (this.sign != realThatSign); ApfloatContext ctx = ApfloatContext.getContext(); AdditionBuilder additionBuilder = ctx.getBuilderFactory().getAdditionBuilder(Double.TYPE); AdditionStrategy additionStrategy = additionBuilder.createAddition(this.radix); int sign; long exponent, precision; DataStorage dataStorage; if (this == that) { if (reallySubtract) { // x - x = 0 return zero(); } else { // x + x = 2 * x sign = this.sign; exponent = this.exponent; precision = this.precision; long size = getSize() + 1; dataStorage = createDataStorage(size); dataStorage.setSize(size); double carry; DataStorage.Iterator src1 = this.dataStorage.iterator(DataStorage.READ, size - 1, 0), src2 = this.dataStorage.iterator(DataStorage.READ, size - 1, 0); // Sub-optimal: could be the same try (DataStorage.Iterator dst = dataStorage.iterator(DataStorage.WRITE, size, 0)) { carry = additionStrategy.add(src1, src2, (double) 0, dst, size - 1); dst.setDouble(carry); } size -= getTrailingZeros(dataStorage, size); // Check if carry occurred int carrySize = (int) carry, leadingZeros = 1 - carrySize; dataStorage = dataStorage.subsequence(leadingZeros, size - leadingZeros); exponent += carrySize; if (this.exponent == MAX_EXPONENT[this.radix] && carrySize > 0) { throw new OverflowException("Overflow"); } if (precision != Apfloat.INFINITE && (carrySize > 0 || getInitialDigits(dataStorage) > getInitialDigits())) { // Carry overflow for most significant digit; number of significant digits increases by one precision++; } } } else { // Now this != that int comparison; if (scale() > that.scale()) { comparison = 1; } else if (scale() < that.scale()) { comparison = -1; } else if (reallySubtract) { comparison = compareMantissaTo(that); // Might be sub-optimal, but a more efficient algorithm would be complicated } else { comparison = 1; // Add equally big numbers; arbitrarily choose one } DoubleApfloatImpl big, small; if (comparison > 0) { big = this; small = that; sign = this.sign; } else if (comparison < 0) { big = that; small = this; sign = realThatSign; } else { // x - x = 0 return zero(); } long scaleDifference = big.scale() - small.scale(), exponentDifference, size, bigSize, smallSize; if (scaleDifference < 0) { // Small number is completely insignificantly small compared to big precision = big.precision; exponent = big.exponent; bigSize = big.getSize(); smallSize = 0; size = bigSize; exponentDifference = bigSize; } else { precision = Math.min(big.precision, Util.ifFinite(small.precision, scaleDifference + small.precision)); // Detects overflow also long basePrecision = Math.min(MAX_EXPONENT[this.radix], getBasePrecision(precision, big.getInitialDigits())); exponent = big.exponent; exponentDifference = big.exponent - small.exponent; size = Math.min(basePrecision, Math.max(big.getSize(), exponentDifference + small.getSize())); bigSize = Math.min(size, big.getSize()); smallSize = Math.max(0, Math.min(size - exponentDifference, small.getSize())); } long dstSize = size + 1; // One extra word for carry overflow dataStorage = createDataStorage(dstSize); dataStorage.setSize(dstSize); double carry = 0; DataStorage.Iterator src1 = big.dataStorage.iterator(DataStorage.READ, bigSize, 0), src2 = small.dataStorage.iterator(DataStorage.READ, smallSize, 0); try (DataStorage.Iterator dst = dataStorage.iterator(DataStorage.WRITE, dstSize, 0)) { // big: XXXXXXXX XXXX // small: XXXXXXXX or XXXX // This part: XXXX XXXX if (size > bigSize) { long blockSize = Math.min(size - bigSize, smallSize); if (reallySubtract) { carry = additionStrategy.subtract(null, src2, carry, dst, blockSize); } else { carry = additionStrategy.add(null, src2, carry, dst, blockSize); } } // big: XXXXXXXXXXXX // small: XXXX // This part: XXXX else if (size > exponentDifference + smallSize) { long blockSize = size - exponentDifference - smallSize; if (reallySubtract) { carry = additionStrategy.subtract(src1, null, carry, dst, blockSize); } else { carry = additionStrategy.add(src1, null, carry, dst, blockSize); } } // big: XXXX // small: XXXX // This part: XXXX if (exponentDifference > bigSize) { long blockSize = exponentDifference - bigSize; if (reallySubtract) { carry = additionStrategy.subtract(null, null, carry, dst, blockSize); } else { carry = additionStrategy.add(null, null, carry, dst, blockSize); } } // big: XXXXXXXX XXXXXXXXXXXX // small: XXXXXXXX or XXXX // This part: XXXX XXXX else if (bigSize > exponentDifference) { long blockSize = Math.min(bigSize - exponentDifference, smallSize); if (reallySubtract) { carry = additionStrategy.subtract(src1, src2, carry, dst, blockSize); } else { carry = additionStrategy.add(src1, src2, carry, dst, blockSize); } } // big: XXXXXXXX XXXXXXXXXXXX XXXX // small: XXXXXXXX or XXXX or XXXX // This part: XXXX XXXX XXXX if (exponentDifference > 0) { long blockSize = Math.min(bigSize, exponentDifference); if (reallySubtract) { carry = additionStrategy.subtract(src1, null, carry, dst, blockSize); } else { carry = additionStrategy.add(src1, null, carry, dst, blockSize); } } // Set most significant word dst.setDouble(carry); } long leadingZeros; if (reallySubtract) { // Get denormalization leadingZeros = getLeadingZeros(dataStorage, 0); assert (leadingZeros <= size); } else { // Check if carry occurred up to and including most significant word leadingZeros = (carry == 0 ? 1 : 0); if (this.exponent == MAX_EXPONENT[this.radix] && leadingZeros == 0) { throw new OverflowException("Overflow"); } } dstSize -= getTrailingZeros(dataStorage, dstSize); dataStorage = dataStorage.subsequence(leadingZeros, dstSize - leadingZeros); exponent += 1 - leadingZeros; if (exponent < -MAX_EXPONENT[this.radix]) { // Underflow return zero(); } if (precision != Apfloat.INFINITE) { // If scale of number changes, the number of significant digits changes accordingly long scaleChange = (1 - leadingZeros) * BASE_DIGITS[this.radix] + getInitialDigits(dataStorage) - big.getInitialDigits(); if (-scaleChange >= precision) { // All significant digits were lost anyway, due to trailing garbage digits return zero(); } precision += scaleChange; precision = (precision <= 0 ? Apfloat.INFINITE : precision); // Detect overflow } } dataStorage.setReadOnly(); return new DoubleApfloatImpl(sign, precision, exponent, dataStorage, this.radix); } @Override public ApfloatImpl multiply(ApfloatImpl x) throws ApfloatRuntimeException { if (!(x instanceof DoubleApfloatImpl)) { throw new ImplementationMismatchException("Wrong operand type: " + x.getClass().getName()); } DoubleApfloatImpl that = (DoubleApfloatImpl) x; if (this.radix != that.radix) { throw new RadixMismatchException("Cannot multiply numbers with different radixes: " + this.radix + " and " + that.radix); } int sign = this.sign * that.sign; if (sign == 0) { return zero(); } long exponent = this.exponent + that.exponent; if (exponent > MAX_EXPONENT[this.radix]) { throw new OverflowException("Overflow"); } else if (exponent < -MAX_EXPONENT[this.radix]) { // Underflow return zero(); } long precision = Math.min(this.precision, that.precision), basePrecision = getBasePrecision(precision, 0), // Round up thisSize = getSize(), thatSize = that.getSize(), size = Math.min(Util.ifFinite(basePrecision, basePrecision + 1), thisSize + thatSize), // Reserve one extra word for carry thisDataSize = Math.min(thisSize, basePrecision), thatDataSize = Math.min(thatSize, basePrecision); DataStorage thisDataStorage = this.dataStorage.subsequence(0, thisDataSize), thatDataStorage = (this.dataStorage == that.dataStorage ? thisDataStorage : // Enable auto-convolution that.dataStorage.subsequence(0, thatDataSize)); ApfloatContext ctx = ApfloatContext.getContext(); ConvolutionBuilder convolutionBuilder = ctx.getBuilderFactory().getConvolutionBuilder(); ConvolutionStrategy convolutionStrategy = convolutionBuilder.createConvolution(this.radix, thisDataSize, thatDataSize, size); // Possibly sub-optimal: could look up trailing zeros of the subsequences DataStorage dataStorage = convolutionStrategy.convolute(thisDataStorage, thatDataStorage, size); // Check if carry occurred up to and including most significant word int leadingZeros = (getMostSignificantWord(dataStorage) == 0 ? 1 : 0); exponent -= leadingZeros; if (exponent < -MAX_EXPONENT[this.radix]) { // Underflow return zero(); } size -= leadingZeros; dataStorage = dataStorage.subsequence(leadingZeros, size); size = Math.min(size, getBasePrecision(precision, getInitialDigits(dataStorage))); size -= getTrailingZeros(dataStorage, size); dataStorage = dataStorage.subsequence(0, size); dataStorage.setReadOnly(); return new DoubleApfloatImpl(sign, precision, exponent, dataStorage, this.radix); } @Override public boolean isShort() throws ApfloatRuntimeException { return (this.sign == 0 || getSize() == 1); } @Override public ApfloatImpl divideShort(ApfloatImpl x) throws ApfloatRuntimeException { if (!(x instanceof DoubleApfloatImpl)) { throw new ImplementationMismatchException("Wrong operand type: " + x.getClass().getName()); } DoubleApfloatImpl that = (DoubleApfloatImpl) x; if (this.radix != that.radix) { throw new RadixMismatchException("Cannot divide numbers with different radixes: " + this.radix + " and " + that.radix); } assert (this.sign != 0); assert (that.sign != 0); int sign = this.sign * that.sign; long exponent = this.exponent - that.exponent + 1; if (exponent > MAX_EXPONENT[this.radix]) { throw new OverflowException("Overflow"); } else if (exponent < -MAX_EXPONENT[this.radix]) { // Underflow return zero(); } long precision = Math.min(this.precision, that.precision), basePrecision = getBasePrecision(), thisDataSize = Math.min(getSize(), basePrecision); DataStorage dataStorage; double divisor = getMostSignificantWord(that.dataStorage); if (divisor == (double) 1) { long size = thisDataSize - getTrailingZeros(this.dataStorage, thisDataSize); dataStorage = this.dataStorage.subsequence(0, size); } else { ApfloatContext ctx = ApfloatContext.getContext(); AdditionBuilder additionBuilder = ctx.getBuilderFactory().getAdditionBuilder(Double.TYPE); AdditionStrategy additionStrategy = additionBuilder.createAddition(this.radix); long size; double carry; // Check for finite or infinite result sequence double dividend = divisor; // Check that the factorization of the divisor consists entirely of factors of the base // E.g. if base is 10=2*5 then the divisor should be 2^n*5^m for (int i = 0; i < RADIX_FACTORS[this.radix].length; i++) { double factor = RADIX_FACTORS[this.radix][i], quotient; // Keep dividing by factor as long as dividend % factor == 0 // that is remove factors of the base from the divisor while ((dividend - factor * (quotient = (double) (long) (dividend / factor))) == 0) { dividend = quotient; } } // Check if the divisor was factored all the way to one by just dividing by factors of the base if (dividend != (double) 1) { // Divisor does not contain only factors of the base; infinite nonzero sequence if (basePrecision == Apfloat.INFINITE) { throw new InfiniteExpansionException("Cannot perform inexact division to infinite precision"); } size = basePrecision; } else { // Divisor contains only factors of the base; calculate maximum sequence length carry = (double) 1; DataStorage.Iterator dummy = new DataStorage.Iterator() { @Override public void setDouble(double value) {} @Override public void next() {} private static final long serialVersionUID = 1L; }; long sequenceSize; for (sequenceSize = 0; carry != 0; sequenceSize++) { carry = additionStrategy.divide(null, divisor, carry, dummy, 1); } size = Math.min(basePrecision, thisDataSize + sequenceSize); } // One extra word for result in case the initial word becomes zero; to avoid loss of precision size++; dataStorage = createDataStorage(size); dataStorage.setSize(size); DataStorage.Iterator src = this.dataStorage.iterator(DataStorage.READ, 0, thisDataSize), dst = dataStorage.iterator(DataStorage.WRITE, 0, size); // Perform actual division carry = additionStrategy.divide(src, divisor, (double) 0, dst, thisDataSize); // Produce the trailing sequence of digits due to inexact division carry = additionStrategy.divide(null, divisor, carry, dst, size - thisDataSize); size -= getTrailingZeros(dataStorage, size); // Check if initial word of result is zero int leadingZeros = (getMostSignificantWord() < divisor ? 1 : 0); dataStorage = dataStorage.subsequence(leadingZeros, size - leadingZeros); exponent -= leadingZeros; if (exponent < -MAX_EXPONENT[this.radix]) { // Underflow return zero(); } dataStorage.setReadOnly(); } return new DoubleApfloatImpl(sign, precision, exponent, dataStorage, this.radix); } @Override public ApfloatImpl absFloor() throws ApfloatRuntimeException { if (this.sign == 0 || this.exponent >= this.dataStorage.getSize()) // Is integer already, with no extra hidden trailing digits { return precision(Apfloat.INFINITE); } else if (this.exponent <= 0) // Is less than one in absolute value { return zero(); } long size = this.exponent; // Size of integer part, now that this.dataStorage.getSize() > this.exponent size -= getTrailingZeros(this.dataStorage, size); DataStorage dataStorage = this.dataStorage.subsequence(0, size); ApfloatImpl apfloatImpl = new DoubleApfloatImpl(this.sign, Apfloat.INFINITE, this.exponent, dataStorage, this.radix); return apfloatImpl; } @Override public ApfloatImpl absCeil() throws ApfloatRuntimeException { if (this.sign == 0) { return this; } long exponent; DataStorage dataStorage; if (this.exponent <= 0) { // Number is < 1 but > 0; result is one int size = 1; dataStorage = createDataStorage(size); dataStorage.setSize(size); try (ArrayAccess arrayAccess = dataStorage.getArray(DataStorage.WRITE, 0, size)) { arrayAccess.getDoubleData()[arrayAccess.getOffset()] = (double) 1; } exponent = 1; } else if (getSize() <= this.exponent || // Check if the fractional part is nonzero findMismatch(getZeroPaddedIterator(this.exponent, getSize()), ZERO_ITERATOR, getSize() - this.exponent) < 0) { // Fractional part is zero; the result is the number itself (to infinite precision) long size = Math.min(this.dataStorage.getSize(), this.exponent); size -= getTrailingZeros(this.dataStorage, size); dataStorage = this.dataStorage.subsequence(0, size); // Ensure truncation exponent = this.exponent; } else { // Fractional part is nonzero; round up ApfloatContext ctx = ApfloatContext.getContext(); AdditionBuilder additionBuilder = ctx.getBuilderFactory().getAdditionBuilder(Double.TYPE); AdditionStrategy additionStrategy = additionBuilder.createAddition(this.radix); long size = this.exponent; // Size of integer part dataStorage = createDataStorage(size + 1); // Reserve room for carry overflow dataStorage.setSize(size + 1); double carry; try (DataStorage.Iterator src = this.dataStorage.iterator(DataStorage.READ, size, 0); DataStorage.Iterator dst = dataStorage.iterator(DataStorage.WRITE, size + 1, 0)) { carry = additionStrategy.add(src, null, (double) 1, dst, size); // Add carry dst.setDouble(carry); // Set leading double as overflow carry } int carrySize = (int) carry; // For adjusting size, if carry did overflow or not size -= getTrailingZeros(dataStorage, size + 1); dataStorage = dataStorage.subsequence(1 - carrySize, size + carrySize); exponent = this.exponent + carrySize; } dataStorage.setReadOnly(); ApfloatImpl apfloatImpl = new DoubleApfloatImpl(this.sign, Apfloat.INFINITE, exponent, dataStorage, this.radix); return apfloatImpl; } @Override public ApfloatImpl frac() throws ApfloatRuntimeException { if (this.sign == 0 || this.exponent <= 0) // Is less than one in absolute value already { return this; } if (this.exponent >= getSize()) // Is an integer, fractional part is zero { return zero(); } long size = this.dataStorage.getSize() - this.exponent; // Size of fractional part, now that getSize() > this.exponent long leadingZeros = getLeadingZeros(this.dataStorage, this.exponent); if (this.exponent + leadingZeros >= getSize()) { // All significant digits were lost, only trailing garbage digits return zero(); } DataStorage dataStorage = this.dataStorage.subsequence(this.exponent + leadingZeros, size - leadingZeros); long precision; if (this.precision != Apfloat.INFINITE) { // Precision is reduced as the integer part is omitted, plus any leading zeros precision = this.precision - getInitialDigits() - (this.exponent + leadingZeros) * BASE_DIGITS[this.radix] + getInitialDigits(dataStorage); if (precision <= 0) { // All significant digits were lost anyway, only trailing garbage digits return zero(); } } else { precision = Apfloat.INFINITE; } long exponent = -leadingZeros; ApfloatImpl apfloatImpl = new DoubleApfloatImpl(this.sign, precision, exponent, dataStorage, this.radix); return apfloatImpl; } private ApfloatImpl zero() { return new DoubleApfloatImpl(0, Apfloat.INFINITE, 0, null, this.radix); } @Override public int radix() { return this.radix; } @Override public long precision() { return this.precision; } @Override public long size() throws ApfloatRuntimeException { assert (this.dataStorage != null); if (this.size == 0) { // Writes and reads of volatile long values are always atomic so multiple threads can read and write this at the same time this.size = getInitialDigits() + (getSize() - 1) * BASE_DIGITS[this.radix] - getLeastZeros(); } return this.size; } // Get number of trailing zeros private long getLeastZeros() throws ApfloatRuntimeException { if (this.leastZeros == UNDEFINED) { // Cache the value // NOTE: This is not synchronized; it's OK if multiple threads set this at the same time // Writes and reads of volatile long values are always atomic so multiple threads can read and write this at the same time long index = getSize() - 1; double word = getWord(index); word = getLeastSignificantWord(index, word); long leastZeros = 0; if (word == 0) { // Usually the last word is nonzero but in case precision was later changed, it might be zero long trailingZeros = getTrailingZeros(this.dataStorage, index) + 1; index -= trailingZeros; word = getWord(index); word = getLeastSignificantWord(index, word); leastZeros += trailingZeros * BASE_DIGITS[this.radix]; } assert (word != 0); while (word % this.radix == 0) { leastZeros++; word /= this.radix; } this.leastZeros = leastZeros; } return this.leastZeros; } @Override public ApfloatImpl precision(long precision) { if (this.sign == 0 || precision == this.precision) { return this; } else { return new DoubleApfloatImpl(this.sign, precision, this.exponent, this.dataStorage, this.radix); } } @Override public long scale() throws ApfloatRuntimeException { assert (this.dataStorage != null); return (this.exponent - 1) * BASE_DIGITS[this.radix] + getInitialDigits(); } @Override public int signum() { return this.sign; } @Override public ApfloatImpl negate() throws ApfloatRuntimeException { return new DoubleApfloatImpl(-this.sign, this.precision, this.exponent, this.dataStorage, this.radix); } @Override public double doubleValue() { if (this.sign == 0) { return 0.0; } double value = 0.0, doubleBase = (double) BASE[this.radix]; int size = (int) Math.min(MAX_DOUBLE_SIZE, getSize()); DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.READ, size, 0); while (iterator.hasNext()) { value += (double) iterator.getDouble(); value /= doubleBase; iterator.next(); } // If the end result fits in a double, any intermediate calculation must not overflow // Note that 1/BASE <= value < 1 if (this.exponent > 0) { return this.sign * value * Math.pow((double) BASE[this.radix], (double) (this.exponent - 1)) * BASE[this.radix]; } else { return this.sign * value * Math.pow((double) BASE[this.radix], (double) this.exponent); } } @Override public long longValue() { if (this.sign == 0 || this.exponent <= 0) { return 0; } else if (this.exponent > MAX_LONG_SIZE) { // Overflow for sure return (this.sign > 0 ? Long.MAX_VALUE : Long.MIN_VALUE); } long value = 0, longBase = (long) BASE[this.radix], maxPrevious = Long.MIN_VALUE / longBase; // Number of words in integer part of the number int size = (int) Math.min(this.exponent, getSize()); try (DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.READ, 0, size)) { for (int i = 0; i < (int) this.exponent; i++) { if (value < maxPrevious) { // Overflow value = 0; break; } value *= longBase; if (i < size) { value -= (long) iterator.getDouble(); // Calculate value negated to handle 0x8000000000000000 iterator.next(); } } } if (value == Long.MIN_VALUE || value >= 0) { // Overflow return (this.sign > 0 ? Long.MAX_VALUE : Long.MIN_VALUE); } else { return -this.sign * value; } } @Override public boolean isOne() throws ApfloatRuntimeException { if (this.isOne == UNDEFINED) { // Cache the value // NOTE: This is not synchronized; it's OK if multiple threads set this at the same time this.isOne = (this.sign == 1 && this.exponent == 1 && getSize() == 1 && getMostSignificantWord() == (double) 1 ? 1 : 0); } return (this.isOne == 1); } @Override public long equalDigits(ApfloatImpl x) throws ApfloatRuntimeException { if (!(x instanceof DoubleApfloatImpl)) { throw new ImplementationMismatchException("Wrong operand type: " + x.getClass().getName()); } DoubleApfloatImpl that = (DoubleApfloatImpl) x; if (this.sign == 0 && that.sign == 0) // Both are zero { return Apfloat.INFINITE; } else if (this.sign != that.sign) // No match { return 0; } else if (this.radix != that.radix) { throw new RadixMismatchException("Cannot compare values with different radixes: " + this.radix + " and " + that.radix); } long thisScale = scale(), thatScale = that.scale(), minScale = Math.min(thisScale, thatScale), maxScale = Math.max(thisScale, thatScale); if (maxScale - 1 > minScale) // No match { return 0; } // Need to compare mantissas long thisSize = getSize(), thatSize = that.getSize(), size = Math.max(thisSize, thatSize), result = Math.min(this.precision, that.precision); // If mantissas are identical try (DataStorage.Iterator thisIterator = getZeroPaddedIterator(0, thisSize); DataStorage.Iterator thatIterator = that.getZeroPaddedIterator(0, thatSize)) { long index; int lastMatchingDigits = -1; // Will be used for deferred comparison hanging in last word, e.g. this = 1.000000000, that = 0.999999999 double carry, base = BASE[this.radix]; if (this.exponent > that.exponent) { // Possible case this = 1.0000000, that = 0.9999999 double value = thisIterator.getDouble(); // Check first word if (value != (double) 1) { // No match return 0; } carry = base; thisIterator.next(); } else if (this.exponent < that.exponent) { // Possible case this = 0.9999999, that = 1.0000000 double value = thatIterator.getDouble(); // Check first word if (value != (double) 1) { // No match return 0; } carry = -base; thatIterator.next(); } else { // Trivial case, e.g. this = 111234, that = 111567 carry = 0; } // Calculate this - that, stopping at first difference for (index = 0; index < size; index++) { double value = thisIterator.getDouble() - thatIterator.getDouble() + carry; if (value == 0) { // Trivial case; words are equal carry = 0; } else if (Math.abs(value) > (double) 1) { // Mismatch found if (Math.abs(value) >= base) { // Deferred comparison, e.g. this = 1.0000000002, that = 0.9999999991 lastMatchingDigits = -1; } else { // Any trivial cases and e.g. this = 1.0000000001, that = 0.9999999992 lastMatchingDigits = BASE_DIGITS[this.radix] - getDigits(Math.abs(value)); } break; } else if (value == (double) 1) { // Case this = 1.0000000..., that = 0.9999999... carry = base; } else if (value == (double) -1) { // Case this = 0.9999999..., that = 1.0000000... carry = -base; } thisIterator.next(); thatIterator.next(); } if (index < size || carry != 0) // Mismatch found { long initialMatchingDigits = (this.exponent == that.exponent ? Math.min(getInitialDigits(), that.getInitialDigits()) : // Normal case, e.g. this = 10, that = 5 BASE_DIGITS[this.radix]); // Special case, e.g. this = 1.0, that = 0.9 // Note that this works even if index == 0 long middleMatchingDigits = (index - 1) * BASE_DIGITS[this.radix]; // This is correct even if exponents are different // Limit by available precision result = Math.min(result, initialMatchingDigits + middleMatchingDigits + lastMatchingDigits); // Handle some cases e.g. 0.15 vs. 0.04 result = Math.max(result, 0); } } return result; } @Override public int compareTo(ApfloatImpl x) throws ApfloatRuntimeException { if (!(x instanceof DoubleApfloatImpl)) { throw new ImplementationMismatchException("Wrong operand type: " + x.getClass().getName()); } DoubleApfloatImpl that = (DoubleApfloatImpl) x; if (this.sign == 0 && that.sign == 0) { return 0; } else if (this.sign < that.sign) // Now we know that not both are zero { return -1; } else if (this.sign > that.sign) { return 1; } else if (this.radix != that.radix) { throw new RadixMismatchException("Cannot compare values with different radixes: " + this.radix + " and " + that.radix); } else if (scale() < that.scale()) // Now we know that both have same sign (which is not zero) { return -this.sign; } else if (scale() > that.scale()) { return this.sign; } // Need to compare mantissas return this.sign * compareMantissaTo(that); } // Returns an iterator for this number's data storage from start to end, // least significant word is correctly truncated with getLeastSignificantWord(), // after that the iterator returns zeros only private DataStorage.Iterator getZeroPaddedIterator(long start, long end) throws ApfloatRuntimeException { DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.READ, start, end); return new DataStorage.Iterator() { @Override public double getDouble() throws ApfloatRuntimeException { double value; if (this.index < end) { value = iterator.getDouble(); if (this.index == end - 1) { value = getLeastSignificantWord(this.index, value); } } else { value = 0; } return value; } @Override public void next() throws ApfloatRuntimeException { if (this.index < end) { iterator.next(); this.index++; } } @Override public void close() throws ApfloatRuntimeException { iterator.close(); } private static final long serialVersionUID = 1L; private long index = start; }; } // Compare absolute values of mantissas private int compareMantissaTo(DoubleApfloatImpl that) throws ApfloatRuntimeException { int result = 0; long thisSize = getSize(), thatSize = that.getSize(), size = Math.max(thisSize, thatSize); try (DataStorage.Iterator thisIterator = getZeroPaddedIterator(0, thisSize); DataStorage.Iterator thatIterator = that.getZeroPaddedIterator(0, thatSize)) { long index = findMismatch(thisIterator, thatIterator, size); if (index >= 0) // Mismatch found { double thisValue = thisIterator.getDouble(), thatValue = thatIterator.getDouble(); if (thisValue < thatValue) { result = -1; } else if (thisValue > thatValue) { result = 1; } } } return result; } // Returns index of first mismatching double, or -1 if mantissas are equal // Iterators are left to point to the mismatching words private long findMismatch(DataStorage.Iterator thisIterator, DataStorage.Iterator thatIterator, long size) throws ApfloatRuntimeException { for (long index = 0; index < size; index++) { double thisValue = thisIterator.getDouble(), thatValue = thatIterator.getDouble(); if (thisValue != thatValue) { return index; } thisIterator.next(); thatIterator.next(); } // All searched words matched exactly return -1; } // Truncate insignificant digits from the last double of the number private double getLeastSignificantWord(long index, double word) throws ApfloatRuntimeException { if (this.precision == Apfloat.INFINITE) { return word; } // Total digits including the specified index long digits = getInitialDigits() + index * BASE_DIGITS[this.radix]; if (this.precision >= digits) { return word; } // Assert that the second array access will not be out of bounds double divisor = MINIMUM_FOR_DIGITS[this.radix][(int) (digits - this.precision)]; return (double) (long) (word / divisor) * divisor; } /** * Compares this object to the specified object. * * @param obj The object to compare with. * * @return true if the objects are equal; false otherwise. */ @Override public boolean equals(Object obj) { if (!(obj instanceof ApfloatImpl)) { return false; } ApfloatImpl thatImpl = (ApfloatImpl) obj; // Special comparisons against Apfloat.ZERO and Apfloat.ONE work regardless of radix or implementation class if (signum() == 0 && thatImpl.signum() == 0) { return true; } else if (isOne() && thatImpl.isOne()) { return true; } if (!(obj instanceof DoubleApfloatImpl)) { return false; } DoubleApfloatImpl that = (DoubleApfloatImpl) obj; if (this.radix != that.radix) { // Limitation: cannot compare values with different radixes return false; } else if (this.sign != that.sign || this.exponent != that.exponent) { return false; } else { // Need to compare mantissas return compareMantissaTo(that) == 0; } } @Override public int hashCode() { if (this.hashCode == 0) { // Cache the value // NOTE: This is not synchronized; it's OK if multiple threads set this at the same time int hashCode = 1 + this.sign + (int) this.exponent + (int) (this.exponent >>> 32); if (this.dataStorage != null) { long size = getSize(); // Scan through log(size) scattered words in the mantissa for (long i = 0; i < size; i = i + i + 1) { double word = getWord(i); if (i == size - 1) { word = getLeastSignificantWord(i, word); } long element = (long) word; hashCode += (int) element + (int) (element >>> 32); } } this.hashCode = hashCode; } return this.hashCode; } @Override public String toString(boolean pretty) throws ApfloatRuntimeException { if (this.sign == 0) { return "0"; } long size = getSize() * BASE_DIGITS[this.radix], // This is a rounded up value length; if (pretty) { long scale = scale(); if (scale <= 0) { length = 2 - scale + size; // Format is 0.xxxx or 0.0000xxx } else if (size > scale) { length = 1 + size; // Format is x.xxx } else { length = scale; // Format is xxxx or xxxx0000 } length += (this.sign < 0 ? 1 : 0); // Room for minus sign } else { length = size + 24; // Sign, "0.", "e", exponent sign and 19 digits of exponent } if (length > Integer.MAX_VALUE || length < 0) // Detect overflow { throw new ApfloatInternalException("Number is too large to fit in a String"); } StringWriter writer = new StringWriter((int) length); try { writeTo(writer, pretty); } catch (IOException ioe) { throw new ApfloatInternalException("Unexpected I/O error writing to StringWriter", ioe); } String value = writer.toString(); assert (value.length() <= length); // Postcondition to ensure performance return value; } private static void writeZeros(Writer out, long count) throws IOException { for (long i = 0; i < count; i++) { out.write('0'); } } @Override public void writeTo(Writer out, boolean pretty) throws IOException, ApfloatRuntimeException { if (this.sign == 0) { out.write('0'); return; } if (this.sign < 0) { out.write('-'); } long integerDigits, // Number of digits to write before the decimal point exponent; // Exponent to print if (pretty) { if (this.exponent <= 0) { out.write("0."); // Output is 0.xxxx writeZeros(out, -scale()); // Print leading zeros after decimal point before first nonzero digit integerDigits = -1; // Decimal point is already written } else { integerDigits = scale(); // Decimal point location } exponent = 0; // Do not print exponent } else { integerDigits = 1; // Always write as x.xxxey exponent = scale() - 1; // Print exponent } boolean leftPadZeros = false; // If the written base unit should be left-padded with zeros long size = getSize(), digitsToWrite = Math.min(this.precision, getInitialDigits() + (size - 1) * BASE_DIGITS[this.radix]), digitsWritten = 0, trailingZeros = 0; DataStorage.Iterator iterator = this.dataStorage.iterator(DataStorage.READ, 0, size); char[] buffer = new char[BASE_DIGITS[this.radix]]; while (size > 0) { int start = (leftPadZeros ? 0 : BASE_DIGITS[this.radix] - getInitialDigits()), digits = (int) Math.min(digitsToWrite, BASE_DIGITS[this.radix] - start); formatWord(buffer, iterator.getDouble()); for (int i = 0; i < digits; i++) { int c = buffer[start + i]; if (c == '0') { trailingZeros++; digitsToWrite--; } else { while (trailingZeros > 0) { if (digitsWritten == integerDigits) { out.write('.'); } out.write('0'); digitsWritten++; trailingZeros--; } if (digitsWritten == integerDigits) { out.write('.'); } out.write(c); digitsWritten++; digitsToWrite--; } } leftPadZeros = true; // Always pad with zeros after first word iterator.next(); size--; } if (!pretty && exponent != 0) { out.write("e" + exponent); } writeZeros(out, integerDigits - digitsWritten); // If format is xxxx0000 } private void formatWord(char[] buffer, double word) { int position = BASE_DIGITS[this.radix]; while (position > 0 && word > 0) { double newWord = (double) (long) (word / this.radix); int digit = (int) (word - newWord * this.radix); word = newWord; position--; buffer[position] = Character.forDigit(digit, this.radix); } // Left pad zeros while (position > 0) { position--; buffer[position] = '0'; } } // Effective size, in doubles private long getSize() throws ApfloatRuntimeException { assert (this.dataStorage != null); return Math.min(getBasePrecision(), this.dataStorage.getSize()); } private static int checkRadix(int radix) throws NumberFormatException { if (radix < Character.MIN_RADIX || radix > Character.MAX_RADIX) { throw new NumberFormatException("Invalid radix " + radix + "; radix must be between " + Character.MIN_RADIX + " and " + Character.MAX_RADIX); } return radix; } // Get the most significant word of this number private double getMostSignificantWord() throws ApfloatRuntimeException { return getMostSignificantWord(this.dataStorage); } // Get the most significant word of the specified data storage private static double getMostSignificantWord(DataStorage dataStorage) throws ApfloatRuntimeException { double msw; try (ArrayAccess arrayAccess = dataStorage.getArray(DataStorage.READ, 0, 1)) { msw = arrayAccess.getDoubleData()[arrayAccess.getOffset()]; } return msw; } // Get number of digits in the most significant word private int getInitialDigits() throws ApfloatRuntimeException { if (this.initialDigits == UNDEFINED) { // Cache the value // NOTE: This is not synchronized; it's OK if multiple threads set this at the same time this.initialDigits = getDigits(getMostSignificantWord()); } return this.initialDigits; } // Get number of digits in the most significant word of specified data private int getInitialDigits(DataStorage dataStorage) throws ApfloatRuntimeException { return getDigits(getMostSignificantWord(dataStorage)); } // Gets the number of digits in the specified double and this number's radix private int getDigits(double x) { assert (x > 0); double[] minimums = MINIMUM_FOR_DIGITS[this.radix]; int i = minimums.length; while (x < minimums[--i]) { } return i + 1; } // Gets the precision in doubles private long getBasePrecision() throws ApfloatRuntimeException { return getBasePrecision(this.precision, getInitialDigits()); } // Gets the precision in doubles, based on specified precision (in digits), // number of digits in most significant word and this number's radix private long getBasePrecision(long precision, int mswDigits) { if (precision == Apfloat.INFINITE) { return Apfloat.INFINITE; } else { return Long.divideUnsigned(precision + BASE_DIGITS[this.radix] - mswDigits - 1, BASE_DIGITS[this.radix]) + 1; } } private double getWord(long index) { double word; try (ArrayAccess arrayAccess = this.dataStorage.getArray(DataStorage.READ, index, 1)) { word = arrayAccess.getDoubleData()[arrayAccess.getOffset()]; } return word; } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { this.leastZeros = UNDEFINED; this.isOne = UNDEFINED; in.defaultReadObject(); } // Gets a new data storage for specified size private static DataStorage createDataStorage(long size) throws ApfloatRuntimeException { ApfloatContext ctx = ApfloatContext.getContext(); DataStorageBuilder dataStorageBuilder = ctx.getBuilderFactory().getDataStorageBuilder(); return dataStorageBuilder.createDataStorage(size * Double.BYTES); } // Gets I/O block size in doubles private static int getBlockSize() { ApfloatContext ctx = ApfloatContext.getContext(); return ctx.getBlockSize() / Double.BYTES; } private static final DataStorage.Iterator ZERO_ITERATOR = new DataStorage.Iterator() { @Override public double getDouble() { return 0; } @Override public void next() { } private static final long serialVersionUID = 1L; }; private static final long serialVersionUID = -4177541592360478544L; private static final int UNDEFINED = 0x80000000; private static final int MAX_LONG_SIZE = 4; private static final int MAX_DOUBLE_SIZE = 4; private int sign; private long precision; private long exponent; private DataStorage dataStorage; private int radix; private int hashCode = 0; private int initialDigits = UNDEFINED; private int isOne = UNDEFINED; private volatile long leastZeros = UNDEFINED; private volatile long size = 0; }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy