0efdde188b
Reviewed-by: bpb
479 lines
17 KiB
Java
479 lines
17 KiB
Java
/*
|
|
* Copyright (c) 2009, 2020, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/*
|
|
* @test
|
|
* @bug 4504839 4215269 6322074 8030814
|
|
* @summary Basic tests for unsigned operations
|
|
* @author Joseph D. Darcy
|
|
*/
|
|
|
|
import java.math.*;
|
|
|
|
public class Unsigned {
|
|
public static void main(String... args) {
|
|
int errors = 0;
|
|
|
|
errors += testRoundtrip();
|
|
errors += testByteToUnsignedLong();
|
|
errors += testShortToUnsignedLong();
|
|
errors += testUnsignedCompare();
|
|
errors += testToStringUnsigned();
|
|
errors += testParseUnsignedLong();
|
|
errors += testDivideAndRemainder();
|
|
|
|
if (errors > 0) {
|
|
throw new RuntimeException(errors + " errors found in unsigned operations.");
|
|
}
|
|
}
|
|
|
|
private static final BigInteger TWO = BigInteger.valueOf(2L);
|
|
|
|
private static int testRoundtrip() {
|
|
int errors = 0;
|
|
|
|
long[] data = {-1L, 0L, 1L};
|
|
|
|
for(long datum : data) {
|
|
if (Long.parseUnsignedLong(Long.toBinaryString(datum), 2) != datum) {
|
|
errors++;
|
|
System.err.println("Bad binary roundtrip conversion of " + datum);
|
|
}
|
|
|
|
if (Long.parseUnsignedLong(Long.toOctalString(datum), 8) != datum) {
|
|
errors++;
|
|
System.err.println("Bad octal roundtrip conversion of " + datum);
|
|
}
|
|
|
|
if (Long.parseUnsignedLong(Long.toHexString(datum), 16) != datum) {
|
|
errors++;
|
|
System.err.println("Bad hex roundtrip conversion of " + datum);
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
private static int testByteToUnsignedLong() {
|
|
int errors = 0;
|
|
|
|
for(int i = Byte.MIN_VALUE; i <= Byte.MAX_VALUE; i++) {
|
|
byte datum = (byte) i;
|
|
long ui = Byte.toUnsignedLong(datum);
|
|
|
|
if ( (ui & (~0xffL)) != 0L ||
|
|
((byte)ui != datum )) {
|
|
errors++;
|
|
System.err.printf("Bad conversion of byte %d to unsigned long %d%n",
|
|
datum, ui);
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
private static int testShortToUnsignedLong() {
|
|
int errors = 0;
|
|
|
|
for(int i = Short.MIN_VALUE; i <= Short.MAX_VALUE; i++) {
|
|
short datum = (short) i;
|
|
long ui = Short.toUnsignedLong(datum);
|
|
|
|
if ( (ui & (~0xffffL)) != 0L ||
|
|
((short)ui != datum )) {
|
|
errors++;
|
|
System.err.printf("Bad conversion of short %d to unsigned long %d%n",
|
|
datum, ui);
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
private static int testUnsignedCompare() {
|
|
int errors = 0;
|
|
|
|
long[] data = {
|
|
0L,
|
|
1L,
|
|
2L,
|
|
3L,
|
|
0x00000000_80000000L,
|
|
0x00000000_FFFFFFFFL,
|
|
0x00000001_00000000L,
|
|
0x80000000_00000000L,
|
|
0x80000000_00000001L,
|
|
0x80000000_00000002L,
|
|
0x80000000_00000003L,
|
|
0x80000000_80000000L,
|
|
0xFFFFFFFF_FFFFFFFEL,
|
|
0xFFFFFFFF_FFFFFFFFL,
|
|
};
|
|
|
|
for(long i : data) {
|
|
for(long j : data) {
|
|
long libraryResult = Long.compareUnsigned(i, j);
|
|
long libraryResultRev = Long.compareUnsigned(j, i);
|
|
long localResult = compUnsigned(i, j);
|
|
|
|
if (i == j) {
|
|
if (libraryResult != 0) {
|
|
errors++;
|
|
System.err.printf("Value 0x%x did not compare as " +
|
|
"an unsigned equal to itself; got %d%n",
|
|
i, libraryResult);
|
|
}
|
|
}
|
|
|
|
if (Long.signum(libraryResult) != Long.signum(localResult)) {
|
|
errors++;
|
|
System.err.printf("Unsigned compare of 0x%x to 0x%x%n:" +
|
|
"\texpected sign of %d, got %d%n",
|
|
i, j, localResult, libraryResult);
|
|
}
|
|
|
|
if (Long.signum(libraryResult) !=
|
|
-Long.signum(libraryResultRev)) {
|
|
errors++;
|
|
System.err.printf("signum(compareUnsigned(x, y)) != -signum(compareUnsigned(y,x))" +
|
|
" for \t0x%x and 0x%x, computed %d and %d%n",
|
|
i, j, libraryResult, libraryResultRev);
|
|
}
|
|
}
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
|
|
private static int compUnsigned(long x, long y) {
|
|
BigInteger big_x = toUnsignedBigInt(x);
|
|
BigInteger big_y = toUnsignedBigInt(y);
|
|
|
|
return big_x.compareTo(big_y);
|
|
}
|
|
|
|
private static BigInteger toUnsignedBigInt(long x) {
|
|
if (x >= 0)
|
|
return BigInteger.valueOf(x);
|
|
else {
|
|
int upper = (int)(((long)x) >> 32);
|
|
int lower = (int) x;
|
|
|
|
BigInteger bi = // (upper << 32) + lower
|
|
(BigInteger.valueOf(Integer.toUnsignedLong(upper))).shiftLeft(32).
|
|
add(BigInteger.valueOf(Integer.toUnsignedLong(lower)));
|
|
|
|
// System.out.printf("%n\t%d%n\t%s%n", x, bi.toString());
|
|
return bi;
|
|
}
|
|
}
|
|
|
|
private static int testToStringUnsigned() {
|
|
int errors = 0;
|
|
|
|
long[] data = {
|
|
0L,
|
|
1L,
|
|
2L,
|
|
3L,
|
|
99999L,
|
|
100000L,
|
|
999999L,
|
|
100000L,
|
|
999999999L,
|
|
1000000000L,
|
|
0x1234_5678L,
|
|
0x8000_0000L,
|
|
0x8000_0001L,
|
|
0x8000_0002L,
|
|
0x8000_0003L,
|
|
0x8765_4321L,
|
|
0xFFFF_FFFEL,
|
|
0xFFFF_FFFFL,
|
|
|
|
// Long-range values
|
|
999_999_999_999L,
|
|
1_000_000_000_000L,
|
|
|
|
999_999_999_999_999_999L,
|
|
1_000_000_000_000_000_000L,
|
|
|
|
0xFFFF_FFFF_FFFF_FFFEL,
|
|
0xFFFF_FFFF_FFFF_FFFFL,
|
|
};
|
|
|
|
for(int radix = Character.MIN_RADIX; radix <= Character.MAX_RADIX; radix++) {
|
|
for(long datum : data) {
|
|
String result1 = Long.toUnsignedString(datum, radix);
|
|
String result2 = toUnsignedBigInt(datum).toString(radix);
|
|
|
|
if (!result1.equals(result2)) {
|
|
errors++;
|
|
System.err.printf("Unexpected string difference converting 0x%x:" +
|
|
"\t%s %s%n",
|
|
datum, result1, result2);
|
|
}
|
|
|
|
if (radix == 10) {
|
|
String result3 = Long.toUnsignedString(datum);
|
|
if (!result2.equals(result3)) {
|
|
errors++;
|
|
System.err.printf("Unexpected string difference converting 0x%x:" +
|
|
"\t%s %s%n",
|
|
datum, result3, result2);
|
|
}
|
|
}
|
|
|
|
long parseResult = Long.parseUnsignedLong(result1, radix);
|
|
|
|
if (parseResult != datum) {
|
|
errors++;
|
|
System.err.printf("Bad roundtrip conversion of %d in base %d" +
|
|
"\tconverting back ''%s'' resulted in %d%n",
|
|
datum, radix, result1, parseResult);
|
|
}
|
|
}
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
|
|
private static int testParseUnsignedLong() {
|
|
int errors = 0;
|
|
long maxUnsignedInt = Integer.toUnsignedLong(0xffff_ffff);
|
|
|
|
// Values include those between signed Long.MAX_VALUE and
|
|
// unsignted Long MAX_VALUE.
|
|
BigInteger[] inRange = {
|
|
BigInteger.valueOf(0L),
|
|
BigInteger.valueOf(1L),
|
|
BigInteger.valueOf(10L),
|
|
BigInteger.valueOf(2147483646L), // Integer.MAX_VALUE - 1
|
|
BigInteger.valueOf(2147483647L), // Integer.MAX_VALUE
|
|
BigInteger.valueOf(2147483648L), // Integer.MAX_VALUE + 1
|
|
|
|
BigInteger.valueOf(maxUnsignedInt - 1L),
|
|
BigInteger.valueOf(maxUnsignedInt),
|
|
|
|
BigInteger.valueOf(Long.MAX_VALUE - 1L),
|
|
BigInteger.valueOf(Long.MAX_VALUE),
|
|
BigInteger.valueOf(Long.MAX_VALUE).add(BigInteger.ONE),
|
|
|
|
TWO.pow(64).subtract(BigInteger.ONE)
|
|
};
|
|
|
|
for(BigInteger value : inRange) {
|
|
for(int radix = Character.MIN_RADIX; radix <= Character.MAX_RADIX; radix++) {
|
|
String bigString = value.toString(radix);
|
|
long longResult = Long.parseUnsignedLong(bigString, radix);
|
|
|
|
if (!toUnsignedBigInt(longResult).equals(value)) {
|
|
errors++;
|
|
System.err.printf("Bad roundtrip conversion of %d in base %d" +
|
|
"\tconverting back ''%s'' resulted in %d%n",
|
|
value, radix, bigString, longResult);
|
|
}
|
|
|
|
// test offset based parse method
|
|
longResult = Long.parseUnsignedLong("prefix" + bigString + "suffix", "prefix".length(),
|
|
"prefix".length() + bigString.length(), radix);
|
|
|
|
if (!toUnsignedBigInt(longResult).equals(value)) {
|
|
errors++;
|
|
System.err.printf("Bad roundtrip conversion of %d in base %d" +
|
|
"\tconverting back ''%s'' resulted in %d%n",
|
|
value, radix, bigString, longResult);
|
|
}
|
|
}
|
|
}
|
|
|
|
String[] outOfRange = {
|
|
null,
|
|
"",
|
|
"-1",
|
|
TWO.pow(64).toString(),
|
|
};
|
|
|
|
for(String s : outOfRange) {
|
|
try {
|
|
long result = Long.parseUnsignedLong(s);
|
|
errors++; // Should not reach here
|
|
System.err.printf("Unexpected got %d from an unsigned conversion of %s",
|
|
result, s);
|
|
} catch(NumberFormatException nfe) {
|
|
; // Correct result
|
|
}
|
|
}
|
|
|
|
// test case known at one time to fail
|
|
errors += testUnsignedOverflow("1234567890abcdef1", 16, true);
|
|
|
|
// largest value with guard = 91 = 13*7; radix = 13
|
|
errors += testUnsignedOverflow("196a78a44c3bba320c", 13, false);
|
|
|
|
// smallest value with guard = 92 = 23*2*2; radix = 23
|
|
errors += testUnsignedOverflow("137060c6g1c1dg0", 23, false);
|
|
|
|
// guard in [92,98]: no overflow
|
|
|
|
// one less than smallest guard value to overflow: guard = 99 = 11*3*3, radix = 33
|
|
errors += testUnsignedOverflow("b1w8p7j5q9r6f", 33, false);
|
|
|
|
// smallest guard value to overflow: guard = 99 = 11*3*3, radix = 33
|
|
errors += testUnsignedOverflow("b1w8p7j5q9r6g", 33, true);
|
|
|
|
// test overflow of overflow
|
|
BigInteger maxUnsignedLong =
|
|
BigInteger.ONE.shiftLeft(64).subtract(BigInteger.ONE);
|
|
for (int radix = Character.MIN_RADIX; radix <= Character.MAX_RADIX; radix++) {
|
|
BigInteger quotient = maxUnsignedLong.divide(BigInteger.valueOf(radix));
|
|
for (int addend = 2; addend <= radix; addend++) {
|
|
BigInteger b = quotient.multiply(BigInteger.valueOf(radix + addend));
|
|
errors += testUnsignedOverflow(b.toString(radix), radix, b.compareTo(maxUnsignedLong) > 0);
|
|
}
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
|
|
// test for missing or unexpected unsigned overflow exception
|
|
private static int testUnsignedOverflow(String s, int radix, boolean exception) {
|
|
int errors = 0;
|
|
long result;
|
|
try {
|
|
result = Long.parseUnsignedLong(s, radix);
|
|
if (exception) {
|
|
System.err.printf("Unexpected result %d for Long.parseUnsignedLong(%s,%d)\n",
|
|
result, s, radix);
|
|
errors++;
|
|
}
|
|
} catch (NumberFormatException nfe) {
|
|
if (!exception) {
|
|
System.err.printf("Unexpected exception %s for Long.parseUnsignedLong(%s,%d)\n",
|
|
nfe.toString(), s, radix);
|
|
errors++;
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
private static int testDivideAndRemainder() {
|
|
int errors = 0;
|
|
long TWO_31 = 1L << Integer.SIZE - 1;
|
|
long TWO_32 = 1L << Integer.SIZE;
|
|
long TWO_33 = 1L << Integer.SIZE + 1;
|
|
BigInteger NINETEEN = BigInteger.valueOf(19L);
|
|
BigInteger TWO_63 = BigInteger.ONE.shiftLeft(Long.SIZE - 1);
|
|
BigInteger TWO_64 = BigInteger.ONE.shiftLeft(Long.SIZE);
|
|
|
|
BigInteger[] inRange = {
|
|
BigInteger.ZERO,
|
|
BigInteger.ONE,
|
|
BigInteger.TEN,
|
|
NINETEEN,
|
|
|
|
BigInteger.valueOf(TWO_31 - 19L),
|
|
BigInteger.valueOf(TWO_31 - 10L),
|
|
BigInteger.valueOf(TWO_31 - 1L),
|
|
BigInteger.valueOf(TWO_31),
|
|
BigInteger.valueOf(TWO_31 + 1L),
|
|
BigInteger.valueOf(TWO_31 + 10L),
|
|
BigInteger.valueOf(TWO_31 + 19L),
|
|
|
|
BigInteger.valueOf(TWO_32 - 19L),
|
|
BigInteger.valueOf(TWO_32 - 10L),
|
|
BigInteger.valueOf(TWO_32 - 1L),
|
|
BigInteger.valueOf(TWO_32),
|
|
BigInteger.valueOf(TWO_32 + 1L),
|
|
BigInteger.valueOf(TWO_32 + 10L),
|
|
BigInteger.valueOf(TWO_32 - 19L),
|
|
|
|
BigInteger.valueOf(TWO_33 - 19L),
|
|
BigInteger.valueOf(TWO_33 - 10L),
|
|
BigInteger.valueOf(TWO_33 - 1L),
|
|
BigInteger.valueOf(TWO_33),
|
|
BigInteger.valueOf(TWO_33 + 1L),
|
|
BigInteger.valueOf(TWO_33 + 10L),
|
|
BigInteger.valueOf(TWO_33 + 19L),
|
|
|
|
TWO_63.subtract(NINETEEN),
|
|
TWO_63.subtract(BigInteger.TEN),
|
|
TWO_63.subtract(BigInteger.ONE),
|
|
TWO_63,
|
|
TWO_63.add(BigInteger.ONE),
|
|
TWO_63.add(BigInteger.TEN),
|
|
TWO_63.add(NINETEEN),
|
|
|
|
TWO_64.subtract(NINETEEN),
|
|
TWO_64.subtract(BigInteger.TEN),
|
|
TWO_64.subtract(BigInteger.ONE),
|
|
};
|
|
|
|
for(BigInteger dividend : inRange) {
|
|
for(BigInteger divisor : inRange) {
|
|
long quotient;
|
|
BigInteger longQuotient;
|
|
|
|
long remainder;
|
|
BigInteger longRemainder;
|
|
|
|
if (divisor.equals(BigInteger.ZERO)) {
|
|
try {
|
|
quotient = Long.divideUnsigned(dividend.longValue(), divisor.longValue());
|
|
errors++;
|
|
} catch(ArithmeticException ea) {
|
|
; // Expected
|
|
}
|
|
|
|
try {
|
|
remainder = Long.remainderUnsigned(dividend.longValue(), divisor.longValue());
|
|
errors++;
|
|
} catch(ArithmeticException ea) {
|
|
; // Expected
|
|
}
|
|
} else {
|
|
quotient = Long.divideUnsigned(dividend.longValue(), divisor.longValue());
|
|
longQuotient = dividend.divide(divisor);
|
|
|
|
if (quotient != longQuotient.longValue()) {
|
|
errors++;
|
|
System.err.printf("Unexpected unsigned divide result %s on %s/%s%n",
|
|
Long.toUnsignedString(quotient),
|
|
Long.toUnsignedString(dividend.longValue()),
|
|
Long.toUnsignedString(divisor.longValue()));
|
|
}
|
|
|
|
remainder = Long.remainderUnsigned(dividend.longValue(), divisor.longValue());
|
|
longRemainder = dividend.remainder(divisor);
|
|
|
|
if (remainder != longRemainder.longValue()) {
|
|
errors++;
|
|
System.err.printf("Unexpected unsigned remainder result %s on %s%%%s%n",
|
|
Long.toUnsignedString(remainder),
|
|
Long.toUnsignedString(dividend.longValue()),
|
|
Long.toUnsignedString(divisor.longValue()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
}
|