e7795851d2
Reviewed-by: lmesnik, dholmes
330 lines
14 KiB
Java
330 lines
14 KiB
Java
/*
|
|
* Copyright (c) 2000, 2022, Oracle and/or its affiliates. All rights reserved.
|
|
* Copyright (c) 2015, Red Hat Inc. 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 8130150 8131779 8139907
|
|
* @summary Verify that the Montgomery multiply and square intrinsic works and correctly checks their arguments.
|
|
* @requires vm.flavor == "server" & !vm.emulatedClient
|
|
* @modules java.base/jdk.internal.misc:open
|
|
* @modules java.base/java.math:open
|
|
* @library /test/lib /
|
|
*
|
|
* @build jdk.test.whitebox.WhiteBox
|
|
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
|
|
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI
|
|
* compiler.intrinsics.bigInteger.MontgomeryMultiplyTest
|
|
*/
|
|
|
|
package compiler.intrinsics.bigInteger;
|
|
|
|
import jdk.test.lib.Platform;
|
|
import jdk.test.whitebox.WhiteBox;
|
|
import compiler.whitebox.CompilerWhiteBoxTest;
|
|
|
|
import java.lang.invoke.MethodHandle;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.reflect.Constructor;
|
|
import java.lang.reflect.Executable;
|
|
import java.lang.reflect.Field;
|
|
import java.lang.reflect.Method;
|
|
import java.math.BigInteger;
|
|
import java.util.Arrays;
|
|
import java.util.Random;
|
|
|
|
public class MontgomeryMultiplyTest {
|
|
|
|
private static final WhiteBox wb = WhiteBox.getWhiteBox();
|
|
|
|
static final MethodHandles.Lookup lookup = MethodHandles.lookup();
|
|
|
|
static final MethodHandle montgomeryMultiplyHandle, montgomerySquareHandle;
|
|
static final MethodHandle bigIntegerConstructorHandle;
|
|
static final Field bigIntegerMagField;
|
|
|
|
static {
|
|
// Use reflection to gain access to the methods we want to test.
|
|
try {
|
|
Method m = BigInteger.class.getDeclaredMethod("montgomeryMultiply",
|
|
/*a*/int[].class, /*b*/int[].class, /*n*/int[].class, /*len*/int.class,
|
|
/*inv*/long.class, /*product*/int[].class);
|
|
m.setAccessible(true);
|
|
montgomeryMultiplyHandle = lookup.unreflect(m);
|
|
|
|
m = BigInteger.class.getDeclaredMethod("montgomerySquare",
|
|
/*a*/int[].class, /*n*/int[].class, /*len*/int.class,
|
|
/*inv*/long.class, /*product*/int[].class);
|
|
m.setAccessible(true);
|
|
montgomerySquareHandle = lookup.unreflect(m);
|
|
|
|
Constructor c
|
|
= BigInteger.class.getDeclaredConstructor(int.class, int[].class);
|
|
c.setAccessible(true);
|
|
bigIntegerConstructorHandle = lookup.unreflectConstructor(c);
|
|
|
|
bigIntegerMagField = BigInteger.class.getDeclaredField("mag");
|
|
bigIntegerMagField.setAccessible(true);
|
|
|
|
} catch (Throwable ex) {
|
|
throw new RuntimeException(ex);
|
|
}
|
|
}
|
|
|
|
/* Obtain executable for the intrinsics tested. Depending on the
|
|
* value of 'isMultiply', the executable corresponding to either
|
|
* implMontgomerMultiply or implMontgomerySqure is returned. */
|
|
static Executable getExecutable(boolean isMultiply) throws RuntimeException {
|
|
try {
|
|
Class aClass = Class.forName("java.math.BigInteger");
|
|
Method aMethod;
|
|
if (isMultiply) {
|
|
aMethod = aClass.getDeclaredMethod("implMontgomeryMultiply",
|
|
int[].class,
|
|
int[].class,
|
|
int[].class,
|
|
int.class,
|
|
long.class,
|
|
int[].class);
|
|
} else {
|
|
aMethod = aClass.getDeclaredMethod("implMontgomerySquare",
|
|
int[].class,
|
|
int[].class,
|
|
int.class,
|
|
long.class,
|
|
int[].class);
|
|
}
|
|
return aMethod;
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException("Test bug, method is unavailable. " + e);
|
|
} catch (ClassNotFoundException e) {
|
|
throw new RuntimeException("Test bug, class is unavailable. " + e);
|
|
}
|
|
}
|
|
|
|
// Invoke either BigInteger.montgomeryMultiply or BigInteger.montgomerySquare.
|
|
int[] montgomeryMultiply(int[] a, int[] b, int[] n, int len, long inv,
|
|
int[] product) throws Throwable {
|
|
int[] result =
|
|
(a == b) ? (int[]) montgomerySquareHandle.invokeExact(a, n, len, inv, product)
|
|
: (int[]) montgomeryMultiplyHandle.invokeExact(a, b, n, len, inv, product);
|
|
return Arrays.copyOf(result, len);
|
|
}
|
|
|
|
// Invoke the private constructor BigInteger(int[]).
|
|
BigInteger newBigInteger(int[] val) throws Throwable {
|
|
return (BigInteger) bigIntegerConstructorHandle.invokeExact(1, val);
|
|
}
|
|
|
|
// Get the private field BigInteger.mag
|
|
int[] mag(BigInteger n) {
|
|
try {
|
|
return (int[]) bigIntegerMagField.get(n);
|
|
} catch (Exception ex) {
|
|
throw new RuntimeException(ex);
|
|
}
|
|
}
|
|
|
|
// Montgomery multiplication
|
|
// Calculate a * b * r^-1 mod n)
|
|
//
|
|
// R is a power of the word size
|
|
// N' = R^-1 mod N
|
|
//
|
|
// T := ab
|
|
// m := (T mod R)N' mod R [so 0 <= m < R]
|
|
// t := (T + mN)/R
|
|
// if t >= N then return t - N else return t
|
|
//
|
|
BigInteger montgomeryMultiply(BigInteger a, BigInteger b, BigInteger N,
|
|
int len, BigInteger n_prime)
|
|
throws Throwable {
|
|
BigInteger T = a.multiply(b);
|
|
BigInteger R = BigInteger.ONE.shiftLeft(len*32);
|
|
BigInteger mask = R.subtract(BigInteger.ONE);
|
|
BigInteger m = (T.and(mask)).multiply(n_prime);
|
|
m = m.and(mask); // i.e. m.mod(R)
|
|
T = T.add(m.multiply(N));
|
|
T = T.shiftRight(len*32); // i.e. T.divide(R)
|
|
if (T.compareTo(N) > 0) {
|
|
T = T.subtract(N);
|
|
}
|
|
return T;
|
|
}
|
|
|
|
// Call the Montgomery multiply intrinsic.
|
|
BigInteger montgomeryMultiply(int[] a_words, int[] b_words, int[] n_words,
|
|
int len, BigInteger inv)
|
|
throws Throwable {
|
|
BigInteger t = montgomeryMultiply(
|
|
newBigInteger(a_words),
|
|
newBigInteger(b_words),
|
|
newBigInteger(n_words),
|
|
len, inv);
|
|
return t;
|
|
}
|
|
|
|
// Check that the Montgomery multiply intrinsic returns the same
|
|
// result as the longhand calculation.
|
|
void check(int[] a_words, int[] b_words, int[] n_words, int len, BigInteger inv)
|
|
throws Throwable {
|
|
BigInteger n = newBigInteger(n_words);
|
|
BigInteger slow = montgomeryMultiply(a_words, b_words, n_words, len, inv);
|
|
BigInteger fast
|
|
= newBigInteger(montgomeryMultiply
|
|
(a_words, b_words, n_words, len, inv.longValue(), null));
|
|
// The intrinsic may not return the same value as the longhand
|
|
// calculation but they must have the same residue mod N.
|
|
if (!slow.mod(n).equals(fast.mod(n))) {
|
|
throw new RuntimeException();
|
|
}
|
|
}
|
|
|
|
Random rnd = new Random(42);
|
|
|
|
// Return a random value of length <= bits in an array of even length
|
|
int[] random_val(int bits) {
|
|
int len = (bits+63)/64; // i.e. length in longs
|
|
int[] val = new int[len*2];
|
|
for (int i = 0; i < val.length; i++)
|
|
val[i] = rnd.nextInt();
|
|
int leadingZeros = 64 - (bits & 64);
|
|
if (leadingZeros >= 32) {
|
|
val[0] = 0;
|
|
val[1] &= ~(-1l << (leadingZeros & 31));
|
|
} else {
|
|
val[0] &= ~(-1l << leadingZeros);
|
|
}
|
|
return val;
|
|
}
|
|
|
|
void testOneLength(int lenInBits, int lenInInts) throws Throwable {
|
|
BigInteger mod = new BigInteger(lenInBits, 2, rnd);
|
|
BigInteger r = BigInteger.ONE.shiftLeft(lenInInts * 32);
|
|
BigInteger n_prime = mod.modInverse(r).negate();
|
|
|
|
// Make n.length even, padding with a zero if necessary
|
|
int[] n = mag(mod);
|
|
if (n.length < lenInInts) {
|
|
int[] x = new int[lenInInts];
|
|
System.arraycopy(n, 0, x, lenInInts-n.length, n.length);
|
|
n = x;
|
|
}
|
|
|
|
for (int i = 0; i < 10000; i++) {
|
|
// multiply
|
|
check(random_val(lenInBits), random_val(lenInBits), n, lenInInts, n_prime);
|
|
// square
|
|
int[] tmp = random_val(lenInBits);
|
|
check(tmp, tmp, n, lenInInts, n_prime);
|
|
}
|
|
}
|
|
|
|
// Test the Montgomery multiply intrinsic with a bunch of random
|
|
// values of varying lengths. Do this for long enough that the
|
|
// caller of the intrinsic is C2-compiled.
|
|
void testResultValues() throws Throwable {
|
|
// Test a couple of interesting edge cases.
|
|
testOneLength(1024, 32);
|
|
testOneLength(1025, 34);
|
|
for (int j = 10; j > 0; j--) {
|
|
// Construct a random prime whose length in words is even
|
|
int lenInBits = rnd.nextInt(2048) + 64;
|
|
int lenInInts = (lenInBits + 63)/64*2;
|
|
testOneLength(lenInBits, lenInInts);
|
|
}
|
|
}
|
|
|
|
// Range checks
|
|
void testOneMontgomeryMultiplyCheck(int[] a, int[] b, int[] n, int len, long inv,
|
|
int[] product, Class klass) {
|
|
try {
|
|
montgomeryMultiply(a, b, n, len, inv, product);
|
|
} catch (Throwable ex) {
|
|
if (klass.isAssignableFrom(ex.getClass()))
|
|
return;
|
|
throw new RuntimeException(klass + " expected, " + ex + " was thrown");
|
|
}
|
|
throw new RuntimeException(klass + " expected, was not thrown");
|
|
}
|
|
|
|
void testOneMontgomeryMultiplyCheck(int[] a, int[] b, BigInteger n, int len, BigInteger inv,
|
|
Class klass) {
|
|
testOneMontgomeryMultiplyCheck(a, b, mag(n), len, inv.longValue(), null, klass);
|
|
}
|
|
|
|
void testOneMontgomeryMultiplyCheck(int[] a, int[] b, BigInteger n, int len, BigInteger inv,
|
|
int[] product, Class klass) {
|
|
testOneMontgomeryMultiplyCheck(a, b, mag(n), len, inv.longValue(), product, klass);
|
|
}
|
|
|
|
void testMontgomeryMultiplyChecks() {
|
|
int[] blah = random_val(40);
|
|
int[] small = random_val(39);
|
|
BigInteger mod = new BigInteger(40*32 , 2, rnd);
|
|
BigInteger r = BigInteger.ONE.shiftLeft(40*32);
|
|
BigInteger n_prime = mod.modInverse(r).negate();
|
|
|
|
// Length out of range: square
|
|
testOneMontgomeryMultiplyCheck(blah, blah, mod, 41, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah, mod, 0, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah, mod, -1, n_prime, IllegalArgumentException.class);
|
|
// As above, but for multiply
|
|
testOneMontgomeryMultiplyCheck(blah, blah.clone(), mod, 41, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah.clone(), mod, 0, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah.clone(), mod, 0, n_prime, IllegalArgumentException.class);
|
|
|
|
// Length odd
|
|
testOneMontgomeryMultiplyCheck(small, small, mod, 39, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, small, mod, 0, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, small, mod, -1, n_prime, IllegalArgumentException.class);
|
|
// As above, but for multiply
|
|
testOneMontgomeryMultiplyCheck(small, small.clone(), mod, 39, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, small.clone(), mod, 0, n_prime, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, small.clone(), mod, -1, n_prime, IllegalArgumentException.class);
|
|
|
|
// array too small
|
|
testOneMontgomeryMultiplyCheck(blah, blah, mod, 40, n_prime, small, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah.clone(), mod, 40, n_prime, small, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, blah, mod, 40, n_prime, blah, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, small, mod, 40, n_prime, blah, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(blah, blah, mod, 40, n_prime, small, IllegalArgumentException.class);
|
|
testOneMontgomeryMultiplyCheck(small, small, mod, 40, n_prime, blah, IllegalArgumentException.class);
|
|
}
|
|
|
|
public static void main(String args[]) {
|
|
if (!Platform.isServer() || Platform.isEmulatedClient()) {
|
|
throw new Error("TESTBUG: Not server mode");
|
|
}
|
|
if (wb.isIntrinsicAvailable(getExecutable(true), CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION) &&
|
|
wb.isIntrinsicAvailable(getExecutable(false), CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION)) {
|
|
try {
|
|
new MontgomeryMultiplyTest().testMontgomeryMultiplyChecks();
|
|
new MontgomeryMultiplyTest().testResultValues();
|
|
} catch (Throwable ex) {
|
|
throw new RuntimeException(ex);
|
|
}
|
|
}
|
|
}
|
|
}
|