From d3042cc4015566e78e4900c735bc3fc3a6a316ea Mon Sep 17 00:00:00 2001 From: Kim Barrett Date: Mon, 25 May 2020 16:21:25 -0400 Subject: [PATCH] 8245594: Remove volatile-qualified member functions and parameters from oop class Remove volatile qualifications in oop and derived classes; use Atomic for access. Reviewed-by: dholmes, coleenp --- .../gc/shenandoah/shenandoahVerifier.hpp | 12 +---- .../share/gc/z/zOopClosures.inline.hpp | 4 +- src/hotspot/share/oops/oopsHierarchy.hpp | 45 ++++++------------- src/hotspot/share/runtime/thread.cpp | 9 +++- src/hotspot/share/runtime/thread.hpp | 4 +- src/hotspot/share/services/memoryManager.cpp | 10 +++-- src/hotspot/share/services/memoryManager.hpp | 4 +- src/hotspot/share/services/memoryPool.cpp | 35 ++++++++------- src/hotspot/share/services/memoryPool.hpp | 4 +- 9 files changed, 56 insertions(+), 71 deletions(-) diff --git a/src/hotspot/share/gc/shenandoah/shenandoahVerifier.hpp b/src/hotspot/share/gc/shenandoah/shenandoahVerifier.hpp index ff80988b7ab..bb5d8ff9708 100644 --- a/src/hotspot/share/gc/shenandoah/shenandoahVerifier.hpp +++ b/src/hotspot/share/gc/shenandoah/shenandoahVerifier.hpp @@ -41,17 +41,7 @@ class ShenandoahVerifierTask { public: ShenandoahVerifierTask(oop o = NULL, int idx = 0): _obj(o) { } ShenandoahVerifierTask(oop o, size_t idx): _obj(o) { } - ShenandoahVerifierTask(const ShenandoahVerifierTask& t): _obj(t._obj) { } - - ShenandoahVerifierTask& operator =(const ShenandoahVerifierTask& t) { - _obj = t._obj; - return *this; - } - volatile ShenandoahVerifierTask& - operator =(const volatile ShenandoahVerifierTask& t) volatile { - (void)const_cast(_obj = t._obj); - return *this; - } + // Trivially copyable. inline oop obj() const { return _obj; } diff --git a/src/hotspot/share/gc/z/zOopClosures.inline.hpp b/src/hotspot/share/gc/z/zOopClosures.inline.hpp index 28c74d34e54..da8f22ff9d8 100644 --- a/src/hotspot/share/gc/z/zOopClosures.inline.hpp +++ b/src/hotspot/share/gc/z/zOopClosures.inline.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2020, 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 @@ -87,7 +87,7 @@ inline void ZPhantomKeepAliveOopClosure::do_oop(narrowOop* p) { inline void ZPhantomCleanOopClosure::do_oop(oop* p) { // Read the oop once, to make sure the liveness check // and the later clearing uses the same value. - const oop obj = *(volatile oop*)p; + const oop obj = Atomic::load(p); if (ZBarrier::is_alive_barrier_on_phantom_oop(obj)) { ZBarrier::keep_alive_barrier_on_phantom_oop_field(p); } else { diff --git a/src/hotspot/share/oops/oopsHierarchy.hpp b/src/hotspot/share/oops/oopsHierarchy.hpp index a359d13946c..ac5b91bc94b 100644 --- a/src/hotspot/share/oops/oopsHierarchy.hpp +++ b/src/hotspot/share/oops/oopsHierarchy.hpp @@ -80,36 +80,28 @@ class oop { void register_oop(); void unregister_oop(); -public: - void set_obj(const void* p) { - _o = (oopDesc*)p; + void register_if_checking() { if (CheckUnhandledOops) register_oop(); } - oop() { set_obj(NULL); } - oop(const oop& o) { set_obj(o.obj()); } - oop(const volatile oop& o) { set_obj(o.obj()); } - oop(const void* p) { set_obj(p); } - ~oop() { +public: + oop() : _o(NULL) { register_if_checking(); } + oop(const oop& o) : _o(o._o) { register_if_checking(); } + oop(const void* p) : _o((oopDesc*)p) { register_if_checking(); } + ~oop() { if (CheckUnhandledOops) unregister_oop(); } - oopDesc* obj() const volatile { return _o; } + oopDesc* obj() const { return _o; } + oopDesc* operator->() const { return _o; } + operator oopDesc* () const { return _o; } - // General access - oopDesc* operator->() const { return obj(); } - bool operator==(const oop o) const { return obj() == o.obj(); } - bool operator==(void *p) const { return obj() == p; } - bool operator!=(const volatile oop o) const { return obj() != o.obj(); } - bool operator!=(void *p) const { return obj() != p; } + bool operator==(const oop& o) const { return _o == o._o; } + bool operator==(void *p) const { return _o == p; } + bool operator!=(const oop& o) const { return _o != o._o; } + bool operator!=(void *p) const { return _o != p; } - // Assignment - oop& operator=(const oop& o) { _o = o.obj(); return *this; } - volatile oop& operator=(const oop& o) volatile { _o = o.obj(); return *this; } - volatile oop& operator=(const volatile oop& o) volatile { _o = o.obj(); return *this; } - - // Explict user conversions - operator oopDesc* () const volatile { return obj(); } + oop& operator=(const oop& o) { _o = o._o; return *this; } }; template<> @@ -128,7 +120,6 @@ struct PrimitiveConversions::Translate : public TrueType { type##Oop() : oop() {} \ type##Oop(const type##Oop& o) : oop(o) {} \ type##Oop(const oop& o) : oop(o) {} \ - type##Oop(const volatile oop& o) : oop(o) {} \ type##Oop(const void* p) : oop(p) {} \ operator type##OopDesc* () const { return (type##OopDesc*)obj(); } \ type##OopDesc* operator->() const { \ @@ -138,14 +129,6 @@ struct PrimitiveConversions::Translate : public TrueType { oop::operator=(o); \ return *this; \ } \ - volatile type##Oop& operator=(const type##Oop& o) volatile { \ - (void)const_cast(oop::operator=(o)); \ - return *this; \ - } \ - volatile type##Oop& operator=(const volatile type##Oop& o) volatile {\ - (void)const_cast(oop::operator=(o)); \ - return *this; \ - } \ }; \ \ template<> \ diff --git a/src/hotspot/share/runtime/thread.cpp b/src/hotspot/share/runtime/thread.cpp index 4c3f916733c..d87602960e6 100644 --- a/src/hotspot/share/runtime/thread.cpp +++ b/src/hotspot/share/runtime/thread.cpp @@ -1665,7 +1665,7 @@ void JavaThread::initialize() { } #endif // INCLUDE_JVMCI _reserved_stack_activation = NULL; // stack base not known yet - (void)const_cast(_exception_oop = oop(NULL)); + set_exception_oop(oop()); _exception_pc = 0; _exception_handler_pc = 0; _is_method_handle_return = 0; @@ -2252,6 +2252,13 @@ bool JavaThread::is_lock_owned(address adr) const { return false; } +oop JavaThread::exception_oop() const { + return Atomic::load(&_exception_oop); +} + +void JavaThread::set_exception_oop(oop o) { + Atomic::store(&_exception_oop, o); +} void JavaThread::add_monitor_chunk(MonitorChunk* chunk) { chunk->set_next(monitor_chunks()); diff --git a/src/hotspot/share/runtime/thread.hpp b/src/hotspot/share/runtime/thread.hpp index dd01a61c57c..b33021d63f8 100644 --- a/src/hotspot/share/runtime/thread.hpp +++ b/src/hotspot/share/runtime/thread.hpp @@ -1574,12 +1574,12 @@ class JavaThread: public Thread { #endif // INCLUDE_JVMCI // Exception handling for compiled methods - oop exception_oop() const { return _exception_oop; } + oop exception_oop() const; address exception_pc() const { return _exception_pc; } address exception_handler_pc() const { return _exception_handler_pc; } bool is_method_handle_return() const { return _is_method_handle_return == 1; } - void set_exception_oop(oop o) { (void)const_cast(_exception_oop = o); } + void set_exception_oop(oop o); void set_exception_pc(address a) { _exception_pc = a; } void set_exception_handler_pc(address a) { _exception_handler_pc = a; } void set_is_method_handle_return(bool value) { _is_method_handle_return = value ? 1 : 0; } diff --git a/src/hotspot/share/services/memoryManager.cpp b/src/hotspot/share/services/memoryManager.cpp index 61041f63509..681ec280c6a 100644 --- a/src/hotspot/share/services/memoryManager.cpp +++ b/src/hotspot/share/services/memoryManager.cpp @@ -38,10 +38,8 @@ #include "services/gcNotifier.hpp" #include "utilities/dtrace.hpp" -MemoryManager::MemoryManager(const char* name) : _name(name) { - _num_pools = 0; - (void)const_cast(_memory_mgr_obj = instanceOop(NULL)); -} +MemoryManager::MemoryManager(const char* name) : + _num_pools(0), _name(name), _memory_mgr_obj() {} int MemoryManager::add_pool(MemoryPool* pool) { int index = _num_pools; @@ -54,6 +52,10 @@ int MemoryManager::add_pool(MemoryPool* pool) { return index; } +bool MemoryManager::is_manager(instanceHandle mh) const { + return mh() == Atomic::load(&_memory_mgr_obj); +} + MemoryManager* MemoryManager::get_code_cache_memory_manager() { return new MemoryManager("CodeCacheManager"); } diff --git a/src/hotspot/share/services/memoryManager.hpp b/src/hotspot/share/services/memoryManager.hpp index 5cc4e20d736..dbd3312c960 100644 --- a/src/hotspot/share/services/memoryManager.hpp +++ b/src/hotspot/share/services/memoryManager.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2020, 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 @@ -70,7 +70,7 @@ public: int add_pool(MemoryPool* pool); - bool is_manager(instanceHandle mh) { return mh() == _memory_mgr_obj; } + bool is_manager(instanceHandle mh) const; virtual instanceOop get_memory_manager_instance(TRAPS); virtual bool is_gc_memory_manager() { return false; } diff --git a/src/hotspot/share/services/memoryPool.cpp b/src/hotspot/share/services/memoryPool.cpp index 1a4dc69a228..32f3261016d 100644 --- a/src/hotspot/share/services/memoryPool.cpp +++ b/src/hotspot/share/services/memoryPool.cpp @@ -42,24 +42,27 @@ MemoryPool::MemoryPool(const char* name, size_t init_size, size_t max_size, bool support_usage_threshold, - bool support_gc_threshold) { - _name = name; - _initial_size = init_size; - _max_size = max_size; - (void)const_cast(_memory_pool_obj = instanceOop(NULL)); - _available_for_allocation = true; - _num_managers = 0; - _type = type; - - // initialize the max and init size of collection usage - _after_gc_usage = MemoryUsage(_initial_size, 0, 0, _max_size); - - _usage_sensor = NULL; - _gc_usage_sensor = NULL; + bool support_gc_threshold) : + _name(name), + _type(type), + _initial_size(init_size), + _max_size(max_size), + _available_for_allocation(true), + _managers(), + _num_managers(0), + _peak_usage(), + _after_gc_usage(init_size, 0, 0, max_size), // usage threshold supports both high and low threshold - _usage_threshold = new ThresholdSupport(support_usage_threshold, support_usage_threshold); + _usage_threshold(new ThresholdSupport(support_usage_threshold, support_usage_threshold)), // gc usage threshold supports only high threshold - _gc_usage_threshold = new ThresholdSupport(support_gc_threshold, support_gc_threshold); + _gc_usage_threshold(new ThresholdSupport(support_gc_threshold, support_gc_threshold)), + _usage_sensor(), + _gc_usage_sensor(), + _memory_pool_obj() +{} + +bool MemoryPool::is_pool(instanceHandle pool) const { + return pool() == Atomic::load(&_memory_pool_obj); } void MemoryPool::add_manager(MemoryManager* mgr) { diff --git a/src/hotspot/share/services/memoryPool.hpp b/src/hotspot/share/services/memoryPool.hpp index 9b46badf44b..70cf6c73e17 100644 --- a/src/hotspot/share/services/memoryPool.hpp +++ b/src/hotspot/share/services/memoryPool.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2020, 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 @@ -95,7 +95,7 @@ class MemoryPool : public CHeapObj { // max size could be changed virtual size_t max_size() const { return _max_size; } - bool is_pool(instanceHandle pool) { return pool() == _memory_pool_obj; } + bool is_pool(instanceHandle pool) const; bool available_for_allocation() { return _available_for_allocation; } bool set_available_for_allocation(bool value) {