8315457: Implement JEP 459: String Templates (Second Preview)

Reviewed-by: jlahoda, alanb, vromero
This commit is contained in:
Jim Laskey 2023-11-17 12:53:49 +00:00
parent 5522656af7
commit 9902d2eb17
16 changed files with 34 additions and 50 deletions

View File

@ -46,7 +46,6 @@ import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.RejectedExecutionException; import java.util.concurrent.RejectedExecutionException;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.javac.PreviewFeature;
import jdk.internal.misc.CarrierThreadLocal; import jdk.internal.misc.CarrierThreadLocal;
import jdk.internal.module.ServicesCatalog; import jdk.internal.module.ServicesCatalog;
import jdk.internal.reflect.ConstantPool; import jdk.internal.reflect.ConstantPool;
@ -420,19 +419,16 @@ public interface JavaLangAccess {
/** /**
* Get the coder for the supplied character. * Get the coder for the supplied character.
*/ */
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES)
long stringConcatCoder(char value); long stringConcatCoder(char value);
/** /**
* Update lengthCoder for StringBuilder. * Update lengthCoder for StringBuilder.
*/ */
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES)
long stringBuilderConcatMix(long lengthCoder, StringBuilder sb); long stringBuilderConcatMix(long lengthCoder, StringBuilder sb);
/** /**
* Prepend StringBuilder content. * Prepend StringBuilder content.
*/ */
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES)
long stringBuilderConcatPrepend(long lengthCoder, byte[] buf, StringBuilder sb); long stringBuilderConcatPrepend(long lengthCoder, byte[] buf, StringBuilder sb);
/** /**

View File

@ -67,8 +67,7 @@ public @interface PreviewFeature {
// not used, but required for interim javac to not warn. // not used, but required for interim javac to not warn.
VIRTUAL_THREADS, VIRTUAL_THREADS,
FOREIGN, FOREIGN,
@JEP(number=459, title="String Templates", status="Second Preview")
@JEP(number=430, title="String Templates")
STRING_TEMPLATES, STRING_TEMPLATES,
@JEP(number=445, title="Unnamed Classes and Instance Main Methods") @JEP(number=445, title="Unnamed Classes and Instance Main Methods")
UNNAMED_CLASSES, UNNAMED_CLASSES,

View File

@ -177,6 +177,7 @@ public interface Tree {
/** /**
* Used for instances of {@link StringTemplateTree}. * Used for instances of {@link StringTemplateTree}.
* @since 21
*/ */
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true) @PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true)
TEMPLATE(StringTemplateTree.class), TEMPLATE(StringTemplateTree.class),

View File

@ -264,6 +264,7 @@ public interface TreeVisitor<R,P> {
* @param node the node being visited * @param node the node being visited
* @param p a parameter value * @param p a parameter value
* @return a result value * @return a result value
* @since 21
*/ */
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true) @PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true)
R visitStringTemplate(StringTemplateTree node, P p); R visitStringTemplate(StringTemplateTree node, P p);

View File

@ -634,6 +634,7 @@ public class SimpleTreeVisitor <R,P> implements TreeVisitor<R,P> {
* @param node {@inheritDoc} * @param node {@inheritDoc}
* @param p {@inheritDoc} * @param p {@inheritDoc}
* @return the result of {@code defaultAction} * @return the result of {@code defaultAction}
* @since 21
*/ */
@Override @Override
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true) @PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true)

View File

@ -783,6 +783,7 @@ public class TreeScanner<R,P> implements TreeVisitor<R,P> {
* @param node {@inheritDoc} * @param node {@inheritDoc}
* @param p {@inheritDoc} * @param p {@inheritDoc}
* @return the result of scanning * @return the result of scanning
* @since 21
*/ */
@Override @Override
@PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true) @PreviewFeature(feature=PreviewFeature.Feature.STRING_TEMPLATES, reflective=true)

View File

@ -5024,12 +5024,11 @@ public class Attr extends JCTree.Visitor {
public void visitStringTemplate(JCStringTemplate tree) { public void visitStringTemplate(JCStringTemplate tree) {
JCExpression processor = tree.processor; JCExpression processor = tree.processor;
Type resultType = syms.stringTemplateType; Type processorType = attribTree(processor, env, new ResultInfo(KindSelector.VAL, Type.noType));
chk.checkProcessorType(processor, processorType, env);
if (processor != null) { Type processMethodType = getProcessMethodType(tree, processorType);
resultType = attribTree(processor, env, new ResultInfo(KindSelector.VAL, Type.noType)); tree.processMethodType = processMethodType;
resultType = chk.checkProcessorType(processor, resultType, env); Type resultType = processMethodType.getReturnType();
}
Env<AttrContext> localEnv = env.dup(tree, env.info.dup()); Env<AttrContext> localEnv = env.dup(tree, env.info.dup());
@ -5039,10 +5038,16 @@ public class Attr extends JCTree.Visitor {
tree.type = resultType; tree.type = resultType;
result = resultType; result = resultType;
check(tree, resultType, KindSelector.VAL, resultInfo); check(tree, resultType, KindSelector.VAL, resultInfo);
} }
private Type getProcessMethodType(JCStringTemplate tree, Type processorType) {
MethodSymbol processSymbol = rs.resolveInternalMethod(tree.pos(),
env, types.skipTypeVars(processorType, false),
names.process, List.of(syms.stringTemplateType), List.nil());
return types.memberType(processorType, processSymbol);
}
public void visitTypeIdent(JCPrimitiveTypeTree tree) { public void visitTypeIdent(JCPrimitiveTypeTree tree) {
result = check(tree, syms.typeOfTag[tree.typetag.ordinal()], KindSelector.TYP, resultInfo); result = check(tree, syms.typeOfTag[tree.typetag.ordinal()], KindSelector.TYP, resultInfo);
} }

View File

@ -4370,8 +4370,7 @@ public class Check {
if (typeArguments.size() == 2) { if (typeArguments.size() == 2) {
resultType = typeArguments.head; resultType = typeArguments.head;
} else { } else {
log.error(DiagnosticFlag.RESOLVE_ERROR, processor.pos, resultType = syms.objectType;
Errors.ProcessorTypeCannotBeARawType(processorType.tsym));
} }
} else { } else {
log.error(DiagnosticFlag.RESOLVE_ERROR, processor.pos, log.error(DiagnosticFlag.RESOLVE_ERROR, processor.pos,

View File

@ -1678,23 +1678,8 @@ public class Flow {
@Override @Override
public void visitStringTemplate(JCStringTemplate tree) { public void visitStringTemplate(JCStringTemplate tree) {
JCExpression processor = tree.processor; for (Type thrown : tree.processMethodType.getThrownTypes()) {
markThrown(tree, thrown);
if (processor != null) {
scan(processor);
Type interfaceType = types.asSuper(processor.type, syms.processorType.tsym);
if (interfaceType != null) {
List<Type> typeArguments = interfaceType.getTypeArguments();
if (typeArguments.size() == 2) {
Type throwType = typeArguments.tail.head;
if (throwType != null) {
markThrown(tree, throwType);
}
}
}
} }
scan(tree.expressions); scan(tree.expressions);

View File

@ -254,7 +254,12 @@ public final class TransLiterals extends TreeTranslator {
} }
boolean isNamedProcessor(Name name) { boolean isNamedProcessor(Name name) {
if (processor instanceof JCIdent ident && ident.sym instanceof VarSymbol varSym) { Symbol sym = switch (processor) {
case JCIdent ident -> ident.sym;
case JCFieldAccess access -> access.sym;
default -> null;
};
if (sym instanceof VarSymbol varSym) {
if (varSym.flags() == (Flags.PUBLIC | Flags.FINAL | Flags.STATIC) && if (varSym.flags() == (Flags.PUBLIC | Flags.FINAL | Flags.STATIC) &&
varSym.name == name && varSym.name == name &&
types.isSameType(varSym.owner.type, syms.stringTemplateType)) { types.isSameType(varSym.owner.type, syms.stringTemplateType)) {
@ -265,8 +270,7 @@ public final class TransLiterals extends TreeTranslator {
} }
boolean isLinkageProcessor() { boolean isLinkageProcessor() {
return processor != null && return !useValuesList &&
!useValuesList &&
types.isSubtype(processor.type, syms.linkageType) && types.isSubtype(processor.type, syms.linkageType) &&
processor.type.isFinal() && processor.type.isFinal() &&
TreeInfo.symbol(processor) instanceof VarSymbol varSymbol && TreeInfo.symbol(processor) instanceof VarSymbol varSymbol &&
@ -278,7 +282,7 @@ public final class TransLiterals extends TreeTranslator {
JCExpression result; JCExpression result;
make.at(tree.pos); make.at(tree.pos);
if (processor == null || isNamedProcessor(names.RAW)) { if (isNamedProcessor(names.RAW)) {
result = newStringTemplate(); result = newStringTemplate();
} else if (isNamedProcessor(names.STR)) { } else if (isNamedProcessor(names.STR)) {
result = concatExpression(fragments, expressions); result = concatExpression(fragments, expressions);

View File

@ -1354,10 +1354,6 @@ compiler.err.text.block.template.is.not.well.formed=\
compiler.err.processor.missing.from.string.template.expression=\ compiler.err.processor.missing.from.string.template.expression=\
processor missing from string template expression processor missing from string template expression
# 0: symbol
compiler.err.processor.type.cannot.be.a.raw.type=\
processor type cannot be a raw type: {0}
# 0: symbol # 0: symbol
compiler.err.not.a.processor.type=\ compiler.err.not.a.processor.type=\
not a processor type: {0} not a processor type: {0}

View File

@ -2488,6 +2488,7 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition {
public JCExpression processor; public JCExpression processor;
public List<String> fragments; public List<String> fragments;
public List<JCExpression> expressions; public List<JCExpression> expressions;
public Type processMethodType;
protected JCStringTemplate(JCExpression processor, protected JCStringTemplate(JCExpression processor,
List<String> fragments, List<String> fragments,

View File

@ -1480,9 +1480,7 @@ public class Pretty extends JCTree.Visitor {
try { try {
JCExpression processor = tree.processor; JCExpression processor = tree.processor;
print("["); print("[");
if (processor != null) {
printExpr(processor); printExpr(processor);
}
print("]"); print("]");
print("\"" + tree.fragments.stream().collect(Collectors.joining("\\{}")) + "\""); print("\"" + tree.fragments.stream().collect(Collectors.joining("\\{}")) + "\"");
print("("); print("(");

View File

@ -548,11 +548,7 @@ public class TreeInfo {
} }
case STRING_TEMPLATE: { case STRING_TEMPLATE: {
JCStringTemplate node = (JCStringTemplate) tree; JCStringTemplate node = (JCStringTemplate) tree;
if (node.processor == null) { return node.processor == null ? node.pos : getStartPos(node.processor);
return node.pos;
} else {
return getStartPos(node.processor);
}
} }
case ERRONEOUS: { case ERRONEOUS: {
JCErroneous node = (JCErroneous)tree; JCErroneous node = (JCErroneous)tree;

View File

@ -22,6 +22,8 @@
*/ */
// key: compiler.note.preview.filename // key: compiler.note.preview.filename
// key: compiler.err.cant.resolve.location.args
// key: compiler.misc.location
// key: compiler.note.preview.recompile // key: compiler.note.preview.recompile
// key: compiler.err.not.a.processor.type // key: compiler.err.not.a.processor.type
// options: --enable-preview -source ${jdk.version} // options: --enable-preview -source ${jdk.version}

View File

@ -25,7 +25,6 @@
// key: compiler.note.preview.recompile // key: compiler.note.preview.recompile
// key: compiler.misc.unexpected.ret.val // key: compiler.misc.unexpected.ret.val
// key: compiler.err.prob.found.req // key: compiler.err.prob.found.req
// key: compiler.err.processor.type.cannot.be.a.raw.type
// options: --enable-preview -source ${jdk.version} // options: --enable-preview -source ${jdk.version}
import java.lang.*; import java.lang.*;