8232613: Move Object.registerNatives into HotSpot

Reviewed-by: dholmes, adinn, coleenp, lfoltan, mchung
This commit is contained in:
Claes Redestad 2019-10-24 09:57:29 +02:00
parent 68e5c40f6f
commit 574263a884
13 changed files with 312 additions and 136 deletions

View File

@ -87,6 +87,21 @@ InjectedField JavaClasses::_injected_fields[] = {
ALL_INJECTED_FIELDS(DECLARE_INJECTED_FIELD) ALL_INJECTED_FIELDS(DECLARE_INJECTED_FIELD)
}; };
// Register native methods of Object
void java_lang_Object::register_natives(TRAPS) {
InstanceKlass* obj = SystemDictionary::Object_klass();
Method::register_native(obj, vmSymbols::hashCode_name(),
vmSymbols::void_int_signature(), (address) &JVM_IHashCode, CHECK);
Method::register_native(obj, vmSymbols::wait_name(),
vmSymbols::long_void_signature(), (address) &JVM_MonitorWait, CHECK);
Method::register_native(obj, vmSymbols::notify_name(),
vmSymbols::void_method_signature(), (address) &JVM_MonitorNotify, CHECK);
Method::register_native(obj, vmSymbols::notifyAll_name(),
vmSymbols::void_method_signature(), (address) &JVM_MonitorNotifyAll, CHECK);
Method::register_native(obj, vmSymbols::clone_name(),
vmSymbols::void_object_signature(), (address) &JVM_Clone, THREAD);
}
int JavaClasses::compute_injected_offset(InjectedFieldID id) { int JavaClasses::compute_injected_offset(InjectedFieldID id) {
return _injected_fields[id].compute_offset(); return _injected_fields[id].compute_offset();
} }

View File

@ -88,6 +88,13 @@
BASIC_JAVA_CLASSES_DO_PART1(f) \ BASIC_JAVA_CLASSES_DO_PART1(f) \
BASIC_JAVA_CLASSES_DO_PART2(f) BASIC_JAVA_CLASSES_DO_PART2(f)
// Interface to java.lang.Object objects
class java_lang_Object : AllStatic {
public:
static void register_natives(TRAPS);
};
// Interface to java.lang.String objects // Interface to java.lang.String objects
class java_lang_String : AllStatic { class java_lang_String : AllStatic {

View File

@ -1972,6 +1972,10 @@ void SystemDictionary::resolve_well_known_classes(TRAPS) {
resolve_wk_klasses_through(WK_KLASS_ENUM_NAME(Class_klass), scan, CHECK); resolve_wk_klasses_through(WK_KLASS_ENUM_NAME(Class_klass), scan, CHECK);
} }
assert(WK_KLASS(Object_klass) != NULL, "well-known classes should now be initialized");
java_lang_Object::register_natives(CHECK);
// Calculate offsets for String and Class classes since they are loaded and // Calculate offsets for String and Class classes since they are loaded and
// can be used after this point. // can be used after this point.
java_lang_String::compute_offsets(); java_lang_String::compute_offsets();

View File

@ -219,7 +219,6 @@ class GCTimer;
\ \
/*end*/ /*end*/
class SystemDictionary : AllStatic { class SystemDictionary : AllStatic {
friend class BootstrapInfo; friend class BootstrapInfo;
friend class VMStructs; friend class VMStructs;
@ -383,7 +382,6 @@ public:
int limit = (int)end_id + 1; int limit = (int)end_id + 1;
resolve_wk_klasses_until((WKID) limit, start_id, THREAD); resolve_wk_klasses_until((WKID) limit, start_id, THREAD);
} }
public: public:
#define WK_KLASS_DECLARE(name, symbol) \ #define WK_KLASS_DECLARE(name, symbol) \
static InstanceKlass* name() { return check_klass(_well_known_klasses[WK_KLASS_ENUM_NAME(name)]); } \ static InstanceKlass* name() { return check_klass(_well_known_klasses[WK_KLASS_ENUM_NAME(name)]); } \
@ -628,21 +626,6 @@ protected:
// Basic find on classes in the midst of being loaded // Basic find on classes in the midst of being loaded
static Symbol* find_placeholder(Symbol* name, ClassLoaderData* loader_data); static Symbol* find_placeholder(Symbol* name, ClassLoaderData* loader_data);
// Add a placeholder for a class being loaded
static void add_placeholder(int index,
Symbol* class_name,
ClassLoaderData* loader_data);
static void remove_placeholder(int index,
Symbol* class_name,
ClassLoaderData* loader_data);
// Performs cleanups after resolve_super_or_fail. This typically needs
// to be called on failure.
// Won't throw, but can block.
static void resolution_cleanups(Symbol* class_name,
ClassLoaderData* loader_data,
TRAPS);
// Resolve well-known classes so they can be used like SystemDictionary::String_klass() // Resolve well-known classes so they can be used like SystemDictionary::String_klass()
static void resolve_well_known_classes(TRAPS); static void resolve_well_known_classes(TRAPS);

View File

@ -25,6 +25,7 @@
#include "precompiled.hpp" #include "precompiled.hpp"
#include "classfile/classLoaderDataGraph.hpp" #include "classfile/classLoaderDataGraph.hpp"
#include "classfile/metadataOnStackMark.hpp" #include "classfile/metadataOnStackMark.hpp"
#include "classfile/symbolTable.hpp"
#include "classfile/systemDictionary.hpp" #include "classfile/systemDictionary.hpp"
#include "code/codeCache.hpp" #include "code/codeCache.hpp"
#include "code/debugInfoRec.hpp" #include "code/debugInfoRec.hpp"
@ -378,7 +379,83 @@ void Method::set_itable_index(int index) {
assert(valid_itable_index(), ""); assert(valid_itable_index(), "");
} }
// The RegisterNatives call being attempted tried to register with a method that
// is not native. Ask JVM TI what prefixes have been specified. Then check
// to see if the native method is now wrapped with the prefixes. See the
// SetNativeMethodPrefix(es) functions in the JVM TI Spec for details.
static Method* find_prefixed_native(Klass* k, Symbol* name, Symbol* signature, TRAPS) {
#if INCLUDE_JVMTI
ResourceMark rm(THREAD);
Method* method;
int name_len = name->utf8_length();
char* name_str = name->as_utf8();
int prefix_count;
char** prefixes = JvmtiExport::get_all_native_method_prefixes(&prefix_count);
for (int i = 0; i < prefix_count; i++) {
char* prefix = prefixes[i];
int prefix_len = (int)strlen(prefix);
// try adding this prefix to the method name and see if it matches another method name
int trial_len = name_len + prefix_len;
char* trial_name_str = NEW_RESOURCE_ARRAY(char, trial_len + 1);
strcpy(trial_name_str, prefix);
strcat(trial_name_str, name_str);
TempNewSymbol trial_name = SymbolTable::probe(trial_name_str, trial_len);
if (trial_name == NULL) {
continue; // no such symbol, so this prefix wasn't used, try the next prefix
}
method = k->lookup_method(trial_name, signature);
if (method == NULL) {
continue; // signature doesn't match, try the next prefix
}
if (method->is_native()) {
method->set_is_prefixed_native();
return method; // wahoo, we found a prefixed version of the method, return it
}
// found as non-native, so prefix is good, add it, probably just need more prefixes
name_len = trial_len;
name_str = trial_name_str;
}
#endif // INCLUDE_JVMTI
return NULL; // not found
}
bool Method::register_native(Klass* k, Symbol* name, Symbol* signature, address entry, TRAPS) {
Method* method = k->lookup_method(name, signature);
if (method == NULL) {
ResourceMark rm(THREAD);
stringStream st;
st.print("Method '");
print_external_name(&st, k, name, signature);
st.print("' name or signature does not match");
THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), false);
}
if (!method->is_native()) {
// trying to register to a non-native method, see if a JVM TI agent has added prefix(es)
method = find_prefixed_native(k, name, signature, THREAD);
if (method == NULL) {
ResourceMark rm(THREAD);
stringStream st;
st.print("Method '");
print_external_name(&st, k, name, signature);
st.print("' is not declared as native");
THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), false);
}
}
if (entry != NULL) {
method->set_native_function(entry, native_bind_event_is_interesting);
} else {
method->clear_native_function();
}
if (PrintJNIResolving) {
ResourceMark rm(THREAD);
tty->print_cr("[Registering JNI native method %s.%s]",
method->method_holder()->external_name(),
method->name()->as_C_string());
}
return true;
}
bool Method::was_executed_more_than(int n) { bool Method::was_executed_more_than(int n) {
// Invocation counter is reset when the Method* is compiled. // Invocation counter is reset when the Method* is compiled.

View File

@ -346,6 +346,12 @@ class Method : public Metadata {
// InterpreterRuntime::exception_handler_for_exception. // InterpreterRuntime::exception_handler_for_exception.
static int fast_exception_handler_bci_for(const methodHandle& mh, Klass* ex_klass, int throw_bci, TRAPS); static int fast_exception_handler_bci_for(const methodHandle& mh, Klass* ex_klass, int throw_bci, TRAPS);
static bool register_native(Klass* k,
Symbol* name,
Symbol* signature,
address entry,
TRAPS);
// method data access // method data access
MethodData* method_data() const { MethodData* method_data() const {
return _method_data; return _method_data;

View File

@ -2909,89 +2909,6 @@ DEFINE_SETSCALARARRAYREGION(T_DOUBLE, jdouble, Double, double
HOTSPOT_JNI_SETDOUBLEARRAYREGION_RETURN()) HOTSPOT_JNI_SETDOUBLEARRAYREGION_RETURN())
//
// Interception of natives
//
// The RegisterNatives call being attempted tried to register with a method that
// is not native. Ask JVM TI what prefixes have been specified. Then check
// to see if the native method is now wrapped with the prefixes. See the
// SetNativeMethodPrefix(es) functions in the JVM TI Spec for details.
static Method* find_prefixed_native(Klass* k, Symbol* name, Symbol* signature, TRAPS) {
#if INCLUDE_JVMTI
ResourceMark rm(THREAD);
Method* method;
int name_len = name->utf8_length();
char* name_str = name->as_utf8();
int prefix_count;
char** prefixes = JvmtiExport::get_all_native_method_prefixes(&prefix_count);
for (int i = 0; i < prefix_count; i++) {
char* prefix = prefixes[i];
int prefix_len = (int)strlen(prefix);
// try adding this prefix to the method name and see if it matches another method name
int trial_len = name_len + prefix_len;
char* trial_name_str = NEW_RESOURCE_ARRAY(char, trial_len + 1);
strcpy(trial_name_str, prefix);
strcat(trial_name_str, name_str);
TempNewSymbol trial_name = SymbolTable::probe(trial_name_str, trial_len);
if (trial_name == NULL) {
continue; // no such symbol, so this prefix wasn't used, try the next prefix
}
method = k->lookup_method(trial_name, signature);
if (method == NULL) {
continue; // signature doesn't match, try the next prefix
}
if (method->is_native()) {
method->set_is_prefixed_native();
return method; // wahoo, we found a prefixed version of the method, return it
}
// found as non-native, so prefix is good, add it, probably just need more prefixes
name_len = trial_len;
name_str = trial_name_str;
}
#endif // INCLUDE_JVMTI
return NULL; // not found
}
static bool register_native(Klass* k, Symbol* name, Symbol* signature, address entry, TRAPS) {
Method* method = k->lookup_method(name, signature);
if (method == NULL) {
ResourceMark rm;
stringStream st;
st.print("Method '");
Method::print_external_name(&st, k, name, signature);
st.print("' name or signature does not match");
THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), false);
}
if (!method->is_native()) {
// trying to register to a non-native method, see if a JVM TI agent has added prefix(es)
method = find_prefixed_native(k, name, signature, THREAD);
if (method == NULL) {
ResourceMark rm;
stringStream st;
st.print("Method '");
Method::print_external_name(&st, k, name, signature);
st.print("' is not declared as native");
THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), false);
}
}
if (entry != NULL) {
method->set_native_function(entry,
Method::native_bind_event_is_interesting);
} else {
method->clear_native_function();
}
if (PrintJNIResolving) {
ResourceMark rm(THREAD);
tty->print_cr("[Registering JNI native method %s.%s]",
method->method_holder()->external_name(),
method->name()->as_C_string());
}
return true;
}
DT_RETURN_MARK_DECL(RegisterNatives, jint DT_RETURN_MARK_DECL(RegisterNatives, jint
, HOTSPOT_JNI_REGISTERNATIVES_RETURN(_ret_ref)); , HOTSPOT_JNI_REGISTERNATIVES_RETURN(_ret_ref));
@ -3024,8 +2941,8 @@ JNI_ENTRY(jint, jni_RegisterNatives(JNIEnv *env, jclass clazz,
THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), -1); THROW_MSG_(vmSymbols::java_lang_NoSuchMethodError(), st.as_string(), -1);
} }
bool res = register_native(k, name, signature, bool res = Method::register_native(k, name, signature,
(address) methods[index].fnPtr, THREAD); (address) methods[index].fnPtr, THREAD);
if (!res) { if (!res) {
ret = -1; ret = -1;
break; break;

View File

@ -38,11 +38,6 @@ import jdk.internal.HotSpotIntrinsicCandidate;
*/ */
public class Object { public class Object {
private static native void registerNatives();
static {
registerNatives();
}
/** /**
* Constructs a new object. * Constructs a new object.
*/ */

View File

@ -39,21 +39,6 @@
#include "java_lang_Object.h" #include "java_lang_Object.h"
static JNINativeMethod methods[] = {
{"hashCode", "()I", (void *)&JVM_IHashCode},
{"wait", "(J)V", (void *)&JVM_MonitorWait},
{"notify", "()V", (void *)&JVM_MonitorNotify},
{"notifyAll", "()V", (void *)&JVM_MonitorNotifyAll},
{"clone", "()Ljava/lang/Object;", (void *)&JVM_Clone},
};
JNIEXPORT void JNICALL
Java_java_lang_Object_registerNatives(JNIEnv *env, jclass cls)
{
(*env)->RegisterNatives(env, cls,
methods, sizeof(methods)/sizeof(methods[0]));
}
JNIEXPORT jclass JNICALL JNIEXPORT jclass JNICALL
Java_java_lang_Object_getClass(JNIEnv *env, jobject this) Java_java_lang_Object_getClass(JNIEnv *env, jobject this)
{ {

View File

@ -34,11 +34,6 @@ public class Object {
@HotSpotIntrinsicCandidate @HotSpotIntrinsicCandidate
public Object() {} public Object() {}
private static native void registerNatives();
static {
registerNatives();
}
@HotSpotIntrinsicCandidate @HotSpotIntrinsicCandidate
public final native Class<?> getClass(); public final native Class<?> getClass();

View File

@ -23,15 +23,27 @@
/* /*
* @test * @test
* @bug 8024804 * @bug 8024804 8028741 8232613
* @bug 8028741
* @summary interface method resolution should skip finding j.l.Object's registerNatives() and succeed in selecting class B's registerNatives() * @summary interface method resolution should skip finding j.l.Object's registerNatives() and succeed in selecting class B's registerNatives()
* @run main RegisterNatives * @run main RegisterNatives
*/ */
public class RegisterNatives { public class RegisterNatives {
interface I { void registerNatives(); } interface I {
void registerNatives();
}
interface J extends I {} interface J extends I {}
static class B implements J { public void registerNatives() { System.out.println("B"); } }
interface K {
default public void registerNatives() { System.out.println("K"); }
}
static class B implements J {
public void registerNatives() { System.out.println("B"); }
}
static class C implements K {}
public static void main(String... args) { public static void main(String... args) {
System.out.println("Regression test for JDK-8024804, crash when InterfaceMethodref resolves to Object.registerNatives\n"); System.out.println("Regression test for JDK-8024804, crash when InterfaceMethodref resolves to Object.registerNatives\n");
J val = new B(); J val = new B();
@ -42,6 +54,14 @@ public class RegisterNatives {
e.printStackTrace(); e.printStackTrace();
throw e; throw e;
} }
C cval = new C();
try {
cval.registerNatives();
} catch (IllegalAccessError e) {
System.out.println("TEST FAILS - a default method named registerNatives should no longer be masked by removed Object.registerNatives\n");
e.printStackTrace();
throw e;
}
System.out.println("TEST PASSES - no IAE resulted\n"); System.out.println("TEST PASSES - no IAE resulted\n");
} }
} }

View File

@ -0,0 +1,177 @@
/*
* 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();
}
}
}

View File

@ -37,11 +37,6 @@ import nsk.jvmti.scenarios.bcinstr.BI04.bi04t002a;
*/ */
public class Object { public class Object {
private static native void registerNatives();
static {
registerNatives();
}
/** /**
* Returns the runtime class of an object. That <tt>Class</tt> * Returns the runtime class of an object. That <tt>Class</tt>
* object is the object that is locked by <tt>static synchronized</tt> * object is the object that is locked by <tt>static synchronized</tt>