9583e3657e
Co-authored-by: Ron Pressler <rpressler@openjdk.org> Co-authored-by: Alan Bateman <alanb@openjdk.org> Co-authored-by: Erik Österlund <eosterlund@openjdk.org> Co-authored-by: Andrew Haley <aph@openjdk.org> Co-authored-by: Rickard Bäckman <rbackman@openjdk.org> Co-authored-by: Markus Grönlund <mgronlun@openjdk.org> Co-authored-by: Leonid Mesnik <lmesnik@openjdk.org> Co-authored-by: Serguei Spitsyn <sspitsyn@openjdk.org> Co-authored-by: Chris Plummer <cjplummer@openjdk.org> Co-authored-by: Coleen Phillimore <coleenp@openjdk.org> Co-authored-by: Robbin Ehn <rehn@openjdk.org> Co-authored-by: Stefan Karlsson <stefank@openjdk.org> Co-authored-by: Thomas Schatzl <tschatzl@openjdk.org> Co-authored-by: Sergey Kuksenko <skuksenko@openjdk.org> Reviewed-by: lancea, eosterlund, rehn, sspitsyn, stefank, tschatzl, dfuchs, lmesnik, dcubed, kevinw, amenkov, dlong, mchung, psandoz, bpb, coleenp, smarks, egahlin, mseledtsov, coffeys, darcy
113 lines
4.4 KiB
Java
113 lines
4.4 KiB
Java
/*
|
|
* Copyright (c) 2016, 2022, 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.
|
|
*/
|
|
|
|
import java.lang.instrument.ClassFileTransformer;
|
|
import java.lang.instrument.IllegalClassFormatException;
|
|
import java.lang.instrument.Instrumentation;
|
|
import java.security.ProtectionDomain;
|
|
import java.util.HashMap;
|
|
|
|
// This is a test utility class used to transform
|
|
// specified classes via initial transformation (ClassFileLoadHook).
|
|
// Names of classes to be transformed are supplied as arguments,
|
|
// the phrase to be transformed is a hard-coded predefined
|
|
// fairly unique phrase.
|
|
|
|
public class TransformerAgent {
|
|
private static String[] classesToTransform;
|
|
|
|
|
|
private static void log(String msg) {
|
|
System.out.println("TransformerAgent: " + msg);
|
|
}
|
|
|
|
|
|
// arguments are comma-separated list of classes to transform
|
|
public static void premain(String agentArguments, Instrumentation instrumentation) {
|
|
log("premain() is called, arguments = " + agentArguments);
|
|
classesToTransform = agentArguments.split(",");
|
|
instrumentation.addTransformer(new SimpleTransformer(), /*canRetransform=*/true);
|
|
}
|
|
|
|
|
|
public static void agentmain(String args, Instrumentation inst) throws Exception {
|
|
log("agentmain() is called");
|
|
premain(args, inst);
|
|
}
|
|
|
|
|
|
static class SimpleTransformer implements ClassFileTransformer {
|
|
public byte[] transform(ClassLoader loader, String name, Class<?> classBeingRedefined,
|
|
ProtectionDomain pd, byte[] buffer) throws IllegalClassFormatException {
|
|
try {
|
|
// Printing cause ClassCircularityError for java/util/concurrent/locks/AbstractQueuedSynchronizer$ExclusiveNode
|
|
// So don't try to print some system classes
|
|
if (!name.startsWith("java/")) {
|
|
log("SimpleTransformer called for: " + name + "@" + incrCounter(name));
|
|
}
|
|
if (!shouldTransform(name))
|
|
return null;
|
|
|
|
log("transforming: class name = " + name);
|
|
int nrOfReplacements = TransformUtil.replace(buffer, TransformUtil.BeforePattern,
|
|
TransformUtil.AfterPattern);
|
|
log("replaced the string, nrOfReplacements = " + nrOfReplacements);
|
|
} catch (Throwable t) {
|
|
// The retransform native code that called this method does not propagate
|
|
// exceptions. Instead of getting an uninformative generic error, catch
|
|
// problems here and print it, then exit.
|
|
log("Transformation failed!");
|
|
t.printStackTrace();
|
|
System.exit(1);
|
|
}
|
|
return buffer;
|
|
}
|
|
|
|
// Check class name pattern, since test should only transform certain classes
|
|
private static boolean shouldTransform(String name) {
|
|
for (String match : classesToTransform) {
|
|
if (name.matches(match)) {
|
|
log("shouldTransform: match-found, match = " + match);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
static HashMap<String, Integer> counterMap = new HashMap<>();
|
|
|
|
static Integer incrCounter(String className) {
|
|
Integer i = counterMap.get(className);
|
|
if (i == null) {
|
|
i = Integer.valueOf(1);
|
|
} else {
|
|
i = Integer.valueOf(i.intValue() + 1);
|
|
}
|
|
counterMap.put(className, i);
|
|
return i;
|
|
}
|
|
}
|