test/hotspot/jtreg/vmTestbase/nsk/share/aod/AbstractJarAgent.java
author jlahoda
Wed, 27 Nov 2019 09:00:01 +0100
changeset 59285 7799a51dbe30
parent 49934 44839fbb20db
permissions -rw-r--r--
8231826: Implement javac changes for pattern matching for instanceof Reviewed-by: mcimadamore Contributed-by: brian.goetz@oracle.com, gavin.bierman@oracle.com, maurizio.cimadamore@oracle.com, srikanth.adayapalam@oracle.com, vicente.romero@oracle.com, jan.lahoda@oracle.com

/*
 * Copyright (c) 2008, 2018, 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 nsk.share.aod;

import java.lang.instrument.*;
import java.io.*;
import nsk.share.*;

/*

AbstractJarAgent is base class for java agents used in AttachOnDemand tests
(tests against Attach API, API from package com.sun.tools.attach).

In all AttachOnDemand tests the same algorithm is used:
    - java application where agent is loaded to (target application) based on
     class nsk.share.aod.TargetApplicationWaitingAgents starts and waits when
     test agents will be loaded

    - special application (nsk.share.jvmti.aod.AgentsAttacher) loads test agents
    in the target application using Attach API

    - when agent is loaded it notifies target application about that and executes
    test-specific actions. When agent execution is completed it also notifies
    target application about that

    - when all test agents finish execution target application checks its status
    (passed or failed) and also finishes

Each java agent should have method 'agentmain' where only agent initialization should be done,
main agent's actions should be executed in special thread started from 'agentmain'.
Class AbstractJarAgent incapsulates actions common for all java agents: agent initialization,
starting thread executing agent's actions and communication with target application.

In most cases test agents should override only method 'agentActions' and its 'agentmain'
should contain only call of the method 'AbstractJarAgent.runJarAgent'.

Typical agent class looks like this:

public class agentExample extends AbstractJarAgent {

    protected void init(String[] args) {
        // parse agent's options and do test-specific initialization
    }

    protected void agentActions() {
        // do test specific actions
    }

    public static void agentmain(final String options, Instrumentation inst) {
        new agentExample().runJarAgent(options, inst);
    }
}
 */
abstract public class AbstractJarAgent {

    private boolean finishedSuccessfully = true;

    private Log log;

    protected void display(String message) {
        log.display(outputPrefix + message);
    }

    protected void complain(String message) {
        log.complain(outputPrefix + message);
    }

    protected void logThrowable(Throwable t) {
        t.printStackTrace(log.getOutStream());
    }

    /*
     * Instrumentation object passed to the 'agentmain' method
     */
    protected Instrumentation inst;

    private String name;

    private String outputPrefix;

    private String pathToNewByteCode;

    protected String pathToNewByteCode() {
        return pathToNewByteCode;
    }

    /*
     * Subclasses should report about test failures using this method
     */
    protected void setStatusFailed(String errorMessage) {
        finishedSuccessfully = false;
        complain("ERROR: " + errorMessage);
    }

    /*
     * Initialization method, called from agentmain before method agentActions is called
     * (it introduced for overriding in subclasses)
     */
    protected void init(String[] args) {
    }

    protected static class AgentOption {
        public String name;
        public String value;
        public AgentOption(String name, String value) {
            this.name = name;
            this.value = value;
        }
    }

    protected AgentOption parseAgentArgument(String arg) {
        int index = arg.indexOf('=');
        if (index <= 0) {
            throw new TestBug("Invalid agent parameters format");
        }
        return new AgentOption(arg.substring(0, index), arg.substring(index + 1));
    }

    static protected final String agentNameOption = "-agentName";

    static protected final String pathToNewByteCodeOption = "-pathToNewByteCode";

    /*
     * Parse agent's options, initialize common parameters
     */
    private void defaultInit(String[] args) {
        for (int i = 0; i < args.length; i++) {
            AgentOption option = parseAgentArgument(args[i]);
            if (option.name.equals(agentNameOption)) {
                name = option.value;
                outputPrefix = name + ": ";
            } else if (option.name.equals(pathToNewByteCodeOption)) {
                pathToNewByteCode = option.value;
            }
        }

        if (name == null)
            throw new TestBug("Agent name wasn't specified");

        log = new Log(System.out, true);
    }

    /*
     * Special thread which is started from agentmain method and executing main
     * agent's actions. When agent completes execution AgentThread notifies
     * target application about that.
     */
    class AgentThread extends Thread {

        AgentThread() {
            super("Jar agent thread (agent: " + name + ")");
        }

        public void run() {
            try {
                agentActions();
            } catch (Throwable t) {
                setStatusFailed("Unexpected exception in the JarAgent: " + t);
                logThrowable(t);
            } finally {
                TargetApplicationWaitingAgents.agentFinished(name, finishedSuccessfully);
            }
        }
    }

    /*
     * This methods parses agent's options, initializes agent, notifies target application
     * that agent is started and starts thread executing main agent's actions.
     * Agents used in AttachOnDemand tests should call this method from its agentmain methods.
     */
    public final void runJarAgent(String options, Instrumentation inst) {
        if (options == null)
            throw new TestBug("Agent options weren't specified");

        this.inst = inst;

        String[] args = options.split(" ");

        // initialize common parameters
        defaultInit(args);

        // test-specific initialization
        init(args);

        // notify target application that agent was loaded and initialized
        TargetApplicationWaitingAgents.agentLoaded(name);

        // start special thread executing test-specific actions
        new AgentThread().start();
    }

    /*
     * Actions specific for test should be realized in this method.
     * This method is called from special thread started from agentmain method
     * after agent initialization
     */
    abstract protected void agentActions() throws Throwable;


    /*
     * Create ClassDefinition object for given class, path to the new class file
     * is specified in the parameter 'pathToByteCode'
     */
    protected static ClassDefinition createClassDefinition(Class<?> klass,
            String pathToByteCode) throws IOException {
        File classFile = new File(pathToByteCode + File.separator +
                klass.getName().replace(".", File.separator) +
                ".class");

        if (classFile.length() > Integer.MAX_VALUE)
            throw new Failure("Class file '" + classFile.getName() + " 'too large");

        byte data[] = new byte[(int)classFile.length()];

        DataInputStream in = null;
        try {
            in = new DataInputStream(new FileInputStream(classFile));
            in.readFully(data);
        } finally {
            if (in != null)
                in.close();
        }

        return new ClassDefinition(klass, data);
    }

    /*
     * Redefine given class using path to byte code specified with options -pathToNewByteCode
     */
    protected void redefineClass(Class<?> klass) throws Throwable {
        if (pathToNewByteCode() == null)
            throw new TestBug("Path to new class files wasn't specified");

        ClassDefinition newClassDef = createClassDefinition(klass, pathToNewByteCode());
        inst.redefineClasses(newClassDef);
    }
}