8288130: compiler error with AP and explicit record accessor

Reviewed-by: jlahoda
This commit is contained in:
Vicente Romero 2022-06-24 21:42:23 +00:00
parent 08288819dd
commit 53b37fe153
5 changed files with 225 additions and 180 deletions

View File

@ -1503,30 +1503,25 @@ public abstract class Symbol extends AnnoConstruct implements PoolConstant, Elem
return null; return null;
} }
public RecordComponent getRecordComponent(JCVariableDecl var, boolean addIfMissing, List<JCAnnotation> annotations) { /* creates a record component if non is related to the given variable and recreates a brand new one
* in other case
*/
public RecordComponent createRecordComponent(JCVariableDecl var, List<JCAnnotation> annotations) {
RecordComponent toRemove = null; RecordComponent toRemove = null;
for (RecordComponent rc : recordComponents) { for (RecordComponent rc : recordComponents) {
/* it could be that a record erroneously declares two record components with the same name, in that /* it could be that a record erroneously declares two record components with the same name, in that
* case we need to use the position to disambiguate * case we need to use the position to disambiguate
*/ */
if (rc.name == var.name && var.pos == rc.pos) { if (rc.name == var.name && var.pos == rc.pos) {
if (rc.type.hasTag(TypeTag.ERROR) && !var.sym.type.hasTag(TypeTag.ERROR)) {
// Found a record component with an erroneous type: save it so that it can be removed later.
// If the class type of the record component is generated by annotation processor, it should
// use the new actual class type and symbol instead of the old dummy ErrorType.
toRemove = rc; toRemove = rc;
} else {
// Found a good record component: just return.
return rc;
}
} }
} }
RecordComponent rc = null; RecordComponent rc = null;
if (toRemove != null) { if (toRemove != null) {
// Found a record component with an erroneous type: remove it and create a new one // Found a record component with an erroneous type: remove it and create a new one
recordComponents = List.filter(recordComponents, toRemove); recordComponents = List.filter(recordComponents, toRemove);
recordComponents = recordComponents.append(rc = new RecordComponent(var.sym, annotations)); recordComponents = recordComponents.append(rc = new RecordComponent(var.sym, toRemove.originalAnnos, toRemove.isVarargs));
} else if (addIfMissing) { } else {
// Didn't find the record component: create one. // Didn't find the record component: create one.
recordComponents = recordComponents.append(rc = new RecordComponent(var.sym, annotations)); recordComponents = recordComponents.append(rc = new RecordComponent(var.sym, annotations));
} }
@ -1809,6 +1804,10 @@ public abstract class Symbol extends AnnoConstruct implements PoolConstant, Elem
} }
public RecordComponent(VarSymbol field, List<JCAnnotation> annotations) { public RecordComponent(VarSymbol field, List<JCAnnotation> annotations) {
this(field, annotations, field.type.hasTag(TypeTag.ARRAY) && ((ArrayType)field.type).isVarargs());
}
public RecordComponent(VarSymbol field, List<JCAnnotation> annotations, boolean isVarargs) {
super(PUBLIC, field.name, field.type, field.owner); super(PUBLIC, field.name, field.type, field.owner);
this.originalAnnos = annotations; this.originalAnnos = annotations;
this.pos = field.pos; this.pos = field.pos;
@ -1817,7 +1816,7 @@ public abstract class Symbol extends AnnoConstruct implements PoolConstant, Elem
* the symbol will be blown out and we won't be able to know if the original * the symbol will be blown out and we won't be able to know if the original
* record component was declared varargs or not. * record component was declared varargs or not.
*/ */
this.isVarargs = type.hasTag(TypeTag.ARRAY) && ((ArrayType)type).isVarargs(); this.isVarargs = isVarargs;
} }
public List<JCAnnotation> getOriginalAnnos() { return originalAnnos; } public List<JCAnnotation> getOriginalAnnos() { return originalAnnos; }

View File

@ -2960,6 +2960,9 @@ public class Check {
/** Check an annotation of a symbol. /** Check an annotation of a symbol.
*/ */
private void validateAnnotation(JCAnnotation a, JCTree declarationTree, Symbol s) { private void validateAnnotation(JCAnnotation a, JCTree declarationTree, Symbol s) {
/** NOTE: if annotation processors are present, annotation processing rounds can happen after this method,
* this can impact in particular records for which annotations are forcibly propagated.
*/
validateAnnotationTree(a); validateAnnotationTree(a);
boolean isRecordMember = ((s.flags_field & RECORD) != 0 || s.enclClass() != null && s.enclClass().isRecord()); boolean isRecordMember = ((s.flags_field & RECORD) != 0 || s.enclClass() != null && s.enclClass().isRecord());

View File

@ -983,7 +983,7 @@ public class TypeEnter implements Completer {
List<JCVariableDecl> fields = TreeInfo.recordFields(tree); List<JCVariableDecl> fields = TreeInfo.recordFields(tree);
memberEnter.memberEnter(fields, env); memberEnter.memberEnter(fields, env);
for (JCVariableDecl field : fields) { for (JCVariableDecl field : fields) {
sym.getRecordComponent(field, true, sym.createRecordComponent(field,
field.mods.annotations.isEmpty() ? field.mods.annotations.isEmpty() ?
List.nil() : List.nil() :
new TreeCopier<JCTree>(make.at(field.pos)).copy(field.mods.annotations)); new TreeCopier<JCTree>(make.at(field.pos)).copy(field.mods.annotations));
@ -1229,10 +1229,37 @@ public class TypeEnter implements Completer {
memberEnter.memberEnter(equals, env); memberEnter.memberEnter(equals, env);
} }
// fields can't be varargs, lets remove the flag /** Some notes regarding the code below. Annotations applied to elements of a record header are propagated
* to other elements which, when applicable, not explicitly declared by the user: the canonical constructor,
* accessors, fields and record components. Of all these the only ones that can't be explicitly declared are
* the fields and the record components.
*
* Now given that annotations are propagated to all possible targets regardless of applicability,
* annotations not applicable to a given element should be removed. See Check::validateAnnotation. Once
* annotations are removed we could lose the whole picture, that's why original annotations are stored in
* the record component, see RecordComponent::originalAnnos, but there is no real AST representing a record
* component so if there is an annotation processing round it could be that we need to reenter a record for
* which we need to re-attribute its annotations. This is why one of the things the code below is doing is
* copying the original annotations from the record component to the corresponding field, again this applies
* only if APs are present.
*
* We need to copy the annotations to the field so that annotations applicable only to the record component
* can be attributed as if declared in the field and then stored in the metadata associated to the record
* component.
*/
List<JCVariableDecl> recordFields = TreeInfo.recordFields(tree); List<JCVariableDecl> recordFields = TreeInfo.recordFields(tree);
for (JCVariableDecl field: recordFields) { for (JCVariableDecl field: recordFields) {
RecordComponent rec = tree.sym.getRecordComponent(field.sym);
TreeCopier<JCTree> tc = new TreeCopier<>(make.at(field.pos));
List<JCAnnotation> originalAnnos = tc.copy(rec.getOriginalAnnos());
field.mods.flags &= ~Flags.VARARGS; field.mods.flags &= ~Flags.VARARGS;
if (originalAnnos.length() != field.mods.annotations.length()) {
field.mods.annotations = originalAnnos;
annotate.annotateLater(originalAnnos, env, field.sym, field.pos());
}
// also here
field.sym.flags_field &= ~Flags.VARARGS; field.sym.flags_field &= ~Flags.VARARGS;
} }
// now lets add the accessors // now lets add the accessors

View File

@ -65,8 +65,8 @@ public class TreeCopier<P> implements TreeVisitor<JCTree,P> {
} }
public <T extends JCTree> List<T> copy(List<T> trees, P p) { public <T extends JCTree> List<T> copy(List<T> trees, P p) {
if (trees == null) if (trees == null || trees.isEmpty())
return null; return trees;
ListBuffer<T> lb = new ListBuffer<>(); ListBuffer<T> lb = new ListBuffer<>();
for (T tree: trees) for (T tree: trees)
lb.append(copy(tree, p)); lb.append(copy(tree, p));

View File

@ -25,7 +25,7 @@
* RecordCompilationTests * RecordCompilationTests
* *
* @test * @test
* @bug 8250629 8252307 8247352 8241151 8246774 8259025 * @bug 8250629 8252307 8247352 8241151 8246774 8259025 8288130
* @summary Negative compilation tests, and positive compilation (smoke) tests for records * @summary Negative compilation tests, and positive compilation (smoke) tests for records
* @library /lib/combo /tools/lib /tools/javac/lib * @library /lib/combo /tools/lib /tools/javac/lib
* @modules * @modules
@ -177,6 +177,23 @@ public class RecordCompilationTests extends CompilationTestCase {
assertOK("@Deprecated record R(int x, int y) { }"); assertOK("@Deprecated record R(int x, int y) { }");
assertOK("record R(@Deprecated int x, int y) { }"); assertOK("record R(@Deprecated int x, int y) { }");
assertOK("record R<T>(T x, T y) { }"); assertOK("record R<T>(T x, T y) { }");
assertOK(
"""
record R<T>(T x) {
public T x() {
return this.x;
}
}
""");
assertOK(
"""
import java.util.List;
record R<T>(List<T> x) {
public List<T> x() {
return this.x;
}
}
""");
} }
public void testGoodMemberDeclarations() { public void testGoodMemberDeclarations() {
@ -1261,9 +1278,8 @@ public class RecordCompilationTests extends CompilationTestCase {
Assert.check(numberOfFieldRefs == 1); Assert.check(numberOfFieldRefs == 1);
} }
/* check that fields are initialized in a canonical constructor in the same declaration order as the corresponding // check that fields are initialized in a canonical constructor in the same declaration order as the corresponding
* record component // record component
*/
public void testCheckInitializationOrderInCompactConstructor() throws Exception { public void testCheckInitializationOrderInCompactConstructor() throws Exception {
int putField1 = -1; int putField1 = -1;
int putField2 = -1; int putField2 = -1;
@ -1304,7 +1320,8 @@ public class RecordCompilationTests extends CompilationTestCase {
public void testAcceptRecordId() { public void testAcceptRecordId() {
String[] previousOptions = getCompileOptions(); String[] previousOptions = getCompileOptions();
String[] testOptions = {/* no options */}; try {
String[] testOptions = {};
setCompileOptions(testOptions); setCompileOptions(testOptions);
assertFail("compiler.err.illegal.start.of.type", assertFail("compiler.err.illegal.start.of.type",
"class R {\n" + "class R {\n" +
@ -1313,18 +1330,20 @@ public class RecordCompilationTests extends CompilationTestCase {
" }\n" + " }\n" +
" class record {}\n" + " class record {}\n" +
"}"); "}");
} finally {
setCompileOptions(previousOptions); setCompileOptions(previousOptions);
} }
}
public void testAnnos() throws Exception { public void testAnnos() throws Exception {
String[] previousOptions = getCompileOptions(); String[] previousOptions = getCompileOptions();
try {
String srcTemplate = String srcTemplate =
""" """
import java.lang.annotation.*; import java.lang.annotation.*;
@Target({#TARGET}) @Target({#TARGET})
@Retention(RetentionPolicy.RUNTIME) @Retention(RetentionPolicy.RUNTIME)
@interface Anno { } @interface Anno { }
record R(@Anno String s) {} record R(@Anno String s) {}
"""; """;
@ -1375,9 +1394,8 @@ public class RecordCompilationTests extends CompilationTestCase {
// field first // field first
Assert.check(classFile.fields.length == 1); Assert.check(classFile.fields.length == 1);
Field field = classFile.fields[0]; Field field = classFile.fields[0];
/* if FIELD is one of the targets then there must be a declaration annotation applied to the field, apart from // if FIELD is one of the targets then there must be a declaration annotation applied to the field, apart from
* the type annotation // the type annotation
*/
if (target.contains("ElementType.FIELD")) { if (target.contains("ElementType.FIELD")) {
checkAnno(classFile, checkAnno(classFile,
(RuntimeAnnotations_attribute) findAttributeOrFail( (RuntimeAnnotations_attribute) findAttributeOrFail(
@ -1401,9 +1419,8 @@ public class RecordCompilationTests extends CompilationTestCase {
// checking for the annotation on the corresponding parameter of the canonical constructor // checking for the annotation on the corresponding parameter of the canonical constructor
Method init = findMethodOrFail(classFile, "<init>"); Method init = findMethodOrFail(classFile, "<init>");
/* if PARAMETER is one of the targets then there must be a declaration annotation applied to the parameter, apart from // if PARAMETER is one of the targets then there must be a declaration annotation applied to the parameter, apart from
* the type annotation // the type annotation
*/
if (target.contains("ElementType.PARAMETER")) { if (target.contains("ElementType.PARAMETER")) {
checkParameterAnno(classFile, checkParameterAnno(classFile,
(RuntimeVisibleParameterAnnotations_attribute) findAttributeOrFail( (RuntimeVisibleParameterAnnotations_attribute) findAttributeOrFail(
@ -1425,9 +1442,8 @@ public class RecordCompilationTests extends CompilationTestCase {
// checking for the annotation in the accessor // checking for the annotation in the accessor
Method accessor = findMethodOrFail(classFile, "s"); Method accessor = findMethodOrFail(classFile, "s");
/* if METHOD is one of the targets then there must be a declaration annotation applied to the accessor, apart from // if METHOD is one of the targets then there must be a declaration annotation applied to the accessor, apart from
* the type annotation // the type annotation
*/
if (target.contains("ElementType.METHOD")) { if (target.contains("ElementType.METHOD")) {
checkAnno(classFile, checkAnno(classFile,
(RuntimeAnnotations_attribute) findAttributeOrFail( (RuntimeAnnotations_attribute) findAttributeOrFail(
@ -1450,9 +1466,8 @@ public class RecordCompilationTests extends CompilationTestCase {
// checking for the annotation in the Record attribute // checking for the annotation in the Record attribute
Record_attribute record = (Record_attribute) findAttributeOrFail(classFile.attributes, Record_attribute.class); Record_attribute record = (Record_attribute) findAttributeOrFail(classFile.attributes, Record_attribute.class);
Assert.check(record.component_count == 1); Assert.check(record.component_count == 1);
/* if RECORD_COMPONENT is one of the targets then there must be a declaration annotation applied to the // if RECORD_COMPONENT is one of the targets then there must be a declaration annotation applied to the
* field, apart from the type annotation // field, apart from the type annotation
*/
if (target.contains("ElementType.RECORD_COMPONENT")) { if (target.contains("ElementType.RECORD_COMPONENT")) {
checkAnno(classFile, checkAnno(classFile,
(RuntimeAnnotations_attribute) findAttributeOrFail( (RuntimeAnnotations_attribute) findAttributeOrFail(
@ -1476,8 +1491,10 @@ public class RecordCompilationTests extends CompilationTestCase {
} }
// let's reset the default compiler options for other tests // let's reset the default compiler options for other tests
} finally {
setCompileOptions(previousOptions); setCompileOptions(previousOptions);
} }
}
private void checkTypeAnno(ClassFile classFile, private void checkTypeAnno(ClassFile classFile,
RuntimeTypeAnnotations_attribute rtAnnos, RuntimeTypeAnnotations_attribute rtAnnos,
@ -1655,7 +1672,7 @@ public class RecordCompilationTests extends CompilationTestCase {
switch (method.getName(classFile.constant_pool)) { switch (method.getName(classFile.constant_pool)) {
case "toString", "equals", "hashCode" -> case "toString", "equals", "hashCode" ->
Assert.check(method.access_flags.is(AccessFlags.ACC_PUBLIC) && method.access_flags.is(AccessFlags.ACC_FINAL)); Assert.check(method.access_flags.is(AccessFlags.ACC_PUBLIC) && method.access_flags.is(AccessFlags.ACC_FINAL));
default -> { /* do nothing */ } default -> {}
} }
} }
} }
@ -1963,9 +1980,8 @@ public class RecordCompilationTests extends CompilationTestCase {
public void testNoWarningForSerializableRecords() { public void testNoWarningForSerializableRecords() {
if (!useAP) { if (!useAP) {
/* dont execute this test when the default annotation processor is on as it will fail due to // dont execute this test when the default annotation processor is on as it will fail due to
* spurious warnings // spurious warnings
*/
appendCompileOptions("-Werror", "-Xlint:serial"); appendCompileOptions("-Werror", "-Xlint:serial");
assertOK( assertOK(
""" """