2007-12-01 00:00:00 +00:00
|
|
|
/*
|
2022-10-07 17:13:30 +00:00
|
|
|
* Copyright (c) 2005, 2022, Oracle and/or its affiliates. All rights reserved.
|
2007-12-01 00:00:00 +00:00
|
|
|
* 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.
|
|
|
|
*
|
2010-05-25 22:58:33 +00:00
|
|
|
* 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.
|
2007-12-01 00:00:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @bug 6263319
|
2016-08-24 18:51:37 +00:00
|
|
|
* @requires ((vm.opt.StartFlightRecording == null) | (vm.opt.StartFlightRecording == false)) & ((vm.opt.FlightRecorder == null) | (vm.opt.FlightRecorder == false))
|
2007-12-01 00:00:00 +00:00
|
|
|
* @summary test setNativeMethodPrefix
|
|
|
|
* @author Robert Field, Sun Microsystems
|
|
|
|
*
|
2015-05-07 07:11:49 +00:00
|
|
|
* @modules java.base/jdk.internal.org.objectweb.asm
|
2015-10-28 03:06:02 +00:00
|
|
|
* java.management
|
|
|
|
* java.instrument
|
2008-03-24 23:04:07 +00:00
|
|
|
* @run shell/timeout=240 MakeJAR2.sh NativeMethodPrefixAgent NativeMethodPrefixApp 'Can-Retransform-Classes: true' 'Can-Set-Native-Method-Prefix: true'
|
2016-03-03 11:34:55 +00:00
|
|
|
* @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:-CheckIntrinsics -javaagent:NativeMethodPrefixAgent.jar NativeMethodPrefixApp
|
2007-12-01 00:00:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import java.lang.instrument.*;
|
|
|
|
import java.security.ProtectionDomain;
|
|
|
|
import java.io.*;
|
|
|
|
|
2014-08-25 16:17:24 +00:00
|
|
|
import asmlib.*;
|
2007-12-01 00:00:00 +00:00
|
|
|
|
|
|
|
class NativeMethodPrefixAgent {
|
|
|
|
|
|
|
|
static ClassFileTransformer t0, t1, t2;
|
|
|
|
static Instrumentation inst;
|
2022-10-07 17:13:30 +00:00
|
|
|
private static Throwable agentError;
|
|
|
|
|
|
|
|
public static void checkErrors() {
|
|
|
|
if (agentError != null) {
|
|
|
|
throw new RuntimeException("Agent error", agentError);
|
|
|
|
}
|
|
|
|
}
|
2007-12-01 00:00:00 +00:00
|
|
|
|
|
|
|
static class Tr implements ClassFileTransformer {
|
|
|
|
final String trname;
|
|
|
|
final int transformId;
|
|
|
|
|
|
|
|
Tr(int transformId) {
|
|
|
|
this.trname = "tr" + transformId;
|
|
|
|
this.transformId = transformId;
|
|
|
|
}
|
|
|
|
|
|
|
|
public byte[]
|
|
|
|
transform(
|
|
|
|
ClassLoader loader,
|
|
|
|
String className,
|
|
|
|
Class<?> classBeingRedefined,
|
|
|
|
ProtectionDomain protectionDomain,
|
|
|
|
byte[] classfileBuffer) {
|
|
|
|
boolean redef = classBeingRedefined != null;
|
|
|
|
System.out.println(trname + ": " +
|
|
|
|
(redef? "Retransforming " : "Loading ") + className);
|
|
|
|
if (className != null) {
|
|
|
|
try {
|
2014-08-25 16:17:24 +00:00
|
|
|
byte[] newcf = Instrumentor.instrFor(classfileBuffer)
|
|
|
|
.addNativeMethodTrackingInjection(
|
|
|
|
"wrapped_" + trname + "_",
|
|
|
|
(h)->{
|
|
|
|
h.push(h.getName());
|
|
|
|
h.push(transformId);
|
|
|
|
h.invokeStatic("bootreporter/StringIdCallbackReporter", "tracker", "(Ljava/lang/String;I)V", false);
|
|
|
|
})
|
|
|
|
.apply();
|
|
|
|
/*** debugging ...
|
|
|
|
if (newcf != null) {
|
|
|
|
String fname = trname + (redef?"_redef" : "") + "/" + className;
|
|
|
|
System.err.println("dumping to: " + fname);
|
|
|
|
write_buffer(fname + "_before.class", classfileBuffer);
|
|
|
|
write_buffer(fname + "_instr.class", newcf);
|
|
|
|
}
|
|
|
|
***/
|
|
|
|
|
2007-12-01 00:00:00 +00:00
|
|
|
return redef? null : newcf;
|
|
|
|
} catch (Throwable ex) {
|
2022-10-07 17:13:30 +00:00
|
|
|
if (agentError == null) {
|
|
|
|
agentError = ex;
|
|
|
|
}
|
2007-12-01 00:00:00 +00:00
|
|
|
System.err.println("ERROR: Injection failure: " + ex);
|
|
|
|
ex.printStackTrace();
|
2022-10-07 17:13:30 +00:00
|
|
|
return null;
|
2007-12-01 00:00:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// for debugging
|
|
|
|
static void write_buffer(String fname, byte[]buffer) {
|
|
|
|
try {
|
2014-08-25 16:17:24 +00:00
|
|
|
File f = new File(fname);
|
|
|
|
if (!f.getParentFile().exists()) {
|
|
|
|
f.getParentFile().mkdirs();
|
|
|
|
}
|
|
|
|
try (FileOutputStream outStream = new FileOutputStream(f)) {
|
|
|
|
outStream.write(buffer, 0, buffer.length);
|
|
|
|
}
|
|
|
|
} catch (IOException ex) {
|
2007-12-01 00:00:00 +00:00
|
|
|
System.err.println("EXCEPTION in write_buffer: " + ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void
|
|
|
|
premain (String agentArgs, Instrumentation instArg)
|
|
|
|
throws IOException, IllegalClassFormatException,
|
|
|
|
ClassNotFoundException, UnmodifiableClassException {
|
|
|
|
inst = instArg;
|
|
|
|
System.out.println("Premain");
|
|
|
|
|
|
|
|
t1 = new Tr(1);
|
|
|
|
t2 = new Tr(2);
|
|
|
|
t0 = new Tr(0);
|
|
|
|
inst.addTransformer(t1, true);
|
|
|
|
inst.addTransformer(t2, false);
|
|
|
|
inst.addTransformer(t0, true);
|
|
|
|
instArg.setNativeMethodPrefix(t0, "wrapped_tr0_");
|
|
|
|
instArg.setNativeMethodPrefix(t1, "wrapped_tr1_");
|
|
|
|
instArg.setNativeMethodPrefix(t2, "wrapped_tr2_");
|
|
|
|
|
|
|
|
// warm up: cause load of transformer classes before used during class load
|
|
|
|
instArg.retransformClasses(Runtime.class);
|
|
|
|
}
|
|
|
|
}
|