8306123: Move InstanceKlass writeable flags
Reviewed-by: iklam, fparain
This commit is contained in:
parent
1a41e12c22
commit
ddb86469e0
@ -223,15 +223,11 @@ class InstanceKlass: public Klass {
|
|||||||
|
|
||||||
volatile u2 _idnum_allocated_count; // JNI/JVMTI: increments with the addition of methods, old ids don't change
|
volatile u2 _idnum_allocated_count; // JNI/JVMTI: increments with the addition of methods, old ids don't change
|
||||||
|
|
||||||
// _is_marked_dependent can be set concurrently, thus cannot be part of the
|
|
||||||
// _misc_flags right now.
|
|
||||||
bool _is_marked_dependent; // used for marking during flushing and deoptimization
|
|
||||||
|
|
||||||
volatile ClassState _init_state; // state of class
|
volatile ClassState _init_state; // state of class
|
||||||
|
|
||||||
u1 _reference_type; // reference type
|
u1 _reference_type; // reference type
|
||||||
|
|
||||||
// State is set while executing, eventually atomically to not disturb other state
|
// State is set either at parse time or while executing, atomically to not disturb other state
|
||||||
InstanceKlassFlags _misc_flags;
|
InstanceKlassFlags _misc_flags;
|
||||||
|
|
||||||
Monitor* _init_monitor; // mutual exclusion to _init_state and _init_thread.
|
Monitor* _init_monitor; // mutual exclusion to _init_state and _init_thread.
|
||||||
@ -531,8 +527,8 @@ public:
|
|||||||
void set_should_verify_class(bool value) { _misc_flags.set_should_verify_class(value); }
|
void set_should_verify_class(bool value) { _misc_flags.set_should_verify_class(value); }
|
||||||
|
|
||||||
// marking
|
// marking
|
||||||
bool is_marked_dependent() const { return _is_marked_dependent; }
|
bool is_marked_dependent() const { return _misc_flags.is_marked_dependent(); }
|
||||||
void set_is_marked_dependent(bool value) { _is_marked_dependent = value; }
|
void set_is_marked_dependent(bool value) { _misc_flags.set_is_marked_dependent(value); }
|
||||||
|
|
||||||
// initialization (virtuals from Klass)
|
// initialization (virtuals from Klass)
|
||||||
bool should_be_initialized() const; // means that initialize should be called
|
bool should_be_initialized() const; // means that initialize should be called
|
||||||
@ -681,16 +677,8 @@ public:
|
|||||||
// Redefinition locking. Class can only be redefined by one thread at a time.
|
// Redefinition locking. Class can only be redefined by one thread at a time.
|
||||||
// The flag is in access_flags so that it can be set and reset using atomic
|
// The flag is in access_flags so that it can be set and reset using atomic
|
||||||
// operations, and not be reset by other misc_flag settings.
|
// operations, and not be reset by other misc_flag settings.
|
||||||
bool is_being_redefined() const {
|
bool is_being_redefined() const { return _misc_flags.is_being_redefined(); }
|
||||||
return _access_flags.is_being_redefined();
|
void set_is_being_redefined(bool value) { _misc_flags.set_is_being_redefined(value); }
|
||||||
}
|
|
||||||
void set_is_being_redefined(bool value) {
|
|
||||||
if (value) {
|
|
||||||
_access_flags.set_is_being_redefined();
|
|
||||||
} else {
|
|
||||||
_access_flags.clear_is_being_redefined();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// RedefineClasses() support for previous versions:
|
// RedefineClasses() support for previous versions:
|
||||||
void add_previous_version(InstanceKlass* ik, int emcp_method_count);
|
void add_previous_version(InstanceKlass* ik, int emcp_method_count);
|
||||||
@ -716,13 +704,8 @@ public:
|
|||||||
bool is_scratch_class() const { return _misc_flags.is_scratch_class(); }
|
bool is_scratch_class() const { return _misc_flags.is_scratch_class(); }
|
||||||
void set_is_scratch_class() { _misc_flags.set_is_scratch_class(true); }
|
void set_is_scratch_class() { _misc_flags.set_is_scratch_class(true); }
|
||||||
|
|
||||||
bool has_resolved_methods() const {
|
bool has_resolved_methods() const { return _misc_flags.has_resolved_methods(); }
|
||||||
return _access_flags.has_resolved_methods();
|
void set_has_resolved_methods() { _misc_flags.set_has_resolved_methods(true); }
|
||||||
}
|
|
||||||
|
|
||||||
void set_has_resolved_methods() {
|
|
||||||
_access_flags.set_has_resolved_methods();
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
#if INCLUDE_JVMTI
|
#if INCLUDE_JVMTI
|
||||||
|
@ -26,9 +26,31 @@
|
|||||||
#include "classfile/classLoader.hpp"
|
#include "classfile/classLoader.hpp"
|
||||||
#include "classfile/classLoaderData.inline.hpp"
|
#include "classfile/classLoaderData.inline.hpp"
|
||||||
#include "oops/instanceKlassFlags.hpp"
|
#include "oops/instanceKlassFlags.hpp"
|
||||||
|
#include "runtime/atomic.hpp"
|
||||||
#include "runtime/safepoint.hpp"
|
#include "runtime/safepoint.hpp"
|
||||||
#include "utilities/macros.hpp"
|
#include "utilities/macros.hpp"
|
||||||
|
|
||||||
|
// This can be removed for the atomic bitset functions, when available.
|
||||||
|
void InstanceKlassFlags::atomic_set_bits(u1 bits) {
|
||||||
|
// Atomically update the status with the bits given
|
||||||
|
u1 old_status, new_status, f;
|
||||||
|
do {
|
||||||
|
old_status = _status;
|
||||||
|
new_status = old_status | bits;
|
||||||
|
f = Atomic::cmpxchg(&_status, old_status, new_status);
|
||||||
|
} while(f != old_status);
|
||||||
|
}
|
||||||
|
|
||||||
|
void InstanceKlassFlags::atomic_clear_bits(u1 bits) {
|
||||||
|
// Atomically update the status with the bits given
|
||||||
|
u1 old_status, new_status, f;
|
||||||
|
do {
|
||||||
|
old_status = _status;
|
||||||
|
new_status = old_status & ~bits;
|
||||||
|
f = Atomic::cmpxchg(&_status, old_status, new_status);
|
||||||
|
} while(f != old_status);
|
||||||
|
}
|
||||||
|
|
||||||
#if INCLUDE_CDS
|
#if INCLUDE_CDS
|
||||||
void InstanceKlassFlags::set_shared_class_loader_type(s2 loader_type) {
|
void InstanceKlassFlags::set_shared_class_loader_type(s2 loader_type) {
|
||||||
switch (loader_type) {
|
switch (loader_type) {
|
||||||
|
@ -27,6 +27,12 @@
|
|||||||
|
|
||||||
class ClassLoaderData;
|
class ClassLoaderData;
|
||||||
|
|
||||||
|
// The InstanceKlassFlags class contains the parse-time and writeable flags associated with
|
||||||
|
// an InstanceKlass, and their associated accessors.
|
||||||
|
// _flags are parse-time and constant in the InstanceKlass after that. _status are set at runtime and
|
||||||
|
// require atomic access.
|
||||||
|
// These flags are JVM internal and not part of the AccessFlags classfile specification.
|
||||||
|
|
||||||
class InstanceKlassFlags {
|
class InstanceKlassFlags {
|
||||||
friend class VMStructs;
|
friend class VMStructs;
|
||||||
friend class JVMCIVMStructs;
|
friend class JVMCIVMStructs;
|
||||||
@ -35,18 +41,15 @@ class InstanceKlassFlags {
|
|||||||
flag(rewritten , 1 << 0) /* methods rewritten. */ \
|
flag(rewritten , 1 << 0) /* methods rewritten. */ \
|
||||||
flag(has_nonstatic_fields , 1 << 1) /* for sizing with UseCompressedOops */ \
|
flag(has_nonstatic_fields , 1 << 1) /* for sizing with UseCompressedOops */ \
|
||||||
flag(should_verify_class , 1 << 2) /* allow caching of preverification */ \
|
flag(should_verify_class , 1 << 2) /* allow caching of preverification */ \
|
||||||
flag(unused , 1 << 3) /* not currently used */ \
|
flag(is_contended , 1 << 3) /* marked with contended annotation */ \
|
||||||
flag(is_contended , 1 << 4) /* marked with contended annotation */ \
|
flag(has_nonstatic_concrete_methods , 1 << 4) /* class/superclass/implemented interfaces has non-static, concrete methods */ \
|
||||||
flag(has_nonstatic_concrete_methods , 1 << 5) /* class/superclass/implemented interfaces has non-static, concrete methods */ \
|
flag(declares_nonstatic_concrete_methods, 1 << 5) /* directly declares non-static, concrete methods */ \
|
||||||
flag(declares_nonstatic_concrete_methods, 1 << 6) /* directly declares non-static, concrete methods */ \
|
flag(shared_loading_failed , 1 << 6) /* class has been loaded from shared archive */ \
|
||||||
flag(has_been_redefined , 1 << 7) /* class has been redefined */ \
|
flag(is_shared_boot_class , 1 << 7) /* defining class loader is boot class loader */ \
|
||||||
flag(shared_loading_failed , 1 << 8) /* class has been loaded from shared archive */ \
|
flag(is_shared_platform_class , 1 << 8) /* defining class loader is platform class loader */ \
|
||||||
flag(is_scratch_class , 1 << 9) /* class is the redefined scratch class */ \
|
flag(is_shared_app_class , 1 << 9) /* defining class loader is app class loader */ \
|
||||||
flag(is_shared_boot_class , 1 << 10) /* defining class loader is boot class loader */ \
|
flag(has_contended_annotations , 1 << 10) /* has @Contended annotation */ \
|
||||||
flag(is_shared_platform_class , 1 << 11) /* defining class loader is platform class loader */ \
|
flag(has_localvariable_table , 1 << 11) /* has localvariable information */
|
||||||
flag(is_shared_app_class , 1 << 12) /* defining class loader is app class loader */ \
|
|
||||||
flag(has_contended_annotations , 1 << 13) /* has @Contended annotation */ \
|
|
||||||
flag(has_localvariable_table , 1 << 14) /* has localvariable information */
|
|
||||||
|
|
||||||
#define IK_FLAGS_ENUM_NAME(name, value) _misc_##name = value,
|
#define IK_FLAGS_ENUM_NAME(name, value) _misc_##name = value,
|
||||||
enum {
|
enum {
|
||||||
@ -54,6 +57,19 @@ class InstanceKlassFlags {
|
|||||||
};
|
};
|
||||||
#undef IK_FLAGS_ENUM_NAME
|
#undef IK_FLAGS_ENUM_NAME
|
||||||
|
|
||||||
|
#define IK_STATUS_DO(status) \
|
||||||
|
status(is_being_redefined , 1 << 0) /* True if the klass is being redefined */ \
|
||||||
|
status(has_resolved_methods , 1 << 1) /* True if the klass has resolved MethodHandle methods */ \
|
||||||
|
status(has_been_redefined , 1 << 2) /* class has been redefined */ \
|
||||||
|
status(is_scratch_class , 1 << 3) /* class is the redefined scratch class */ \
|
||||||
|
status(is_marked_dependent , 1 << 4) /* class is the redefined scratch class */
|
||||||
|
|
||||||
|
#define IK_STATUS_ENUM_NAME(name, value) _misc_##name = value,
|
||||||
|
enum {
|
||||||
|
IK_STATUS_DO(IK_STATUS_ENUM_NAME)
|
||||||
|
};
|
||||||
|
#undef IK_STATUS_ENUM_NAME
|
||||||
|
|
||||||
u2 shared_loader_type_bits() const {
|
u2 shared_loader_type_bits() const {
|
||||||
return _misc_is_shared_boot_class|_misc_is_shared_platform_class|_misc_is_shared_app_class;
|
return _misc_is_shared_boot_class|_misc_is_shared_platform_class|_misc_is_shared_app_class;
|
||||||
}
|
}
|
||||||
@ -61,6 +77,9 @@ class InstanceKlassFlags {
|
|||||||
// These flags are write-once before the class is published and then read-only so don't require atomic updates.
|
// These flags are write-once before the class is published and then read-only so don't require atomic updates.
|
||||||
u2 _flags;
|
u2 _flags;
|
||||||
|
|
||||||
|
// These flags are written during execution so require atomic stores
|
||||||
|
u1 _status;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
InstanceKlassFlags() : _flags(0) {}
|
InstanceKlassFlags() : _flags(0) {}
|
||||||
@ -87,6 +106,26 @@ class InstanceKlassFlags {
|
|||||||
|
|
||||||
void assign_class_loader_type(const ClassLoaderData* cld);
|
void assign_class_loader_type(const ClassLoaderData* cld);
|
||||||
void assert_is_safe(bool set) NOT_DEBUG_RETURN;
|
void assert_is_safe(bool set) NOT_DEBUG_RETURN;
|
||||||
|
|
||||||
|
// Create getters and setters for the status values.
|
||||||
|
#define IK_STATUS_GET(name, ignore) \
|
||||||
|
bool name() const { return (_status & _misc_##name) != 0; }
|
||||||
|
IK_STATUS_DO(IK_STATUS_GET)
|
||||||
|
#undef IK_STATUS_GET
|
||||||
|
|
||||||
|
#define IK_STATUS_SET(name, ignore) \
|
||||||
|
void set_##name(bool b) { \
|
||||||
|
if (b) { \
|
||||||
|
atomic_set_bits(_misc_##name); \
|
||||||
|
} else { \
|
||||||
|
atomic_clear_bits(_misc_##name); \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
IK_STATUS_DO(IK_STATUS_SET)
|
||||||
|
#undef IK_STATUS_SET
|
||||||
|
|
||||||
|
void atomic_set_bits(u1 bits);
|
||||||
|
void atomic_clear_bits(u1 bits);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // SHARE_OOPS_INSTANCEKLASSFLAGS_HPP
|
#endif // SHARE_OOPS_INSTANCEKLASSFLAGS_HPP
|
||||||
|
@ -236,7 +236,6 @@
|
|||||||
nonstatic_field(InstanceKlass, _static_field_size, int) \
|
nonstatic_field(InstanceKlass, _static_field_size, int) \
|
||||||
nonstatic_field(InstanceKlass, _static_oop_field_count, u2) \
|
nonstatic_field(InstanceKlass, _static_oop_field_count, u2) \
|
||||||
nonstatic_field(InstanceKlass, _nonstatic_oop_map_size, int) \
|
nonstatic_field(InstanceKlass, _nonstatic_oop_map_size, int) \
|
||||||
nonstatic_field(InstanceKlass, _is_marked_dependent, bool) \
|
|
||||||
volatile_nonstatic_field(InstanceKlass, _init_state, InstanceKlass::ClassState) \
|
volatile_nonstatic_field(InstanceKlass, _init_state, InstanceKlass::ClassState) \
|
||||||
volatile_nonstatic_field(InstanceKlass, _init_thread, JavaThread*) \
|
volatile_nonstatic_field(InstanceKlass, _init_thread, JavaThread*) \
|
||||||
nonstatic_field(InstanceKlass, _itable_len, int) \
|
nonstatic_field(InstanceKlass, _itable_len, int) \
|
||||||
|
@ -68,8 +68,6 @@ enum {
|
|||||||
JVM_ACC_HAS_FINAL_METHOD = 0x01000000, // True if klass has final method
|
JVM_ACC_HAS_FINAL_METHOD = 0x01000000, // True if klass has final method
|
||||||
JVM_ACC_IS_HIDDEN_CLASS = 0x04000000, // True if klass is hidden
|
JVM_ACC_IS_HIDDEN_CLASS = 0x04000000, // True if klass is hidden
|
||||||
JVM_ACC_IS_VALUE_BASED_CLASS = 0x08000000, // True if klass is marked as a ValueBased class
|
JVM_ACC_IS_VALUE_BASED_CLASS = 0x08000000, // True if klass is marked as a ValueBased class
|
||||||
JVM_ACC_IS_BEING_REDEFINED = 0x00100000, // True if the klass is being redefined.
|
|
||||||
JVM_ACC_HAS_RESOLVED_METHODS = 0x00200000, // True if the klass has resolved methods
|
|
||||||
|
|
||||||
// Method* flags
|
// Method* flags
|
||||||
JVM_ACC_HAS_LOCAL_VARIABLE_TABLE= 0x00400000,
|
JVM_ACC_HAS_LOCAL_VARIABLE_TABLE= 0x00400000,
|
||||||
@ -133,13 +131,6 @@ class AccessFlags {
|
|||||||
void set_has_localvariable_table() { atomic_set_bits(JVM_ACC_HAS_LOCAL_VARIABLE_TABLE); }
|
void set_has_localvariable_table() { atomic_set_bits(JVM_ACC_HAS_LOCAL_VARIABLE_TABLE); }
|
||||||
void clear_has_localvariable_table() { atomic_clear_bits(JVM_ACC_HAS_LOCAL_VARIABLE_TABLE); }
|
void clear_has_localvariable_table() { atomic_clear_bits(JVM_ACC_HAS_LOCAL_VARIABLE_TABLE); }
|
||||||
|
|
||||||
bool is_being_redefined() const { return (_flags & JVM_ACC_IS_BEING_REDEFINED) != 0; }
|
|
||||||
void set_is_being_redefined() { atomic_set_bits(JVM_ACC_IS_BEING_REDEFINED); }
|
|
||||||
void clear_is_being_redefined() { atomic_clear_bits(JVM_ACC_IS_BEING_REDEFINED); }
|
|
||||||
|
|
||||||
bool has_resolved_methods() const { return (_flags & JVM_ACC_HAS_RESOLVED_METHODS) != 0; }
|
|
||||||
void set_has_resolved_methods() { atomic_set_bits(JVM_ACC_HAS_RESOLVED_METHODS); }
|
|
||||||
|
|
||||||
bool on_stack() const { return (_flags & JVM_ACC_ON_STACK) != 0; }
|
bool on_stack() const { return (_flags & JVM_ACC_ON_STACK) != 0; }
|
||||||
|
|
||||||
// get .class file flags
|
// get .class file flags
|
||||||
|
@ -80,7 +80,6 @@ public class InstanceKlass extends Klass {
|
|||||||
staticFieldSize = new CIntField(type.getCIntegerField("_static_field_size"), 0);
|
staticFieldSize = new CIntField(type.getCIntegerField("_static_field_size"), 0);
|
||||||
staticOopFieldCount = new CIntField(type.getCIntegerField("_static_oop_field_count"), 0);
|
staticOopFieldCount = new CIntField(type.getCIntegerField("_static_oop_field_count"), 0);
|
||||||
nonstaticOopMapSize = new CIntField(type.getCIntegerField("_nonstatic_oop_map_size"), 0);
|
nonstaticOopMapSize = new CIntField(type.getCIntegerField("_nonstatic_oop_map_size"), 0);
|
||||||
isMarkedDependent = new CIntField(type.getCIntegerField("_is_marked_dependent"), 0);
|
|
||||||
initState = new CIntField(type.getCIntegerField("_init_state"), 0);
|
initState = new CIntField(type.getCIntegerField("_init_state"), 0);
|
||||||
itableLen = new CIntField(type.getCIntegerField("_itable_len"), 0);
|
itableLen = new CIntField(type.getCIntegerField("_itable_len"), 0);
|
||||||
if (VM.getVM().isJvmtiSupported()) {
|
if (VM.getVM().isJvmtiSupported()) {
|
||||||
@ -145,7 +144,6 @@ public class InstanceKlass extends Klass {
|
|||||||
private static CIntField staticFieldSize;
|
private static CIntField staticFieldSize;
|
||||||
private static CIntField staticOopFieldCount;
|
private static CIntField staticOopFieldCount;
|
||||||
private static CIntField nonstaticOopMapSize;
|
private static CIntField nonstaticOopMapSize;
|
||||||
private static CIntField isMarkedDependent;
|
|
||||||
private static CIntField initState;
|
private static CIntField initState;
|
||||||
private static CIntField itableLen;
|
private static CIntField itableLen;
|
||||||
private static AddressField breakpoints;
|
private static AddressField breakpoints;
|
||||||
@ -373,7 +371,6 @@ public class InstanceKlass extends Klass {
|
|||||||
public long getNonstaticFieldSize() { return nonstaticFieldSize.getValue(this); }
|
public long getNonstaticFieldSize() { return nonstaticFieldSize.getValue(this); }
|
||||||
public long getStaticOopFieldCount() { return staticOopFieldCount.getValue(this); }
|
public long getStaticOopFieldCount() { return staticOopFieldCount.getValue(this); }
|
||||||
public long getNonstaticOopMapSize() { return nonstaticOopMapSize.getValue(this); }
|
public long getNonstaticOopMapSize() { return nonstaticOopMapSize.getValue(this); }
|
||||||
public boolean getIsMarkedDependent() { return isMarkedDependent.getValue(this) != 0; }
|
|
||||||
public long getItableLen() { return itableLen.getValue(this); }
|
public long getItableLen() { return itableLen.getValue(this); }
|
||||||
public long majorVersion() { return getConstants().majorVersion(); }
|
public long majorVersion() { return getConstants().majorVersion(); }
|
||||||
public long minorVersion() { return getConstants().minorVersion(); }
|
public long minorVersion() { return getConstants().minorVersion(); }
|
||||||
@ -571,7 +568,6 @@ public class InstanceKlass extends Klass {
|
|||||||
visitor.doCInt(staticFieldSize, true);
|
visitor.doCInt(staticFieldSize, true);
|
||||||
visitor.doCInt(staticOopFieldCount, true);
|
visitor.doCInt(staticOopFieldCount, true);
|
||||||
visitor.doCInt(nonstaticOopMapSize, true);
|
visitor.doCInt(nonstaticOopMapSize, true);
|
||||||
visitor.doCInt(isMarkedDependent, true);
|
|
||||||
visitor.doCInt(initState, true);
|
visitor.doCInt(initState, true);
|
||||||
visitor.doCInt(itableLen, true);
|
visitor.doCInt(itableLen, true);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user