From 7aca8d6fe4fdf869da6f82e724b3f2c30b7cbf80 Mon Sep 17 00:00:00 2001 From: Dmitrij Pochepko Date: Thu, 23 Jun 2016 17:39:06 +0300 Subject: [PATCH] 8158650: [jittester] when generating tests with default parameters, generation hangs after 98 test Reviewed-by: kvn, iignatyev --- .../src/jdk/test/lib/jittester/IRNode.java | 25 ++++++++++++++++ .../ClassDefinitionBlockFactory.java | 30 +++++++++---------- .../jittester/factories/MainKlassFactory.java | 30 +++++++++---------- 3 files changed, 54 insertions(+), 31 deletions(-) diff --git a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/IRNode.java b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/IRNode.java index 3ffa03e8669..c532d0cd3f8 100644 --- a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/IRNode.java +++ b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/IRNode.java @@ -198,6 +198,31 @@ public abstract class IRNode { return result; } + public static long getModifiableNodesCount(List nodes) { + return nodes.stream() + .map(IRNode::getStackableLeaves) + .mapToInt(List::size) + .filter(i -> i > 0) + .count(); + } + + public static boolean tryToReduceNodesDepth(List nodes, int maxDepth) { + boolean allSucceed = true; + for (IRNode child : nodes) { + for (IRNode leaf : child.getDeviantBlocks(Math.max(child.countDepth(), maxDepth + 1))) { + if (child.countDepth() > maxDepth) { + // doesn't remove control deviation block. Just some parts. + leaf.removeSelf(); + boolean successfull = child.countDepth() > maxDepth; + allSucceed &= successfull; + } else { + break; + } + } + } + return allSucceed; + } + // TODO: add field instead this function public boolean isCFDeviation() { return this instanceof If || this instanceof Switch diff --git a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/ClassDefinitionBlockFactory.java b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/ClassDefinitionBlockFactory.java index 822c99fada8..f2c5af8ebc5 100644 --- a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/ClassDefinitionBlockFactory.java +++ b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/ClassDefinitionBlockFactory.java @@ -102,11 +102,16 @@ class ClassDefinitionBlockFactory extends Factory { addMoreChildren(childs, content, minDepth); } - private void addMoreChildren(List childs, Collection content, int minDepth) - throws ProductionFailedException { - while (!childs.isEmpty() && IRNode.countDepth(content) < minDepth) { - PseudoRandom.shuffle(childs); - IRNode randomChild = childs.get(0); + private void addMoreChildren(List children, Collection content, int minDepth) + throws ProductionFailedException { + /* check situation when no stackable leaves available in all children */ + if (IRNode.getModifiableNodesCount(children) == 0L) { + return; + } + /* now let's try to add children */ + while (!children.isEmpty() && IRNode.countDepth(content) < minDepth) { + PseudoRandom.shuffle(children); + IRNode randomChild = children.get(0); List leaves = randomChild.getStackableLeaves(); if (!leaves.isEmpty()) { Block randomLeaf = (Block) leaves.get(PseudoRandom.randomNotNegative(leaves.size())); @@ -131,18 +136,11 @@ class ClassDefinitionBlockFactory extends Factory { private void ensureMaxDepth(Collection content) { int maxDepth = ProductionParams.maxCfgDepth.value(); - List childs = content.stream() + List childrenClasses = content.stream() .filter(c -> c instanceof Klass && c.countDepth() > maxDepth) .collect(Collectors.toList()); - for (IRNode ch : childs) { - List leaves; - do { - long depth = Math.max(ch.countDepth(), maxDepth + 1); - leaves = ch.getDeviantBlocks(depth); - if(leaves.size() > 0) { - leaves.get(0).removeSelf(); - } - } while (!leaves.isEmpty() && ch.countDepth() > maxDepth); - } + /* now attempt to reduce depth by removing optional parts of control deviation + blocks in case IRTree has oversized depth */ + IRNode.tryToReduceNodesDepth(childrenClasses, maxDepth); } } diff --git a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/MainKlassFactory.java b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/MainKlassFactory.java index 81a19675ca9..54237c5f1cc 100644 --- a/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/MainKlassFactory.java +++ b/hotspot/test/testlibrary/jittester/src/jdk/test/lib/jittester/factories/MainKlassFactory.java @@ -112,19 +112,14 @@ class MainKlassFactory extends Factory { functionDefinitions, testFunction, printVariables); } - private void ensureMaxDepth(List childs) { + private void ensureMaxDepth(List children) { int maxDepth = ProductionParams.maxCfgDepth.value(); - List filtered = childs.stream() - .filter(c -> c.isCFDeviation() && c.countDepth() > maxDepth) - .collect(Collectors.toList()); - for (IRNode child : filtered) { - List leaves; - do { - long depth = Math.max(child.countDepth(), maxDepth + 1); - leaves = child.getDeviantBlocks(depth); - leaves.get(0).removeSelf(); - } while (!leaves.isEmpty() && child.countDepth() > maxDepth); - } + List filtered = children.stream() + .filter(c -> c.isCFDeviation() && c.countDepth() > maxDepth) + .collect(Collectors.toList()); + /* Now attempt to reduce depth by removing optional parts of control deviation + blocks in case IRTree has oversized depth */ + IRNode.tryToReduceNodesDepth(filtered, maxDepth); } private void ensureMinDepth(List childs, IRNodeBuilder builder) @@ -134,10 +129,15 @@ class MainKlassFactory extends Factory { addMoreChildren(filtered, minDepth, builder); } - private void addMoreChildren(List childs, int minDepth, IRNodeBuilder builder) + private void addMoreChildren(List children, int minDepth, IRNodeBuilder builder) throws ProductionFailedException { - while (!childs.isEmpty() && IRNode.countDepth(childs) < minDepth) { - IRNode randomChild = childs.get(PseudoRandom.randomNotNegative(childs.size())); + /* check situation when no stackable leaves available in all children */ + if (IRNode.getModifiableNodesCount(children) == 0L) { + return; + } + /* now let's try to add children */ + while (!children.isEmpty() && IRNode.countDepth(children) < minDepth) { + IRNode randomChild = children.get(PseudoRandom.randomNotNegative(children.size())); List leaves = randomChild.getStackableLeaves(); if (!leaves.isEmpty()) { Block randomLeaf = (Block) leaves.get(PseudoRandom.randomNotNegative(leaves.size()));