Skip to content

Commit beea530

Browse files
committedMay 14, 2024
8331907: BigInteger and BigDecimal should use optimized division
Reviewed-by: rgiulietti, bpb
1 parent 440782e commit beea530

File tree

4 files changed

+83
-180
lines changed

4 files changed

+83
-180
lines changed
 

‎src/java.base/share/classes/java/math/BigDecimal.java

+5-56
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1996, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1996, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -5680,18 +5680,8 @@ private static BigDecimal divideAndRound128(final long dividendHi, final long di
56805680

56815681
tmp = (dividendHi << shift) | (dividendLo >>> 64 - shift);
56825682
long u2 = tmp & LONG_MASK;
5683-
long q1, r_tmp;
5684-
if (v1 == 1) {
5685-
q1 = tmp;
5686-
r_tmp = 0;
5687-
} else if (tmp >= 0) {
5688-
q1 = tmp / v1;
5689-
r_tmp = tmp - q1 * v1;
5690-
} else {
5691-
long[] rq = divRemNegativeLong(tmp, v1);
5692-
q1 = rq[1];
5693-
r_tmp = rq[0];
5694-
}
5683+
long q1 = Long.divideUnsigned(tmp, v1);
5684+
long r_tmp = Long.remainderUnsigned(tmp, v1);
56955685

56965686
while(q1 >= DIV_NUM_BASE || unsignedLongCompare(q1*v0, make64(r_tmp, u1))) {
56975687
q1--;
@@ -5702,18 +5692,8 @@ private static BigDecimal divideAndRound128(final long dividendHi, final long di
57025692

57035693
tmp = mulsub(u2,u1,v1,v0,q1);
57045694
u1 = tmp & LONG_MASK;
5705-
long q0;
5706-
if (v1 == 1) {
5707-
q0 = tmp;
5708-
r_tmp = 0;
5709-
} else if (tmp >= 0) {
5710-
q0 = tmp / v1;
5711-
r_tmp = tmp - q0 * v1;
5712-
} else {
5713-
long[] rq = divRemNegativeLong(tmp, v1);
5714-
q0 = rq[1];
5715-
r_tmp = rq[0];
5716-
}
5695+
long q0 = Long.divideUnsigned(tmp, v1);
5696+
r_tmp = Long.remainderUnsigned(tmp, v1);
57175697

57185698
while(q0 >= DIV_NUM_BASE || unsignedLongCompare(q0*v0,make64(r_tmp,u0))) {
57195699
q0--;
@@ -5793,37 +5773,6 @@ static BigDecimal scaledTenPow(int n, int sign, int scale) {
57935773
}
57945774
}
57955775

5796-
/**
5797-
* Calculate the quotient and remainder of dividing a negative long by
5798-
* another long.
5799-
*
5800-
* @param n the numerator; must be negative
5801-
* @param d the denominator; must not be unity
5802-
* @return a two-element {@code long} array with the remainder and quotient in
5803-
* the initial and final elements, respectively
5804-
*/
5805-
private static long[] divRemNegativeLong(long n, long d) {
5806-
assert n < 0 : "Non-negative numerator " + n;
5807-
assert d != 1 : "Unity denominator";
5808-
5809-
// Approximate the quotient and remainder
5810-
long q = (n >>> 1) / (d >>> 1);
5811-
long r = n - q * d;
5812-
5813-
// Correct the approximation
5814-
while (r < 0) {
5815-
r += d;
5816-
q--;
5817-
}
5818-
while (r >= d) {
5819-
r -= d;
5820-
q++;
5821-
}
5822-
5823-
// n - q*d == r && 0 <= r < d, hence we're done.
5824-
return new long[] {r, q};
5825-
}
5826-
58275776
private static long make64(long hi, long lo) {
58285777
return hi<<32 | lo;
58295778
}

‎src/java.base/share/classes/java/math/MutableBigInteger.java

+16-92
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1999, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1999, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -1092,9 +1092,9 @@ int divideOneWord(int divisor, MutableBigInteger quotient) {
10921092

10931093
// Special case of one word dividend
10941094
if (intLen == 1) {
1095-
long dividendValue = value[offset] & LONG_MASK;
1096-
int q = (int) (dividendValue / divisorLong);
1097-
int r = (int) (dividendValue - q * divisorLong);
1095+
int dividendValue = value[offset];
1096+
int q = Integer.divideUnsigned(dividendValue, divisor);
1097+
int r = Integer.remainderUnsigned(dividendValue, divisor);
10981098
quotient.value[0] = q;
10991099
quotient.intLen = (q == 0) ? 0 : 1;
11001100
quotient.offset = 0;
@@ -1106,41 +1106,17 @@ int divideOneWord(int divisor, MutableBigInteger quotient) {
11061106
quotient.offset = 0;
11071107
quotient.intLen = intLen;
11081108

1109-
// Normalize the divisor
1110-
int shift = Integer.numberOfLeadingZeros(divisor);
1111-
1112-
int rem = value[offset];
1113-
long remLong = rem & LONG_MASK;
1114-
if (remLong < divisorLong) {
1115-
quotient.value[0] = 0;
1116-
} else {
1117-
quotient.value[0] = (int)(remLong / divisorLong);
1118-
rem = (int) (remLong - (quotient.value[0] * divisorLong));
1119-
remLong = rem & LONG_MASK;
1120-
}
1121-
int xlen = intLen;
1122-
while (--xlen > 0) {
1123-
long dividendEstimate = (remLong << 32) |
1109+
long rem = 0;
1110+
for (int xlen = intLen; xlen > 0; xlen--) {
1111+
long dividendEstimate = (rem << 32) |
11241112
(value[offset + intLen - xlen] & LONG_MASK);
1125-
int q;
1126-
if (dividendEstimate >= 0) {
1127-
q = (int) (dividendEstimate / divisorLong);
1128-
rem = (int) (dividendEstimate - q * divisorLong);
1129-
} else {
1130-
long tmp = divWord(dividendEstimate, divisor);
1131-
q = (int) (tmp & LONG_MASK);
1132-
rem = (int) (tmp >>> 32);
1133-
}
1113+
int q = (int) Long.divideUnsigned(dividendEstimate, divisorLong);
1114+
rem = Long.remainderUnsigned(dividendEstimate, divisorLong);
11341115
quotient.value[intLen - xlen] = q;
1135-
remLong = rem & LONG_MASK;
11361116
}
11371117

11381118
quotient.normalize();
1139-
// Unnormalize
1140-
if (shift > 0)
1141-
return rem % divisor;
1142-
else
1143-
return rem;
1119+
return (int)rem;
11441120
}
11451121

11461122
/**
@@ -1557,14 +1533,8 @@ private MutableBigInteger divideMagnitude(MutableBigInteger div,
15571533
skipCorrection = qrem + 0x80000000 < nh2;
15581534
} else {
15591535
long nChunk = (((long)nh) << 32) | (nm & LONG_MASK);
1560-
if (nChunk >= 0) {
1561-
qhat = (int) (nChunk / dhLong);
1562-
qrem = (int) (nChunk - (qhat * dhLong));
1563-
} else {
1564-
long tmp = divWord(nChunk, dh);
1565-
qhat = (int) (tmp & LONG_MASK);
1566-
qrem = (int) (tmp >>> 32);
1567-
}
1536+
qhat = (int) Long.divideUnsigned(nChunk, dhLong);
1537+
qrem = (int) Long.remainderUnsigned(nChunk, dhLong);
15681538
}
15691539

15701540
if (qhat == 0)
@@ -1616,14 +1586,8 @@ private MutableBigInteger divideMagnitude(MutableBigInteger div,
16161586
skipCorrection = qrem + 0x80000000 < nh2;
16171587
} else {
16181588
long nChunk = (((long) nh) << 32) | (nm & LONG_MASK);
1619-
if (nChunk >= 0) {
1620-
qhat = (int) (nChunk / dhLong);
1621-
qrem = (int) (nChunk - (qhat * dhLong));
1622-
} else {
1623-
long tmp = divWord(nChunk, dh);
1624-
qhat = (int) (tmp & LONG_MASK);
1625-
qrem = (int) (tmp >>> 32);
1626-
}
1589+
qhat = (int) Long.divideUnsigned(nChunk, dhLong);
1590+
qrem = (int) Long.remainderUnsigned(nChunk, dhLong);
16271591
}
16281592
if (qhat != 0) {
16291593
if (!skipCorrection) { // Correct qhat
@@ -1732,14 +1696,8 @@ private MutableBigInteger divideLongMagnitude(long ldivisor, MutableBigInteger q
17321696
skipCorrection = qrem + 0x80000000 < nh2;
17331697
} else {
17341698
long nChunk = (((long) nh) << 32) | (nm & LONG_MASK);
1735-
if (nChunk >= 0) {
1736-
qhat = (int) (nChunk / dhLong);
1737-
qrem = (int) (nChunk - (qhat * dhLong));
1738-
} else {
1739-
long tmp = divWord(nChunk, dh);
1740-
qhat =(int)(tmp & LONG_MASK);
1741-
qrem = (int)(tmp>>>32);
1742-
}
1699+
qhat = (int) Long.divideUnsigned(nChunk, dhLong);
1700+
qrem = (int) Long.remainderUnsigned(nChunk, dhLong);
17431701
}
17441702

17451703
if (qhat == 0)
@@ -1834,40 +1792,6 @@ private boolean unsignedLongCompare(long one, long two) {
18341792
return (one+Long.MIN_VALUE) > (two+Long.MIN_VALUE);
18351793
}
18361794

1837-
/**
1838-
* This method divides a long quantity by an int to estimate
1839-
* qhat for two multi precision numbers. It is used when
1840-
* the signed value of n is less than zero.
1841-
* Returns long value where high 32 bits contain remainder value and
1842-
* low 32 bits contain quotient value.
1843-
*/
1844-
static long divWord(long n, int d) {
1845-
long dLong = d & LONG_MASK;
1846-
long r;
1847-
long q;
1848-
if (dLong == 1) {
1849-
q = (int)n;
1850-
r = 0;
1851-
return (r << 32) | (q & LONG_MASK);
1852-
}
1853-
1854-
// Approximate the quotient and remainder
1855-
q = (n >>> 1) / (dLong >>> 1);
1856-
r = n - q*dLong;
1857-
1858-
// Correct the approximation
1859-
while (r < 0) {
1860-
r += dLong;
1861-
q--;
1862-
}
1863-
while (r >= dLong) {
1864-
r -= dLong;
1865-
q++;
1866-
}
1867-
// n - q*dlong == r && 0 <= r <dLong, hence we're done.
1868-
return (r << 32) | (q & LONG_MASK);
1869-
}
1870-
18711795
/**
18721796
* Calculate the integer square root {@code floor(sqrt(this))} where
18731797
* {@code sqrt(.)} denotes the mathematical square root. The contents of

‎test/micro/org/openjdk/bench/java/math/BigDecimals.java

+28-31
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -133,15 +133,6 @@ public void testConstructorWithDouble(Blackhole bh) {
133133
}
134134
}
135135

136-
/** Invokes the toString method of BigDecimal with various different values. */
137-
@Benchmark
138-
@OperationsPerInvocation(TEST_SIZE)
139-
public void testToString(Blackhole bh) {
140-
for (BigDecimal s : bigDecimals) {
141-
bh.consume(s.toString());
142-
}
143-
}
144-
145136
/**
146137
* Invokes the setScale method of BigDecimal with various different values.
147138
*/
@@ -194,40 +185,46 @@ public void testMultiply(Blackhole bh) {
194185
bh.consume(tmp);
195186
}
196187

197-
/** Invokes the compareTo method of BigDecimal with various different values. */
188+
/** Test divide with huge/small numbers */
198189
@Benchmark
199-
@OperationsPerInvocation(TEST_SIZE - 1)
200-
public void testCompareTo(Blackhole bh) {
201-
BigDecimal c = bigDecimals[0];
202-
for (BigDecimal s : bigDecimals) {
203-
bh.consume(c.compareTo(s));
190+
@OperationsPerInvocation(TEST_SIZE * TEST_SIZE)
191+
public void testHugeSmallDivide(Blackhole bh) {
192+
for (BigDecimal s : hugeArray) {
193+
for (BigDecimal t : smallArray) {
194+
bh.consume(s.divide(t, RoundingMode.DOWN));
195+
}
204196
}
205197
}
206198

207-
/** Test BigDecimal.toString() with huge numbers larger than MAX_LONG */
199+
/** Test divide with large/small numbers */
208200
@Benchmark
209-
@OperationsPerInvocation(TEST_SIZE)
210-
public void testHugeToString(Blackhole bh) {
211-
for (BigDecimal s : hugeArray) {
212-
bh.consume(s.toString());
201+
@OperationsPerInvocation(TEST_SIZE * TEST_SIZE)
202+
public void testLargeSmallDivide(Blackhole bh) {
203+
for (BigDecimal s : largeArray) {
204+
for (BigDecimal t : smallArray) {
205+
bh.consume(s.divide(t, RoundingMode.DOWN));
206+
}
213207
}
214208
}
215209

216-
/** Test BigDecimal.toString() with large numbers less than MAX_LONG but larger than MAX_INT */
210+
/** Test divide with huge/large numbers */
217211
@Benchmark
218-
@OperationsPerInvocation(TEST_SIZE)
219-
public void testLargeToString(Blackhole bh) {
220-
for (BigDecimal s : largeArray) {
221-
bh.consume(s.toString());
212+
@OperationsPerInvocation(TEST_SIZE * TEST_SIZE)
213+
public void testHugeLargeDivide(Blackhole bh) {
214+
for (BigDecimal s : hugeArray) {
215+
for (BigDecimal t : largeArray) {
216+
bh.consume(s.divide(t, RoundingMode.DOWN));
217+
}
222218
}
223219
}
224220

225-
/** Test BigDecimal.toString() with small numbers less than MAX_INT */
221+
/** Invokes the compareTo method of BigDecimal with various different values. */
226222
@Benchmark
227-
@OperationsPerInvocation(TEST_SIZE)
228-
public void testSmallToString(Blackhole bh) {
229-
for (BigDecimal s : smallArray) {
230-
bh.consume(s.toString());
223+
@OperationsPerInvocation(TEST_SIZE - 1)
224+
public void testCompareTo(Blackhole bh) {
225+
BigDecimal c = bigDecimals[0];
226+
for (BigDecimal s : bigDecimals) {
227+
bh.consume(c.compareTo(s));
231228
}
232229
}
233230
}

‎test/micro/org/openjdk/bench/java/math/BigIntegers.java

+34-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -132,6 +132,39 @@ public void testMultiply(Blackhole bh) {
132132
bh.consume(tmp);
133133
}
134134

135+
/** Test divide with huge/small numbers */
136+
@Benchmark
137+
@OperationsPerInvocation(TESTSIZE * TESTSIZE)
138+
public void testHugeSmallDivide(Blackhole bh) {
139+
for (BigInteger s : hugeArray) {
140+
for (BigInteger t : smallArray) {
141+
bh.consume(s.divide(t));
142+
}
143+
}
144+
}
145+
146+
/** Test divide with large/small numbers */
147+
@Benchmark
148+
@OperationsPerInvocation(TESTSIZE * TESTSIZE)
149+
public void testLargeSmallDivide(Blackhole bh) {
150+
for (BigInteger s : largeArray) {
151+
for (BigInteger t : smallArray) {
152+
bh.consume(s.divide(t));
153+
}
154+
}
155+
}
156+
157+
/** Test divide with huge/large numbers */
158+
@Benchmark
159+
@OperationsPerInvocation(TESTSIZE * TESTSIZE)
160+
public void testHugeLargeDivide(Blackhole bh) {
161+
for (BigInteger s : hugeArray) {
162+
for (BigInteger t : largeArray) {
163+
bh.consume(s.divide(t));
164+
}
165+
}
166+
}
167+
135168
/** Invokes the multiply method of BigInteger with various different values. */
136169
@Benchmark
137170
@OperationsPerInvocation(TESTSIZE)

0 commit comments

Comments
 (0)
Please sign in to comment.