blob: cfbec4aabe934cdd5bc61db1cc734dfa7e962eeb [file] [log] [blame]
/*
* Copyright (c) 2015, 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.
*/
package jdk.test.lib.jittester.factories;
import java.util.ArrayList;
import java.util.List;
import jdk.test.lib.jittester.Block;
import jdk.test.lib.jittester.CatchBlock;
import jdk.test.lib.jittester.ProductionFailedException;
import jdk.test.lib.jittester.TryCatchBlock;
import jdk.test.lib.jittester.Type;
import jdk.test.lib.jittester.TypeList;
import jdk.test.lib.jittester.utils.TypeUtil;
import jdk.test.lib.jittester.types.TypeKlass;
import jdk.test.lib.jittester.utils.PseudoRandom;
class TryCatchBlockFactory extends Factory<TryCatchBlock> {
private final static double CATCH_SELECTION_COEF = 0.1d;
private final Type returnType;
private final long complexityLimit;
private final int statementLimit, operatorLimit;
private final boolean subBlock;
private final boolean canHaveBreaks;
private final boolean canHaveContinues;
private final boolean canHaveReturn;
private final int level;
private final TypeKlass ownerClass;
TryCatchBlockFactory(TypeKlass ownerClass, Type returnType,
long complexityLimit, int statementLimit, int operatorLimit,
int level, boolean subBlock, boolean canHaveBreaks,
boolean canHaveContinues, boolean canHaveReturn) {
this.ownerClass = ownerClass;
this.returnType = returnType;
this.complexityLimit = complexityLimit;
this.statementLimit = statementLimit;
this.operatorLimit = operatorLimit;
this.level = level;
this.subBlock = subBlock;
this.canHaveBreaks = canHaveBreaks;
this.canHaveContinues = canHaveContinues;
this.canHaveReturn = canHaveReturn;
}
@Override
public TryCatchBlock produce() throws ProductionFailedException {
if (complexityLimit < 1 || statementLimit < 1) {
throw new ProductionFailedException();
}
List<Type> uncheckedThrowables = getUncheckedThrowables();
IRNodeBuilder builder = new IRNodeBuilder().setOwnerKlass(ownerClass)
.setResultType(returnType)
.setOperatorLimit(operatorLimit)
.setLevel(level)
.setSubBlock(subBlock)
.setCanHaveReturn(canHaveReturn)
.setCanHaveContinues(canHaveContinues)
.setCanHaveBreaks(canHaveBreaks);
Block body = getBlock(builder, 0.6);
int catchBlocksCount = (int) (CATCH_SELECTION_COEF
* PseudoRandom.random() * uncheckedThrowables.size());
List<CatchBlock> catchBlocks = new ArrayList<>();
List<Type> caught = new ArrayList<>();
for (int i = 0; i < catchBlocksCount; i++) {
List<Type> whatToCatch = new ArrayList<>();
int throwableLimit = 1 + (int) ((1/(2*CATCH_SELECTION_COEF))
* PseudoRandom.random());
for (int j = 0; j < throwableLimit; j++) {
whatToCatch.add(selectUniqueThrowable(uncheckedThrowables, caught));
}
catchBlocks.add(new CatchBlock(getBlock(builder, 0.3/catchBlocksCount),
whatToCatch, level));
}
Block finallyBody = PseudoRandom.randomBoolean() || catchBlocksCount == 0 ? getBlock(builder, 0.1) : null;
return new TryCatchBlock(body, finallyBody, catchBlocks, level);
}
private Type selectUniqueThrowable(List<Type> variants, List<Type> caught) {
Type selected;
do {
int randomIndex = PseudoRandom.randomNotZero(variants.size()) - 1;
selected = variants.get(randomIndex);
} while (caught.contains(selected));
caught.add(selected);
return selected;
}
private Block getBlock(IRNodeBuilder builder, double weight)
throws ProductionFailedException {
long actualComplexityLim = (long) (weight * PseudoRandom.random()
* complexityLimit);
int actualStatementLim = (int) (weight * PseudoRandom.random()
* statementLimit);
return builder.setStatementLimit(actualStatementLim)
.setComplexityLimit(actualComplexityLim)
.getBlockFactory()
.produce();
}
private List<Type> getUncheckedThrowables() {
List<Type> result = new ArrayList<>();
result.addAll(TypeUtil.getImplicitlyCastable(TypeList.getAll(),
new TypeKlass("java.lang.Error")));
result.addAll(TypeUtil.getImplicitlyCastable(TypeList.getAll(),
new TypeKlass("java.lang.RuntimeException")));
return result;
}
}