fe008ae27a
Reviewed-by: darcy, weijun
161 lines
6.4 KiB
Java
161 lines
6.4 KiB
Java
/*
|
|
* Copyright (c) 2006, 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 6364692
|
|
* @summary When the "java.rmi.server.randomIDs" system property is
|
|
* not defined, the ObjID() constructor should behave as if it were
|
|
* set to "true" and generate random object numbers; if the property
|
|
* is defined to something other than "true" (ignoring case), then
|
|
* ObjID() should still generate sequential object numbers.
|
|
* @author Peter Jones
|
|
*
|
|
* @build RandomIDs
|
|
* @run main/othervm RandomIDs random
|
|
* @run main/othervm -Djava.rmi.server.randomIDs=true RandomIDs random
|
|
* @run main/othervm -Djava.rmi.server.randomIDs=false RandomIDs sequential
|
|
*/
|
|
|
|
import java.io.IOException;
|
|
import java.io.ObjectOutputStream;
|
|
import java.io.OutputStream;
|
|
import java.rmi.server.ObjID;
|
|
|
|
public class RandomIDs {
|
|
private static final int COUNT = 10000;
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
boolean shouldBeRandom = false;
|
|
boolean shouldBeSequential = false;
|
|
|
|
String usage = "Usage: java RandomIDs [random|sequential]";
|
|
if (args.length != 1) {
|
|
System.err.println(usage);
|
|
throw new Error("wrong number of arguments");
|
|
} else if (args[0].equals("random")) {
|
|
shouldBeRandom = true;
|
|
} else if (args[0].equals("sequential")) {
|
|
shouldBeSequential = true;
|
|
} else {
|
|
System.err.println(usage);
|
|
throw new Error("invalid argument");
|
|
}
|
|
|
|
System.err.println("\nRegression test for bug 6364692\n");
|
|
String propertyValue = System.getProperty("java.rmi.server.randomIDs");
|
|
System.err.println(
|
|
"Value of java.rmi.server.randomIDs system property: " +
|
|
(propertyValue != null ? "\"" + propertyValue + "\"" : null));
|
|
System.err.println(
|
|
"Expecting object numbers of unique ObjIDs to be: " + args[0]);
|
|
|
|
/*
|
|
* Get the 64-bit "object number" component of COUNT number of
|
|
* unique (not "well-known") ObjID instances created in
|
|
* sequence, by writing each to a dummy ObjectOutputStream and
|
|
* trapping the first writeLong invocation on the stream.
|
|
*/
|
|
final long[] objnums = new long[COUNT];
|
|
for (int i = 0; i < COUNT; i++) {
|
|
final int j = i;
|
|
class Escape extends RuntimeException { }
|
|
try {
|
|
new ObjID().write(new ObjectOutputStream(new OutputStream() {
|
|
public void write(int b) { }
|
|
}) {
|
|
public void writeLong(long val) throws IOException {
|
|
objnums[j] = val;
|
|
throw new Escape();
|
|
}
|
|
});
|
|
throw new Error("writeLong not invoked");
|
|
} catch (Escape e) {
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If the object numbers should be random, then verify that
|
|
* they are. (This verification is certainly not a thorough
|
|
* evaluation of randomness, but it performs a couple of
|
|
* simple checks to catch mistakes in ObjID's application of a
|
|
* CSPRNG: are roughly half the bits set, and can the sequence
|
|
* be used to get a rough Monte Carlo estimate of pi. Errors
|
|
* up to 5% are tolerated for both checks.)
|
|
*/
|
|
if (shouldBeRandom) {
|
|
int bitCount = 0;
|
|
int piHitCount = 0;
|
|
for (int i = 0; i < COUNT; i++) {
|
|
bitCount += Long.bitCount(objnums[i]);
|
|
double x = ((double) (objnums[i] >>> 32)) / (1L << 32);
|
|
double y = ((double) (objnums[i] & 0xFFFFFFFFL)) / (1L << 32);
|
|
if (((x * x) + (y * y)) <= 1.0) {
|
|
piHitCount++;
|
|
}
|
|
}
|
|
|
|
int bitCountTarget = COUNT * 32;
|
|
double bitCountError =
|
|
((double) (bitCount - bitCountTarget)) / bitCountTarget;
|
|
if (Math.abs(bitCountError) > 0.05) { // tolerate 5% error
|
|
throw new Error("TEST FAILED: " +
|
|
"bitCount == " + bitCount);
|
|
}
|
|
|
|
double piEstimate = ((double) piHitCount / COUNT) * 4.0;
|
|
double piEstimateError = (piEstimate - Math.PI) / Math.PI;
|
|
if (Math.abs(piEstimateError) > 0.05) { // tolerate 5% error
|
|
throw new Error("TEST FAILED: " +
|
|
"piEstimate == " + piEstimate);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* If the object numbers should be sequential, then verify
|
|
* that they are.
|
|
*/
|
|
if (shouldBeSequential) {
|
|
long first = objnums[0];
|
|
/*
|
|
* This test currently verifies that the first object
|
|
* number is zero, but that could be false if one or more
|
|
* remote objects get exported as part of VM startup-- if
|
|
* that starts happening, this check could be relaxed.
|
|
*/
|
|
if (first != 0) {
|
|
throw new Error("TEST FAILED: " +
|
|
"first object number == " + first +
|
|
" (not zero)");
|
|
}
|
|
for (int i = 1; i < COUNT; i++) {
|
|
if (objnums[i] != first + i) {
|
|
throw new Error("TEST FAILED: first == " + first + ", " +
|
|
"objnums[" + i + "] == " + objnums[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
System.err.println("TEST PASSED");
|
|
}
|
|
}
|