Skip to content

Commit 2001da3

Browse files
author
Mandy Chung
committedDec 11, 2020
8257596: Clarify trusted final fields for record classes
Reviewed-by: hseigel, chegar, psandoz
1 parent b1afed7 commit 2001da3

File tree

4 files changed

+45
-30
lines changed

4 files changed

+45
-30
lines changed
 

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

+6
Original file line numberDiff line numberDiff line change
@@ -739,6 +739,12 @@ void InstanceKlass::deallocate_contents(ClassLoaderData* loader_data) {
739739
}
740740
}
741741

742+
bool InstanceKlass::is_record() const {
743+
return _record_components != NULL &&
744+
is_final() &&
745+
java_super() == SystemDictionary::Record_klass();
746+
}
747+
742748
bool InstanceKlass::is_sealed() const {
743749
return _permitted_subclasses != NULL &&
744750
_permitted_subclasses != Universe::the_empty_short_array();

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -473,7 +473,7 @@ class InstanceKlass: public Klass {
473473
void set_record_components(Array<RecordComponent*>* record_components) {
474474
_record_components = record_components;
475475
}
476-
bool is_record() const { return _record_components != NULL; }
476+
bool is_record() const;
477477

478478
// permitted subclasses
479479
Array<u2>* permitted_subclasses() const { return _permitted_subclasses; }

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

+25-22
Original file line numberDiff line numberDiff line change
@@ -1851,6 +1851,9 @@ JVM_ENTRY(jobjectArray, JVM_GetClassDeclaredFields(JNIEnv *env, jclass ofClass,
18511851
}
18521852
JVM_END
18531853

1854+
1855+
// A class is a record if and only if it is final and a direct subclass of
1856+
// java.lang.Record and has a Record attribute; otherwise, it is not a record.
18541857
JVM_ENTRY(jboolean, JVM_IsRecord(JNIEnv *env, jclass cls))
18551858
{
18561859
JVMWrapper("JVM_IsRecord");
@@ -1864,38 +1867,38 @@ JVM_ENTRY(jboolean, JVM_IsRecord(JNIEnv *env, jclass cls))
18641867
}
18651868
JVM_END
18661869

1870+
// Returns an array containing the components of the Record attribute,
1871+
// or NULL if the attribute is not present.
1872+
//
1873+
// Note that this function returns the components of the Record attribute
1874+
// even if the class is not a record.
18671875
JVM_ENTRY(jobjectArray, JVM_GetRecordComponents(JNIEnv* env, jclass ofClass))
18681876
{
18691877
JVMWrapper("JVM_GetRecordComponents");
18701878
Klass* c = java_lang_Class::as_Klass(JNIHandles::resolve_non_null(ofClass));
18711879
assert(c->is_instance_klass(), "must be");
18721880
InstanceKlass* ik = InstanceKlass::cast(c);
18731881

1874-
if (ik->is_record()) {
1875-
Array<RecordComponent*>* components = ik->record_components();
1876-
assert(components != NULL, "components should not be NULL");
1877-
{
1878-
JvmtiVMObjectAllocEventCollector oam;
1879-
constantPoolHandle cp(THREAD, ik->constants());
1880-
int length = components->length();
1881-
assert(length >= 0, "unexpected record_components length");
1882-
objArrayOop record_components =
1883-
oopFactory::new_objArray(SystemDictionary::RecordComponent_klass(), length, CHECK_NULL);
1884-
objArrayHandle components_h (THREAD, record_components);
1885-
1886-
for (int x = 0; x < length; x++) {
1887-
RecordComponent* component = components->at(x);
1888-
assert(component != NULL, "unexpected NULL record component");
1889-
oop component_oop = java_lang_reflect_RecordComponent::create(ik, component, CHECK_NULL);
1890-
components_h->obj_at_put(x, component_oop);
1891-
}
1892-
return (jobjectArray)JNIHandles::make_local(THREAD, components_h());
1882+
Array<RecordComponent*>* components = ik->record_components();
1883+
if (components != NULL) {
1884+
JvmtiVMObjectAllocEventCollector oam;
1885+
constantPoolHandle cp(THREAD, ik->constants());
1886+
int length = components->length();
1887+
assert(length >= 0, "unexpected record_components length");
1888+
objArrayOop record_components =
1889+
oopFactory::new_objArray(SystemDictionary::RecordComponent_klass(), length, CHECK_NULL);
1890+
objArrayHandle components_h (THREAD, record_components);
1891+
1892+
for (int x = 0; x < length; x++) {
1893+
RecordComponent* component = components->at(x);
1894+
assert(component != NULL, "unexpected NULL record component");
1895+
oop component_oop = java_lang_reflect_RecordComponent::create(ik, component, CHECK_NULL);
1896+
components_h->obj_at_put(x, component_oop);
18931897
}
1898+
return (jobjectArray)JNIHandles::make_local(THREAD, components_h());
18941899
}
18951900

1896-
// Return empty array if ofClass is not a record.
1897-
objArrayOop result = oopFactory::new_objArray(SystemDictionary::RecordComponent_klass(), 0, CHECK_NULL);
1898-
return (jobjectArray)JNIHandles::make_local(THREAD, result);
1901+
return NULL;
18991902
}
19001903
JVM_END
19011904

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

+13-7
Original file line numberDiff line numberDiff line change
@@ -2383,11 +2383,7 @@ public RecordComponent[] getRecordComponents() {
23832383
if (!isRecord()) {
23842384
return null;
23852385
}
2386-
RecordComponent[] recordComponents = getRecordComponents0();
2387-
if (recordComponents == null) {
2388-
return new RecordComponent[0];
2389-
}
2390-
return recordComponents;
2386+
return getRecordComponents0();
23912387
}
23922388

23932389
/**
@@ -3577,9 +3573,17 @@ private static <U> Constructor<U>[] copyConstructors(Constructor<U>[] arg) {
35773573
private native Field[] getDeclaredFields0(boolean publicOnly);
35783574
private native Method[] getDeclaredMethods0(boolean publicOnly);
35793575
private native Constructor<T>[] getDeclaredConstructors0(boolean publicOnly);
3580-
private native Class<?>[] getDeclaredClasses0();
3576+
private native Class<?>[] getDeclaredClasses0();
3577+
3578+
/*
3579+
* Returns an array containing the components of the Record attribute,
3580+
* or null if the attribute is not present.
3581+
*
3582+
* Note that this method returns non-null array on a class with
3583+
* the Record attribute even if this class is not a record.
3584+
*/
35813585
private native RecordComponent[] getRecordComponents0();
3582-
private native boolean isRecord0();
3586+
private native boolean isRecord0();
35833587

35843588
/**
35853589
* Helper method to get the method name from arguments.
@@ -3706,6 +3710,8 @@ public boolean isEnum() {
37063710
* @since 16
37073711
*/
37083712
public boolean isRecord() {
3713+
// this superclass and final modifier check is not strictly necessary
3714+
// they are intrinsified and serve as a fast-path check
37093715
return getSuperclass() == java.lang.Record.class &&
37103716
(this.getModifiers() & Modifier.FINAL) != 0 &&
37113717
isRecord0();

0 commit comments

Comments
 (0)
Please sign in to comment.