Claes Redestad 574263a884 8232613: Move Object.registerNatives into HotSpot
Reviewed-by: dholmes, adinn, coleenp, lfoltan, mchung
2019-10-24 09:57:29 +02:00

178 lines
6.6 KiB
Java

/*
* Copyright (c) 2019, 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.
*/
/*
* @test
* @bug 8232613
* @summary Ensure Object natives stay registered after redefinition
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.base/jdk.internal.org.objectweb.asm
* java.compiler
* java.instrument
* jdk.jartool/sun.tools.jar
* @run main RedefineObject buildagent
* @run main/othervm -javaagent:redefineagent.jar RedefineObject
*/
import static jdk.test.lib.Asserts.assertTrue;
import java.io.FileNotFoundException;
import java.io.PrintWriter;
import java.lang.RuntimeException;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.lang.instrument.Instrumentation;
import java.lang.instrument.UnmodifiableClassException;
import java.security.ProtectionDomain;
import java.util.Arrays;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.org.objectweb.asm.ClassVisitor;
import jdk.internal.org.objectweb.asm.ClassWriter;
import static jdk.internal.org.objectweb.asm.Opcodes.ASM6;
import static jdk.internal.org.objectweb.asm.Opcodes.V1_8;
public class RedefineObject {
static Instrumentation inst;
public static void premain(String agentArgs, Instrumentation inst) {
RedefineObject.inst = inst;
}
static class Transformer implements ClassFileTransformer {
public byte[] asm(ClassLoader loader, String className,
Class<?> classBeingRedefined,
ProtectionDomain protectionDomain, byte[] classfileBuffer)
throws IllegalClassFormatException {
ClassWriter cw = new ClassWriter(0);
// Force an older ASM to force a bytecode update
ClassVisitor cv = new DummyClassVisitor(ASM6, cw) { };
ClassReader cr = new ClassReader(classfileBuffer);
cr.accept(cv, 0);
byte[] bytes = cw.toByteArray();
return bytes;
}
public class DummyClassVisitor extends ClassVisitor {
public DummyClassVisitor(int api, ClassVisitor cv) {
super(api, cv);
}
public void visit(
final int version,
final int access,
final String name,
final String signature,
final String superName,
final String[] interfaces) {
// Artificially lower to JDK 8 version to force a redefine
cv.visit(V1_8, access, name, signature, superName, interfaces);
}
}
@Override public byte[] transform(ClassLoader loader, String className,
Class<?> classBeingRedefined,
ProtectionDomain protectionDomain, byte[] classfileBuffer)
throws IllegalClassFormatException {
if (className.contains("java/lang/Object")) {
try {
// Here we remove and re-add the dummy fields. This shuffles the constant pool
return asm(loader, className, classBeingRedefined, protectionDomain, classfileBuffer);
} catch (Throwable e) {
// 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.
e.printStackTrace();
System.exit(1);
}
}
return null;
}
}
private static void buildAgent() {
try {
ClassFileInstaller.main("RedefineObject");
} catch (Exception e) {
throw new RuntimeException("Could not write agent classfile", e);
}
try {
PrintWriter pw = new PrintWriter("MANIFEST.MF");
pw.println("Premain-Class: RedefineObject");
pw.println("Agent-Class: RedefineObject");
pw.println("Can-Retransform-Classes: true");
pw.close();
} catch (FileNotFoundException e) {
throw new RuntimeException("Could not write manifest file for the agent", e);
}
sun.tools.jar.Main jarTool = new sun.tools.jar.Main(System.out, System.err, "jar");
if (!jarTool.run(new String[] { "-cmf", "MANIFEST.MF", "redefineagent.jar", "RedefineObject.class" })) {
throw new RuntimeException("Could not write the agent jar file");
}
}
public static void main(String[] args) throws Exception {
int objHash = System.identityHashCode(Object.class);
System.out.println("Object hashCode: " + objHash);
if (args.length == 1 && args[0].equals("buildagent")) {
buildAgent();
return;
}
if (inst == null) {
throw new RuntimeException("Instrumentation object was null");
}
try {
inst.addTransformer(new RedefineObject.Transformer(), true);
inst.retransformClasses(Object.class);
} catch (UnmodifiableClassException e) {
throw new RuntimeException(e);
}
// Exercise native methods on Object after transform
Object b = new Object();
b.hashCode();
C c = new C();
assertTrue(c.hashCode() != c.clone().hashCode() || c != c.clone());
assertTrue(c.clone() instanceof C);
c = (C)c.clone(); // native method on new Object
}
private static class C implements Cloneable {
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
}