Skip to content

Commit c9cadbd

Browse files
committedFeb 10, 2025
8346567: Make Class.getModifiers() non-native
Reviewed-by: alanb, vlivanov, yzheng, dlong
1 parent 5589892 commit c9cadbd

File tree

31 files changed

+90
-146
lines changed

31 files changed

+90
-146
lines changed
 

‎src/hotspot/share/c1/c1_Canonicalizer.cpp

-16
Original file line numberDiff line numberDiff line change
@@ -547,22 +547,6 @@ void Canonicalizer::do_Intrinsic (Intrinsic* x) {
547547
}
548548
break;
549549
}
550-
case vmIntrinsics::_getModifiers: {
551-
assert(x->number_of_arguments() == 1, "wrong type");
552-
553-
// Optimize for Foo.class.getModifier()
554-
InstanceConstant* c = x->argument_at(0)->type()->as_InstanceConstant();
555-
if (c != nullptr && !c->value()->is_null_object()) {
556-
ciType* t = c->value()->java_mirror_type();
557-
if (t->is_klass()) {
558-
set_constant(t->as_klass()->modifier_flags());
559-
} else {
560-
assert(t->is_primitive_type(), "should be a primitive type");
561-
set_constant(JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC);
562-
}
563-
}
564-
break;
565-
}
566550
default:
567551
break;
568552
}

‎src/hotspot/share/c1/c1_Compiler.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,6 @@ bool Compiler::is_intrinsic_supported(vmIntrinsics::ID id) {
156156
case vmIntrinsics::_getClass:
157157
case vmIntrinsics::_isInstance:
158158
case vmIntrinsics::_isPrimitive:
159-
case vmIntrinsics::_getModifiers:
160159
case vmIntrinsics::_currentCarrierThread:
161160
case vmIntrinsics::_currentThread:
162161
case vmIntrinsics::_scopedValueCache:

‎src/hotspot/share/c1/c1_LIRGenerator.cpp

-37
Original file line numberDiff line numberDiff line change
@@ -1297,42 +1297,6 @@ void LIRGenerator::do_isPrimitive(Intrinsic* x) {
12971297
__ cmove(lir_cond_notEqual, LIR_OprFact::intConst(0), LIR_OprFact::intConst(1), result, T_BOOLEAN);
12981298
}
12991299

1300-
// Example: Foo.class.getModifiers()
1301-
void LIRGenerator::do_getModifiers(Intrinsic* x) {
1302-
assert(x->number_of_arguments() == 1, "wrong type");
1303-
1304-
LIRItem receiver(x->argument_at(0), this);
1305-
receiver.load_item();
1306-
LIR_Opr result = rlock_result(x);
1307-
1308-
CodeEmitInfo* info = nullptr;
1309-
if (x->needs_null_check()) {
1310-
info = state_for(x);
1311-
}
1312-
1313-
// While reading off the universal constant mirror is less efficient than doing
1314-
// another branch and returning the constant answer, this branchless code runs into
1315-
// much less risk of confusion for C1 register allocator. The choice of the universe
1316-
// object here is correct as long as it returns the same modifiers we would expect
1317-
// from the primitive class itself. See spec for Class.getModifiers that provides
1318-
// the typed array klasses with similar modifiers as their component types.
1319-
1320-
Klass* univ_klass = Universe::byteArrayKlass();
1321-
assert(univ_klass->modifier_flags() == (JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC), "Sanity");
1322-
LIR_Opr prim_klass = LIR_OprFact::metadataConst(univ_klass);
1323-
1324-
LIR_Opr recv_klass = new_register(T_METADATA);
1325-
__ move(new LIR_Address(receiver.result(), java_lang_Class::klass_offset(), T_ADDRESS), recv_klass, info);
1326-
1327-
// Check if this is a Java mirror of primitive type, and select the appropriate klass.
1328-
LIR_Opr klass = new_register(T_METADATA);
1329-
__ cmp(lir_cond_equal, recv_klass, LIR_OprFact::metadataConst(nullptr));
1330-
__ cmove(lir_cond_equal, prim_klass, recv_klass, klass, T_ADDRESS);
1331-
1332-
// Get the answer.
1333-
__ move(new LIR_Address(klass, in_bytes(Klass::modifier_flags_offset()), T_CHAR), result);
1334-
}
1335-
13361300
void LIRGenerator::do_getObjectSize(Intrinsic* x) {
13371301
assert(x->number_of_arguments() == 3, "wrong type");
13381302
LIR_Opr result_reg = rlock_result(x);
@@ -2951,7 +2915,6 @@ void LIRGenerator::do_Intrinsic(Intrinsic* x) {
29512915
case vmIntrinsics::_Object_init: do_RegisterFinalizer(x); break;
29522916
case vmIntrinsics::_isInstance: do_isInstance(x); break;
29532917
case vmIntrinsics::_isPrimitive: do_isPrimitive(x); break;
2954-
case vmIntrinsics::_getModifiers: do_getModifiers(x); break;
29552918
case vmIntrinsics::_getClass: do_getClass(x); break;
29562919
case vmIntrinsics::_getObjectSize: do_getObjectSize(x); break;
29572920
case vmIntrinsics::_currentCarrierThread: do_currentCarrierThread(x); break;

‎src/hotspot/share/c1/c1_LIRGenerator.hpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2005, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2005, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -255,7 +255,6 @@ class LIRGenerator: public InstructionVisitor, public BlockClosure {
255255
void do_RegisterFinalizer(Intrinsic* x);
256256
void do_isInstance(Intrinsic* x);
257257
void do_isPrimitive(Intrinsic* x);
258-
void do_getModifiers(Intrinsic* x);
259258
void do_getClass(Intrinsic* x);
260259
void do_getObjectSize(Intrinsic* x);
261260
void do_currentCarrierThread(Intrinsic* x);

‎src/hotspot/share/ci/ciKlass.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -119,7 +119,7 @@ class ciKlass : public ciType {
119119
// Get the instance of java.lang.Class corresponding to this klass.
120120
ciInstance* java_mirror();
121121

122-
// Fetch Klass::modifier_flags.
122+
// Fetch modifier flags.
123123
jint modifier_flags();
124124

125125
// Fetch Klass::access_flags.

‎src/hotspot/share/classfile/classFileParser.cpp

-5
Original file line numberDiff line numberDiff line change
@@ -3747,11 +3747,6 @@ void ClassFileParser::apply_parsed_class_metadata(
37473747
this_klass->set_permitted_subclasses(_permitted_subclasses);
37483748
this_klass->set_record_components(_record_components);
37493749

3750-
// Initialize cached modifier_flags to support Class.getModifiers().
3751-
// This must follow setting inner_class attributes.
3752-
u2 computed_modifiers = this_klass->compute_modifier_flags();
3753-
this_klass->set_modifier_flags(computed_modifiers);
3754-
37553750
// Delay the setting of _local_interfaces and _transitive_interfaces until after
37563751
// initialize_supers() in fill_instance_klass(). It is because the _local_interfaces could
37573752
// be shared with _transitive_interfaces and _transitive_interfaces may be shared with

‎src/hotspot/share/classfile/javaClasses.cpp

+18-1
Original file line numberDiff line numberDiff line change
@@ -867,6 +867,7 @@ int java_lang_Class::_source_file_offset;
867867
int java_lang_Class::_classData_offset;
868868
int java_lang_Class::_classRedefinedCount_offset;
869869
int java_lang_Class::_reflectionData_offset;
870+
int java_lang_Class::_modifiers_offset;
870871

871872
bool java_lang_Class::_offsets_computed = false;
872873
GrowableArray<Klass*>* java_lang_Class::_fixup_mirror_list = nullptr;
@@ -1060,6 +1061,10 @@ void java_lang_Class::allocate_mirror(Klass* k, bool is_scratch, Handle protecti
10601061
// Setup indirection from mirror->klass
10611062
set_klass(mirror(), k);
10621063

1064+
// Set the modifiers flag.
1065+
int computed_modifiers = k->compute_modifier_flags();
1066+
set_modifiers(mirror(), computed_modifiers);
1067+
10631068
InstanceMirrorKlass* mk = InstanceMirrorKlass::cast(mirror->klass());
10641069
assert(oop_size(mirror()) == mk->instance_size(k), "should have been set");
10651070

@@ -1355,6 +1360,7 @@ oop java_lang_Class::create_basic_type_mirror(const char* basic_type_name, Basic
13551360
InstanceMirrorKlass* mk = InstanceMirrorKlass::cast(vmClasses::Class_klass());
13561361
assert(static_oop_field_count(java_class) == 0, "should have been zeroed by allocation");
13571362
#endif
1363+
set_modifiers(java_class, JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC);
13581364
return java_class;
13591365
}
13601366

@@ -1493,7 +1499,8 @@ oop java_lang_Class::primitive_mirror(BasicType t) {
14931499
macro(_name_offset, k, "name", string_signature, false); \
14941500
macro(_classData_offset, k, "classData", object_signature, false); \
14951501
macro(_reflectionData_offset, k, "reflectionData", java_lang_ref_SoftReference_signature, false); \
1496-
macro(_signers_offset, k, "signers", object_array_signature, false);
1502+
macro(_signers_offset, k, "signers", object_array_signature, false); \
1503+
macro(_modifiers_offset, k, vmSymbols::modifiers_name(), int_signature, false);
14971504

14981505
void java_lang_Class::compute_offsets() {
14991506
if (_offsets_computed) {
@@ -1527,6 +1534,16 @@ void java_lang_Class::set_classRedefinedCount(oop the_class_mirror, int value) {
15271534
the_class_mirror->int_field_put(_classRedefinedCount_offset, value);
15281535
}
15291536

1537+
int java_lang_Class::modifiers(oop the_class_mirror) {
1538+
assert(_modifiers_offset != 0, "offsets should have been initialized");
1539+
return the_class_mirror->int_field(_modifiers_offset);
1540+
}
1541+
1542+
void java_lang_Class::set_modifiers(oop the_class_mirror, int value) {
1543+
assert(_modifiers_offset != 0, "offsets should have been initialized");
1544+
the_class_mirror->int_field_put(_modifiers_offset, value);
1545+
}
1546+
15301547

15311548
// Note: JDK1.1 and before had a privateInfo_offset field which was used for the
15321549
// platform thread structure, and a eetop offset which was used for thread

‎src/hotspot/share/classfile/javaClasses.hpp

+4
Original file line numberDiff line numberDiff line change
@@ -257,6 +257,7 @@ class java_lang_Class : AllStatic {
257257
static int _classData_offset;
258258
static int _classRedefinedCount_offset;
259259
static int _reflectionData_offset;
260+
static int _modifiers_offset;
260261

261262
static bool _offsets_computed;
262263

@@ -337,6 +338,9 @@ class java_lang_Class : AllStatic {
337338
static oop source_file(oop java_class);
338339
static void set_source_file(oop java_class, oop source_file);
339340

341+
static int modifiers(oop java_class);
342+
static void set_modifiers(oop java_class, int value);
343+
340344
static size_t oop_size(oop java_class);
341345
static void set_oop_size(HeapWord* java_class, size_t size);
342346
static int static_oop_field_count(oop java_class);

‎src/hotspot/share/classfile/vmIntrinsics.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -256,7 +256,6 @@ bool vmIntrinsics::disabled_by_jvm_flags(vmIntrinsics::ID id) {
256256
switch (id) {
257257
case vmIntrinsics::_isInstance:
258258
case vmIntrinsics::_isAssignableFrom:
259-
case vmIntrinsics::_getModifiers:
260259
case vmIntrinsics::_isInterface:
261260
case vmIntrinsics::_isArray:
262261
case vmIntrinsics::_isPrimitive:

‎src/hotspot/share/classfile/vmIntrinsics.hpp

-2
Original file line numberDiff line numberDiff line change
@@ -304,8 +304,6 @@ class methodHandle;
304304
do_name( isAssignableFrom_name, "isAssignableFrom") \
305305
do_intrinsic(_isInstance, java_lang_Class, isInstance_name, object_boolean_signature, F_RN) \
306306
do_name( isInstance_name, "isInstance") \
307-
do_intrinsic(_getModifiers, java_lang_Class, getModifiers_name, void_int_signature, F_RN) \
308-
do_name( getModifiers_name, "getModifiers") \
309307
do_intrinsic(_isInterface, java_lang_Class, isInterface_name, void_boolean_signature, F_RN) \
310308
do_name( isInterface_name, "isInterface") \
311309
do_intrinsic(_isArray, java_lang_Class, isArray_name, void_boolean_signature, F_RN) \

‎src/hotspot/share/jvmci/vmStructs_jvmci.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -272,7 +272,6 @@
272272
nonstatic_field(Klass, _name, Symbol*) \
273273
volatile_nonstatic_field(Klass, _next_sibling, Klass*) \
274274
nonstatic_field(Klass, _java_mirror, OopHandle) \
275-
nonstatic_field(Klass, _modifier_flags, u2) \
276275
nonstatic_field(Klass, _access_flags, AccessFlags) \
277276
nonstatic_field(Klass, _class_loader_data, ClassLoaderData*) \
278277
nonstatic_field(Klass, _secondary_supers_bitmap, uintx) \

‎src/hotspot/share/oops/klass.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,12 @@ jint Klass::array_layout_helper(BasicType etype) {
328328
return lh;
329329
}
330330

331+
int Klass::modifier_flags() const {
332+
int mods = java_lang_Class::modifiers(java_mirror());
333+
assert(mods == compute_modifier_flags(), "should be same");
334+
return mods;
335+
}
336+
331337
bool Klass::can_be_primary_super_slow() const {
332338
if (super() == nullptr)
333339
return true;

‎src/hotspot/share/oops/klass.hpp

+1-8
Original file line numberDiff line numberDiff line change
@@ -123,9 +123,6 @@ class Klass : public Metadata {
123123
AccessFlags _access_flags; // Access flags. The class/interface distinction is stored here.
124124
// Some flags created by the JVM, not in the class file itself,
125125
// are in _misc_flags below.
126-
// Processed access flags, for use by Class.getModifiers.
127-
u2 _modifier_flags;
128-
129126
KlassFlags _misc_flags;
130127

131128
// The fields _super_check_offset, _secondary_super_cache, _secondary_supers
@@ -292,10 +289,6 @@ class Klass : public Metadata {
292289
// This leaves the OopHandle in the CLD, but that's ok, you can't release them.
293290
void clear_java_mirror_handle() { _java_mirror = OopHandle(); }
294291

295-
// modifier flags
296-
u2 modifier_flags() const { return _modifier_flags; }
297-
void set_modifier_flags(u2 flags) { _modifier_flags = flags; }
298-
299292
// size helper
300293
int layout_helper() const { return _layout_helper; }
301294
void set_layout_helper(int lh) { _layout_helper = lh; }
@@ -448,7 +441,6 @@ class Klass : public Metadata {
448441
static ByteSize secondary_supers_offset() { return byte_offset_of(Klass, _secondary_supers); }
449442
static ByteSize java_mirror_offset() { return byte_offset_of(Klass, _java_mirror); }
450443
static ByteSize class_loader_data_offset() { return byte_offset_of(Klass, _class_loader_data); }
451-
static ByteSize modifier_flags_offset() { return byte_offset_of(Klass, _modifier_flags); }
452444
static ByteSize layout_helper_offset() { return byte_offset_of(Klass, _layout_helper); }
453445
static ByteSize access_flags_offset() { return byte_offset_of(Klass, _access_flags); }
454446
#if INCLUDE_JVMCI
@@ -758,6 +750,7 @@ class Klass : public Metadata {
758750

759751
public:
760752
virtual u2 compute_modifier_flags() const = 0;
753+
int modifier_flags() const;
761754

762755
// JVMTI support
763756
virtual jint jvmti_class_status() const;

‎src/hotspot/share/oops/objArrayKlass.cpp

-3
Original file line numberDiff line numberDiff line change
@@ -139,9 +139,6 @@ ObjArrayKlass::ObjArrayKlass(int n, Klass* element_klass, Symbol* name) : ArrayK
139139
set_layout_helper(array_layout_helper(T_OBJECT));
140140
assert(is_array_klass(), "sanity");
141141
assert(is_objArray_klass(), "sanity");
142-
143-
// Compute modifier flags after bottom_klass and element_klass are initialized.
144-
set_modifier_flags(compute_modifier_flags());
145142
}
146143

147144
size_t ObjArrayKlass::oop_size(oop obj) const {

‎src/hotspot/share/oops/typeArrayKlass.cpp

-3
Original file line numberDiff line numberDiff line change
@@ -87,9 +87,6 @@ TypeArrayKlass::TypeArrayKlass(BasicType type, Symbol* name) : ArrayKlass(name,
8787
assert(size() >= TypeArrayKlass::header_size(), "bad size");
8888

8989
set_class_loader_data(ClassLoaderData::the_null_class_loader_data());
90-
91-
// Compute modifier flags.
92-
set_modifier_flags(compute_modifier_flags());
9390
}
9491

9592
typeArrayOop TypeArrayKlass::allocate_common(int length, bool do_zero, TRAPS) {

‎src/hotspot/share/oops/typeArrayKlass.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -51,6 +51,8 @@ class TypeArrayKlass : public ArrayKlass {
5151
jint max_length() { return _max_length; }
5252
void set_max_length(jint m) { _max_length = m; }
5353

54+
u2 compute_modifier_flags() const;
55+
5456
// testers
5557
DEBUG_ONLY(bool is_typeArray_klass_slow() const { return true; })
5658

@@ -73,8 +75,6 @@ class TypeArrayKlass : public ArrayKlass {
7375
// Copying
7476
void copy_array(arrayOop s, int src_pos, arrayOop d, int dst_pos, int length, TRAPS);
7577

76-
u2 compute_modifier_flags() const;
77-
7878
// Oop iterators. Since there are no oops in TypeArrayKlasses,
7979
// these functions only return the size of the object.
8080

‎src/hotspot/share/opto/c2compiler.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -749,7 +749,6 @@ bool C2Compiler::is_intrinsic_supported(vmIntrinsics::ID id) {
749749
case vmIntrinsics::_clone:
750750
case vmIntrinsics::_isAssignableFrom:
751751
case vmIntrinsics::_isInstance:
752-
case vmIntrinsics::_getModifiers:
753752
case vmIntrinsics::_isInterface:
754753
case vmIntrinsics::_isArray:
755754
case vmIntrinsics::_isPrimitive:

‎src/hotspot/share/opto/compile.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -1699,8 +1699,6 @@ Compile::AliasType* Compile::find_alias_type(const TypePtr* adr_type, bool no_cr
16991699
}
17001700
if (flat->offset() == in_bytes(Klass::super_check_offset_offset()))
17011701
alias_type(idx)->set_rewritable(false);
1702-
if (flat->offset() == in_bytes(Klass::modifier_flags_offset()))
1703-
alias_type(idx)->set_rewritable(false);
17041702
if (flat->offset() == in_bytes(Klass::access_flags_offset()))
17051703
alias_type(idx)->set_rewritable(false);
17061704
if (flat->offset() == in_bytes(Klass::misc_flags_offset()))

‎src/hotspot/share/opto/library_call.cpp

-10
Original file line numberDiff line numberDiff line change
@@ -515,7 +515,6 @@ bool LibraryCallKit::try_to_inline(int predicate) {
515515
case vmIntrinsics::_isAssignableFrom: return inline_native_subtype_check();
516516

517517
case vmIntrinsics::_isInstance:
518-
case vmIntrinsics::_getModifiers:
519518
case vmIntrinsics::_isInterface:
520519
case vmIntrinsics::_isArray:
521520
case vmIntrinsics::_isPrimitive:
@@ -3891,10 +3890,6 @@ bool LibraryCallKit::inline_native_Class_query(vmIntrinsics::ID id) {
38913890
prim_return_value = intcon(0);
38923891
obj = argument(1);
38933892
break;
3894-
case vmIntrinsics::_getModifiers:
3895-
prim_return_value = intcon(JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC);
3896-
return_type = TypeInt::CHAR;
3897-
break;
38983893
case vmIntrinsics::_isInterface:
38993894
prim_return_value = intcon(0);
39003895
break;
@@ -3974,11 +3969,6 @@ bool LibraryCallKit::inline_native_Class_query(vmIntrinsics::ID id) {
39743969
query_value = gen_instanceof(obj, kls, safe_for_replace);
39753970
break;
39763971

3977-
case vmIntrinsics::_getModifiers:
3978-
p = basic_plus_adr(kls, in_bytes(Klass::modifier_flags_offset()));
3979-
query_value = make_load(nullptr, p, TypeInt::CHAR, T_CHAR, MemNode::unordered);
3980-
break;
3981-
39823972
case vmIntrinsics::_isInterface:
39833973
// (To verify this code sequence, check the asserts in JVM_IsInterface.)
39843974
if (generate_interface_guard(kls, region) != nullptr)

‎src/hotspot/share/opto/memnode.cpp

+1-7
Original file line numberDiff line numberDiff line change
@@ -1972,12 +1972,6 @@ LoadNode::load_array_final_field(const TypeKlassPtr *tkls,
19721972
ciKlass* klass) const {
19731973
assert(!UseCompactObjectHeaders || tkls->offset() != in_bytes(Klass::prototype_header_offset()),
19741974
"must not happen");
1975-
if (tkls->offset() == in_bytes(Klass::modifier_flags_offset())) {
1976-
// The field is Klass::_modifier_flags. Return its (constant) value.
1977-
// (Folds up the 2nd indirection in aClassConstant.getModifiers().)
1978-
assert(Opcode() == Op_LoadUS, "must load an unsigned short from _modifier_flags");
1979-
return TypeInt::make(klass->modifier_flags());
1980-
}
19811975
if (tkls->offset() == in_bytes(Klass::access_flags_offset())) {
19821976
// The field is Klass::_access_flags. Return its (constant) value.
19831977
// (Folds up the 2nd indirection in Reflection.getClassAccessFlags(aClassConstant).)
@@ -2455,7 +2449,7 @@ const Type* LoadNode::klass_value_common(PhaseGVN* phase) const {
24552449
// a primitive Class (e.g., int.class) has null for a klass field
24562450
return TypePtr::NULL_PTR;
24572451
}
2458-
// (Folds up the 1st indirection in aClassConstant.getModifiers().)
2452+
// Fold up the load of the hidden field
24592453
return TypeKlassPtr::make(t->as_klass(), Type::trust_interfaces);
24602454
}
24612455
// non-constant mirror, so we can't tell what's going on

‎src/hotspot/share/prims/jvm.cpp

-16
Original file line numberDiff line numberDiff line change
@@ -1287,22 +1287,6 @@ JVM_ENTRY(jboolean, JVM_IsPrimitiveClass(JNIEnv *env, jclass cls))
12871287
JVM_END
12881288

12891289

1290-
JVM_ENTRY(jint, JVM_GetClassModifiers(JNIEnv *env, jclass cls))
1291-
oop mirror = JNIHandles::resolve_non_null(cls);
1292-
if (java_lang_Class::is_primitive(mirror)) {
1293-
// Primitive type
1294-
return JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC;
1295-
}
1296-
1297-
Klass* k = java_lang_Class::as_Klass(mirror);
1298-
debug_only(u2 computed_modifiers = k->compute_modifier_flags());
1299-
assert(k->modifier_flags() == computed_modifiers, "modifiers cache is OK");
1300-
return k->modifier_flags();
1301-
JVM_END
1302-
1303-
1304-
// Inner class reflection ///////////////////////////////////////////////////////////////////////////////
1305-
13061290
JVM_ENTRY(jobjectArray, JVM_GetDeclaredClasses(JNIEnv *env, jclass ofClass))
13071291
JvmtiVMObjectAllocEventCollector oam;
13081292
// ofClass is a reference to a java_lang_Class object. The mirror object

‎src/hotspot/share/prims/jvmtiEnv.cpp

+2-11
Original file line numberDiff line numberDiff line change
@@ -2717,19 +2717,10 @@ JvmtiEnv::GetSourceFileName(oop k_mirror, char** source_name_ptr) {
27172717
// modifiers_ptr - pre-checked for null
27182718
jvmtiError
27192719
JvmtiEnv::GetClassModifiers(oop k_mirror, jint* modifiers_ptr) {
2720-
JavaThread* current_thread = JavaThread::current();
2721-
jint result = 0;
2720+
jint result = java_lang_Class::modifiers(k_mirror);
27222721
if (!java_lang_Class::is_primitive(k_mirror)) {
2723-
Klass* k = java_lang_Class::as_Klass(k_mirror);
2724-
NULL_CHECK(k, JVMTI_ERROR_INVALID_CLASS);
2725-
result = k->modifier_flags();
2726-
27272722
// Reset the deleted ACC_SUPER bit (deleted in compute_modifier_flags()).
2728-
if (k->is_super()) {
2729-
result |= JVM_ACC_SUPER;
2730-
}
2731-
} else {
2732-
result = (JVM_ACC_ABSTRACT | JVM_ACC_FINAL | JVM_ACC_PUBLIC);
2723+
result |= JVM_ACC_SUPER;
27332724
}
27342725
*modifiers_ptr = result;
27352726

‎src/hotspot/share/runtime/vmStructs.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -262,7 +262,6 @@
262262
nonstatic_field(Klass, _secondary_supers, Array<Klass*>*) \
263263
nonstatic_field(Klass, _primary_supers[0], Klass*) \
264264
nonstatic_field(Klass, _java_mirror, OopHandle) \
265-
nonstatic_field(Klass, _modifier_flags, u2) \
266265
nonstatic_field(Klass, _super, Klass*) \
267266
volatile_nonstatic_field(Klass, _subklass, Klass*) \
268267
nonstatic_field(Klass, _layout_helper, jint) \

‎src/java.base/share/classes/java/lang/Class.java

+4-3
Original file line numberDiff line numberDiff line change
@@ -236,11 +236,12 @@ private static void runtimeSetup() {
236236
* This constructor is not used and prevents the default constructor being
237237
* generated.
238238
*/
239-
private Class(ClassLoader loader, Class<?> arrayComponentType) {
239+
private Class(ClassLoader loader, Class<?> arrayComponentType, int mods) {
240240
// Initialize final field for classLoader. The initialization value of non-null
241241
// prevents future JIT optimizations from assuming this final field is null.
242242
classLoader = loader;
243243
componentType = arrayComponentType;
244+
modifiers = mods;
244245
}
245246

246247
/**
@@ -1000,6 +1001,7 @@ public Module getModule() {
10001001

10011002
private transient Object classData; // Set by VM
10021003
private transient Object[] signers; // Read by VM, mutable
1004+
private final transient int modifiers; // Set by the VM
10031005

10041006
// package-private
10051007
Object getClassData() {
@@ -1344,8 +1346,7 @@ private Class<?> elementType() {
13441346
* @jls 9.1.1 Interface Modifiers
13451347
* @jvms 4.1 The {@code ClassFile} Structure
13461348
*/
1347-
@IntrinsicCandidate
1348-
public native int getModifiers();
1349+
public int getModifiers() { return modifiers; }
13491350

13501351
/**
13511352
* {@return an unmodifiable set of the {@linkplain AccessFlag access

‎src/java.base/share/classes/jdk/internal/reflect/Reflection.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2001, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2001, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -56,7 +56,7 @@ public class Reflection {
5656
fieldFilterMap = Map.of(
5757
Reflection.class, ALL_MEMBERS,
5858
AccessibleObject.class, ALL_MEMBERS,
59-
Class.class, Set.of("classLoader", "classData"),
59+
Class.class, Set.of("classLoader", "classData", "modifiers"),
6060
ClassLoader.class, ALL_MEMBERS,
6161
Constructor.class, ALL_MEMBERS,
6262
Field.class, ALL_MEMBERS,

‎src/java.base/share/native/libjava/Class.c

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1994, 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1994, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -61,7 +61,6 @@ static JNINativeMethod methods[] = {
6161
{"isArray", "()Z", (void *)&JVM_IsArrayClass},
6262
{"isHidden", "()Z", (void *)&JVM_IsHiddenClass},
6363
{"isPrimitive", "()Z", (void *)&JVM_IsPrimitiveClass},
64-
{"getModifiers", "()I", (void *)&JVM_GetClassModifiers},
6564
{"getDeclaredFields0","(Z)[" FLD, (void *)&JVM_GetClassDeclaredFields},
6665
{"getDeclaredMethods0","(Z)[" MHD, (void *)&JVM_GetClassDeclaredMethods},
6766
{"getDeclaredConstructors0","(Z)[" CTR, (void *)&JVM_GetClassDeclaredConstructors},

‎test/hotspot/jtreg/compiler/jvmci/jdk.vm.ci.runtime.test/src/jdk/vm/ci/runtime/test/TestResolvedJavaType.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -929,7 +929,7 @@ private static boolean isHiddenFromReflection(ResolvedJavaField f) {
929929
return true;
930930
}
931931
if (f.getDeclaringClass().equals(metaAccess.lookupJavaType(Class.class))) {
932-
return f.getName().equals("classLoader") || f.getName().equals("classData");
932+
return f.getName().equals("classLoader") || f.getName().equals("classData") || f.getName().equals("modifiers");
933933
}
934934
if (f.getDeclaringClass().equals(metaAccess.lookupJavaType(Lookup.class))) {
935935
return f.getName().equals("allowedModes") || f.getName().equals("lookupClass");

‎test/jdk/java/lang/reflect/AccessibleObject/ModuleSetAccessibleTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -147,7 +147,7 @@ public void testJavaLangClass() throws Exception {
147147

148148
// non-public constructor
149149
Constructor<?> ctor
150-
= Class.class.getDeclaredConstructor(ClassLoader.class, Class.class);
150+
= Class.class.getDeclaredConstructor(ClassLoader.class, Class.class, int.class);
151151
AccessibleObject[] ctors = { ctor };
152152

153153
try {

‎test/jdk/java/lang/reflect/AccessibleObject/TrySetAccessibleTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2017, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -193,7 +193,7 @@ public void testJavaLangClass() throws Exception {
193193

194194
// non-public constructor
195195
Constructor<?> ctor
196-
= Class.class.getDeclaredConstructor(ClassLoader.class, Class.class);
196+
= Class.class.getDeclaredConstructor(ClassLoader.class, Class.class, int.class);
197197
AccessibleObject[] ctors = { ctor };
198198

199199
assertFalse(ctor.trySetAccessible());

‎test/jdk/jdk/internal/reflect/Reflection/Filtering.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2018, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -23,7 +23,7 @@
2323

2424
/**
2525
* @test
26-
* @bug 8210496
26+
* @bug 8210496 8346567
2727
* @modules java.base/jdk.internal.reflect
2828
* @run testng Filtering
2929
* @summary Test that security sensitive fields that filtered by core reflection
@@ -55,6 +55,7 @@ private Object[][] sensitiveFields() {
5555
{ AccessibleObject.class, "override" },
5656
{ Class.class, "classLoader" },
5757
{ Class.class, "classData" },
58+
{ Class.class, "modifiers" },
5859
{ ClassLoader.class, "parent" },
5960
{ Field.class, "clazz" },
6061
{ Field.class, "modifiers" },

‎test/micro/org/openjdk/bench/java/lang/reflect/Clazz.java

+39-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2014, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -33,8 +33,11 @@
3333
import org.openjdk.jmh.annotations.Mode;
3434
import org.openjdk.jmh.annotations.OutputTimeUnit;
3535
import org.openjdk.jmh.annotations.Warmup;
36+
import org.openjdk.jmh.annotations.Scope;
37+
import org.openjdk.jmh.annotations.State;
3638

3739
@BenchmarkMode(Mode.AverageTime)
40+
@State(Scope.Benchmark)
3841
@OutputTimeUnit(TimeUnit.NANOSECONDS)
3942
@Warmup(iterations = 10, time = 1, timeUnit = TimeUnit.SECONDS)
4043
@Measurement(iterations = 5, time = 2, timeUnit = TimeUnit.SECONDS)
@@ -52,6 +55,41 @@ public Constructor getConstructor() throws NoSuchMethodException {
5255
return Clazz.class.getConstructor();
5356
}
5457

58+
/**
59+
* Get modifiers for this class through reflection
60+
*
61+
* @return
62+
* @throws NoSuchMethodException
63+
*/
64+
@Benchmark
65+
public int getModifiers() {
66+
return Clazz.class.getModifiers();
67+
}
68+
69+
Clazz[] clazzArray = new Clazz[1];
70+
@Benchmark
71+
public int getAppArrayModifiers() {
72+
return clazzArray.getClass().getModifiers();
73+
}
74+
75+
static final Clazz[] clazzArrayFinal = new Clazz[1];
76+
@Benchmark
77+
public int getAppArrayModifiersFinal() {
78+
return clazzArrayFinal.getClass().getModifiers();
79+
}
80+
81+
/**
82+
* Get modifiers for an primitive array class through reflection
83+
*
84+
* @return
85+
* @throws NoSuchMethodException
86+
*/
87+
@Benchmark
88+
public int getArrayModifiers() {
89+
return int[].class.getModifiers();
90+
}
91+
92+
5593
/**
5694
* Get constructor for the String class through reflection, forcing full
5795
* security check

0 commit comments

Comments
 (0)
Please sign in to comment.