Skip to content

Commit c60474b

Browse files
liachasotona
authored andcommittedMay 3, 2024
8323707: Adjust Classfile API's type arg model to better represent the embodied type
Reviewed-by: asotona
1 parent f78fa05 commit c60474b

File tree

6 files changed

+105
-83
lines changed

6 files changed

+105
-83
lines changed
 

‎src/java.base/share/classes/java/lang/classfile/Signature.java

+57-38
Original file line numberDiff line numberDiff line change
@@ -185,88 +185,107 @@ public static ClassTypeSig of(ClassTypeSig outerType, String className, TypeArg.
185185
/**
186186
* Models the type argument.
187187
*
188+
* @sealedGraph
188189
* @since 22
189190
*/
190191
@PreviewFeature(feature = PreviewFeature.Feature.CLASSFILE_API)
191-
public sealed interface TypeArg
192-
permits SignaturesImpl.TypeArgImpl {
192+
public sealed interface TypeArg {
193193

194194
/**
195-
* Indicator for whether a wildcard has default bound, no bound,
196-
* an upper bound, or a lower bound
197-
*
198-
* @since 22
195+
* Models an unbounded type argument {@code *}.
196+
* @since 23
199197
*/
200198
@PreviewFeature(feature = PreviewFeature.Feature.CLASSFILE_API)
201-
public enum WildcardIndicator {
202-
203-
/**
204-
* default bound wildcard (empty)
205-
*/
206-
DEFAULT,
207-
208-
/**
209-
* unbounded indicator {@code *}
210-
*/
211-
UNBOUNDED,
199+
public sealed interface Unbounded extends TypeArg permits SignaturesImpl.UnboundedTypeArgImpl {
200+
}
212201

213-
/**
214-
* upper-bounded indicator {@code +}
215-
*/
216-
EXTENDS,
202+
/**
203+
* Models a type argument with an explicit bound type.
204+
* @since 23
205+
*/
206+
@PreviewFeature(feature = PreviewFeature.Feature.CLASSFILE_API)
207+
public sealed interface Bounded extends TypeArg permits SignaturesImpl.TypeArgImpl {
217208

218209
/**
219-
* lower-bounded indicator {@code -}
210+
* Models a type argument's wildcard indicator.
211+
* @since 23
220212
*/
221-
SUPER;
213+
@PreviewFeature(feature = PreviewFeature.Feature.CLASSFILE_API)
214+
public enum WildcardIndicator {
215+
216+
/**
217+
* No wildcard (empty), an exact type. Also known as
218+
* {@index invariant}.
219+
*/
220+
NONE,
221+
222+
/**
223+
* Upper-bound indicator {@code +}. Also known as
224+
* {@index covariant}.
225+
*/
226+
EXTENDS,
227+
228+
/**
229+
* Lower-bound indicator {@code -}. Also known as
230+
* {@index contravariant}.
231+
*/
232+
SUPER;
233+
}
234+
235+
/** {@return the kind of wildcard} */
236+
WildcardIndicator wildcardIndicator();
237+
238+
/** {@return the signature of the type bound} */
239+
RefTypeSig boundType();
222240
}
223241

224-
/** {@return the wildcard indicator} */
225-
WildcardIndicator wildcardIndicator();
226-
227-
/** {@return the signature of the type bound, if any} */
228-
Optional<RefTypeSig> boundType();
229-
230242
/**
231243
* {@return a bounded type arg}
232244
* @param boundType the bound
245+
* @since 23
233246
*/
234-
public static TypeArg of(RefTypeSig boundType) {
247+
public static TypeArg.Bounded of(RefTypeSig boundType) {
235248
requireNonNull(boundType);
236-
return of(WildcardIndicator.DEFAULT, Optional.of(boundType));
249+
return bounded(Bounded.WildcardIndicator.NONE, boundType);
237250
}
238251

239252
/**
240253
* {@return an unbounded type arg}
254+
* @since 23
241255
*/
242-
public static TypeArg unbounded() {
243-
return of(WildcardIndicator.UNBOUNDED, Optional.empty());
256+
public static TypeArg.Unbounded unbounded() {
257+
return SignaturesImpl.UnboundedTypeArgImpl.INSTANCE;
244258
}
245259

246260
/**
247261
* {@return an upper-bounded type arg}
248262
* @param boundType the upper bound
263+
* @since 23
249264
*/
250-
public static TypeArg extendsOf(RefTypeSig boundType) {
265+
public static TypeArg.Bounded extendsOf(RefTypeSig boundType) {
251266
requireNonNull(boundType);
252-
return of(WildcardIndicator.EXTENDS, Optional.of(boundType));
267+
return bounded(Bounded.WildcardIndicator.EXTENDS, boundType);
253268
}
254269

255270
/**
256271
* {@return a lower-bounded type arg}
257272
* @param boundType the lower bound
273+
* @since 23
258274
*/
259-
public static TypeArg superOf(RefTypeSig boundType) {
275+
public static TypeArg.Bounded superOf(RefTypeSig boundType) {
260276
requireNonNull(boundType);
261-
return of(WildcardIndicator.SUPER, Optional.of(boundType));
277+
return bounded(Bounded.WildcardIndicator.SUPER, boundType);
262278
}
263279

264280
/**
265281
* {@return a bounded type arg}
266282
* @param wildcard the wild card
267283
* @param boundType optional bound type
284+
* @since 23
268285
*/
269-
public static TypeArg of(WildcardIndicator wildcard, Optional<RefTypeSig> boundType) {
286+
public static TypeArg.Bounded bounded(Bounded.WildcardIndicator wildcard, RefTypeSig boundType) {
287+
requireNonNull(wildcard);
288+
requireNonNull(boundType);
270289
return new SignaturesImpl.TypeArgImpl(wildcard, boundType);
271290
}
272291
}

‎src/java.base/share/classes/jdk/internal/classfile/impl/ClassRemapperImpl.java

+5-5
Original file line numberDiff line numberDiff line change
@@ -369,11 +369,11 @@ <S extends Signature> S mapSignature(S signature) {
369369
Signature.ClassTypeSig.of(
370370
cts.outerType().map(this::mapSignature).orElse(null),
371371
map(cts.classDesc()),
372-
cts.typeArgs().stream()
373-
.map(ta -> Signature.TypeArg.of(
374-
ta.wildcardIndicator(),
375-
ta.boundType().map(this::mapSignature)))
376-
.toArray(Signature.TypeArg[]::new));
372+
cts.typeArgs().stream().map(ta -> switch (ta) {
373+
case Signature.TypeArg.Unbounded u -> u;
374+
case Signature.TypeArg.Bounded bta -> Signature.TypeArg.bounded(
375+
bta.wildcardIndicator(), mapSignature(bta.boundType()));
376+
}).toArray(Signature.TypeArg[]::new));
377377
default -> signature;
378378
};
379379
}

‎src/java.base/share/classes/jdk/internal/classfile/impl/SignaturesImpl.java

+16-11
Original file line numberDiff line numberDiff line change
@@ -280,24 +280,29 @@ public String signatureString() {
280280
if (!typeArgs.isEmpty()) {
281281
var sb = new StringBuilder();
282282
sb.append('<');
283-
for (var ta : typeArgs)
284-
sb.append(((TypeArgImpl)ta).signatureString());
283+
for (var ta : typeArgs) {
284+
switch (ta) {
285+
case TypeArg.Bounded b -> {
286+
switch (b.wildcardIndicator()) {
287+
case SUPER -> sb.append('-');
288+
case EXTENDS -> sb.append('+');
289+
}
290+
sb.append(b.boundType().signatureString());
291+
}
292+
case TypeArg.Unbounded _ -> sb.append('*');
293+
}
294+
}
285295
suffix = sb.append(">;").toString();
286296
}
287297
return prefix + className + suffix;
288298
}
289299
}
290300

291-
public static record TypeArgImpl(WildcardIndicator wildcardIndicator, Optional<RefTypeSig> boundType) implements Signature.TypeArg {
301+
public static enum UnboundedTypeArgImpl implements TypeArg.Unbounded {
302+
INSTANCE;
303+
}
292304

293-
public String signatureString() {
294-
return switch (wildcardIndicator) {
295-
case DEFAULT -> boundType.get().signatureString();
296-
case EXTENDS -> "+" + boundType.get().signatureString();
297-
case SUPER -> "-" + boundType.get().signatureString();
298-
case UNBOUNDED -> "*";
299-
};
300-
}
305+
public static record TypeArgImpl(WildcardIndicator wildcardIndicator, RefTypeSig boundType) implements Signature.TypeArg.Bounded {
301306
}
302307

303308
public static record TypeParamImpl(String identifier, Optional<RefTypeSig> classBound, List<RefTypeSig> interfaceBounds)

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

+14-10
Original file line numberDiff line numberDiff line change
@@ -366,16 +366,20 @@ private void print(StringBuilder sb, Signature sig) {
366366
}
367367

368368
private void print(StringBuilder sb, Signature.TypeArg ta) {
369-
switch (ta.wildcardIndicator()) {
370-
case DEFAULT -> print(sb, ta.boundType().get());
371-
case UNBOUNDED -> sb.append('?');
372-
case EXTENDS -> {
373-
sb.append("? extends ");
374-
print(sb, ta.boundType().get());
375-
}
376-
case SUPER -> {
377-
sb.append("? super ");
378-
print(sb, ta.boundType().get());
369+
switch (ta) {
370+
case Signature.TypeArg.Unbounded _ -> sb.append('?');
371+
case Signature.TypeArg.Bounded bta -> {
372+
switch (bta.wildcardIndicator()) {
373+
case NONE -> print(sb, bta.boundType());
374+
case EXTENDS -> {
375+
sb.append("? extends ");
376+
print(sb, bta.boundType());
377+
}
378+
case SUPER -> {
379+
sb.append("? super ");
380+
print(sb, bta.boundType());
381+
}
382+
}
379383
}
380384
}
381385
}

‎test/jdk/jdk/classfile/SignaturesTest.java

+4-3
Original file line numberDiff line numberDiff line change
@@ -183,9 +183,10 @@ static class Observer extends ArrayList<Outer<String>.Inner<Long>>{}
183183
void testClassSignatureClassDesc() throws IOException {
184184
var observerCf = ClassFile.of().parse(Path.of(System.getProperty("test.classes"), "SignaturesTest$Observer.class"));
185185
var sig = observerCf.findAttribute(Attributes.SIGNATURE).orElseThrow().asClassSignature();
186-
var innerSig = (ClassTypeSig) sig.superclassSignature() // ArrayList
187-
.typeArgs().getFirst() // Outer<String>.Inner<Long>
188-
.boundType().orElseThrow(); // assert it's exact bound
186+
var arrayListSig = sig.superclassSignature(); // ArrayList
187+
var arrayListTypeArg = (TypeArg.Bounded) arrayListSig.typeArgs().getFirst(); // Outer<String>.Inner<Long>
188+
assertEquals(TypeArg.Bounded.WildcardIndicator.NONE, arrayListTypeArg.wildcardIndicator());
189+
var innerSig = (ClassTypeSig) arrayListTypeArg.boundType();
189190
assertEquals("Inner", innerSig.className(), "simple name in signature");
190191
assertEquals(Outer.Inner.class.describeConstable().orElseThrow(), innerSig.classDesc(),
191192
"ClassDesc derived from signature");

‎test/langtools/tools/javap/classfile/6888367/T6888367.java

+9-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2009, 2015, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2009, 2024, 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
@@ -312,21 +312,14 @@ public String visitTypeParamType(Signature.TypeParam type) {
312312
}
313313

314314
public String visitWildcardType(Signature.TypeArg type) {
315-
switch (type.wildcardIndicator()) {
316-
case UNBOUNDED -> {
317-
return "W{?}";
318-
}
319-
case EXTENDS -> {
320-
return "W{e," + print(type.boundType().get()) + "}";
321-
}
322-
case SUPER -> {
323-
return "W{s," + print(type.boundType().get()) + "}";
324-
}
325-
default -> {
326-
if (type.boundType().isPresent()) return print(type.boundType().get());
327-
else throw new AssertionError();
328-
}
329-
}
315+
return switch (type) {
316+
case Signature.TypeArg.Unbounded _ -> "W{?}";
317+
case Signature.TypeArg.Bounded b -> switch (b.wildcardIndicator()) {
318+
case EXTENDS -> "W{e," + print(b.boundType()) + "}";
319+
case SUPER -> "W{s," + print(b.boundType()) + "}";
320+
case NONE -> print(b.boundType());
321+
};
322+
};
330323
}
331324

332325
};

0 commit comments

Comments
 (0)
Please sign in to comment.