From 3798a62adb960d44bc1164189a2be81311fffe41 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Erik=20=C3=96sterlund?= Date: Mon, 5 Nov 2018 12:36:23 +0100 Subject: [PATCH] 8212996: Use AS_NO_KEEPALIVE when accessing dead java.lang.invoke.CallSites during nmethod unloading Reviewed-by: coleenp, pliden --- src/hotspot/share/classfile/javaClasses.cpp | 4 ++-- src/hotspot/share/classfile/javaClasses.hpp | 2 +- src/hotspot/share/code/nmethod.cpp | 8 ++++++++ src/hotspot/share/code/nmethod.hpp | 2 +- src/hotspot/share/prims/methodHandles.cpp | 6 +++--- 5 files changed, 15 insertions(+), 7 deletions(-) diff --git a/src/hotspot/share/classfile/javaClasses.cpp b/src/hotspot/share/classfile/javaClasses.cpp index dfa094b3f76..63f16bbb51b 100644 --- a/src/hotspot/share/classfile/javaClasses.cpp +++ b/src/hotspot/share/classfile/javaClasses.cpp @@ -3732,10 +3732,10 @@ void java_lang_invoke_CallSite::serialize_offsets(SerializeClosure* f) { } #endif -oop java_lang_invoke_CallSite::context(oop call_site) { +oop java_lang_invoke_CallSite::context_no_keepalive(oop call_site) { assert(java_lang_invoke_CallSite::is_instance(call_site), ""); - oop dep_oop = call_site->obj_field(_context_offset); + oop dep_oop = call_site->obj_field_access(_context_offset); return dep_oop; } diff --git a/src/hotspot/share/classfile/javaClasses.hpp b/src/hotspot/share/classfile/javaClasses.hpp index bae23364f3a..7c0d8e56218 100644 --- a/src/hotspot/share/classfile/javaClasses.hpp +++ b/src/hotspot/share/classfile/javaClasses.hpp @@ -1186,7 +1186,7 @@ public: static void set_target( oop site, oop target); static void set_target_volatile( oop site, oop target); - static oop context(oop site); + static oop context_no_keepalive(oop site); // Testers static bool is_subclass(Klass* klass) { diff --git a/src/hotspot/share/code/nmethod.cpp b/src/hotspot/share/code/nmethod.cpp index 1d5ef75e107..839ca5b35b8 100644 --- a/src/hotspot/share/code/nmethod.cpp +++ b/src/hotspot/share/code/nmethod.cpp @@ -42,6 +42,7 @@ #include "logging/logStream.hpp" #include "memory/allocation.inline.hpp" #include "memory/resourceArea.hpp" +#include "oops/access.inline.hpp" #include "oops/method.inline.hpp" #include "oops/methodData.hpp" #include "oops/oop.inline.hpp" @@ -1334,6 +1335,13 @@ void nmethod::flush() { CodeCache::free(this); } +oop nmethod::oop_at(int index) const { + if (index == 0) { + return NULL; + } + return NativeAccess::oop_load(oop_addr_at(index)); +} + // // Notify all classes this nmethod is dependent on that it is no // longer dependent. This should only be called in two situations. diff --git a/src/hotspot/share/code/nmethod.hpp b/src/hotspot/share/code/nmethod.hpp index 053b70b4576..67bc38e5b64 100644 --- a/src/hotspot/share/code/nmethod.hpp +++ b/src/hotspot/share/code/nmethod.hpp @@ -365,7 +365,7 @@ class nmethod : public CompiledMethod { // Support for oops in scopes and relocs: // Note: index 0 is reserved for null. - oop oop_at(int index) const { return index == 0 ? (oop) NULL: *oop_addr_at(index); } + oop oop_at(int index) const; oop* oop_addr_at(int index) const { // for GC // relocation indexes are biased by 1 (because 0 is reserved) assert(index > 0 && index <= oops_count(), "must be a valid non-zero index"); diff --git a/src/hotspot/share/prims/methodHandles.cpp b/src/hotspot/share/prims/methodHandles.cpp index 7a04f42da2b..a519d19f181 100644 --- a/src/hotspot/share/prims/methodHandles.cpp +++ b/src/hotspot/share/prims/methodHandles.cpp @@ -1075,7 +1075,7 @@ static bool safe_to_expunge() { void MethodHandles::add_dependent_nmethod(oop call_site, nmethod* nm) { assert_locked_or_safepoint(CodeCache_lock); - oop context = java_lang_invoke_CallSite::context(call_site); + oop context = java_lang_invoke_CallSite::context_no_keepalive(call_site); DependencyContext deps = java_lang_invoke_MethodHandleNatives_CallSiteContext::vmdependencies(context); // Try to purge stale entries on updates. // Since GC doesn't clean dependency contexts rooted at CallSiteContext objects, @@ -1088,7 +1088,7 @@ void MethodHandles::add_dependent_nmethod(oop call_site, nmethod* nm) { void MethodHandles::remove_dependent_nmethod(oop call_site, nmethod* nm) { assert_locked_or_safepoint(CodeCache_lock); - oop context = java_lang_invoke_CallSite::context(call_site); + oop context = java_lang_invoke_CallSite::context_no_keepalive(call_site); DependencyContext deps = java_lang_invoke_MethodHandleNatives_CallSiteContext::vmdependencies(context); deps.remove_dependent_nmethod(nm, /*expunge_stale_entries=*/safe_to_expunge()); } @@ -1102,7 +1102,7 @@ void MethodHandles::flush_dependent_nmethods(Handle call_site, Handle target) { NoSafepointVerifier nsv; MutexLockerEx mu2(CodeCache_lock, Mutex::_no_safepoint_check_flag); - oop context = java_lang_invoke_CallSite::context(call_site()); + oop context = java_lang_invoke_CallSite::context_no_keepalive(call_site()); DependencyContext deps = java_lang_invoke_MethodHandleNatives_CallSiteContext::vmdependencies(context); marked = deps.mark_dependent_nmethods(changes); }