/* * 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 compiler.compilercontrol.share.scenario; import compiler.compilercontrol.share.method.MethodDescriptor; import compiler.compilercontrol.share.processors.CommandProcessor; import compiler.compilercontrol.share.processors.LogProcessor; import compiler.compilercontrol.share.processors.PrintProcessor; import compiler.compilercontrol.share.processors.QuietProcessor; import jdk.test.lib.Asserts; import jdk.test.lib.OutputAnalyzer; import jdk.test.lib.Pair; import pool.PoolHelper; import java.lang.reflect.Executable; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.Callable; import java.util.function.Consumer; /** * Test scenario */ public final class Scenario { private final boolean isValid; private final Map states; private final List> processors; private final Executor executor; private Scenario(boolean isValid, List vmopts, Map states, List compileCommands, List jcmdCommands) { this.isValid = isValid; this.states = states; processors = new ArrayList<>(); processors.add(new LogProcessor(states)); processors.add(new PrintProcessor(states)); List nonQuieted = new ArrayList<>(); List quieted = new ArrayList<>(); boolean metQuiet = false; for (CompileCommand cc : compileCommands) { metQuiet |= cc.command == Command.QUIET; if (metQuiet) { quieted.add(cc); } else { nonQuieted.add(cc); } } processors.add(new CommandProcessor(nonQuieted)); processors.add(new QuietProcessor(quieted)); List jcmdExecCommands = new ArrayList<>(); boolean addCommandMet = false; for (JcmdCommand cmd : jcmdCommands) { switch (cmd.jcmdType) { case ADD: if (!addCommandMet) { jcmdExecCommands.add(JcmdType.ADD.command); } addCommandMet = true; break; default: jcmdExecCommands.add(cmd.jcmdType.command); break; } } executor = new Executor(isValid, vmopts, states, jcmdExecCommands); } /** * Executes scenario */ public void execute() { OutputAnalyzer output = executor.execute(); if (isValid) { output.shouldHaveExitValue(0); processors.forEach(processor -> processor.accept(output)); } else { Asserts.assertNE(output.getExitValue(), 0, "VM should exit with " + "error for incorrect directives"); output.shouldContain("Parsing of compiler directives failed"); } } /** * Gets states of methods for this scenario * * @return pairs of executable and its state */ public Map getStates() { return states; } public static enum Compiler { C1("c1"), C2("c2"); public final String name; Compiler(String name) { this.name = name; } } /** * Type of diagnostic (jcmd) command */ public static enum JcmdType { ADD("Compiler.directives_add " + Type.JCMD.fileName), PRINT("Compiler.directives_print"), CLEAR("Compiler.directives_clear"), REMOVE("Compiler.directives_remove"); public final String command; private JcmdType(String command) { this.command = command; } } /** * Type of the compile command */ public static enum Type { OPTION(""), FILE("command_file"), DIRECTIVE("directives.json"), JCMD("jcmd_directives.json") { @Override public CompileCommand createCompileCommand(Command command, MethodDescriptor md, Compiler compiler) { return new JcmdCommand(command, md, compiler, this, JcmdType.ADD); } }; public final String fileName; public CompileCommand createCompileCommand(Command command, MethodDescriptor md, Compiler compiler) { return new CompileCommand(command, md, compiler, this); } private Type(String fileName) { this.fileName = fileName; } } public static Builder getBuilder() { return new Builder(); } public static class Builder { private final Set vmopts = new LinkedHashSet<>(); private final Map> builders = new HashMap<>(); private final JcmdStateBuilder jcmdStateBuilder; public Builder() { builders.put(Type.FILE, new CommandFileBuilder(Type.FILE.fileName)); builders.put(Type.OPTION, new CommandOptionsBuilder()); builders.put(Type.DIRECTIVE, new DirectiveBuilder( Type.DIRECTIVE.fileName)); jcmdStateBuilder = new JcmdStateBuilder(Type.JCMD.fileName); } public void add(CompileCommand compileCommand) { String[] vmOptions = compileCommand.command.vmOpts; Collections.addAll(vmopts, vmOptions); if (compileCommand.type == Type.JCMD) { jcmdStateBuilder.add((JcmdCommand) compileCommand); } else { StateBuilder builder = builders.get( compileCommand.type); if (builder == null) { throw new Error("TESTBUG: Missing builder for the type: " + compileCommand.type); } builder.add(compileCommand); } } public Scenario build() { boolean isValid = true; // Get states from each of the state builders Map commandFileStates = builders.get(Type.FILE).getStates(); Map commandOptionStates = builders.get(Type.OPTION).getStates(); Map directiveFileStates = builders.get(Type.DIRECTIVE).getStates(); // get all jcmd commands List jcmdCommands = jcmdStateBuilder .getCompileCommands(); boolean isClearedState = false; if (jcmdClearedState(jcmdCommands)) { isClearedState = true; } // Merge states List>> methods = new PoolHelper() .getAllMethods(); Map finalStates = new HashMap<>(); Map jcmdStates = jcmdStateBuilder.getStates(); for (Pair> pair : methods) { Executable x = pair.first; State commandOptionState = commandOptionStates.get(x); State commandFileState = commandFileStates.get(x); State st = State.merge(commandOptionState, commandFileState); if (!isClearedState) { State directiveState = directiveFileStates.get(x); if (directiveState != null) { st = directiveState; } } State jcmdState = jcmdStates.get(x); st = State.merge(st, jcmdState); finalStates.put(x, st); } /* * Create a list of commands from options and file * to handle quiet command */ List ccList = new ArrayList<>(); ccList.addAll(builders.get(Type.OPTION).getCompileCommands()); ccList.addAll(builders.get(Type.FILE).getCompileCommands()); // Get all VM options after we build all states and files List options = new ArrayList<>(); options.addAll(vmopts); for (StateBuilder builder : builders.values()) { options.addAll(builder.getOptions()); isValid &= builder.isValid(); } options.addAll(jcmdStateBuilder.getOptions()); return new Scenario(isValid, options, finalStates, ccList, jcmdCommands); } // shows if jcmd have passed a clear command private boolean jcmdClearedState(List jcmdCommands) { for (JcmdCommand jcmdCommand : jcmdCommands) { if (jcmdCommand.jcmdType == JcmdType.CLEAR) { return true; } } return false; } } }