From e4c96dea46446f736c05616475fa556e517a1a42 Mon Sep 17 00:00:00 2001 From: Robbin Ehn Date: Thu, 19 Dec 2019 16:47:59 +0100 Subject: [PATCH] 8235912: JvmtiBreakpoint remove oops_do and metadata_do Reviewed-by: coleenp, sspitsyn --- .../share/classfile/metadataOnStackMark.cpp | 1 - src/hotspot/share/prims/jvmtiExport.cpp | 1 - src/hotspot/share/prims/jvmtiImpl.cpp | 126 ++++-------------- src/hotspot/share/prims/jvmtiImpl.hpp | 42 +----- 4 files changed, 29 insertions(+), 141 deletions(-) diff --git a/src/hotspot/share/classfile/metadataOnStackMark.cpp b/src/hotspot/share/classfile/metadataOnStackMark.cpp index 4a7f0c8c89e..51c53ce8f95 100644 --- a/src/hotspot/share/classfile/metadataOnStackMark.cpp +++ b/src/hotspot/share/classfile/metadataOnStackMark.cpp @@ -70,7 +70,6 @@ MetadataOnStackMark::MetadataOnStackMark(bool walk_all_metadata, bool redefiniti CodeCache::old_nmethods_do(&md_on_stack); } CompileBroker::mark_on_stack(); - JvmtiCurrentBreakpoints::metadata_do(Metadata::mark_on_stack); ThreadService::metadata_do(Metadata::mark_on_stack); #if INCLUDE_JVMCI JVMCI::metadata_do(Metadata::mark_on_stack); diff --git a/src/hotspot/share/prims/jvmtiExport.cpp b/src/hotspot/share/prims/jvmtiExport.cpp index cc75c20c371..182d3b061ab 100644 --- a/src/hotspot/share/prims/jvmtiExport.cpp +++ b/src/hotspot/share/prims/jvmtiExport.cpp @@ -2600,7 +2600,6 @@ void JvmtiExport::clear_detected_exception(JavaThread* thread) { } void JvmtiExport::oops_do(OopClosure* f) { - JvmtiCurrentBreakpoints::oops_do(f); JvmtiObjectAllocEventCollector::oops_do_for_all_threads(f); } diff --git a/src/hotspot/share/prims/jvmtiImpl.cpp b/src/hotspot/share/prims/jvmtiImpl.cpp index 29934809841..f9670182761 100644 --- a/src/hotspot/share/prims/jvmtiImpl.cpp +++ b/src/hotspot/share/prims/jvmtiImpl.cpp @@ -26,6 +26,8 @@ #include "classfile/symbolTable.hpp" #include "classfile/systemDictionary.hpp" #include "code/nmethod.hpp" +#include "gc/shared/oopStorage.hpp" +#include "gc/shared/oopStorageSet.hpp" #include "interpreter/interpreter.hpp" #include "interpreter/oopMapCache.hpp" #include "jvmtifiles/jvmtiEnv.hpp" @@ -171,24 +173,6 @@ void GrowableCache::append(GrowableElement* e) { recache(); } -// insert a copy of the element using lessthan() -void GrowableCache::insert(GrowableElement* e) { - GrowableElement *new_e = e->clone(); - _elements->append(new_e); - - int n = length()-2; - for (int i=n; i>=0; i--) { - GrowableElement *e1 = _elements->at(i); - GrowableElement *e2 = _elements->at(i+1); - if (e2->lessThan(e1)) { - _elements->at_put(i+1, e1); - _elements->at_put(i, e2); - } - } - - recache(); -} - // remove the element at index void GrowableCache::remove (int index) { GrowableElement *e = _elements->at(index); @@ -209,58 +193,40 @@ void GrowableCache::clear() { recache(); } -void GrowableCache::oops_do(OopClosure* f) { - int len = _elements->length(); - for (int i=0; iat(i); - e->oops_do(f); - } -} - -void GrowableCache::metadata_do(void f(Metadata*)) { - int len = _elements->length(); - for (int i=0; iat(i); - e->metadata_do(f); - } -} - // // class JvmtiBreakpoint // -JvmtiBreakpoint::JvmtiBreakpoint() { - _method = NULL; - _bci = 0; - _class_holder = NULL; +JvmtiBreakpoint::JvmtiBreakpoint(Method* m_method, jlocation location) + : _method(m_method), _bci((int)location), _class_holder(NULL) { + assert(_method != NULL, "No method for breakpoint."); + assert(_bci >= 0, "Negative bci for breakpoint."); + oop class_holder_oop = _method->method_holder()->klass_holder(); + _class_holder = OopStorageSet::vm_global()->allocate(); + if (_class_holder == NULL) { + vm_exit_out_of_memory(sizeof(oop), OOM_MALLOC_ERROR, + "Cannot create breakpoint oop handle"); + } + NativeAccess<>::oop_store(_class_holder, class_holder_oop); } -JvmtiBreakpoint::JvmtiBreakpoint(Method* m_method, jlocation location) { - _method = m_method; - _class_holder = _method->method_holder()->klass_holder(); -#ifdef CHECK_UNHANDLED_OOPS - // _class_holder can't be wrapped in a Handle, because JvmtiBreakpoints are - // sometimes allocated on the heap. - // - // The code handling JvmtiBreakpoints allocated on the stack can't be - // interrupted by a GC until _class_holder is reachable by the GC via the - // oops_do method. - Thread::current()->allow_unhandled_oop(&_class_holder); -#endif // CHECK_UNHANDLED_OOPS - assert(_method != NULL, "_method != NULL"); - _bci = (int) location; - assert(_bci >= 0, "_bci >= 0"); +JvmtiBreakpoint::~JvmtiBreakpoint() { + if (_class_holder != NULL) { + NativeAccess<>::oop_store(_class_holder, (oop)NULL); + OopStorageSet::vm_global()->release(_class_holder); + } } void JvmtiBreakpoint::copy(JvmtiBreakpoint& bp) { _method = bp._method; _bci = bp._bci; - _class_holder = bp._class_holder; -} - -bool JvmtiBreakpoint::lessThan(JvmtiBreakpoint& bp) { - Unimplemented(); - return false; + _class_holder = OopStorageSet::vm_global()->allocate(); + if (_class_holder == NULL) { + vm_exit_out_of_memory(sizeof(oop), OOM_MALLOC_ERROR, + "Cannot create breakpoint oop handle"); + } + oop resolved_ch = NativeAccess<>::oop_load(bp._class_holder); + NativeAccess<>::oop_store(_class_holder, resolved_ch); } bool JvmtiBreakpoint::equals(JvmtiBreakpoint& bp) { @@ -268,12 +234,6 @@ bool JvmtiBreakpoint::equals(JvmtiBreakpoint& bp) { && _bci == bp._bci; } -bool JvmtiBreakpoint::is_valid() { - // class loader can be NULL - return _method != NULL && - _bci >= 0; -} - address JvmtiBreakpoint::getBcp() const { return _method->bcp_from(_bci); } @@ -347,21 +307,6 @@ void VM_ChangeBreakpoints::doit() { } } -void VM_ChangeBreakpoints::oops_do(OopClosure* f) { - // The JvmtiBreakpoints in _breakpoints will be visited via - // JvmtiExport::oops_do. - if (_bp != NULL) { - _bp->oops_do(f); - } -} - -void VM_ChangeBreakpoints::metadata_do(void f(Metadata*)) { - // Walk metadata in breakpoints to keep from being deallocated with RedefineClasses - if (_bp != NULL) { - _bp->metadata_do(f); - } -} - // // class JvmtiBreakpoints // @@ -374,14 +319,6 @@ JvmtiBreakpoints::JvmtiBreakpoints(void listener_fun(void *,address *)) { JvmtiBreakpoints:: ~JvmtiBreakpoints() {} -void JvmtiBreakpoints::oops_do(OopClosure* f) { - _bps.oops_do(f); -} - -void JvmtiBreakpoints::metadata_do(void f(Metadata*)) { - _bps.metadata_do(f); -} - void JvmtiBreakpoints::print() { #ifndef PRODUCT LogTarget(Trace, jvmti) log; @@ -490,19 +427,6 @@ void JvmtiCurrentBreakpoints::listener_fun(void *this_obj, address *cache) { set_breakpoint_list(cache); } - -void JvmtiCurrentBreakpoints::oops_do(OopClosure* f) { - if (_jvmti_breakpoints != NULL) { - _jvmti_breakpoints->oops_do(f); - } -} - -void JvmtiCurrentBreakpoints::metadata_do(void f(Metadata*)) { - if (_jvmti_breakpoints != NULL) { - _jvmti_breakpoints->metadata_do(f); - } -} - /////////////////////////////////////////////////////////////// // // class VM_GetOrSetLocal diff --git a/src/hotspot/share/prims/jvmtiImpl.hpp b/src/hotspot/share/prims/jvmtiImpl.hpp index ad42682d108..d23b707acef 100644 --- a/src/hotspot/share/prims/jvmtiImpl.hpp +++ b/src/hotspot/share/prims/jvmtiImpl.hpp @@ -68,10 +68,7 @@ public: virtual ~GrowableElement() {} virtual address getCacheValue() =0; virtual bool equals(GrowableElement* e) =0; - virtual bool lessThan(GrowableElement *e)=0; virtual GrowableElement *clone() =0; - virtual void oops_do(OopClosure* f) =0; - virtual void metadata_do(void f(Metadata*)) =0; }; class GrowableCache { @@ -110,16 +107,10 @@ public: int find(GrowableElement* e); // append a copy of the element to the end of the collection, notify listener void append(GrowableElement* e); - // insert a copy of the element using lessthan(), notify listener - void insert(GrowableElement* e); // remove the element at index, notify listener void remove (int index); // clear out all elements and release all heap space, notify listener void clear(); - // apply f to every element and update the cache - void oops_do(OopClosure* f); - // walk metadata to preserve for RedefineClasses - void metadata_do(void f(Metadata*)); }; @@ -141,7 +132,7 @@ public: ~JvmtiBreakpointCache() {} void initialize(void *this_obj, void listener_fun(void *, address*) ) { - _cache.initialize(this_obj,listener_fun); + _cache.initialize(this_obj, listener_fun); } int length() { return _cache.length(); } @@ -149,9 +140,6 @@ public: int find(JvmtiBreakpoint& e) { return _cache.find((GrowableElement *) &e); } void append(JvmtiBreakpoint& e) { _cache.append((GrowableElement *) &e); } void remove (int index) { _cache.remove(index); } - void clear() { _cache.clear(); } - void oops_do(OopClosure* f) { _cache.oops_do(f); } - void metadata_do(void f(Metadata*)) { _cache.metadata_do(f); } }; @@ -171,16 +159,14 @@ class JvmtiBreakpoint : public GrowableElement { private: Method* _method; int _bci; - Bytecodes::Code _orig_bytecode; - oop _class_holder; // keeps _method memory from being deallocated + oop* _class_holder; // keeps _method memory from being deallocated public: - JvmtiBreakpoint(); + JvmtiBreakpoint() : _method(NULL), _bci(0), _class_holder(NULL) {} JvmtiBreakpoint(Method* m_method, jlocation location); + virtual ~JvmtiBreakpoint(); bool equals(JvmtiBreakpoint& bp); - bool lessThan(JvmtiBreakpoint &bp); void copy(JvmtiBreakpoint& bp); - bool is_valid(); address getBcp() const; void each_method_version_do(method_action meth_act); void set(); @@ -191,17 +177,7 @@ public: // GrowableElement implementation address getCacheValue() { return getBcp(); } - bool lessThan(GrowableElement* e) { Unimplemented(); return false; } bool equals(GrowableElement* e) { return equals((JvmtiBreakpoint&) *e); } - void oops_do(OopClosure* f) { - // Mark the method loader as live so the Method* class loader doesn't get - // unloaded and Method* memory reclaimed. - f->do_oop(&_class_holder); - } - void metadata_do(void f(Metadata*)) { - // walk metadata to preserve for RedefineClasses - f(_method); - } GrowableElement *clone() { JvmtiBreakpoint *bp = new JvmtiBreakpoint(); @@ -240,15 +216,11 @@ private: void set_at_safepoint(JvmtiBreakpoint& bp); void clear_at_safepoint(JvmtiBreakpoint& bp); - static void do_element(GrowableElement *e); - public: JvmtiBreakpoints(void listener_fun(void *, address *)); ~JvmtiBreakpoints(); int length(); - void oops_do(OopClosure* f); - void metadata_do(void f(Metadata*)); void print(); int set(JvmtiBreakpoint& bp); @@ -281,7 +253,6 @@ private: // It exists only to make is_breakpoint fast. static address *_breakpoint_list; static inline void set_breakpoint_list(address *breakpoint_list) { _breakpoint_list = breakpoint_list; } - static inline address *get_breakpoint_list() { return _breakpoint_list; } // Listener for the GrowableCache in _jvmti_breakpoints, updates _breakpoint_list. static void listener_fun(void *this_obj, address *cache); @@ -292,9 +263,6 @@ public: // lazily create _jvmti_breakpoints and _breakpoint_list static JvmtiBreakpoints& get_jvmti_breakpoints(); - - static void oops_do(OopClosure* f); - static void metadata_do(void f(Metadata*)) NOT_JVMTI_RETURN; }; /////////////////////////////////////////////////////////////// @@ -326,8 +294,6 @@ public: VMOp_Type type() const { return VMOp_ChangeBreakpoints; } void doit(); - void oops_do(OopClosure* f); - void metadata_do(void f(Metadata*)); };