2018-06-01 15:48:55 -07:00
|
|
|
/*
|
2020-04-30 08:07:36 -07:00
|
|
|
* Copyright (c) 1998, 2020, Oracle and/or its affiliates. All rights reserved.
|
2018-06-01 15:48:55 -07:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package jit.graph;
|
2019-10-11 09:43:41 -07:00
|
|
|
|
|
|
|
import jdk.test.lib.Utils;
|
|
|
|
import jtreg.SkippedException;
|
2018-06-01 15:48:55 -07:00
|
|
|
import nsk.share.TestFailure;
|
|
|
|
import nsk.share.test.StressOptions;
|
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
import java.lang.reflect.InvocationTargetException;
|
|
|
|
import java.util.Vector;
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
public class CGT {
|
|
|
|
private static StressOptions stressOptions = new StressOptions();
|
|
|
|
private static String ClistPath = "";
|
|
|
|
private static long finishTime;
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
private final Vector summation = new Vector(100000);
|
|
|
|
private final Vector idList = new Vector(100000);
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
public CGT(String[] args) {
|
|
|
|
parse(args);
|
|
|
|
Globals.initialize(ClistPath);
|
|
|
|
outputStats(args);
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
public static void main(String[] args) {
|
|
|
|
stressOptions.parseCommandLine(args);
|
|
|
|
new CGT(args).run();
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
public void outputStats(String[] args) {
|
|
|
|
System.out.println("CGT command line options:");
|
|
|
|
for (String arg : args) {
|
|
|
|
System.out.println("# " + arg);
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
System.out.println();
|
|
|
|
|
|
|
|
System.out.println("CGT parameters");
|
|
|
|
System.out.println("Seed: " + Utils.SEED);
|
|
|
|
System.out.println("Number of Random Loop iterations: " + Globals.RANDOM_LOOP);
|
|
|
|
System.out.println("Number of Static Loop iterations: " + Globals.STATIC_LOOP);
|
|
|
|
System.out.println("Max number of Methods in the Graph: " + Globals.NUM_TEST_CLASSES);
|
|
|
|
System.out.println("Verbose function calls: " + Globals.VERBOSE);
|
|
|
|
|
|
|
|
System.out.println();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void run() {
|
|
|
|
finishTime = System.currentTimeMillis() + stressOptions.getTime() * 1000;
|
|
|
|
Long numFcalls = Globals.RANDOM_LOOP - 1;
|
|
|
|
Integer staticFcalls = Globals.STATIC_LOOP;
|
|
|
|
MethodData methodCallStr = Globals.nextRandomMethod();
|
|
|
|
Globals.addFunctionIDToVector(methodCallStr.id, idList);
|
|
|
|
Throwable invocationExcept;
|
|
|
|
|
|
|
|
try {
|
|
|
|
methodCallStr.nextMethod.invoke(methodCallStr.instance, summation, idList, numFcalls, staticFcalls);
|
|
|
|
} catch (IllegalAccessException e) {
|
|
|
|
throw new TestFailure("Illegal Access Exception", e);
|
|
|
|
} catch (InvocationTargetException e) {
|
|
|
|
System.out.println("Invocation Target Exception");
|
|
|
|
invocationExcept = e.getTargetException();
|
|
|
|
System.out.println(invocationExcept);
|
|
|
|
if (invocationExcept.getClass() == e.getClass()) {
|
|
|
|
System.out.println("Processing Exception Invocation Target Exception");
|
|
|
|
while (invocationExcept.getClass() == e.getClass()) {
|
|
|
|
invocationExcept = ((InvocationTargetException) invocationExcept).getTargetException();
|
|
|
|
}
|
|
|
|
System.out.println(invocationExcept);
|
|
|
|
}
|
|
|
|
if (invocationExcept instanceof StackOverflowError) {
|
|
|
|
throw new SkippedException("stack overflow: skipping verification.", invocationExcept);
|
|
|
|
} else if (invocationExcept instanceof OutOfMemoryError) {
|
|
|
|
throw new SkippedException("test devoured heap ;), skipping verification.", invocationExcept);
|
|
|
|
} else {
|
|
|
|
throw new TestFailure(invocationExcept);
|
|
|
|
}
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
verify();
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
private void verify() {
|
|
|
|
long oldsum = 0;
|
|
|
|
long newsum;
|
|
|
|
System.out.println("begin call stack validation");
|
|
|
|
if (summation.size() != idList.size()) {
|
|
|
|
throw new TestFailure("Vector Length's Do Not Match, VERIFY ERROR : Summation Element Count = " + summation.size() + " ID Element Count = " + idList.size());
|
|
|
|
}
|
|
|
|
long vectorSize = summation.size();
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
while (!summation.isEmpty()) {
|
|
|
|
if (CGT.shouldFinish()) {
|
|
|
|
throw new SkippedException("skipping verification due to timeout");
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
newsum = (Long) summation.firstElement();
|
|
|
|
summation.removeElementAt(0);
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
int functionID = (Integer) idList.firstElement();
|
|
|
|
idList.removeElementAt(0);
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
if ((newsum - oldsum) != (functionID)) {
|
|
|
|
throw new TestFailure("Function Call structure invalid, VERIFY ERROR. Expected = " + (newsum - oldsum) + " Actual = " + functionID);
|
|
|
|
}
|
|
|
|
oldsum = newsum;
|
2018-06-01 15:48:55 -07:00
|
|
|
}
|
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
System.out.println("function call structure validated successfully (" + vectorSize + " calls validated)");
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean shouldFinish() {
|
|
|
|
return System.currentTimeMillis() >= finishTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void parse(String args[]) {
|
|
|
|
for (int i = 0; i < args.length; i++) {
|
|
|
|
String arg = args[i].toLowerCase();
|
|
|
|
switch (arg) {
|
|
|
|
case "-help":
|
|
|
|
case "-h":
|
|
|
|
case "-?": {
|
|
|
|
usage();
|
|
|
|
System.exit(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "-staticloop": {
|
|
|
|
int argIndex = i + 1;
|
|
|
|
if (argIndex < args.length) {
|
|
|
|
try {
|
|
|
|
Globals.STATIC_LOOP = Math.abs(Integer.parseInt(args[argIndex])) * stressOptions.getIterationsFactor();
|
|
|
|
} catch (NumberFormatException e) {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i] + " " + args[argIndex], e);
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
} else {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "-randomloop": {
|
|
|
|
int argIndex = i + 1;
|
|
|
|
if (argIndex < args.length) {
|
|
|
|
try {
|
|
|
|
Globals.RANDOM_LOOP = Math.abs(Long.parseLong(args[argIndex])) * stressOptions.getIterationsFactor();
|
|
|
|
} catch (NumberFormatException e) {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i] + " " + args[argIndex], e);
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
} else {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i]);
|
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "-numtestclass": {
|
|
|
|
int argIndex = i + 1;
|
|
|
|
if (argIndex < args.length) {
|
|
|
|
try {
|
|
|
|
Globals.NUM_TEST_CLASSES = Math.abs(Integer.parseInt(args[argIndex]));
|
|
|
|
} catch (NumberFormatException e) {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i] + " " + args[argIndex], e);
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
} else {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "-verbose":
|
|
|
|
case "-v": {
|
|
|
|
Globals.VERBOSE = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "-path": {
|
|
|
|
int argIndex = i + 1;
|
|
|
|
if (argIndex < args.length) {
|
|
|
|
ClistPath = args[argIndex];
|
|
|
|
i++;
|
|
|
|
} else {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Improper Argument: " + args[i]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
if (!arg.startsWith("-stress")) {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: Invalid Argument: " + args[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
|
2019-10-11 09:43:41 -07:00
|
|
|
if ("".equals(ClistPath)) {
|
|
|
|
usage();
|
|
|
|
throw new Error("TESTBUG: class list path not defined");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void usage() {
|
|
|
|
System.out.println("usage: java CGT [options]");
|
|
|
|
System.out.println(" -help prints out this message");
|
|
|
|
System.out.println(" -numTestClass # limits the number of \"Test Methods\" to #");
|
|
|
|
System.out.println(" -randomcLoop # # of random function calls");
|
|
|
|
System.out.println(" -staticLoop # # of non-random static function calls");
|
|
|
|
System.out.println(" -v -verbose turn on verbose mode");
|
|
|
|
System.out.println(" -path <path to classlist> required, argument so program can find classes");
|
|
|
|
}
|
2018-06-01 15:48:55 -07:00
|
|
|
}
|