提交 435eb992 编写于 作者: S sspitsyn

8046246: the constantPoolCacheOopDesc::adjust_method_entries() used in...

8046246: the constantPoolCacheOopDesc::adjust_method_entries() used in RedefineClasses does not scale
Summary: optimize the adjust_method_entries functions by using the orig_method_idnum() function
Reviewed-by: coleenp, dcubed
上级 eeb79d0f
/* /*
* Copyright (c) 2012, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2012, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -1091,6 +1091,7 @@ static void merge_in_new_methods(InstanceKlass* klass, ...@@ -1091,6 +1091,7 @@ static void merge_in_new_methods(InstanceKlass* klass,
} }
// update idnum for new location // update idnum for new location
merged_methods->at(i)->set_method_idnum(i); merged_methods->at(i)->set_method_idnum(i);
merged_methods->at(i)->set_orig_method_idnum(i);
} }
// Verify correct order // Verify correct order
......
/* /*
* Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -222,6 +222,7 @@ private: ...@@ -222,6 +222,7 @@ private:
u2 _max_stack; // Maximum number of entries on the expression stack u2 _max_stack; // Maximum number of entries on the expression stack
u2 _max_locals; // Number of local variables used by this method u2 _max_locals; // Number of local variables used by this method
u2 _size_of_parameters; // size of the parameter block (receiver + arguments) in words u2 _size_of_parameters; // size of the parameter block (receiver + arguments) in words
u2 _orig_method_idnum; // Original unique identification number for the method
// Constructor // Constructor
ConstMethod(int byte_code_size, ConstMethod(int byte_code_size,
...@@ -475,6 +476,9 @@ public: ...@@ -475,6 +476,9 @@ public:
u2 method_idnum() const { return _method_idnum; } u2 method_idnum() const { return _method_idnum; }
void set_method_idnum(u2 idnum) { _method_idnum = idnum; } void set_method_idnum(u2 idnum) { _method_idnum = idnum; }
u2 orig_method_idnum() const { return _orig_method_idnum; }
void set_orig_method_idnum(u2 idnum) { _orig_method_idnum = idnum; }
// max stack // max stack
int max_stack() const { return _max_stack; } int max_stack() const { return _max_stack; }
void set_max_stack(int size) { _max_stack = size; } void set_max_stack(int size) { _max_stack = size; }
......
...@@ -449,7 +449,6 @@ bool ConstantPoolCacheEntry::adjust_method_entry(Method* old_method, ...@@ -449,7 +449,6 @@ bool ConstantPoolCacheEntry::adjust_method_entry(Method* old_method,
new_method->name()->as_C_string(), new_method->name()->as_C_string(),
new_method->signature()->as_C_string())); new_method->signature()->as_C_string()));
} }
return true; return true;
} }
...@@ -477,7 +476,6 @@ bool ConstantPoolCacheEntry::adjust_method_entry(Method* old_method, ...@@ -477,7 +476,6 @@ bool ConstantPoolCacheEntry::adjust_method_entry(Method* old_method,
new_method->name()->as_C_string(), new_method->name()->as_C_string(),
new_method->signature()->as_C_string())); new_method->signature()->as_C_string()));
} }
return true; return true;
} }
...@@ -504,36 +502,33 @@ bool ConstantPoolCacheEntry::check_no_old_or_obsolete_entries() { ...@@ -504,36 +502,33 @@ bool ConstantPoolCacheEntry::check_no_old_or_obsolete_entries() {
(!f1_as_method()->is_old() && !f1_as_method()->is_obsolete()))); (!f1_as_method()->is_old() && !f1_as_method()->is_obsolete())));
} }
bool ConstantPoolCacheEntry::is_interesting_method_entry(Klass* k) { Method* ConstantPoolCacheEntry::get_interesting_method_entry(Klass* k) {
if (!is_method_entry()) { if (!is_method_entry()) {
// not a method entry so not interesting by default // not a method entry so not interesting by default
return false; return NULL;
} }
Method* m = NULL; Method* m = NULL;
if (is_vfinal()) { if (is_vfinal()) {
// virtual and final so _f2 contains method ptr instead of vtable index // virtual and final so _f2 contains method ptr instead of vtable index
m = f2_as_vfinal_method(); m = f2_as_vfinal_method();
} else if (is_f1_null()) { } else if (is_f1_null()) {
// NULL _f1 means this is a virtual entry so also not interesting // NULL _f1 means this is a virtual entry so also not interesting
return false; return NULL;
} else { } else {
if (!(_f1->is_method())) { if (!(_f1->is_method())) {
// _f1 can also contain a Klass* for an interface // _f1 can also contain a Klass* for an interface
return false; return NULL;
} }
m = f1_as_method(); m = f1_as_method();
} }
assert(m != NULL && m->is_method(), "sanity check"); assert(m != NULL && m->is_method(), "sanity check");
if (m == NULL || !m->is_method() || (k != NULL && m->method_holder() != k)) { if (m == NULL || !m->is_method() || (k != NULL && m->method_holder() != k)) {
// robustness for above sanity checks or method is not in // robustness for above sanity checks or method is not in
// the interesting class // the interesting class
return false; return NULL;
} }
// the method is in the interesting class so the entry is interesting // the method is in the interesting class so the entry is interesting
return true; return m;
} }
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
...@@ -621,7 +616,7 @@ void ConstantPoolCache::adjust_method_entries(Method** old_methods, Method** new ...@@ -621,7 +616,7 @@ void ConstantPoolCache::adjust_method_entries(Method** old_methods, Method** new
Klass* old_holder = old_methods[0]->method_holder(); Klass* old_holder = old_methods[0]->method_holder();
for (int i = 0; i < length(); i++) { for (int i = 0; i < length(); i++) {
if (!entry_at(i)->is_interesting_method_entry(old_holder)) { if (entry_at(i)->get_interesting_method_entry(old_holder) == NULL) {
// skip uninteresting methods // skip uninteresting methods
continue; continue;
} }
...@@ -645,10 +640,33 @@ void ConstantPoolCache::adjust_method_entries(Method** old_methods, Method** new ...@@ -645,10 +640,33 @@ void ConstantPoolCache::adjust_method_entries(Method** old_methods, Method** new
} }
} }
// If any entry of this ConstantPoolCache points to any of
// old_methods, replace it with the corresponding new_method.
void ConstantPoolCache::adjust_method_entries(InstanceKlass* holder, bool * trace_name_printed) {
for (int i = 0; i < length(); i++) {
ConstantPoolCacheEntry* entry = entry_at(i);
Method* old_method = entry->get_interesting_method_entry(holder);
if (old_method == NULL || !old_method->is_old()) {
continue; // skip uninteresting entries
}
if (old_method->is_deleted()) {
// clean up entries with deleted methods
entry->initialize_entry(entry->constant_pool_index());
continue;
}
Method* new_method = holder->method_with_idnum(old_method->orig_method_idnum());
assert(new_method != NULL, "method_with_idnum() should not be NULL");
assert(old_method != new_method, "sanity check");
entry_at(i)->adjust_method_entry(old_method, new_method, trace_name_printed);
}
}
// the constant pool cache should never contain old or obsolete methods // the constant pool cache should never contain old or obsolete methods
bool ConstantPoolCache::check_no_old_or_obsolete_entries() { bool ConstantPoolCache::check_no_old_or_obsolete_entries() {
for (int i = 1; i < length(); i++) { for (int i = 1; i < length(); i++) {
if (entry_at(i)->is_interesting_method_entry(NULL) && if (entry_at(i)->get_interesting_method_entry(NULL) != NULL &&
!entry_at(i)->check_no_old_or_obsolete_entries()) { !entry_at(i)->check_no_old_or_obsolete_entries()) {
return false; return false;
} }
...@@ -658,7 +676,7 @@ bool ConstantPoolCache::check_no_old_or_obsolete_entries() { ...@@ -658,7 +676,7 @@ bool ConstantPoolCache::check_no_old_or_obsolete_entries() {
void ConstantPoolCache::dump_cache() { void ConstantPoolCache::dump_cache() {
for (int i = 1; i < length(); i++) { for (int i = 1; i < length(); i++) {
if (entry_at(i)->is_interesting_method_entry(NULL)) { if (entry_at(i)->get_interesting_method_entry(NULL) != NULL) {
entry_at(i)->print(tty, i); entry_at(i)->print(tty, i);
} }
} }
......
/* /*
* Copyright (c) 1998, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1998, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -378,9 +378,9 @@ class ConstantPoolCacheEntry VALUE_OBJ_CLASS_SPEC { ...@@ -378,9 +378,9 @@ class ConstantPoolCacheEntry VALUE_OBJ_CLASS_SPEC {
// printed the klass name so that other routines in the adjust_* // printed the klass name so that other routines in the adjust_*
// group don't print the klass name. // group don't print the klass name.
bool adjust_method_entry(Method* old_method, Method* new_method, bool adjust_method_entry(Method* old_method, Method* new_method,
bool * trace_name_printed); bool* trace_name_printed);
bool check_no_old_or_obsolete_entries(); bool check_no_old_or_obsolete_entries();
bool is_interesting_method_entry(Klass* k); Method* get_interesting_method_entry(Klass* k);
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
// Debugging & Printing // Debugging & Printing
...@@ -477,7 +477,8 @@ class ConstantPoolCache: public MetaspaceObj { ...@@ -477,7 +477,8 @@ class ConstantPoolCache: public MetaspaceObj {
// printed the klass name so that other routines in the adjust_* // printed the klass name so that other routines in the adjust_*
// group don't print the klass name. // group don't print the klass name.
void adjust_method_entries(Method** old_methods, Method** new_methods, void adjust_method_entries(Method** old_methods, Method** new_methods,
int methods_length, bool * trace_name_printed); int methods_length, bool* trace_name_printed);
void adjust_method_entries(InstanceKlass* holder, bool* trace_name_printed);
bool check_no_old_or_obsolete_entries(); bool check_no_old_or_obsolete_entries();
void dump_cache(); void dump_cache();
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
......
/* /*
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -2798,16 +2798,21 @@ Method* InstanceKlass::method_at_itable(Klass* holder, int index, TRAPS) { ...@@ -2798,16 +2798,21 @@ Method* InstanceKlass::method_at_itable(Klass* holder, int index, TRAPS) {
// not yet in the vtable due to concurrent subclass define and superinterface // not yet in the vtable due to concurrent subclass define and superinterface
// redefinition // redefinition
// Note: those in the vtable, should have been updated via adjust_method_entries // Note: those in the vtable, should have been updated via adjust_method_entries
void InstanceKlass::adjust_default_methods(Method** old_methods, Method** new_methods, void InstanceKlass::adjust_default_methods(InstanceKlass* holder, bool* trace_name_printed) {
int methods_length, bool* trace_name_printed) {
// search the default_methods for uses of either obsolete or EMCP methods // search the default_methods for uses of either obsolete or EMCP methods
if (default_methods() != NULL) { if (default_methods() != NULL) {
for (int j = 0; j < methods_length; j++) {
Method* old_method = old_methods[j];
Method* new_method = new_methods[j];
for (int index = 0; index < default_methods()->length(); index ++) { for (int index = 0; index < default_methods()->length(); index ++) {
if (default_methods()->at(index) == old_method) { Method* old_method = default_methods()->at(index);
if (old_method == NULL || old_method->method_holder() != holder || !old_method->is_old()) {
continue; // skip uninteresting entries
}
assert(!old_method->is_deleted(), "default methods may not be deleted");
Method* new_method = holder->method_with_idnum(old_method->orig_method_idnum());
assert(new_method != NULL, "method_with_idnum() should not be NULL");
assert(old_method != new_method, "sanity check");
default_methods()->at_put(index, new_method); default_methods()->at_put(index, new_method);
if (RC_TRACE_IN_RANGE(0x00100000, 0x00400000)) { if (RC_TRACE_IN_RANGE(0x00100000, 0x00400000)) {
if (!(*trace_name_printed)) { if (!(*trace_name_printed)) {
...@@ -2823,8 +2828,6 @@ void InstanceKlass::adjust_default_methods(Method** old_methods, Method** new_me ...@@ -2823,8 +2828,6 @@ void InstanceKlass::adjust_default_methods(Method** old_methods, Method** new_me
} }
} }
} }
}
}
} }
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
......
/* /*
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -959,8 +959,7 @@ class InstanceKlass: public Klass { ...@@ -959,8 +959,7 @@ class InstanceKlass: public Klass {
Method* method_at_itable(Klass* holder, int index, TRAPS); Method* method_at_itable(Klass* holder, int index, TRAPS);
#if INCLUDE_JVMTI #if INCLUDE_JVMTI
void adjust_default_methods(Method** old_methods, Method** new_methods, void adjust_default_methods(InstanceKlass* holder, bool* trace_name_printed);
int methods_length, bool* trace_name_printed);
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
// Garbage collection // Garbage collection
......
/* /*
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -860,20 +860,22 @@ bool klassVtable::adjust_default_method(int vtable_index, Method* old_method, Me ...@@ -860,20 +860,22 @@ bool klassVtable::adjust_default_method(int vtable_index, Method* old_method, Me
} }
return updated; return updated;
} }
void klassVtable::adjust_method_entries(Method** old_methods, Method** new_methods,
int methods_length, bool * trace_name_printed) { // search the vtable for uses of either obsolete or EMCP methods
// search the vtable for uses of either obsolete or EMCP methods void klassVtable::adjust_method_entries(InstanceKlass* holder, bool * trace_name_printed) {
for (int j = 0; j < methods_length; j++) { int prn_enabled = 0;
Method* old_method = old_methods[j];
Method* new_method = new_methods[j];
// In the vast majority of cases we could get the vtable index
// by using: old_method->vtable_index()
// However, there are rare cases, eg. sun.awt.X11.XDecoratedPeer.getX()
// in sun.awt.X11.XFramePeer where methods occur more than once in the
// vtable, so, alas, we must do an exhaustive search.
for (int index = 0; index < length(); index++) { for (int index = 0; index < length(); index++) {
if (unchecked_method_at(index) == old_method) { Method* old_method = unchecked_method_at(index);
if (old_method == NULL || old_method->method_holder() != holder || !old_method->is_old()) {
continue; // skip uninteresting entries
}
assert(!old_method->is_deleted(), "vtable methods may not be deleted");
Method* new_method = holder->method_with_idnum(old_method->orig_method_idnum());
assert(new_method != NULL, "method_with_idnum() should not be NULL");
assert(old_method != new_method, "sanity check");
put_method_at(new_method, index); put_method_at(new_method, index);
// For default methods, need to update the _default_methods array // For default methods, need to update the _default_methods array
// which can only have one method entry for a given signature // which can only have one method entry for a given signature
...@@ -896,9 +898,6 @@ void klassVtable::adjust_method_entries(Method** old_methods, Method** new_metho ...@@ -896,9 +898,6 @@ void klassVtable::adjust_method_entries(Method** old_methods, Method** new_metho
new_method->signature()->as_C_string(), new_method->signature()->as_C_string(),
updated_default ? "true" : "false")); updated_default ? "true" : "false"));
} }
// cannot 'break' here; see for-loop comment above.
}
}
} }
} }
...@@ -1190,20 +1189,22 @@ void klassItable::initialize_with_method(Method* m) { ...@@ -1190,20 +1189,22 @@ void klassItable::initialize_with_method(Method* m) {
} }
#if INCLUDE_JVMTI #if INCLUDE_JVMTI
void klassItable::adjust_method_entries(Method** old_methods, Method** new_methods, // search the itable for uses of either obsolete or EMCP methods
int methods_length, bool * trace_name_printed) { void klassItable::adjust_method_entries(InstanceKlass* holder, bool * trace_name_printed) {
// search the itable for uses of either obsolete or EMCP methods
for (int j = 0; j < methods_length; j++) {
Method* old_method = old_methods[j];
Method* new_method = new_methods[j];
itableMethodEntry* ime = method_entry(0); itableMethodEntry* ime = method_entry(0);
for (int i = 0; i < _size_method_table; i++, ime++) {
Method* old_method = ime->method();
if (old_method == NULL || old_method->method_holder() != holder || !old_method->is_old()) {
continue; // skip uninteresting entries
}
assert(!old_method->is_deleted(), "itable methods may not be deleted");
Method* new_method = holder->method_with_idnum(old_method->orig_method_idnum());
assert(new_method != NULL, "method_with_idnum() should not be NULL");
assert(old_method != new_method, "sanity check");
// The itable can describe more than one interface and the same
// method signature can be specified by more than one interface.
// This means we have to do an exhaustive search to find all the
// old_method references.
for (int i = 0; i < _size_method_table; i++) {
if (ime->method() == old_method) {
ime->initialize(new_method); ime->initialize(new_method);
if (RC_TRACE_IN_RANGE(0x00100000, 0x00400000)) { if (RC_TRACE_IN_RANGE(0x00100000, 0x00400000)) {
...@@ -1218,10 +1219,6 @@ void klassItable::adjust_method_entries(Method** old_methods, Method** new_metho ...@@ -1218,10 +1219,6 @@ void klassItable::adjust_method_entries(Method** old_methods, Method** new_metho
new_method->name()->as_C_string(), new_method->name()->as_C_string(),
new_method->signature()->as_C_string())); new_method->signature()->as_C_string()));
} }
// cannot 'break' here; see for-loop comment above.
}
ime++;
}
} }
} }
......
/* /*
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -98,8 +98,7 @@ class klassVtable : public ResourceObj { ...@@ -98,8 +98,7 @@ class klassVtable : public ResourceObj {
// printed the klass name so that other routines in the adjust_* // printed the klass name so that other routines in the adjust_*
// group don't print the klass name. // group don't print the klass name.
bool adjust_default_method(int vtable_index, Method* old_method, Method* new_method); bool adjust_default_method(int vtable_index, Method* old_method, Method* new_method);
void adjust_method_entries(Method** old_methods, Method** new_methods, void adjust_method_entries(InstanceKlass* holder, bool * trace_name_printed);
int methods_length, bool * trace_name_printed);
bool check_no_old_or_obsolete_entries(); bool check_no_old_or_obsolete_entries();
void dump_vtable(); void dump_vtable();
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
...@@ -288,8 +287,7 @@ class klassItable : public ResourceObj { ...@@ -288,8 +287,7 @@ class klassItable : public ResourceObj {
// trace_name_printed is set to true if the current call has // trace_name_printed is set to true if the current call has
// printed the klass name so that other routines in the adjust_* // printed the klass name so that other routines in the adjust_*
// group don't print the klass name. // group don't print the klass name.
void adjust_method_entries(Method** old_methods, Method** new_methods, void adjust_method_entries(InstanceKlass* holder, bool * trace_name_printed);
int methods_length, bool * trace_name_printed);
bool check_no_old_or_obsolete_entries(); bool check_no_old_or_obsolete_entries();
void dump_itable(); void dump_itable();
#endif // INCLUDE_JVMTI #endif // INCLUDE_JVMTI
......
/* /*
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
...@@ -1419,6 +1419,7 @@ void Method::sort_methods(Array<Method*>* methods, bool idempotent, bool set_idn ...@@ -1419,6 +1419,7 @@ void Method::sort_methods(Array<Method*>* methods, bool idempotent, bool set_idn
for (int i = 0; i < length; i++) { for (int i = 0; i < length; i++) {
Method* m = methods->at(i); Method* m = methods->at(i);
m->set_method_idnum(i); m->set_method_idnum(i);
m->set_orig_method_idnum(i);
} }
} }
} }
......
...@@ -265,6 +265,9 @@ class Method : public Metadata { ...@@ -265,6 +265,9 @@ class Method : public Metadata {
u2 method_idnum() const { return constMethod()->method_idnum(); } u2 method_idnum() const { return constMethod()->method_idnum(); }
void set_method_idnum(u2 idnum) { constMethod()->set_method_idnum(idnum); } void set_method_idnum(u2 idnum) { constMethod()->set_method_idnum(idnum); }
u2 orig_method_idnum() const { return constMethod()->orig_method_idnum(); }
void set_orig_method_idnum(u2 idnum) { constMethod()->set_orig_method_idnum(idnum); }
// code size // code size
int code_size() const { return constMethod()->code_size(); } int code_size() const { return constMethod()->code_size(); }
......
...@@ -777,9 +777,13 @@ jvmtiError VM_RedefineClasses::compare_and_normalize_class_versions( ...@@ -777,9 +777,13 @@ jvmtiError VM_RedefineClasses::compare_and_normalize_class_versions(
Method* idnum_owner = scratch_class->method_with_idnum(old_num); Method* idnum_owner = scratch_class->method_with_idnum(old_num);
if (idnum_owner != NULL) { if (idnum_owner != NULL) {
// There is already a method assigned this idnum -- switch them // There is already a method assigned this idnum -- switch them
// Take current and original idnum from the new_method
idnum_owner->set_method_idnum(new_num); idnum_owner->set_method_idnum(new_num);
idnum_owner->set_orig_method_idnum(k_new_method->orig_method_idnum());
} }
// Take current and original idnum from the old_method
k_new_method->set_method_idnum(old_num); k_new_method->set_method_idnum(old_num);
k_new_method->set_orig_method_idnum(k_old_method->orig_method_idnum());
if (thread->has_pending_exception()) { if (thread->has_pending_exception()) {
return JVMTI_ERROR_OUT_OF_MEMORY; return JVMTI_ERROR_OUT_OF_MEMORY;
} }
...@@ -812,9 +816,12 @@ jvmtiError VM_RedefineClasses::compare_and_normalize_class_versions( ...@@ -812,9 +816,12 @@ jvmtiError VM_RedefineClasses::compare_and_normalize_class_versions(
Method* idnum_owner = scratch_class->method_with_idnum(num); Method* idnum_owner = scratch_class->method_with_idnum(num);
if (idnum_owner != NULL) { if (idnum_owner != NULL) {
// There is already a method assigned this idnum -- switch them // There is already a method assigned this idnum -- switch them
// Take current and original idnum from the new_method
idnum_owner->set_method_idnum(new_num); idnum_owner->set_method_idnum(new_num);
idnum_owner->set_orig_method_idnum(k_new_method->orig_method_idnum());
} }
k_new_method->set_method_idnum(num); k_new_method->set_method_idnum(num);
k_new_method->set_orig_method_idnum(num);
if (thread->has_pending_exception()) { if (thread->has_pending_exception()) {
return JVMTI_ERROR_OUT_OF_MEMORY; return JVMTI_ERROR_OUT_OF_MEMORY;
} }
...@@ -3322,6 +3329,7 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { ...@@ -3322,6 +3329,7 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) {
// This is a very busy routine. We don't want too much tracing // This is a very busy routine. We don't want too much tracing
// printed out. // printed out.
bool trace_name_printed = false; bool trace_name_printed = false;
InstanceKlass *the_class = InstanceKlass::cast(_the_class_oop);
// Very noisy: only enable this call if you are trying to determine // Very noisy: only enable this call if you are trying to determine
// that a specific class gets found by this routine. // that a specific class gets found by this routine.
...@@ -3333,10 +3341,8 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { ...@@ -3333,10 +3341,8 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) {
// If the class being redefined is java.lang.Object, we need to fix all // If the class being redefined is java.lang.Object, we need to fix all
// array class vtables also // array class vtables also
if (k->oop_is_array() && _the_class_oop == SystemDictionary::Object_klass()) { if (k->oop_is_array() && _the_class_oop == SystemDictionary::Object_klass()) {
k->vtable()->adjust_method_entries(_matching_old_methods, k->vtable()->adjust_method_entries(the_class, &trace_name_printed);
_matching_new_methods,
_matching_methods_length,
&trace_name_printed);
} else if (k->oop_is_instance()) { } else if (k->oop_is_instance()) {
HandleMark hm(_thread); HandleMark hm(_thread);
InstanceKlass *ik = InstanceKlass::cast(k); InstanceKlass *ik = InstanceKlass::cast(k);
...@@ -3376,14 +3382,9 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { ...@@ -3376,14 +3382,9 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) {
|| ik->is_subtype_of(_the_class_oop))) { || ik->is_subtype_of(_the_class_oop))) {
// ik->vtable() creates a wrapper object; rm cleans it up // ik->vtable() creates a wrapper object; rm cleans it up
ResourceMark rm(_thread); ResourceMark rm(_thread);
ik->vtable()->adjust_method_entries(_matching_old_methods,
_matching_new_methods, ik->vtable()->adjust_method_entries(the_class, &trace_name_printed);
_matching_methods_length, ik->adjust_default_methods(the_class, &trace_name_printed);
&trace_name_printed);
ik->adjust_default_methods(_matching_old_methods,
_matching_new_methods,
_matching_methods_length,
&trace_name_printed);
} }
// If the current class has an itable and we are either redefining an // If the current class has an itable and we are either redefining an
...@@ -3396,10 +3397,8 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { ...@@ -3396,10 +3397,8 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) {
|| ik->is_subclass_of(_the_class_oop))) { || ik->is_subclass_of(_the_class_oop))) {
// ik->itable() creates a wrapper object; rm cleans it up // ik->itable() creates a wrapper object; rm cleans it up
ResourceMark rm(_thread); ResourceMark rm(_thread);
ik->itable()->adjust_method_entries(_matching_old_methods,
_matching_new_methods, ik->itable()->adjust_method_entries(the_class, &trace_name_printed);
_matching_methods_length,
&trace_name_printed);
} }
// The constant pools in other classes (other_cp) can refer to // The constant pools in other classes (other_cp) can refer to
...@@ -3423,10 +3422,7 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { ...@@ -3423,10 +3422,7 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) {
other_cp = constantPoolHandle(ik->constants()); other_cp = constantPoolHandle(ik->constants());
cp_cache = other_cp->cache(); cp_cache = other_cp->cache();
if (cp_cache != NULL) { if (cp_cache != NULL) {
cp_cache->adjust_method_entries(_matching_old_methods, cp_cache->adjust_method_entries(the_class, &trace_name_printed);
_matching_new_methods,
_matching_methods_length,
&trace_name_printed);
} }
} }
...@@ -3555,6 +3551,7 @@ void VM_RedefineClasses::check_methods_and_mark_as_obsolete( ...@@ -3555,6 +3551,7 @@ void VM_RedefineClasses::check_methods_and_mark_as_obsolete(
// obsolete methods need a unique idnum so they become new entries in // obsolete methods need a unique idnum so they become new entries in
// the jmethodID cache in InstanceKlass // the jmethodID cache in InstanceKlass
assert(old_method->method_idnum() == new_method->method_idnum(), "must match");
u2 num = InstanceKlass::cast(_the_class_oop)->next_method_idnum(); u2 num = InstanceKlass::cast(_the_class_oop)->next_method_idnum();
if (num != ConstMethod::UNSET_IDNUM) { if (num != ConstMethod::UNSET_IDNUM) {
old_method->set_method_idnum(num); old_method->set_method_idnum(num);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册