Skip to content

Commit ab748dd

Browse files
author
Srikanth Adayapalam
committedJan 19, 2022
8280164: [lworld] Generate Preload attribute to enumerate value classes encountered
1 parent 01bd664 commit ab748dd

File tree

11 files changed

+267
-6
lines changed

11 files changed

+267
-6
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/ClassWriter.java

+18-2
Original file line numberDiff line numberDiff line change
@@ -846,6 +846,17 @@ void writeInnerClasses() {
846846
endAttr(alenIdx);
847847
}
848848

849+
/** Write out "Preload" attribute by enumerating the value classes encountered during this compilation.
850+
*/
851+
void writeValueClasses() {
852+
int alenIdx = writeAttr(names.Preload);
853+
databuf.appendChar(poolWriter.valueClasses.size());
854+
for (ClassSymbol c : poolWriter.valueClasses) {
855+
databuf.appendChar(poolWriter.putClass(c));
856+
}
857+
endAttr(alenIdx);
858+
}
859+
849860
int writeRecordAttribute(ClassSymbol csym) {
850861
int alenIdx = writeAttr(names.Record);
851862
Scope s = csym.members();
@@ -1576,14 +1587,14 @@ public void writeClassFile(OutputStream out, ClassSymbol c)
15761587
case VAR: fieldsCount++; break;
15771588
case MTH: if ((sym.flags() & HYPOTHETICAL) == 0) methodsCount++;
15781589
break;
1579-
case TYP: poolWriter.enterInner((ClassSymbol)sym); break;
1590+
case TYP: poolWriter.enterInnerAndValueClass((ClassSymbol)sym); break;
15801591
default : Assert.error();
15811592
}
15821593
}
15831594

15841595
if (c.trans_local != null) {
15851596
for (ClassSymbol local : c.trans_local) {
1586-
poolWriter.enterInner(local);
1597+
poolWriter.enterInnerAndValueClass(local);
15871598
}
15881599
}
15891600

@@ -1674,6 +1685,11 @@ public void writeClassFile(OutputStream out, ClassSymbol c)
16741685
acount++;
16751686
}
16761687

1688+
if (!poolWriter.valueClasses.isEmpty()) {
1689+
writeValueClasses();
1690+
acount++;
1691+
}
1692+
16771693
endAttrs(acountIdx, acount);
16781694

16791695
out.write(poolbuf.elems, 0, poolbuf.length);

‎src/jdk.compiler/share/classes/com/sun/tools/javac/jvm/PoolWriter.java

+30-4
Original file line numberDiff line numberDiff line change
@@ -51,9 +51,11 @@
5151
import java.io.OutputStream;
5252
import java.util.ArrayDeque;
5353
import java.util.HashMap;
54+
import java.util.HashSet;
5455
import java.util.LinkedHashMap;
5556
import java.util.LinkedHashSet;
5657
import java.util.Map;
58+
import java.util.Set;
5759

5860
import static com.sun.tools.javac.code.Kinds.Kind.TYP;
5961
import static com.sun.tools.javac.code.TypeTag.ARRAY;
@@ -94,6 +96,8 @@ public class PoolWriter {
9496
/** The inner classes to be written, as an ordered set (enclosing first). */
9597
LinkedHashSet<ClassSymbol> innerClasses = new LinkedHashSet<>();
9698

99+
Set<ClassSymbol> valueClasses = new HashSet<>();
100+
97101
/** The list of entries in the BootstrapMethods attribute. */
98102
Map<BsmKey, Integer> bootstrapMethods = new LinkedHashMap<>();
99103

@@ -235,17 +239,38 @@ int putModule(ModuleSymbol mod) {
235239
/**
236240
* Enter an inner class into the `innerClasses' set.
237241
*/
238-
void enterInner(ClassSymbol c) {
242+
void enterInnerClass(ClassSymbol c) {
239243
if (c.type.isCompound()) {
240244
throw new AssertionError("Unexpected intersection type: " + c.type);
241245
}
242246
c.complete();
243247
if (c.owner.enclClass() != null && !innerClasses.contains(c)) {
244-
enterInner(c.owner.enclClass());
248+
enterInnerClass(c.owner.enclClass());
245249
innerClasses.add(c);
246250
}
247251
}
248252

253+
/** Enter a value class into the `valueClasses' set.
254+
*/
255+
void enterValueClass(ClassSymbol c) {
256+
if (c.type.isCompound()) {
257+
throw new AssertionError("Unexpected intersection type: " + c.type);
258+
}
259+
c.complete();
260+
if (c.isValueClass() && !c.isPrimitiveClass()) {
261+
valueClasses.add(c);
262+
}
263+
if (c.owner.enclClass() != null) {
264+
enterValueClass(c.owner.enclClass());
265+
}
266+
}
267+
268+
269+
void enterInnerAndValueClass(ClassSymbol c) {
270+
enterInnerClass(c);
271+
enterValueClass(c);
272+
}
273+
249274
/**
250275
* Create a new Utf8 entry representing a descriptor for given (member) symbol.
251276
*/
@@ -331,7 +356,7 @@ protected void append(Name name) {
331356

332357
@Override
333358
protected void classReference(ClassSymbol c) {
334-
enterInner(c);
359+
enterInnerAndValueClass(c);
335360
}
336361

337362
protected void reset() {
@@ -384,7 +409,7 @@ void writeConstant(PoolConstant c) {
384409
poolbuf.appendByte(tag);
385410
poolbuf.appendChar(putName(name));
386411
if (ct.hasTag(CLASS)) {
387-
enterInner((ClassSymbol)ct.tsym);
412+
enterInnerAndValueClass((ClassSymbol)ct.tsym);
388413
}
389414
break;
390415
}
@@ -523,6 +548,7 @@ private Name classSig(Type t) {
523548

524549
void reset() {
525550
innerClasses.clear();
551+
valueClasses.clear();
526552
bootstrapMethods.clear();
527553
pool.reset();
528554
}

‎src/jdk.compiler/share/classes/com/sun/tools/javac/util/Names.java

+2
Original file line numberDiff line numberDiff line change
@@ -155,6 +155,7 @@ public static Names instance(Context context) {
155155
public final Name ModuleResolution;
156156
public final Name NestHost;
157157
public final Name NestMembers;
158+
public final Name Preload;
158159
public final Name Record;
159160
public final Name RuntimeInvisibleAnnotations;
160161
public final Name RuntimeInvisibleParameterAnnotations;
@@ -350,6 +351,7 @@ public Names(Context context) {
350351
ModuleResolution = fromString("ModuleResolution");
351352
NestHost = fromString("NestHost");
352353
NestMembers = fromString("NestMembers");
354+
Preload = fromString("Preload");
353355
Record = fromString("Record");
354356
RuntimeInvisibleAnnotations = fromString("RuntimeInvisibleAnnotations");
355357
RuntimeInvisibleParameterAnnotations = fromString("RuntimeInvisibleParameterAnnotations");

‎src/jdk.jdeps/share/classes/com/sun/tools/classfile/Attribute.java

+3
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,7 @@ public abstract class Attribute {
6060
public static final String ModuleTarget = "ModuleTarget";
6161
public static final String NestHost = "NestHost";
6262
public static final String NestMembers = "NestMembers";
63+
public static final String Preload = "Preload";
6364
public static final String Record = "Record";
6465
public static final String RuntimeVisibleAnnotations = "RuntimeVisibleAnnotations";
6566
public static final String RuntimeInvisibleAnnotations = "RuntimeInvisibleAnnotations";
@@ -136,6 +137,7 @@ protected void init() {
136137
standardAttributes.put(ModuleTarget, ModuleTarget_attribute.class);
137138
standardAttributes.put(NestHost, NestHost_attribute.class);
138139
standardAttributes.put(NestMembers, NestMembers_attribute.class);
140+
standardAttributes.put(Preload, Preload_attribute.class);
139141
standardAttributes.put(Record, Record_attribute.class);
140142
standardAttributes.put(RuntimeInvisibleAnnotations, RuntimeInvisibleAnnotations_attribute.class);
141143
standardAttributes.put(RuntimeInvisibleParameterAnnotations, RuntimeInvisibleParameterAnnotations_attribute.class);
@@ -203,6 +205,7 @@ public interface Visitor<R,P> {
203205
R visitModuleTarget(ModuleTarget_attribute attr, P p);
204206
R visitNestHost(NestHost_attribute attr, P p);
205207
R visitNestMembers(NestMembers_attribute attr, P p);
208+
R visitPreload(Preload_attribute attr, P p);
206209
R visitRecord(Record_attribute attr, P p);
207210
R visitRuntimeVisibleAnnotations(RuntimeVisibleAnnotations_attribute attr, P p);
208211
R visitRuntimeInvisibleAnnotations(RuntimeInvisibleAnnotations_attribute attr, P p);

‎src/jdk.jdeps/share/classes/com/sun/tools/classfile/ClassWriter.java

+8
Original file line numberDiff line numberDiff line change
@@ -754,6 +754,14 @@ public Void visitSynthetic(Synthetic_attribute attr, ClassOutputStream out) {
754754
return null;
755755
}
756756

757+
@Override
758+
public Void visitPreload(Preload_attribute attr, ClassOutputStream out) {
759+
out.writeShort(attr.value_class_info_index.length);
760+
for (int index: attr.value_class_info_index)
761+
out.writeShort(index);
762+
return null;
763+
}
764+
757765
protected void writeAccessFlags(AccessFlags flags, ClassOutputStream out) {
758766
out.writeShort(flags.flags);
759767
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation. Oracle designates this
8+
* particular file as subject to the "Classpath" exception as provided
9+
* by Oracle in the LICENSE file that accompanied this code.
10+
*
11+
* This code is distributed in the hope that it will be useful, but WITHOUT
12+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14+
* version 2 for more details (a copy is included in the LICENSE file that
15+
* accompanied this code).
16+
*
17+
* You should have received a copy of the GNU General Public License version
18+
* 2 along with this work; if not, write to the Free Software Foundation,
19+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20+
*
21+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22+
* or visit www.oracle.com if you need additional information or have any
23+
* questions.
24+
*/
25+
26+
package com.sun.tools.classfile;
27+
28+
import java.io.IOException;
29+
30+
/**
31+
* <p><b>This is NOT part of any supported API.
32+
* If you write code that depends on this, you do so at your own risk.
33+
* This code and its internal interfaces are subject to change or
34+
* deletion without notice.</b>
35+
*/
36+
public class Preload_attribute extends Attribute {
37+
Preload_attribute(ClassReader cr, int name_index, int length) throws IOException {
38+
super(name_index, length);
39+
number_of_classes = cr.readUnsignedShort();
40+
value_class_info_index = new int[number_of_classes];
41+
for (int i = 0; i < number_of_classes; i++)
42+
value_class_info_index[i] = cr.readUnsignedShort();
43+
}
44+
45+
public <R, D> R accept(Visitor<R, D> visitor, D data) {
46+
return visitor.visitPreload(this, data);
47+
}
48+
49+
public final int number_of_classes;
50+
public final int value_class_info_index[];
51+
}

‎src/jdk.jdeps/share/classes/com/sun/tools/javap/AttributeWriter.java

+22
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,7 @@
8080

8181
import static com.sun.tools.classfile.AccessFlags.*;
8282

83+
import com.sun.tools.classfile.Preload_attribute;
8384
import com.sun.tools.javac.util.Assert;
8485
import com.sun.tools.javac.util.StringUtils;
8586

@@ -1112,6 +1113,27 @@ public Void visitSynthetic(Synthetic_attribute attr, Void ignore) {
11121113
return null;
11131114
}
11141115

1116+
@Override
1117+
public Void visitPreload(Preload_attribute attr, Void ignore) {
1118+
boolean first = true;
1119+
for (int index : attr.value_class_info_index) {
1120+
if (first) {
1121+
println("Classes to be preloaded:");
1122+
indent(+1);
1123+
first = false;
1124+
}
1125+
print("#" + index);
1126+
print(";");
1127+
tab();
1128+
print("// value ");
1129+
constantWriter.write(index);
1130+
println();
1131+
}
1132+
if (!first)
1133+
indent(-1);
1134+
return null;
1135+
}
1136+
11151137
static String getJavaName(String name) {
11161138
return name.replace('/', '.');
11171139
}

‎test/langtools/lib/annotations/annotations/classfile/ClassfileInspector.java

+5
Original file line numberDiff line numberDiff line change
@@ -1371,6 +1371,11 @@ public Void visitPermittedSubclasses(PermittedSubclasses_attribute attr, T p) {
13711371
public Void visitRecord(Record_attribute attr, T p) {
13721372
return null;
13731373
}
1374+
1375+
@Override
1376+
public Void visitPreload(Preload_attribute attr, T p) {
1377+
return null;
1378+
}
13741379
}
13751380

13761381
private static final Attribute.Visitor<Void, ExpectedTypeAnnotation> typeAnnoMatcher

‎test/langtools/tools/javac/MethodParameters/AttributeVisitor.java

+1
Original file line numberDiff line numberDiff line change
@@ -66,4 +66,5 @@ class AttributeVisitor<R, P> implements Attribute.Visitor<R, P> {
6666
public R visitSynthetic(Synthetic_attribute attr, P p) { return null; }
6767
public R visitPermittedSubclasses(PermittedSubclasses_attribute attr, P p) { return null; }
6868
public R visitRecord(Record_attribute attr, P p) { return null; }
69+
public R visitPreload(Preload_attribute attr, P p) { return null; }
6970
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
final value class V1 {}
25+
final value class V2 {}
26+
final value class V3 {}
27+
final value class V4 {}
28+
final value class V5 {}
29+
final value class V6 {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test
26+
* @bug 8280164
27+
* @summary Check emission of Preload attribute
28+
* @modules jdk.jdeps/com.sun.tools.classfile
29+
* @compile MultiValues.java
30+
* @compile -g PreloadAttributeTest.java
31+
* @run main PreloadAttributeTest
32+
*/
33+
34+
import com.sun.tools.classfile.*;
35+
import com.sun.tools.classfile.ConstantPool.CONSTANT_Class_info;
36+
37+
public class PreloadAttributeTest {
38+
39+
static final value class X {
40+
final V1 [] v1 = null; // field descriptor
41+
V2[] foo() { // method descriptor encoding value type
42+
return null;
43+
}
44+
void foo(V3 v3) { // method descriptor encoding value type
45+
}
46+
void foo(int x) {
47+
V4 [] v4 = null; // local variable.
48+
}
49+
void goo() {
50+
V5 [] v5 = null;
51+
if (v5 == null) {
52+
// ...
53+
} else {
54+
V5 [] v52 = null;
55+
}
56+
}
57+
}
58+
59+
public static void main(String[] args) throws Exception {
60+
ClassFile cls = ClassFile.read(PreloadAttributeTest.class.getResourceAsStream("PreloadAttributeTest$X.class"));
61+
62+
if (cls == null) {
63+
throw new AssertionError("Could not locate the class files");
64+
}
65+
66+
/* Check emission of Preload attribute */
67+
Preload_attribute preloads = (Preload_attribute) cls.attributes.get(Attribute.Preload);
68+
if (preloads == null) {
69+
throw new AssertionError("Missing Preload attribute!");
70+
}
71+
if (preloads.number_of_classes != 6) {
72+
throw new AssertionError("Incorrect number of Preload classes");
73+
}
74+
75+
int mask = 0x3F;
76+
for (int i = 0; i < preloads.number_of_classes; i++) {
77+
CONSTANT_Class_info clsInfo = cls.constant_pool.getClassInfo(
78+
preloads.value_class_info_index[i]);
79+
switch (clsInfo.getName()) {
80+
case "V1":
81+
mask &= ~1; break;
82+
case "V2":
83+
mask &= ~2; break;
84+
case "V3":
85+
mask &= ~4; break;
86+
case "V4":
87+
mask &= ~8; break;
88+
case "V5":
89+
mask &= ~16; break;
90+
case "PreloadAttributeTest$X":
91+
mask &= ~32; break;
92+
}
93+
}
94+
if (mask != 0) {
95+
throw new AssertionError("Some Preload class entries are missing!");
96+
}
97+
}
98+
}

0 commit comments

Comments
 (0)
Please sign in to comment.