diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Kinds.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Kinds.java index e98a1a42be4..ddde4894390 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Kinds.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Kinds.java @@ -71,6 +71,7 @@ public class Kinds { HIDDEN(Category.RESOLUTION_TARGET), // not overloaded non-target STATICERR(Category.RESOLUTION_TARGET), // overloaded? target MISSING_ENCL(Category.RESOLUTION), // not overloaded non-target + BAD_VAR(Category.RESOLUTION), // not overloaded non-target ABSENT_VAR(Category.RESOLUTION_TARGET, KindName.VAR), // not overloaded non-target WRONG_MTHS(Category.RESOLUTION_TARGET, KindName.METHOD), // overloaded target WRONG_MTH(Category.RESOLUTION_TARGET, KindName.METHOD), // not overloaded target diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java index 9be0a08c3a1..a1882abdf32 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java @@ -227,6 +227,7 @@ public enum Source { return compareTo(JDK1_8) <= 0; } public boolean allowPrivateInterfaceMethods() { return compareTo(JDK1_9) >= 0; } + public boolean allowLocalVariableTypeInference() { return compareTo(JDK1_10) >= 0; } public static SourceVersion toSourceVersion(Source source) { switch(source) { case JDK1_2: diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java index 71d5ed284f7..15f0ddebdd6 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Type.java @@ -1616,6 +1616,7 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { public TypeVar(Name name, Symbol owner, Type lower) { super(null, TypeMetadata.EMPTY); + Assert.checkNonNull(lower); tsym = new TypeVariableSymbol(0, name, this, owner); this.bound = null; this.lower = lower; @@ -1628,6 +1629,7 @@ public abstract class Type extends AnnoConstruct implements TypeMirror { public TypeVar(TypeSymbol tsym, Type bound, Type lower, TypeMetadata metadata) { super(tsym, metadata); + Assert.checkNonNull(lower); this.bound = bound; this.lower = lower; } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java index f0cef96947f..0f77ad316cd 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/TypeAnnotations.java @@ -1247,7 +1247,9 @@ public class TypeAnnotations { final TypeAnnotationPosition pos = TypeAnnotationPosition.localVariable(currentLambda, tree.pos); - separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos); + if (!tree.isImplicitlyTyped()) { + separateAnnotationsKinds(tree.vartype, tree.sym.type, tree.sym, pos); + } } else if (tree.sym.getKind() == ElementKind.EXCEPTION_PARAMETER) { final TypeAnnotationPosition pos = TypeAnnotationPosition.exceptionParameter(currentLambda, diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java index 5156f10ac32..4be03fdfc1d 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/code/Types.java @@ -190,6 +190,245 @@ public class Types { } // + // + + /** + * A projection kind. See {@link TypeProjection} + */ + enum ProjectionKind { + UPWARDS() { + @Override + ProjectionKind complement() { + return DOWNWARDS; + } + }, + DOWNWARDS() { + @Override + ProjectionKind complement() { + return UPWARDS; + } + }; + + abstract ProjectionKind complement(); + } + + /** + * This visitor performs upwards and downwards projections on types. + * + * A projection is defined as a function that takes a type T, a set of type variables V and that + * produces another type S. + * + * An upwards projection maps a type T into a type S such that (i) T has no variables in V, + * and (ii) S is an upper bound of T. + * + * A downwards projection maps a type T into a type S such that (i) T has no variables in V, + * and (ii) S is a lower bound of T. + * + * Note that projections are only allowed to touch variables in V. Theferore it is possible for + * a projection to leave its input type unchanged if it does not contain any variables in V. + * + * Moreover, note that while an upwards projection is always defined (every type as an upper bound), + * a downwards projection is not always defined. + * + * Examples: + * + * {@code upwards(List<#CAP1>, [#CAP1]) = List, where #CAP1 <: String } + * {@code downwards(List<#CAP2>, [#CAP2]) = List, where #CAP2 :> String } + * {@code upwards(List<#CAP1>, [#CAP2]) = List<#CAP1> } + * {@code downwards(List<#CAP1>, [#CAP1]) = not defined } + */ + class TypeProjection extends StructuralTypeMapping { + + List vars; + Set seen = new HashSet<>(); + + public TypeProjection(List vars) { + this.vars = vars; + } + + @Override + public Type visitClassType(ClassType t, ProjectionKind pkind) { + if (t.isCompound()) { + List components = directSupertypes(t); + List components1 = components.map(c -> c.map(this, pkind)); + if (components == components1) return t; + else return makeIntersectionType(components1); + } else { + Type outer = t.getEnclosingType(); + Type outer1 = visit(outer, pkind); + List typarams = t.getTypeArguments(); + List typarams1 = typarams.map(ta -> mapTypeArgument(ta, pkind)); + if (typarams1.stream().anyMatch(ta -> ta.hasTag(BOT))) { + //not defined + return syms.botType; + } + if (outer1 == outer && typarams1 == typarams) return t; + else return new ClassType(outer1, typarams1, t.tsym, t.getMetadata()) { + @Override + protected boolean needsStripping() { + return true; + } + }; + } + } + + protected Type makeWildcard(Type upper, Type lower) { + BoundKind bk; + Type bound; + if (upper.hasTag(BOT)) { + upper = syms.objectType; + } + boolean isUpperObject = isSameType(upper, syms.objectType); + if (!lower.hasTag(BOT) && isUpperObject) { + bound = lower; + bk = SUPER; + } else { + bound = upper; + bk = isUpperObject ? UNBOUND : EXTENDS; + } + return new WildcardType(bound, bk, syms.boundClass); + } + + @Override + public Type visitTypeVar(TypeVar t, ProjectionKind pkind) { + if (vars.contains(t)) { + try { + if (seen.add(t)) { + final Type bound; + switch (pkind) { + case UPWARDS: + bound = t.getUpperBound(); + break; + case DOWNWARDS: + bound = (t.getLowerBound() == null) ? + syms.botType : + t.getLowerBound(); + break; + default: + Assert.error(); + return null; + } + return bound.map(this, pkind); + } else { + //cycle + return syms.objectType; + } + } finally { + seen.remove(t); + } + } else { + return t; + } + } + + @Override + public Type visitWildcardType(WildcardType wt, ProjectionKind pkind) { + switch (pkind) { + case UPWARDS: + return wt.isExtendsBound() ? + wt.type.map(this, pkind) : + syms.objectType; + case DOWNWARDS: + return wt.isSuperBound() ? + wt.type.map(this, pkind) : + syms.botType; + default: + Assert.error(); + return null; + } + } + + private Type mapTypeArgument(Type t, ProjectionKind pkind) { + if (!t.containsAny(vars)) { + return t; + } else if (!t.hasTag(WILDCARD) && pkind == ProjectionKind.DOWNWARDS) { + //not defined + return syms.botType; + } else { + Type upper = t.map(this, pkind); + Type lower = t.map(this, pkind.complement()); + return makeWildcard(upper, lower); + } + } + } + + /** + * Computes an upward projection of given type, and vars. See {@link TypeProjection}. + * + * @param t the type to be projected + * @param vars the set of type variables to be mapped + * @return the type obtained as result of the projection + */ + public Type upward(Type t, List vars) { + return t.map(new TypeProjection(vars), ProjectionKind.UPWARDS); + } + + /** + * Computes the set of captured variables mentioned in a given type. See {@link CaptureScanner}. + * This routine is typically used to computed the input set of variables to be used during + * an upwards projection (see {@link Types#upward(Type, List)}). + * + * @param t the type where occurrences of captured variables have to be found + * @return the set of captured variables found in t + */ + public List captures(Type t) { + CaptureScanner cs = new CaptureScanner(); + Set captures = new HashSet<>(); + cs.visit(t, captures); + return List.from(captures); + } + + /** + * This visitor scans a type recursively looking for occurrences of captured type variables. + */ + class CaptureScanner extends SimpleVisitor> { + + @Override + public Void visitType(Type t, Set types) { + return null; + } + + @Override + public Void visitClassType(ClassType t, Set seen) { + if (t.isCompound()) { + directSupertypes(t).forEach(s -> visit(s, seen)); + } else { + t.allparams().forEach(ta -> visit(ta, seen)); + } + return null; + } + + @Override + public Void visitArrayType(ArrayType t, Set seen) { + return visit(t.elemtype, seen); + } + + @Override + public Void visitWildcardType(WildcardType t, Set seen) { + visit(t.type, seen); + return null; + } + + @Override + public Void visitTypeVar(TypeVar t, Set seen) { + if ((t.tsym.flags() & Flags.SYNTHETIC) != 0 && seen.add(t)) { + visit(t.getUpperBound(), seen); + } + return null; + } + + @Override + public Void visitCapturedType(CapturedType t, Set seen) { + if (seen.add(t)) { + visit(t.getUpperBound(), seen); + visit(t.getLowerBound(), seen); + } + return null; + } + } + + // + // /** * Checks that all the arguments to a class are unbounded diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java index 84fd1dedeff..590266591ad 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Analyzer.java @@ -83,6 +83,7 @@ import com.sun.tools.javac.util.DiagnosticSource; import static com.sun.tools.javac.code.Flags.GENERATEDCONSTR; import static com.sun.tools.javac.code.TypeTag.CLASS; import static com.sun.tools.javac.tree.JCTree.Tag.APPLY; +import static com.sun.tools.javac.tree.JCTree.Tag.FOREACHLOOP; import static com.sun.tools.javac.tree.JCTree.Tag.LABELLED; import static com.sun.tools.javac.tree.JCTree.Tag.METHODDEF; import static com.sun.tools.javac.tree.JCTree.Tag.NEWCLASS; @@ -139,7 +140,8 @@ public class Analyzer { enum AnalyzerMode { DIAMOND("diamond", Source::allowDiamond), LAMBDA("lambda", Source::allowLambda), - METHOD("method", Source::allowGraphInference); + METHOD("method", Source::allowGraphInference), + LOCAL("local", Source::allowLocalVariableTypeInference); final String opt; final Predicate sourceFilter; @@ -341,11 +343,91 @@ public class Analyzer { } } + /** + * Base class for local variable inference analyzers. + */ + abstract class RedundantLocalVarTypeAnalyzerBase extends StatementAnalyzer { + + RedundantLocalVarTypeAnalyzerBase(JCTree.Tag tag) { + super(AnalyzerMode.LOCAL, tag); + } + + /** + * Map a variable tree into a new declaration using implicit type. + */ + JCVariableDecl mapVar(JCVariableDecl oldTree, JCVariableDecl newTree){ + newTree.vartype = null; + return newTree; + } + + /** + * Analyze results of local variable inference. + */ + void processVar(JCVariableDecl oldTree, JCVariableDecl newTree, boolean hasErrors){ + if (!hasErrors) { + if (types.isSameType(oldTree.type, newTree.type)) { + log.warning(oldTree, Warnings.LocalRedundantType); + } + } + } + } + + /** + * This analyzer checks if a local variable declaration has redundant type. + */ + class RedundantLocalVarTypeAnalyzer extends RedundantLocalVarTypeAnalyzerBase { + + RedundantLocalVarTypeAnalyzer() { + super(VARDEF); + } + + boolean match(JCVariableDecl tree){ + return tree.sym.owner.kind == Kind.MTH && + tree.init != null && !tree.isImplicitlyTyped() && + attr.canInferLocalVarType(tree) == null; + } + @Override + JCVariableDecl map(JCVariableDecl oldTree, JCVariableDecl newTree){ + return mapVar(oldTree, newTree); + } + @Override + void process(JCVariableDecl oldTree, JCVariableDecl newTree, boolean hasErrors){ + processVar(oldTree, newTree, hasErrors); + } + } + + /** + * This analyzer checks if a for each variable declaration has redundant type. + */ + class RedundantLocalVarTypeAnalyzerForEach extends RedundantLocalVarTypeAnalyzerBase { + + RedundantLocalVarTypeAnalyzerForEach() { + super(FOREACHLOOP); + } + + @Override + boolean match(JCEnhancedForLoop tree){ + return !tree.var.isImplicitlyTyped(); + } + @Override + JCEnhancedForLoop map(JCEnhancedForLoop oldTree, JCEnhancedForLoop newTree){ + newTree.var = mapVar(oldTree.var, newTree.var); + newTree.body = make.Block(0, List.nil()); //ignore body for analysis purpose + return newTree; + } + @Override + void process(JCEnhancedForLoop oldTree, JCEnhancedForLoop newTree, boolean hasErrors){ + processVar(oldTree.var, newTree.var, hasErrors); + } + } + @SuppressWarnings({"unchecked", "rawtypes"}) StatementAnalyzer[] analyzers = new StatementAnalyzer[] { new DiamondInitializer(), new LambdaAnalyzer(), - new RedundantTypeArgAnalyzer() + new RedundantTypeArgAnalyzer(), + new RedundantLocalVarTypeAnalyzer(), + new RedundantLocalVarTypeAnalyzerForEach() }; /** diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Annotate.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Annotate.java index ad7c7e2eb6f..b8dcdd7fc45 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Annotate.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Annotate.java @@ -28,9 +28,11 @@ package com.sun.tools.javac.comp; import com.sun.tools.javac.code.*; import com.sun.tools.javac.code.Attribute.Compound; import com.sun.tools.javac.code.Attribute.TypeCompound; +import com.sun.tools.javac.code.Kinds.KindSelector; import com.sun.tools.javac.code.Scope.WriteableScope; import com.sun.tools.javac.code.Symbol.*; import com.sun.tools.javac.code.TypeMetadata.Entry.Kind; +import com.sun.tools.javac.comp.Check.CheckContext; import com.sun.tools.javac.resources.CompilerProperties.Errors; import com.sun.tools.javac.tree.JCTree; import com.sun.tools.javac.tree.JCTree.*; @@ -602,7 +604,7 @@ public class Annotate { } private Attribute getAnnotationEnumValue(Type expectedElementType, JCExpression tree, Env env) { - Type result = attr.attribExpr(tree, env, expectedElementType); + Type result = attr.attribTree(tree, env, annotationValueInfo(expectedElementType)); Symbol sym = TreeInfo.symbol(tree); if (sym == null || TreeInfo.nonstaticSelect(tree) || @@ -616,7 +618,7 @@ public class Annotate { } private Attribute getAnnotationClassValue(Type expectedElementType, JCExpression tree, Env env) { - Type result = attr.attribExpr(tree, env, expectedElementType); + Type result = attr.attribTree(tree, env, annotationValueInfo(expectedElementType)); if (result.isErroneous()) { // Does it look like an unresolved class literal? if (TreeInfo.name(tree) == names._class && @@ -642,7 +644,7 @@ public class Annotate { } private Attribute getAnnotationPrimitiveValue(Type expectedElementType, JCExpression tree, Env env) { - Type result = attr.attribExpr(tree, env, expectedElementType); + Type result = attr.attribTree(tree, env, annotationValueInfo(expectedElementType)); if (result.isErroneous()) return new Attribute.Error(result.getOriginalType()); if (result.constValue() == null) { @@ -653,6 +655,22 @@ public class Annotate { return new Attribute.Constant(expectedElementType, result.constValue()); } + private Attr.ResultInfo annotationValueInfo(Type pt) { + return attr.unknownExprInfo.dup(pt, new AnnotationValueContext(attr.unknownExprInfo.checkContext)); + } + + class AnnotationValueContext extends Check.NestedCheckContext { + AnnotationValueContext(CheckContext enclosingContext) { + super(enclosingContext); + } + + @Override + public boolean compatible(Type found, Type req, Warner warn) { + //handle non-final implicitly-typed vars (will be rejected later on) + return found.hasTag(TypeTag.NONE) || super.compatible(found, req, warn); + } + } + private Attribute getAnnotationArrayValue(Type expectedElementType, JCExpression tree, Env env) { // Special case, implicit array if (!tree.hasTag(NEWARRAY)) { diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java index b9cefce188d..d7b10617f5f 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Attr.java @@ -36,7 +36,6 @@ import com.sun.source.tree.MemberSelectTree; import com.sun.source.tree.TreeVisitor; import com.sun.source.util.SimpleTreeVisitor; import com.sun.tools.javac.code.*; -import com.sun.tools.javac.code.Directive.RequiresFlag; import com.sun.tools.javac.code.Lint.LintCategory; import com.sun.tools.javac.code.Scope.WriteableScope; import com.sun.tools.javac.code.Symbol.*; @@ -46,7 +45,6 @@ import com.sun.tools.javac.code.Types.FunctionDescriptorLookupError; import com.sun.tools.javac.comp.ArgumentAttr.LocalCacheContext; import com.sun.tools.javac.comp.Check.CheckContext; import com.sun.tools.javac.comp.DeferredAttr.AttrMode; -import com.sun.tools.javac.comp.Infer.FreeTypeListener; import com.sun.tools.javac.jvm.*; import static com.sun.tools.javac.resources.CompilerProperties.Fragments.Diamond; import static com.sun.tools.javac.resources.CompilerProperties.Fragments.DiamondInvalidArg; @@ -830,6 +828,10 @@ public class Attr extends JCTree.Visitor { final JavaFileObject prevSource = log.useSource(env.toplevel.sourcefile); try { Type itype = attribExpr(variable.init, env, type); + if (variable.isImplicitlyTyped()) { + //fixup local variable type + type = variable.type = variable.sym.type = chk.checkLocalVarType(variable, itype.baseType(), variable.name); + } if (itype.constValue() != null) { return coerce(itype, type).constValue(); } else { @@ -1108,6 +1110,21 @@ public class Attr extends JCTree.Visitor { // parameters have already been entered env.info.scope.enter(tree.sym); } else { + if (tree.isImplicitlyTyped() && (tree.getModifiers().flags & PARAMETER) == 0) { + if (tree.init == null) { + //cannot use 'var' without initializer + log.error(tree, Errors.CantInferLocalVarType(tree.name, Fragments.LocalMissingInit)); + tree.vartype = make.Erroneous(); + } else { + Fragment msg = canInferLocalVarType(tree); + if (msg != null) { + //cannot use 'var' with initializer which require an explicit target + //(e.g. lambda, method reference, array initializer). + log.error(tree, Errors.CantInferLocalVarType(tree.name, msg)); + tree.vartype = make.Erroneous(); + } + } + } try { annotate.blockAnnotations(); memberEnter.memberEnter(tree, env); @@ -1131,7 +1148,7 @@ public class Attr extends JCTree.Visitor { boolean isImplicitLambdaParameter = env.tree.hasTag(LAMBDA) && ((JCLambda)env.tree).paramKind == JCLambda.ParameterKind.IMPLICIT && (tree.sym.flags() & PARAMETER) != 0; - chk.validate(tree.vartype, env, !isImplicitLambdaParameter); + chk.validate(tree.vartype, env, !isImplicitLambdaParameter && !tree.isImplicitlyTyped()); try { v.getConstValue(); // ensure compile-time constant initializer is evaluated @@ -1152,6 +1169,10 @@ public class Attr extends JCTree.Visitor { // marking the variable as undefined. initEnv.info.enclVar = v; attribExpr(tree.init, initEnv, v.type); + if (tree.isImplicitlyTyped()) { + //fixup local variable type + v.type = chk.checkLocalVarType(tree, tree.init.type.baseType(), tree.name); + } } } result = tree.type = v.type; @@ -1161,6 +1182,71 @@ public class Attr extends JCTree.Visitor { } } + Fragment canInferLocalVarType(JCVariableDecl tree) { + LocalInitScanner lis = new LocalInitScanner(); + lis.scan(tree.init); + return lis.badInferenceMsg; + } + + static class LocalInitScanner extends TreeScanner { + Fragment badInferenceMsg = null; + boolean needsTarget = true; + + @Override + public void visitNewArray(JCNewArray tree) { + if (tree.elemtype == null && needsTarget) { + badInferenceMsg = Fragments.LocalArrayMissingTarget; + } + } + + @Override + public void visitLambda(JCLambda tree) { + if (needsTarget) { + badInferenceMsg = Fragments.LocalLambdaMissingTarget; + } + } + + @Override + public void visitTypeCast(JCTypeCast tree) { + boolean prevNeedsTarget = needsTarget; + try { + needsTarget = false; + super.visitTypeCast(tree); + } finally { + needsTarget = prevNeedsTarget; + } + } + + @Override + public void visitReference(JCMemberReference tree) { + if (needsTarget) { + badInferenceMsg = Fragments.LocalMrefMissingTarget; + } + } + + @Override + public void visitNewClass(JCNewClass tree) { + boolean prevNeedsTarget = needsTarget; + try { + needsTarget = false; + super.visitNewClass(tree); + } finally { + needsTarget = prevNeedsTarget; + } + } + + @Override + public void visitApply(JCMethodInvocation tree) { + boolean prevNeedsTarget = needsTarget; + try { + needsTarget = false; + super.visitApply(tree); + } finally { + needsTarget = prevNeedsTarget; + } + } + } + public void visitSkip(JCSkip tree) { result = null; } @@ -1243,7 +1329,6 @@ public class Attr extends JCTree.Visitor { //attributing the for-each expression; we mimick this by attributing //the for-each expression first (against original scope). Type exprType = types.cvarUpperBound(attribExpr(tree.expr, loopEnv)); - attribStat(tree.var, loopEnv); chk.checkNonVoid(tree.pos(), exprType); Type elemtype = types.elemtype(exprType); // perhaps expr is an array? if (elemtype == null) { @@ -1261,6 +1346,15 @@ public class Attr extends JCTree.Visitor { : types.wildUpperBound(iterableParams.head); } } + if (tree.var.isImplicitlyTyped()) { + Type inferredType = chk.checkLocalVarType(tree.var, elemtype, tree.var.name); + if (inferredType.isErroneous()) { + tree.var.vartype = make.at(tree.var.vartype).Erroneous(); + } else { + tree.var.vartype = make.at(tree.var.vartype).Type(inferredType); + } + } + attribStat(tree.var, loopEnv); chk.checkType(tree.expr.pos(), elemtype, tree.var.sym.type); loopEnv.tree = tree; // before, we were not in loop! attribStat(tree.body, loopEnv); @@ -2379,7 +2473,8 @@ public class Attr extends JCTree.Visitor { if (pt().hasTag(ARRAY)) { elemtype = types.elemtype(pt()); } else { - if (!pt().hasTag(ERROR)) { + if (!pt().hasTag(ERROR) && + (env.info.enclVar == null || !env.info.enclVar.type.isErroneous())) { log.error(tree.pos(), Errors.IllegalInitializerForType(pt())); } @@ -2404,7 +2499,7 @@ public class Attr extends JCTree.Visitor { @Override public void visitLambda(final JCLambda that) { if (pt().isErroneous() || (pt().hasTag(NONE) && pt() != Type.recoveryType)) { - if (pt().hasTag(NONE)) { + if (pt().hasTag(NONE) && (env.info.enclVar == null || !env.info.enclVar.type.isErroneous())) { //lambda only allowed in assignment or method invocation/cast context log.error(that.pos(), Errors.UnexpectedLambda); } @@ -2837,7 +2932,7 @@ public class Attr extends JCTree.Visitor { @Override public void visitReference(final JCMemberReference that) { if (pt().isErroneous() || (pt().hasTag(NONE) && pt() != Type.recoveryType)) { - if (pt().hasTag(NONE)) { + if (pt().hasTag(NONE) && (env.info.enclVar == null || !env.info.enclVar.type.isErroneous())) { //method reference only allowed in assignment or method invocation/cast context log.error(that.pos(), Errors.UnexpectedMref); } @@ -3811,6 +3906,14 @@ public class Attr extends JCTree.Visitor { break; case VAR: VarSymbol v = (VarSymbol)sym; + + if (env.info.enclVar != null + && v.type.hasTag(NONE)) { + //self reference to implicitly typed variable declaration + log.error(TreeInfo.positionFor(v, env.enclClass), Errors.CantInferLocalVarType(v.name, Fragments.LocalSelfRef)); + return v.type = types.createErrorType(v.type); + } + // Test (4): if symbol is an instance field of a raw type, // which is being assigned to, issue an unchecked warning if // its type changes under erasure. @@ -4135,6 +4238,9 @@ public class Attr extends JCTree.Visitor { public void visitTypeArray(JCArrayTypeTree tree) { Type etype = attribType(tree.elemtype, env); Type type = new ArrayType(etype, syms.arrayClass); + if (etype.isErroneous()) { + type = types.createErrorType(type); + } result = check(tree, type, KindSelector.TYP, resultInfo); } @@ -4776,7 +4882,7 @@ public class Attr extends JCTree.Visitor { } public void visitVarDef(final JCVariableDecl tree) { //System.err.println("validateTypeAnnotations.visitVarDef " + tree); - if (tree.sym != null && tree.sym.type != null) + if (tree.sym != null && tree.sym.type != null && !tree.isImplicitlyTyped()) validateAnnotatedType(tree.vartype, tree.sym.type); scan(tree.mods); scan(tree.vartype); @@ -4904,17 +5010,16 @@ public class Attr extends JCTree.Visitor { repeat = false; } else if (enclTr.hasTag(JCTree.Tag.WILDCARD)) { JCWildcard wc = (JCWildcard) enclTr; - if (wc.getKind() == JCTree.Kind.EXTENDS_WILDCARD) { - validateAnnotatedType(wc.getBound(), ((WildcardType)enclTy).getExtendsBound()); - } else if (wc.getKind() == JCTree.Kind.SUPER_WILDCARD) { - validateAnnotatedType(wc.getBound(), ((WildcardType)enclTy).getSuperBound()); + if (wc.getKind() == JCTree.Kind.EXTENDS_WILDCARD || + wc.getKind() == JCTree.Kind.SUPER_WILDCARD) { + validateAnnotatedType(wc.getBound(), wc.getBound().type); } else { // Nothing to do for UNBOUND } repeat = false; } else if (enclTr.hasTag(TYPEARRAY)) { JCArrayTypeTree art = (JCArrayTypeTree) enclTr; - validateAnnotatedType(art.getType(), ((ArrayType)enclTy).getComponentType()); + validateAnnotatedType(art.getType(), art.elemtype.type); repeat = false; } else if (enclTr.hasTag(TYPEUNION)) { JCTypeUnion ut = (JCTypeUnion) enclTr; diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java index c763edb3bc9..df606822698 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Check.java @@ -843,26 +843,30 @@ public class Check { List checkDiamondDenotable(ClassType t) { ListBuffer buf = new ListBuffer<>(); for (Type arg : t.allparams()) { - if (!diamondTypeChecker.visit(arg, null)) { + if (!checkDenotable(arg)) { buf.append(arg); } } return buf.toList(); } + + boolean checkDenotable(Type t) { + return denotableChecker.visit(t, null); + } // where /** diamondTypeChecker: A type visitor that descends down the given type looking for non-denotable * types. The visit methods return false as soon as a non-denotable type is encountered and true * otherwise. */ - private static final Types.SimpleVisitor diamondTypeChecker = new Types.SimpleVisitor() { + private static final Types.SimpleVisitor denotableChecker = new Types.SimpleVisitor() { @Override public Boolean visitType(Type t, Void s) { return true; } @Override public Boolean visitClassType(ClassType t, Void s) { - if (t.isCompound()) { + if (t.isUnion() || t.isIntersection()) { return false; } for (Type targ : t.allparams()) { @@ -878,7 +882,7 @@ public class Check { /* Any type variable mentioned in the inferred type must have been declared as a type parameter (i.e cannot have been produced by inference (18.4)) */ - return t.tsym.owner.type.getTypeArguments().contains(t); + return (t.tsym.flags() & SYNTHETIC) == 0; } @Override @@ -941,6 +945,17 @@ public class Check { (allowPrivateSafeVarargs ? PRIVATE : 0) )) != 0); } + Type checkLocalVarType(DiagnosticPosition pos, Type t, Name name) { + //upward project the initializer type + t = types.upward(t, types.captures(t)); + //check that resulting type is not the null type + if (t.hasTag(BOT)) { + log.error(pos, Errors.CantInferLocalVarType(name, Fragments.LocalCantInferNull)); + return types.createErrorType(t); + } + return t; + } + Type checkMethod(final Type mtype, final Symbol sym, final Env env, @@ -3159,7 +3174,10 @@ public class Check { if (s.kind == PCK) return true; } else if (target == names.TYPE_USE) { - if (s.kind == TYP || s.kind == VAR || + if (s.kind == VAR && s.owner.kind == MTH && s.type.hasTag(NONE)) { + //cannot type annotate implictly typed locals + return false; + } else if (s.kind == TYP || s.kind == VAR || (s.kind == MTH && !s.isConstructor() && !s.type.getReturnType().hasTag(VOID)) || (s.kind == MTH && s.isConstructor())) { diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java index 08c5140e894..03a537acae9 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Infer.java @@ -529,7 +529,7 @@ public class Infer { List upperBounds = uv.getBounds(InferenceBound.UPPER); if (Type.containsAny(upperBounds, vars)) { TypeSymbol fresh_tvar = new TypeVariableSymbol(Flags.SYNTHETIC, uv.qtype.tsym.name, null, uv.qtype.tsym.owner); - fresh_tvar.type = new TypeVar(fresh_tvar, types.makeIntersectionType(uv.getBounds(InferenceBound.UPPER)), null); + fresh_tvar.type = new TypeVar(fresh_tvar, types.makeIntersectionType(uv.getBounds(InferenceBound.UPPER)), syms.botType); todo.append(uv); uv.setInst(fresh_tvar.type); } else if (upperBounds.nonEmpty()) { diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/MemberEnter.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/MemberEnter.java index 476370ff9e2..775c314a81e 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/MemberEnter.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/MemberEnter.java @@ -259,7 +259,7 @@ public class MemberEnter extends JCTree.Visitor { try { if (TreeInfo.isEnumInit(tree)) { attr.attribIdentAsEnumType(localEnv, (JCIdent)tree.vartype); - } else { + } else if (!tree.isImplicitlyTyped()) { attr.attribType(tree.vartype, localEnv); if (TreeInfo.isReceiverParam(tree)) checkReceiver(tree, localEnv); @@ -279,8 +279,8 @@ public class MemberEnter extends JCTree.Visitor { tree.vartype.type = atype.makeVarargs(); } WriteableScope enclScope = enter.enterScope(env); - VarSymbol v = - new VarSymbol(0, tree.name, tree.vartype.type, enclScope.owner); + Type vartype = tree.isImplicitlyTyped() ? Type.noType : tree.vartype.type; + VarSymbol v = new VarSymbol(0, tree.name, vartype, enclScope.owner); v.flags_field = chk.checkFlags(tree.pos(), tree.mods.flags, v, tree); tree.sym = v; if (tree.init != null) { @@ -298,7 +298,9 @@ public class MemberEnter extends JCTree.Visitor { } annotate.annotateLater(tree.mods.annotations, localEnv, v, tree.pos()); - annotate.queueScanTreeAndTypeAnnotate(tree.vartype, localEnv, v, tree.pos()); + if (!tree.isImplicitlyTyped()) { + annotate.queueScanTreeAndTypeAnnotate(tree.vartype, localEnv, v, tree.pos()); + } v.pos = tree.pos; } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java index b1b5bd86303..f0e8f859ce5 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java @@ -105,6 +105,7 @@ public class Resolve { public final boolean allowModules; public final boolean checkVarargsAccessAfterResolution; private final boolean compactMethodDiags; + private final boolean allowLocalVariableTypeInference; final EnumSet verboseResolutionMode; WriteableScope polymorphicSignatureScope; @@ -136,6 +137,7 @@ public class Resolve { Target target = Target.instance(context); allowMethodHandles = target.hasMethodHandles(); allowFunctionalInterfaceMostSpecific = source.allowFunctionalInterfaceMostSpecific(); + allowLocalVariableTypeInference = source.allowLocalVariableTypeInference(); checkVarargsAccessAfterResolution = source.allowPostApplicabilityVarargsAccessCheck(); polymorphicSignatureScope = WriteableScope.create(syms.noSymbol); @@ -2325,6 +2327,10 @@ public class Resolve { * (a subset of VAL, TYP, PCK). */ Symbol findIdent(Env env, Name name, KindSelector kind) { + return checkVarType(findIdentInternal(env, name, kind), name); + } + + Symbol findIdentInternal(Env env, Name name, KindSelector kind) { Symbol bestSoFar = typeNotFound; Symbol sym; @@ -2354,6 +2360,11 @@ public class Resolve { */ Symbol findIdentInPackage(Env env, TypeSymbol pck, Name name, KindSelector kind) { + return checkVarType(findIdentInPackageInternal(env, pck, name, kind), name); + } + + Symbol findIdentInPackageInternal(Env env, TypeSymbol pck, + Name name, KindSelector kind) { Name fullname = TypeSymbol.formFullName(name, pck); Symbol bestSoFar = typeNotFound; if (kind.contains(KindSelector.TYP)) { @@ -2383,6 +2394,11 @@ public class Resolve { */ Symbol findIdentInType(Env env, Type site, Name name, KindSelector kind) { + return checkVarType(findIdentInTypeInternal(env, site, name, kind), name); + } + + Symbol findIdentInTypeInternal(Env env, Type site, + Name name, KindSelector kind) { Symbol bestSoFar = typeNotFound; Symbol sym; if (kind.contains(KindSelector.VAL)) { @@ -2399,6 +2415,14 @@ public class Resolve { return bestSoFar; } + private Symbol checkVarType(Symbol bestSoFar, Name name) { + if (allowLocalVariableTypeInference && name.equals(names.var) && + (bestSoFar.kind == TYP || bestSoFar.kind == ABSENT_TYP)) { + bestSoFar = new BadVarTypeError(); + } + return bestSoFar; + } + /* *************************************************************************** * Access checking * The following methods convert ResolveErrors to ErrorSymbols, issuing @@ -3774,6 +3798,17 @@ public class Resolve { } } + class BadVarTypeError extends ResolveError { + BadVarTypeError() { + super(Kind.BAD_VAR, "bad var use"); + } + + @Override + JCDiagnostic getDiagnostic(DiagnosticType dkind, DiagnosticPosition pos, Symbol location, Type site, Name name, List argtypes, List typeargtypes) { + return diags.create(dkind, log.currentSource(), pos, "illegal.ref.to.var.type", name); + } + } + /** * InvalidSymbolError error class indicating that a symbol matching a * given name does not exists in a given site. diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java index 8f80ee52f59..b477b49e3ea 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java @@ -179,6 +179,7 @@ public class JavacParser implements Parser { this.allowAnnotationsAfterTypeParams = source.allowAnnotationsAfterTypeParams(); this.allowUnderscoreIdentifier = source.allowUnderscoreIdentifier(); this.allowPrivateInterfaceMethods = source.allowPrivateInterfaceMethods(); + this.allowLocalVariableTypeInference = source.allowLocalVariableTypeInference(); this.keepDocComments = keepDocComments; this.parseModuleInfo = parseModuleInfo; docComments = newDocCommentTable(keepDocComments, fac); @@ -270,11 +271,14 @@ public class JavacParser implements Parser { */ boolean allowThisIdent; + /** Switch: is local variable inference allowed? + */ + boolean allowLocalVariableTypeInference; + /** The type of the method receiver, as specified by a first "this" parameter. */ JCVariableDecl receiverParam; - /** When terms are parsed, the mode determines which is expected: * mode = EXPR : an expression * mode = TYPE : a type @@ -808,12 +812,16 @@ public class JavacParser implements Parser { * parsing annotations. */ public JCExpression parseType() { - List annotations = typeAnnotationsOpt(); - return parseType(annotations); + return parseType(false); } - public JCExpression parseType(List annotations) { - JCExpression result = unannotatedType(); + public JCExpression parseType(boolean allowVar) { + List annotations = typeAnnotationsOpt(); + return parseType(allowVar, annotations); + } + + public JCExpression parseType(boolean allowVar, List annotations) { + JCExpression result = unannotatedType(allowVar); if (annotations.nonEmpty()) { result = insertAnnotationsToMostInner(result, annotations, false); @@ -822,10 +830,18 @@ public class JavacParser implements Parser { return result; } - public JCExpression unannotatedType() { - return term(TYPE); + public JCExpression unannotatedType(boolean allowVar) { + JCExpression result = term(TYPE); + + if (!allowVar && isRestrictedLocalVarTypeName(result)) { + syntaxError(result.pos, "var.not.allowed.here"); + } + + return result; } + + protected JCExpression term(int newmode) { int prevmode = mode; mode = newmode; @@ -1152,11 +1168,11 @@ public class JavacParser implements Parser { accept(LPAREN); mode = TYPE; int pos1 = pos; - List targets = List.of(t = term3()); + List targets = List.of(t = parseType()); while (token.kind == AMP) { checkIntersectionTypesInCast(); accept(AMP); - targets = targets.prepend(term3()); + targets = targets.prepend(parseType()); } if (targets.length() > 1) { t = toP(F.at(pos1).TypeIntersection(targets.reverse())); @@ -1912,7 +1928,7 @@ public class JavacParser implements Parser { */ JCExpression typeArgument() { List annotations = typeAnnotationsOpt(); - if (token.kind != QUES) return parseType(annotations); + if (token.kind != QUES) return parseType(false, annotations); int pos = token.pos; nextToken(); JCExpression result; @@ -2425,13 +2441,8 @@ public class JavacParser implements Parser { token.kind == ENUM) { return List.of(classOrInterfaceOrEnumDeclaration(mods, dc)); } else { - JCExpression t = parseType(); - ListBuffer stats = - variableDeclarators(mods, t, new ListBuffer()); - // A "LocalVariableDeclarationStatement" subsumes the terminating semicolon - accept(SEMI); - storeEnd(stats.last(), S.prevToken().endPos); - return stats.toList(); + JCExpression t = parseType(true); + return localVariableDeclarations(mods, t); } } case ABSTRACT: case STRICTFP: { @@ -2458,12 +2469,7 @@ public class JavacParser implements Parser { pos = token.pos; JCModifiers mods = F.at(Position.NOPOS).Modifiers(0); F.at(pos); - ListBuffer stats = - variableDeclarators(mods, t, new ListBuffer()); - // A "LocalVariableDeclarationStatement" subsumes the terminating semicolon - accept(SEMI); - storeEnd(stats.last(), S.prevToken().endPos); - return stats.toList(); + return localVariableDeclarations(mods, t); } else { // This Exec is an "ExpressionStatement"; it subsumes the terminating semicolon t = checkExprStat(t); @@ -2473,6 +2479,15 @@ public class JavacParser implements Parser { } } } + //where + private List localVariableDeclarations(JCModifiers mods, JCExpression type) { + ListBuffer stats = + variableDeclarators(mods, type, new ListBuffer<>(), true); + // A "LocalVariableDeclarationStatement" subsumes the terminating semicolon + accept(SEMI); + storeEnd(stats.last(), S.prevToken().endPos); + return stats.toList(); + } /** Statement = * Block @@ -2766,11 +2781,11 @@ public class JavacParser implements Parser { ListBuffer stats = new ListBuffer<>(); int pos = token.pos; if (token.kind == FINAL || token.kind == MONKEYS_AT) { - return variableDeclarators(optFinal(0), parseType(), stats).toList(); + return variableDeclarators(optFinal(0), parseType(true), stats, true).toList(); } else { JCExpression t = term(EXPR | TYPE); if ((lastmode & TYPE) != 0 && LAX_IDENTIFIER.accepts(token.kind)) { - return variableDeclarators(modifiersOpt(), t, stats).toList(); + return variableDeclarators(modifiersOpt(), t, stats, true).toList(); } else if ((lastmode & TYPE) != 0 && token.kind == COLON) { error(pos, "bad.initializer", "for-loop"); return List.of((JCStatement)F.at(pos).VarDef(null, null, t, null)); @@ -2989,9 +3004,10 @@ public class JavacParser implements Parser { */ public > T variableDeclarators(JCModifiers mods, JCExpression type, - T vdefs) + T vdefs, + boolean localDecl) { - return variableDeclaratorsRest(token.pos, mods, type, ident(), false, null, vdefs); + return variableDeclaratorsRest(token.pos, mods, type, ident(), false, null, vdefs, localDecl); } /** VariableDeclaratorsRest = VariableDeclaratorRest { "," VariableDeclarator } @@ -3006,14 +3022,20 @@ public class JavacParser implements Parser { Name name, boolean reqInit, Comment dc, - T vdefs) + T vdefs, + boolean localDecl) { - vdefs.append(variableDeclaratorRest(pos, mods, type, name, reqInit, dc)); + JCVariableDecl head = variableDeclaratorRest(pos, mods, type, name, reqInit, dc, localDecl); + boolean implicit = allowLocalVariableTypeInference && head.vartype == null; + vdefs.append(head); while (token.kind == COMMA) { + if (implicit) { + reportSyntaxError(pos, "var.not.allowed.compound"); + } // All but last of multiple declarators subsume a comma storeEnd((JCTree)vdefs.last(), token.endPos); nextToken(); - vdefs.append(variableDeclarator(mods, type, reqInit, dc)); + vdefs.append(variableDeclarator(mods, type, reqInit, dc, localDecl)); } return vdefs; } @@ -3021,8 +3043,8 @@ public class JavacParser implements Parser { /** VariableDeclarator = Ident VariableDeclaratorRest * ConstantDeclarator = Ident ConstantDeclaratorRest */ - JCVariableDecl variableDeclarator(JCModifiers mods, JCExpression type, boolean reqInit, Comment dc) { - return variableDeclaratorRest(token.pos, mods, type, ident(), reqInit, dc); + JCVariableDecl variableDeclarator(JCModifiers mods, JCExpression type, boolean reqInit, Comment dc, boolean localDecl) { + return variableDeclaratorRest(token.pos, mods, type, ident(), reqInit, dc, localDecl); } /** VariableDeclaratorRest = BracketsOpt ["=" VariableInitializer] @@ -3032,7 +3054,7 @@ public class JavacParser implements Parser { * @param dc The documentation comment for the variable declarations, or null. */ JCVariableDecl variableDeclaratorRest(int pos, JCModifiers mods, JCExpression type, Name name, - boolean reqInit, Comment dc) { + boolean reqInit, Comment dc, boolean localDecl) { type = bracketsOpt(type); JCExpression init = null; if (token.kind == EQ) { @@ -3040,12 +3062,40 @@ public class JavacParser implements Parser { init = variableInitializer(); } else if (reqInit) syntaxError(token.pos, "expected", EQ); + JCTree elemType = TreeInfo.innermostType(type, true); + if (allowLocalVariableTypeInference && elemType.hasTag(IDENT)) { + Name typeName = ((JCIdent)elemType).name; + if (isRestrictedLocalVarTypeName(typeName)) { + if (type.hasTag(TYPEARRAY)) { + //error - 'var' and arrays + reportSyntaxError(pos, "var.not.allowed.array"); + } else { + //implicit type + type = null; + } + } + } JCVariableDecl result = toP(F.at(pos).VarDef(mods, name, type, init)); attach(result, dc); return result; } + boolean isRestrictedLocalVarTypeName(JCExpression e) { + switch (e.getTag()) { + case IDENT: + return isRestrictedLocalVarTypeName(((JCIdent)e).name); + case TYPEARRAY: + return isRestrictedLocalVarTypeName(((JCArrayTypeTree)e).elemtype); + default: + return false; + } + } + + boolean isRestrictedLocalVarTypeName(Name name) { + return allowLocalVariableTypeInference && name == names.var; + } + /** VariableDeclaratorId = Ident BracketsOpt */ JCVariableDecl variableDeclaratorId(JCModifiers mods, JCExpression type) { @@ -3111,13 +3161,13 @@ public class JavacParser implements Parser { int startPos = token.pos; if (token.kind == FINAL || token.kind == MONKEYS_AT) { JCModifiers mods = optFinal(Flags.FINAL); - JCExpression t = parseType(); - return variableDeclaratorRest(token.pos, mods, t, ident(), true, null); + JCExpression t = parseType(true); + return variableDeclaratorRest(token.pos, mods, t, ident(), true, null, true); } JCExpression t = term(EXPR | TYPE); if ((lastmode & TYPE) != 0 && LAX_IDENTIFIER.accepts(token.kind)) { JCModifiers mods = toP(F.at(startPos).Modifiers(Flags.FINAL)); - return variableDeclaratorRest(token.pos, mods, t, ident(), true, null); + return variableDeclaratorRest(token.pos, mods, t, ident(), true, null, true); } else { checkVariableInTryWithResources(startPos); if (!t.hasTag(IDENT) && !t.hasTag(SELECT)) { @@ -3397,7 +3447,7 @@ public class JavacParser implements Parser { protected JCClassDecl classDeclaration(JCModifiers mods, Comment dc) { int pos = token.pos; accept(CLASS); - Name name = ident(); + Name name = typeName(); List typarams = typeParametersOpt(); @@ -3418,6 +3468,15 @@ public class JavacParser implements Parser { return result; } + Name typeName() { + int pos = token.pos; + Name name = ident(); + if (isRestrictedLocalVarTypeName(name)) { + reportSyntaxError(pos, "var.not.allowed", name); + } + return name; + } + /** InterfaceDeclaration = INTERFACE Ident TypeParametersOpt * [EXTENDS TypeList] InterfaceBody * @param mods The modifiers starting the interface declaration @@ -3426,7 +3485,8 @@ public class JavacParser implements Parser { protected JCClassDecl interfaceDeclaration(JCModifiers mods, Comment dc) { int pos = token.pos; accept(INTERFACE); - Name name = ident(); + + Name name = typeName(); List typarams = typeParametersOpt(); @@ -3449,7 +3509,8 @@ public class JavacParser implements Parser { protected JCClassDecl enumDeclaration(JCModifiers mods, Comment dc) { int pos = token.pos; accept(ENUM); - Name name = ident(); + + Name name = typeName(); List implementing = List.nil(); if (token.kind == IMPLEMENTS) { @@ -3647,7 +3708,7 @@ public class JavacParser implements Parser { nextToken(); } else { // method returns types are un-annotated types - type = unannotatedType(); + type = unannotatedType(false); } if (token.kind == LPAREN && !isInterface && type.hasTag(IDENT)) { if (isInterface || tk.name() != className) @@ -3667,7 +3728,7 @@ public class JavacParser implements Parser { } else if (!isVoid && typarams.isEmpty()) { List defs = variableDeclaratorsRest(pos, mods, type, name, isInterface, dc, - new ListBuffer()).toList(); + new ListBuffer(), false).toList(); accept(SEMI); storeEnd(defs.last(), S.prevToken().endPos); return defs; @@ -3809,7 +3870,7 @@ public class JavacParser implements Parser { JCTypeParameter typeParameter() { int pos = token.pos; List annos = typeAnnotationsOpt(); - Name name = ident(); + Name name = typeName(); ListBuffer bounds = new ListBuffer<>(); if (token.kind == EXTENDS) { nextToken(); diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties index 0333cf7c1bb..c04e2560486 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties @@ -1190,6 +1190,47 @@ compiler.err.io.exception=\ compiler.err.undef.label=\ undefined label: {0} +# 0: name (type) +compiler.err.illegal.ref.to.var.type=\ + illegal reference to restricted type ''{0}'' + +# 0: token +compiler.err.var.not.allowed=\ + ''{0}'' not allowed here\n\ + as of release 10, ''{0}'' is a restricted local variable type and cannot be used for type declarations + +# 0: name (variable), 1: message segment +compiler.err.cant.infer.local.var.type=\ + cannot infer type for local variable {0}\n\ + ({1}) + +compiler.err.var.not.allowed.here=\ + ''var'' is not allowed here + +compiler.err.var.not.allowed.array=\ + ''var'' is not allowed as an element type of an array + +compiler.err.var.not.allowed.compound=\ + ''var'' is not allowed in a compound declaration + +compiler.misc.local.cant.infer.null=\ + variable initializer is ''null'' + +compiler.misc.local.missing.init=\ + cannot use ''var'' on variable without initializer + +compiler.misc.local.lambda.missing.target=\ + lambda expression needs an explicit target-type + +compiler.misc.local.mref.missing.target=\ + method reference needs an explicit target-type + +compiler.misc.local.array.missing.target=\ + array initializer needs an explicit target-type + +compiler.misc.local.self.ref=\ + cannot use ''var'' on self-referencing variable + # 0: message segment, 1: unused compiler.err.cant.apply.diamond=\ cannot infer type arguments for {0} @@ -1873,6 +1914,9 @@ compiler.warn.raw.class.use=\ compiler.warn.diamond.redundant.args=\ Redundant type arguments in new expression (use diamond operator instead). +compiler.warn.local.redundant.type=\ + Redundant type for local variable (replace explicit type with ''var''). + compiler.warn.potential.lambda.found=\ This anonymous inner class creation can be turned into a lambda expression. diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/JCTree.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/JCTree.java index 054a5b6ba34..afbfef6a283 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/JCTree.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/JCTree.java @@ -946,6 +946,10 @@ public abstract class JCTree implements Tree, Cloneable, DiagnosticPosition { } } + public boolean isImplicitlyTyped() { + return vartype == null; + } + @Override public void accept(Visitor v) { v.visitVarDef(this); } diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/Pretty.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/Pretty.java index a2258e9c4e4..cc46e4e75a8 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/Pretty.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/Pretty.java @@ -1359,7 +1359,7 @@ public class Pretty extends JCTree.Visitor { // Prints the inner element type of a nested array private void printBaseElementType(JCTree tree) throws IOException { - printExpr(TreeInfo.innermostType(tree)); + printExpr(TreeInfo.innermostType(tree, false)); } // prints the brackets of a nested array in reverse order diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/TreeInfo.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/TreeInfo.java index 1796c4dd890..4dbbe0c64b3 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/TreeInfo.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/tree/TreeInfo.java @@ -1136,7 +1136,7 @@ public class TreeInfo { * For an array that contains an annotated type, return that annotated type. * TODO: currently only used by Pretty. Describe behavior better. */ - public static JCTree innermostType(JCTree type) { + public static JCTree innermostType(JCTree type, boolean skipAnnos) { JCTree lastAnnotatedType = null; JCTree cur = type; loop: while (true) { @@ -1157,7 +1157,7 @@ public class TreeInfo { break loop; } } - if (lastAnnotatedType!=null) { + if (!skipAnnos && lastAnnotatedType!=null) { return lastAnnotatedType; } else { return cur; diff --git a/src/jdk.compiler/share/classes/com/sun/tools/javac/util/Names.java b/src/jdk.compiler/share/classes/com/sun/tools/javac/util/Names.java index afd3112693e..c3eeaf5c29b 100644 --- a/src/jdk.compiler/share/classes/com/sun/tools/javac/util/Names.java +++ b/src/jdk.compiler/share/classes/com/sun/tools/javac/util/Names.java @@ -63,6 +63,7 @@ public class Names { public final Name _default; public final Name _super; public final Name _this; + public final Name var; public final Name exports; public final Name opens; public final Name module; @@ -224,6 +225,7 @@ public class Names { _default = fromString("default"); _super = fromString("super"); _this = fromString("this"); + var = fromString("var"); exports = fromString("exports"); opens = fromString("opens"); module = fromString("module"); diff --git a/src/jdk.compiler/share/classes/module-info.java b/src/jdk.compiler/share/classes/module-info.java index 8283f5a0acc..95605477473 100644 --- a/src/jdk.compiler/share/classes/module-info.java +++ b/src/jdk.compiler/share/classes/module-info.java @@ -106,7 +106,8 @@ module jdk.compiler { exports com.sun.tools.javac.jvm to jdk.javadoc; exports com.sun.tools.javac.main to - jdk.javadoc; + jdk.javadoc, + jdk.jshell; exports com.sun.tools.javac.model to jdk.javadoc; exports com.sun.tools.javac.parser to diff --git a/src/jdk.jshell/share/classes/jdk/jshell/Eval.java b/src/jdk.jshell/share/classes/jdk/jshell/Eval.java index e50d594d901..267bce7fb1d 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/Eval.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/Eval.java @@ -40,6 +40,7 @@ import com.sun.source.tree.ExpressionTree; import com.sun.source.tree.IdentifierTree; import com.sun.source.tree.MethodTree; import com.sun.source.tree.ModifiersTree; +import com.sun.source.tree.NewClassTree; import com.sun.source.tree.Tree; import com.sun.source.tree.VariableTree; import com.sun.tools.javac.tree.JCTree; @@ -59,6 +60,7 @@ import jdk.jshell.TaskFactory.AnalyzeTask; import jdk.jshell.TaskFactory.BaseTask; import jdk.jshell.TaskFactory.CompileTask; import jdk.jshell.TaskFactory.ParseTask; +import jdk.jshell.Wrap.CompoundWrap; import jdk.jshell.Wrap.Range; import jdk.jshell.Snippet.Status; import jdk.jshell.spi.ExecutionControl.ClassBytecodes; @@ -274,26 +276,119 @@ class Eval { for (Tree unitTree : units) { VariableTree vt = (VariableTree) unitTree; String name = vt.getName().toString(); - String typeName = EvalPretty.prettyExpr((JCTree) vt.getType(), false); - Tree baseType = vt.getType(); + String typeName; + String fullTypeName; TreeDependencyScanner tds = new TreeDependencyScanner(); - tds.scan(baseType); // Not dependent on initializer + Wrap typeWrap; + Wrap anonDeclareWrap = null; + Wrap winit = null; StringBuilder sbBrackets = new StringBuilder(); - while (baseType instanceof ArrayTypeTree) { - //TODO handle annotations too - baseType = ((ArrayTypeTree) baseType).getType(); - sbBrackets.append("[]"); + Tree baseType = vt.getType(); + if (baseType != null) { + tds.scan(baseType); // Not dependent on initializer + fullTypeName = typeName = EvalPretty.prettyExpr((JCTree) vt.getType(), false); + while (baseType instanceof ArrayTypeTree) { + //TODO handle annotations too + baseType = ((ArrayTypeTree) baseType).getType(); + sbBrackets.append("[]"); + } + Range rtype = dis.treeToRange(baseType); + typeWrap = Wrap.rangeWrap(compileSource, rtype); + } else { + Tree init = vt.getInitializer(); + if (init != null) { + Range rinit = dis.treeToRange(init); + String initCode = rinit.part(compileSource); + ExpressionInfo ei = + ExpressionToTypeInfo.localVariableTypeForInitializer(initCode, state); + typeName = ei == null ? "java.lang.Object" : ei.typeName; + fullTypeName = ei == null ? "java.lang.Object" : ei.fullTypeName; + if (ei != null && init.getKind() == Tree.Kind.NEW_CLASS && + ((NewClassTree) init).getClassBody() != null) { + NewClassTree nct = (NewClassTree) init; + StringBuilder constructor = new StringBuilder(); + constructor.append(fullTypeName).append("("); + String sep = ""; + if (ei.enclosingInstanceType != null) { + constructor.append(ei.enclosingInstanceType); + constructor.append(" encl"); + sep = ", "; + } + int idx = 0; + for (String type : ei.parameterTypes) { + constructor.append(sep); + constructor.append(type); + constructor.append(" "); + constructor.append("arg" + idx++); + sep = ", "; + } + if (ei.enclosingInstanceType != null) { + constructor.append(") { encl.super ("); + } else { + constructor.append(") { super ("); + } + sep = ""; + for (int i = 0; i < idx; i++) { + constructor.append(sep); + constructor.append("arg" + i++); + sep = ", "; + } + constructor.append("); }"); + List members = nct.getClassBody().getMembers(); + Range bodyRange = dis.treeListToRange(members); + Wrap bodyWrap; + + if (bodyRange != null) { + bodyWrap = Wrap.rangeWrap(compileSource, bodyRange); + } else { + bodyWrap = Wrap.simpleWrap(" "); + } + + Range argRange = dis.treeListToRange(nct.getArguments()); + Wrap argWrap; + + if (argRange != null) { + argWrap = Wrap.rangeWrap(compileSource, argRange); + } else { + argWrap = Wrap.simpleWrap(" "); + } + + if (ei.enclosingInstanceType != null) { + Range enclosingRanges = + dis.treeToRange(nct.getEnclosingExpression()); + Wrap enclosingWrap = Wrap.rangeWrap(compileSource, enclosingRanges); + argWrap = argRange != null ? new CompoundWrap(enclosingWrap, + Wrap.simpleWrap(","), + argWrap) + : enclosingWrap; + } + Wrap hwrap = Wrap.simpleWrap("public static class " + fullTypeName + + (ei.isClass ? " extends " : " implements ") + + typeName + " { " + constructor); + anonDeclareWrap = new CompoundWrap(hwrap, bodyWrap, Wrap.simpleWrap("}")); + winit = new CompoundWrap("new " + fullTypeName + "(", argWrap, ")"); + + String superType = typeName; + + typeName = fullTypeName; + fullTypeName = ei.isClass ? "" + : ""; + } + tds.scan(init); + } else { + fullTypeName = typeName = "java.lang.Object"; + } + typeWrap = Wrap.identityWrap(typeName); } - Range rtype = dis.treeToRange(baseType); Range runit = dis.treeToRange(vt); runit = new Range(runit.begin, runit.end - 1); ExpressionTree it = vt.getInitializer(); - Range rinit = null; int nameMax = runit.end - 1; SubKind subkind; if (it != null) { subkind = SubKind.VAR_DECLARATION_WITH_INITIALIZER_SUBKIND; - rinit = dis.treeToRange(it); + Range rinit = dis.treeToRange(it); + winit = winit == null ? Wrap.rangeWrap(compileSource, rinit) : winit; nameMax = rinit.begin - 1; } else { subkind = SubKind.VAR_DECLARATION_SUBKIND; @@ -304,10 +399,11 @@ class Eval { } int nameEnd = nameStart + name.length(); Range rname = new Range(nameStart, nameEnd); - Wrap guts = Wrap.varWrap(compileSource, rtype, sbBrackets.toString(), rname, rinit); - DiagList modDiag = modifierDiagnostics(vt.getModifiers(), dis, true); + Wrap guts = Wrap.varWrap(compileSource, typeWrap, sbBrackets.toString(), rname, + winit, anonDeclareWrap); + DiagList modDiag = modifierDiagnostics(vt.getModifiers(), dis, true); Snippet snip = new VarSnippet(state.keyMap.keyForVariable(name), userSource, guts, - name, subkind, typeName, + name, subkind, fullTypeName, tds.declareReferences(), modDiag); snippets.add(snip); } diff --git a/src/jdk.jshell/share/classes/jdk/jshell/ExpressionToTypeInfo.java b/src/jdk.jshell/share/classes/jdk/jshell/ExpressionToTypeInfo.java index f33c37133a6..86480ab394b 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/ExpressionToTypeInfo.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/ExpressionToTypeInfo.java @@ -29,14 +29,20 @@ import com.sun.source.tree.ReturnTree; import com.sun.source.tree.ClassTree; import com.sun.source.tree.CompilationUnitTree; import com.sun.source.tree.ConditionalExpressionTree; +import com.sun.source.tree.ExpressionStatementTree; import com.sun.source.tree.ExpressionTree; +import com.sun.source.tree.MethodInvocationTree; import com.sun.source.tree.MethodTree; +import com.sun.source.tree.NewClassTree; import com.sun.source.tree.Tree; +import com.sun.source.tree.Tree.Kind; +import com.sun.source.tree.VariableTree; import com.sun.source.util.TreePath; import com.sun.source.util.TreePathScanner; import com.sun.tools.javac.code.Symtab; import com.sun.tools.javac.code.Type; import com.sun.tools.javac.code.Types; +import com.sun.tools.javac.util.List; import jdk.jshell.TaskFactory.AnalyzeTask; /** @@ -63,6 +69,10 @@ class ExpressionToTypeInfo { public static class ExpressionInfo { ExpressionTree tree; String typeName; + String fullTypeName; + List parameterTypes; + String enclosingInstanceType; + boolean isClass; boolean isNonVoid; } @@ -111,6 +121,16 @@ class ExpressionToTypeInfo { return null; } } + + @Override + public TreePath visitVariable(VariableTree node, Boolean isTargetContext) { + if (isTargetContext) { + throw new Result(getCurrentPath()); + } else { + return null; + } + } + } private Type pathToType(TreePath tp) { @@ -156,6 +176,30 @@ class ExpressionToTypeInfo { } } + /** + * Entry method: get expression info corresponding to a local variable declaration if its type + * has been inferred automatically from the given initializer. + * @param code the initializer as a string + * @param state a JShell instance + * @return type information + */ + public static ExpressionInfo localVariableTypeForInitializer(String code, JShell state) { + if (code == null || code.isEmpty()) { + return null; + } + try { + OuterWrap codeWrap = state.outerMap.wrapInTrialClass(Wrap.methodWrap("var $$$ = " + code)); + AnalyzeTask at = state.taskFactory.new AnalyzeTask(codeWrap); + CompilationUnitTree cu = at.firstCuTree(); + if (at.hasErrors() || cu == null) { + return null; + } + return new ExpressionToTypeInfo(at, cu, state).typeOfExpression(); + } catch (Exception ex) { + return null; + } + } + private ExpressionInfo typeOfExpression() { return treeToInfo(findExpressionPath()); } @@ -172,9 +216,11 @@ class ExpressionToTypeInfo { private ExpressionInfo treeToInfo(TreePath tp) { if (tp != null) { Tree tree = tp.getLeaf(); - if (tree instanceof ExpressionTree) { + boolean isExpression = tree instanceof ExpressionTree; + if (isExpression || tree.getKind() == Kind.VARIABLE) { ExpressionInfo ei = new ExpressionInfo(); - ei.tree = (ExpressionTree) tree; + if (isExpression) + ei.tree = (ExpressionTree) tree; Type type = pathToType(tp, tree); if (type != null) { switch (type.getKind()) { @@ -189,27 +235,56 @@ class ExpressionToTypeInfo { break; default: { ei.isNonVoid = true; - ei.typeName = varTypeName(type); - if (ei.typeName == null) { - ei.typeName = OBJECT_TYPE_NAME; - } + ei.typeName = varTypeName(type, false); + ei.fullTypeName = varTypeName(type, true); break; } } } + if (tree.getKind() == Tree.Kind.VARIABLE) { + Tree init = ((VariableTree) tree).getInitializer(); + if (init.getKind() == Tree.Kind.NEW_CLASS && + ((NewClassTree) init).getClassBody() != null) { + NewClassTree nct = (NewClassTree) init; + ClassTree clazz = nct.getClassBody(); + MethodTree constructor = (MethodTree) clazz.getMembers().get(0); + ExpressionStatementTree superCallStatement = + (ExpressionStatementTree) constructor.getBody().getStatements().get(0); + MethodInvocationTree superCall = + (MethodInvocationTree) superCallStatement.getExpression(); + TreePath superCallPath = + at.trees().getPath(tp.getCompilationUnit(), superCall.getMethodSelect()); + Type constrType = pathToType(superCallPath); + ei.parameterTypes = constrType.getParameterTypes() + .stream() + .map(t -> varTypeName(t, false)) + .collect(List.collector()); + if (nct.getEnclosingExpression() != null) { + TreePath enclPath = new TreePath(tp, nct.getEnclosingExpression()); + ei.enclosingInstanceType = varTypeName(pathToType(enclPath), false); + } + ei.isClass = at.task.getTypes().directSupertypes(type).size() == 1; + } + } return ei; } } return null; } - private String varTypeName(Type type) { + private String varTypeName(Type type, boolean printIntersectionTypes) { try { - TypePrinter tp = new VarTypePrinter(at.messages(), - state.maps::fullClassNameAndPackageToClass, syms, types); - return tp.toString(type); + TypePrinter tp = new TypePrinter(at.messages(), + state.maps::fullClassNameAndPackageToClass, printIntersectionTypes); + List captures = types.captures(type); + String res = tp.toString(types.upward(type, captures)); + + if (res == null) + res = OBJECT_TYPE_NAME; + + return res; } catch (Exception ex) { - return null; + return OBJECT_TYPE_NAME; } } diff --git a/src/jdk.jshell/share/classes/jdk/jshell/ReplParser.java b/src/jdk.jshell/share/classes/jdk/jshell/ReplParser.java index 1d98f6ad3b7..14c54b86507 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/ReplParser.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/ReplParser.java @@ -232,7 +232,7 @@ class ReplParser extends JavacParser { //mods.flags |= Flags.STATIC; List defs = variableDeclaratorsRest(pos, mods, t, name, false, dc, - new ListBuffer()).toList(); + new ListBuffer(), true).toList(); accept(SEMI); storeEnd(defs.last(), S.prevToken().endPos); return defs; diff --git a/src/jdk.jshell/share/classes/jdk/jshell/SourceCodeAnalysisImpl.java b/src/jdk.jshell/share/classes/jdk/jshell/SourceCodeAnalysisImpl.java index 913dfbe53b8..957ad76643b 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/SourceCodeAnalysisImpl.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/SourceCodeAnalysisImpl.java @@ -134,6 +134,8 @@ import static jdk.jshell.TreeDissector.printType; import static java.util.stream.Collectors.joining; +import javax.lang.model.type.IntersectionType; + /** * The concrete implementation of SourceCodeAnalysis. * @author Robert Field @@ -715,6 +717,13 @@ class SourceCodeAnalysisImpl extends SourceCodeAnalysis { return Collections.emptyList(); switch (site.getKind()) { + case INTERSECTION: { + List result = new ArrayList<>(); + for (TypeMirror bound : ((IntersectionType) site).getBounds()) { + result.addAll(membersOf(at, bound, shouldGenerateDotClassItem)); + } + return result; + } case DECLARED: { TypeElement element = (TypeElement) at.getTypes().asElement(site); List result = new ArrayList<>(); diff --git a/src/jdk.jshell/share/classes/jdk/jshell/TaskFactory.java b/src/jdk.jshell/share/classes/jdk/jshell/TaskFactory.java index 53ed54c73e6..d81d8483cc6 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/TaskFactory.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/TaskFactory.java @@ -61,7 +61,20 @@ import javax.lang.model.util.Elements; import javax.tools.FileObject; import jdk.jshell.MemoryFileManager.SourceMemoryJavaFileObject; import java.lang.Runtime.Version; +import java.nio.CharBuffer; import com.sun.source.tree.Tree.Kind; +import com.sun.tools.javac.code.Kinds; +import com.sun.tools.javac.code.Symbol.ClassSymbol; +import com.sun.tools.javac.code.Symbol.VarSymbol; +import com.sun.tools.javac.parser.Parser; +import com.sun.tools.javac.tree.JCTree.JCClassDecl; +import com.sun.tools.javac.tree.JCTree.JCCompilationUnit; +import com.sun.tools.javac.tree.JCTree.JCExpression; +import com.sun.tools.javac.tree.JCTree.JCTypeCast; +import com.sun.tools.javac.tree.JCTree.Tag; +import com.sun.tools.javac.util.Context.Factory; +import com.sun.tools.javac.util.Log.DiscardDiagnosticHandler; +import jdk.jshell.Snippet.Status; /** * The primary interface to the compiler API. Parsing, analysis, and @@ -355,6 +368,7 @@ class TaskFactory { Iterable compilationUnits = inputs .map(in -> sh.sourceToFileObject(fileManager, in)) .collect(Collectors.toList()); + JShellJavaCompiler.preRegister(context, state); this.task = (JavacTaskImpl) ((JavacTool) compiler).getTask(null, fileManager, diagnostics, options, null, compilationUnits, context); @@ -464,4 +478,57 @@ class TaskFactory { } } + private static final class JShellJavaCompiler extends com.sun.tools.javac.main.JavaCompiler { + + public static void preRegister(Context c, JShell state) { + c.put(compilerKey, (Factory) i -> new JShellJavaCompiler(i, state)); + } + + private final JShell state; + + public JShellJavaCompiler(Context context, JShell state) { + super(context); + this.state = state; + } + + @Override + public void processAnnotations(com.sun.tools.javac.util.List roots, Collection classnames) { + super.processAnnotations(roots, classnames); + state.maps + .snippetList() + .stream() + .filter(s -> s.status() == Status.VALID) + .filter(s -> s.kind() == Snippet.Kind.VAR) + .filter(s -> s.subKind() == Snippet.SubKind.VAR_DECLARATION_WITH_INITIALIZER_SUBKIND) + .forEach(s -> setVariableType(roots, (VarSnippet) s)); + } + + private void setVariableType(com.sun.tools.javac.util.List roots, VarSnippet s) { + ClassSymbol clazz = syms.getClass(syms.unnamedModule, names.fromString(s.classFullName())); + if (clazz == null || !clazz.isCompleted()) + return; + VarSymbol field = (VarSymbol) clazz.members().findFirst(names.fromString(s.name()), sym -> sym.kind == Kinds.Kind.VAR); + if (field != null) { + JavaFileObject prev = log.useSource(null); + DiscardDiagnosticHandler h = new DiscardDiagnosticHandler(log); + try { + String typeName = s.typeName(); + CharBuffer buf = CharBuffer.wrap(("(" + typeName +")x\u0000").toCharArray(), 0, typeName.length() + 3); + Parser parser = parserFactory.newParser(buf, false, false, false); + JCExpression expr = parser.parseExpression(); + if (expr.hasTag(Tag.TYPECAST)) { + JCTypeCast tree = (JCTypeCast) expr; + if (tree.clazz.hasTag(Tag.TYPEINTERSECTION)) { + field.type = attr.attribType(tree.clazz, + ((JCClassDecl) roots.head.getTypeDecls().head).sym); + } + } + } finally { + log.popDiagnosticHandler(h); + log.useSource(prev); + } + } + } + } + } diff --git a/src/jdk.jshell/share/classes/jdk/jshell/TreeDissector.java b/src/jdk.jshell/share/classes/jdk/jshell/TreeDissector.java index b177bf2db02..da1dff1b204 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/TreeDissector.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/TreeDissector.java @@ -227,7 +227,7 @@ class TreeDissector { Type typeImpl = (Type) type; try { TypePrinter tp = new TypePrinter(at.messages(), - state.maps::fullClassNameAndPackageToClass); + state.maps::fullClassNameAndPackageToClass, true); return tp.toString(typeImpl); } catch (Exception ex) { return null; diff --git a/src/jdk.jshell/share/classes/jdk/jshell/TypePrinter.java b/src/jdk.jshell/share/classes/jdk/jshell/TypePrinter.java index b5ae176fbf5..5a622460947 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/TypePrinter.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/TypePrinter.java @@ -32,9 +32,11 @@ import com.sun.tools.javac.code.Symbol.ClassSymbol; import com.sun.tools.javac.code.Symbol.PackageSymbol; import com.sun.tools.javac.code.Type; import com.sun.tools.javac.code.Type.ClassType; +import com.sun.tools.javac.code.Type.IntersectionClassType; import com.sun.tools.javac.util.JavacMessages; import java.util.Locale; import java.util.function.BinaryOperator; +import java.util.stream.Collectors; /** * Print types in source form. @@ -45,10 +47,14 @@ class TypePrinter extends Printer { private final JavacMessages messages; private final BinaryOperator fullClassNameAndPackageToClass; + private final boolean printEnhancedTypes; - TypePrinter(JavacMessages messages, BinaryOperator fullClassNameAndPackageToClass) { + TypePrinter(JavacMessages messages, + BinaryOperator fullClassNameAndPackageToClass, + boolean printEnhancedTypes) { this.messages = messages; this.fullClassNameAndPackageToClass = fullClassNameAndPackageToClass; + this.printEnhancedTypes = printEnhancedTypes; } String toString(Type t) { @@ -92,8 +98,18 @@ class TypePrinter extends Printer { protected String className(ClassType t, boolean longform, Locale locale) { Symbol sym = t.tsym; if (sym.name.length() == 0 && (sym.flags() & COMPOUND) != 0) { - return OBJECT; + if (printEnhancedTypes) { + return ((IntersectionClassType) t).getExplicitComponents() + .stream() + .map(i -> visit(i, locale)) + .collect(Collectors.joining("&")); + } else { + return OBJECT; + } } else if (sym.name.length() == 0) { + if (printEnhancedTypes) { + return t.tsym.flatName().toString().substring(t.tsym.outermostClass().flatName().length()); + } // Anonymous String s; ClassType norm = (ClassType) t.tsym.type; diff --git a/src/jdk.jshell/share/classes/jdk/jshell/VarTypePrinter.java b/src/jdk.jshell/share/classes/jdk/jshell/VarTypePrinter.java deleted file mode 100644 index 4d69b1ab71a..00000000000 --- a/src/jdk.jshell/share/classes/jdk/jshell/VarTypePrinter.java +++ /dev/null @@ -1,265 +0,0 @@ -/* - * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. - * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. - * - * This code is free software; you can redistribute it and/or modify it - * under the terms of the GNU General Public License version 2 only, as - * published by the Free Software Foundation. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * This code is distributed in the hope that it will be useful, but WITHOUT - * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or - * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License - * version 2 for more details (a copy is included in the LICENSE file that - * accompanied this code). - * - * You should have received a copy of the GNU General Public License version - * 2 along with this work; if not, write to the Free Software Foundation, - * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. - * - * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA - * or visit www.oracle.com if you need additional information or have any - * questions. - */ - -package jdk.jshell; - -import java.util.HashSet; -import com.sun.tools.javac.code.Type; -import com.sun.tools.javac.code.Type.ClassType; -import com.sun.tools.javac.util.JavacMessages; -import java.util.Locale; -import java.util.Set; -import java.util.function.BinaryOperator; -import com.sun.tools.javac.code.BoundKind; -import com.sun.tools.javac.code.Flags; -import com.sun.tools.javac.code.Symtab; -import com.sun.tools.javac.code.Type.CapturedType; -import com.sun.tools.javac.code.Type.StructuralTypeMapping; -import com.sun.tools.javac.code.Type.TypeVar; -import com.sun.tools.javac.code.Type.WildcardType; -import com.sun.tools.javac.code.Types; -import com.sun.tools.javac.code.Types.SimpleVisitor; -import com.sun.tools.javac.util.List; -import static com.sun.tools.javac.code.BoundKind.EXTENDS; -import static com.sun.tools.javac.code.BoundKind.SUPER; -import static com.sun.tools.javac.code.BoundKind.UNBOUND; -import static com.sun.tools.javac.code.Type.ArrayType; -import static com.sun.tools.javac.code.TypeTag.BOT; -import static com.sun.tools.javac.code.TypeTag.WILDCARD; - -/** - * Print variable types in source form. - * TypeProjection and CaptureScanner are copied from Types in the JEP-286 - * Sandbox by Maurizio. The checks for Non-Denotable in TypePrinter are - * cribbed from denotableChecker of the same source. - * - * @author Maurizio Cimadamore - * @author Robert Field - */ -class VarTypePrinter extends TypePrinter { - private static final String WILD = "?"; - - private final Symtab syms; - private final Types types; - - VarTypePrinter(JavacMessages messages, BinaryOperator fullClassNameAndPackageToClass, - Symtab syms, Types types) { - super(messages, fullClassNameAndPackageToClass); - this.syms = syms; - this.types = types; - } - - @Override - String toString(Type t) { - return super.toString(upward(t)); - } - - @Override - public String visitTypeVar(TypeVar t, Locale locale) { - /* Any type variable mentioned in the inferred type must have been declared as a type parameter - (i.e cannot have been produced by inference (18.4)) - */ - // and beyond that, there are no global type vars, so if there are any - // type variables left, they need to be eliminated - return WILD; // Non-denotable - } - - @Override - public String visitCapturedType(CapturedType t, Locale locale) { - /* Any type variable mentioned in the inferred type must have been declared as a type parameter - (i.e cannot have been produced by capture conversion (5.1.10)) - */ - return WILD; // Non-denotable - } - - public Type upward(Type t) { - List captures = captures(t); - return upward(t, captures); - } - - /************* Following from JEP-286 Types.java ***********/ - - public Type upward(Type t, List vars) { - return t.map(new TypeProjection(vars), true); - } - - public List captures(Type t) { - CaptureScanner cs = new CaptureScanner(); - Set captures = new HashSet<>(); - cs.visit(t, captures); - return List.from(captures); - } - - class CaptureScanner extends SimpleVisitor> { - - @Override - public Void visitType(Type t, Set types) { - return null; - } - - @Override - public Void visitClassType(ClassType t, Set seen) { - if (t.isCompound()) { - types.directSupertypes(t).forEach(s -> visit(s, seen)); - } else { - t.allparams().forEach(ta -> visit(ta, seen)); - } - return null; - } - - @Override - public Void visitArrayType(ArrayType t, Set seen) { - return visit(t.elemtype, seen); - } - - @Override - public Void visitWildcardType(WildcardType t, Set seen) { - visit(t.type, seen); - return null; - } - - @Override - public Void visitTypeVar(TypeVar t, Set seen) { - if ((t.tsym.flags() & Flags.SYNTHETIC) != 0 && seen.add(t)) { - visit(t.getUpperBound(), seen); - } - return null; - } - - @Override - public Void visitCapturedType(CapturedType t, Set seen) { - if (seen.add(t)) { - visit(t.getUpperBound(), seen); - visit(t.getLowerBound(), seen); - } - return null; - } - } - - class TypeProjection extends StructuralTypeMapping { - - List vars; - Set seen = new HashSet<>(); - - public TypeProjection(List vars) { - this.vars = vars; - } - - @Override - public Type visitClassType(ClassType t, Boolean upward) { - if (upward && !t.isCompound() && t.tsym.name.isEmpty()) { - //lift anonymous class type to first supertype (class or interface) - return types.directSupertypes(t).last(); - } else if (t.isCompound()) { - List components = types.directSupertypes(t); - List components1 = components.map(c -> c.map(this, upward)); - if (components == components1) return t; - else return types.makeIntersectionType(components1); - } else { - Type outer = t.getEnclosingType(); - Type outer1 = visit(outer, upward); - List typarams = t.getTypeArguments(); - List typarams1 = typarams.map(ta -> mapTypeArgument(ta, upward)); - if (typarams1.stream().anyMatch(ta -> ta.hasTag(BOT))) { - //not defined - return syms.botType; - } - if (outer1 == outer && typarams1 == typarams) return t; - else return new ClassType(outer1, typarams1, t.tsym, t.getMetadata()) { - @Override - protected boolean needsStripping() { - return true; - } - }; - } - } - - protected Type makeWildcard(Type upper, Type lower) { - BoundKind bk; - Type bound; - if (upper.hasTag(BOT)) { - upper = syms.objectType; - } - boolean isUpperObject = types.isSameType(upper, syms.objectType); - if (!lower.hasTag(BOT) && isUpperObject) { - bound = lower; - bk = SUPER; - } else { - bound = upper; - bk = isUpperObject ? UNBOUND : EXTENDS; - } - return new WildcardType(bound, bk, syms.boundClass); - } - - @Override - public Type visitTypeVar(TypeVar t, Boolean upward) { - if (vars.contains(t)) { - try { - if (seen.add(t)) { - return (upward ? - t.getUpperBound() : - (t.getLowerBound() == null) ? - syms.botType : - t.getLowerBound()) - .map(this, upward); - } else { - //cycle - return syms.objectType; - } - } finally { - seen.remove(t); - } - } else { - return t; - } - } - - @Override - public Type visitWildcardType(WildcardType wt, Boolean upward) { - if (upward) { - return wt.isExtendsBound() ? - wt.type.map(this, upward) : - syms.objectType; - } else { - return wt.isSuperBound() ? - wt.type.map(this, upward) : - syms.botType; - } - } - - private Type mapTypeArgument(Type t, boolean upward) { - if (!t.containsAny(vars)) { - return t; - } else if (!t.hasTag(WILDCARD) && !upward) { - //not defined - return syms.botType; - } else { - Type upper = t.map(this, upward); - Type lower = t.map(this, !upward); - return makeWildcard(upper, lower); - } - } - } -} diff --git a/src/jdk.jshell/share/classes/jdk/jshell/Wrap.java b/src/jdk.jshell/share/classes/jdk/jshell/Wrap.java index 053e9858079..05998afe263 100644 --- a/src/jdk.jshell/share/classes/jdk/jshell/Wrap.java +++ b/src/jdk.jshell/share/classes/jdk/jshell/Wrap.java @@ -74,16 +74,15 @@ abstract class Wrap implements GeneralWrap { * @param rdecl Type name and name * @return */ - public static Wrap varWrap(String source, Range rtype, String brackets, Range rname, Range rinit) { + public static Wrap varWrap(String source, Wrap wtype, String brackets, + Range rname, Wrap winit, Wrap anonDeclareWrap) { RangeWrap wname = new RangeWrap(source, rname); - RangeWrap wtype = new RangeWrap(source, rtype); Wrap wVarDecl = new VarDeclareWrap(wtype, brackets, wname); Wrap wmeth; - if (rinit == null) { + if (winit == null) { wmeth = new CompoundWrap(new NoWrap(" "), " return null;\n"); } else { - RangeWrap winit = new RangeWrap(source, rinit); // int x = y // int x_ = y; return x = x_; // decl + "_ = " + init ; + "return " + name + "=" + name + "_ ;" @@ -93,7 +92,8 @@ abstract class Wrap implements GeneralWrap { ); } Wrap wInitMeth = new DoitMethodWrap(wmeth); - return new CompoundWrap(wVarDecl, wInitMeth); + return anonDeclareWrap != null ? new CompoundWrap(wVarDecl, wInitMeth, anonDeclareWrap) + : new CompoundWrap(wVarDecl, wInitMeth); } public static Wrap tempVarWrap(String source, String typename, String name) { @@ -112,6 +112,14 @@ abstract class Wrap implements GeneralWrap { return new NoWrap(source); } + public static Wrap identityWrap(String source) { + return new NoWrap(source); + } + + public static Wrap rangeWrap(String source, Range range) { + return new RangeWrap(source, range); + } + public static Wrap classMemberWrap(String source) { Wrap w = new NoWrap(source); return new CompoundWrap(" public static\n ", w); diff --git a/test/langtools/jdk/jshell/CompletionSuggestionTest.java b/test/langtools/jdk/jshell/CompletionSuggestionTest.java index 402fb92cd0f..912de602014 100644 --- a/test/langtools/jdk/jshell/CompletionSuggestionTest.java +++ b/test/langtools/jdk/jshell/CompletionSuggestionTest.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8131025 8141092 8153761 8145263 8131019 8175886 8176184 8176241 8176110 + * @bug 8131025 8141092 8153761 8145263 8131019 8175886 8176184 8176241 8176110 8177466 * @summary Test Completion and Documentation * @library /tools/lib * @modules jdk.compiler/com.sun.tools.javac.api @@ -645,6 +645,22 @@ public class CompletionSuggestionTest extends KullaTesting { assertCompletion("Foo.m(new Baz<>(|", true, "str"); } + public void testIntersection() { + assertEval(" Z get() { return null; }"); + assertEval("var v = get();"); + assertCompletionIncludesExcludes("v.|", true, Set.of("run()", "length()"), Set.of()); + assertCompletion("Runnable r = |", true, "get()", "v"); + assertCompletion("CharSequence r = |", true, "get()", "v"); + assertCompletion("Number r = |", true); + } + + public void testAnonymous() { + assertEval("var v = new Runnable() { public void run() { } public int length() { return 0; } };"); + assertCompletionIncludesExcludes("v.|", true, Set.of("run()", "length()"), Set.of()); + assertCompletion("Runnable r = |", true, "v"); + assertCompletion("CharSequence r = |", true); + } + @BeforeMethod public void setUp() { super.setUp(); diff --git a/test/langtools/jdk/jshell/ToolSimpleTest.java b/test/langtools/jdk/jshell/ToolSimpleTest.java index 56fd594273b..426eee3974d 100644 --- a/test/langtools/jdk/jshell/ToolSimpleTest.java +++ b/test/langtools/jdk/jshell/ToolSimpleTest.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8153716 8143955 8151754 8150382 8153920 8156910 8131024 8160089 8153897 8167128 8154513 8170015 8170368 8172102 8172103 8165405 8173073 8173848 8174041 8173916 8174028 8174262 8174797 8177079 8180508 + * @bug 8153716 8143955 8151754 8150382 8153920 8156910 8131024 8160089 8153897 8167128 8154513 8170015 8170368 8172102 8172103 8165405 8173073 8173848 8174041 8173916 8174028 8174262 8174797 8177079 8180508 8177466 * @summary Simple jshell tool tests * @modules jdk.compiler/com.sun.tools.javac.api * jdk.compiler/com.sun.tools.javac.main @@ -740,4 +740,26 @@ public class ToolSimpleTest extends ReplToolTesting { (a) -> assertCommandOutputContains(a, "1234", "==> 1234") ); } + + @Test + public void testIntersection() { + test( + (a) -> assertCommandOutputContains(a, " Z get1() { return null; }", "get1()"), + (a) -> assertCommandOutputContains(a, "var g1 = get1()", "g1"), + (a) -> assertCommand(a, "/vars g1", "| CharSequence&Runnable g1 = null"), + (a) -> assertCommandOutputContains(a, " Z get2() { return null; }", "get2()"), + (a) -> assertCommandOutputContains(a, "var g2 = get2()", "g2"), + (a) -> assertCommand(a, "/vars g2", "| Number&CharSequence g2 = null") + ); + } + + @Test + public void testAnonymous() { + test( + (a) -> assertCommandOutputContains(a, "var r1 = new Object() {}", "r1"), + (a) -> assertCommandOutputContains(a, "/vars r1", "| r1 = "), + (a) -> assertCommandOutputContains(a, "var r2 = new Runnable() { public void run() { } }", "r2"), + (a) -> assertCommandOutputContains(a, "/vars r2", "| r2 = ") + ); + } } diff --git a/test/langtools/jdk/jshell/VariablesTest.java b/test/langtools/jdk/jshell/VariablesTest.java index de5fc1fb973..cc7edae0a6d 100644 --- a/test/langtools/jdk/jshell/VariablesTest.java +++ b/test/langtools/jdk/jshell/VariablesTest.java @@ -23,7 +23,7 @@ /* * @test - * @bug 8144903 + * @bug 8144903 8177466 * @summary Tests for EvaluationState.variables * @build KullaTesting TestingInputStream ExpectedDiagnostic * @run testng VariablesTest @@ -337,4 +337,30 @@ public class VariablesTest extends KullaTesting { assertEquals(unr.get(0), "class undefined"); assertVariables(variable("undefined", "d")); } + + public void lvti() { + assertEval("var d = 234;", "234"); + assertEval("class Test { T[][] get() { return null; } }", added(VALID)); + assertEval("Test test() { return new Test<>(); }", added(VALID)); + assertEval("var t = test().get();", added(VALID)); + assertEval(" Z get1() { return null; }", added(VALID)); + assertEval("var i1 = get1();", added(VALID)); + assertEval("void t1() { i1.run(); i1.length(); }", added(VALID)); + assertEval("i1 = 1;", DiagCheck.DIAG_ERROR, DiagCheck.DIAG_OK, ste(MAIN_SNIPPET, NONEXISTENT, REJECTED, false, null)); + assertEval(" Z get2() { return null; }", added(VALID)); + assertEval("var i2 = get2();", added(VALID)); + assertEval("void t2() { i2.length(); }", added(VALID)); + assertEval("var r1 = new Runnable() { public void run() { } public String get() { return \"good\"; } };", added(VALID)); + assertEval("Runnable r2 = r1;"); + assertEval("r1.get()", "\"good\""); + assertEval("var v = r1.get();", "\"good\""); + assertEval("var r3 = new java.util.ArrayList(42) { public String get() { return \"good\"; } };", added(VALID)); + assertEval("r3.get()", "\"good\""); + assertEval("class O { public class Inner { public String test() { return \"good\"; } } }"); + assertEval("var r4 = new O().new Inner() { public String get() { return \"good\"; } };"); + assertEval("r4.get()", "\"good\""); + assertEval("class O2 { public class Inner { public Inner(int i) { } public String test() { return \"good\"; } } }"); + assertEval("var r5 = new O2().new Inner(1) { public String get() { return \"good\"; } };"); + assertEval("r5.get()", "\"good\""); + } } diff --git a/test/langtools/tools/javac/diags/examples/IllegalRefToVarType.java b/test/langtools/tools/javac/diags/examples/IllegalRefToVarType.java new file mode 100644 index 00000000000..6de5c58db67 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/IllegalRefToVarType.java @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.illegal.ref.to.var.type + +import java.util.List; + +class IllegalRefToVarType { + var list() { return null; } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalArrayMissingTarget.java b/test/langtools/tools/javac/diags/examples/LocalArrayMissingTarget.java new file mode 100644 index 00000000000..0719bc5c603 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalArrayMissingTarget.java @@ -0,0 +1,31 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.array.missing.target + +class LocalArrayMissingTarget { + void test() { + var x = { 1, 2, 3 }; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalCantInferNull.java b/test/langtools/tools/javac/diags/examples/LocalCantInferNull.java new file mode 100644 index 00000000000..69e41fb245a --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalCantInferNull.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.cant.infer.null + +class LocalCantInferNull { + void test() { + var s = null; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalLambdaMissingTarget.java b/test/langtools/tools/javac/diags/examples/LocalLambdaMissingTarget.java new file mode 100644 index 00000000000..49c58653c06 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalLambdaMissingTarget.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.lambda.missing.target + +class LocalLambdaMissingTarget { + void test() { + var x = () -> { }; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalMissingInit.java b/test/langtools/tools/javac/diags/examples/LocalMissingInit.java new file mode 100644 index 00000000000..eb0ced4c23e --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalMissingInit.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.missing.init + +class LocalMissingInit { + void test() { + var s; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalMrefMissingTarget.java b/test/langtools/tools/javac/diags/examples/LocalMrefMissingTarget.java new file mode 100644 index 00000000000..f05b490c5cc --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalMrefMissingTarget.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.mref.missing.target + +class LocalMrefMissingTarget { + void test() { + var x = this::test; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalRedundantType.java b/test/langtools/tools/javac/diags/examples/LocalRedundantType.java new file mode 100644 index 00000000000..6e7d4e524e2 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalRedundantType.java @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// options: -XDfind=local +// key: compiler.warn.local.redundant.type + +class LocalRedundantType { + void test() { + String s = ""; + } +} diff --git a/test/langtools/tools/javac/diags/examples/LocalSelfRef.java b/test/langtools/tools/javac/diags/examples/LocalSelfRef.java new file mode 100644 index 00000000000..bcbf8cb75bf --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/LocalSelfRef.java @@ -0,0 +1,35 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.cant.infer.local.var.type +// key: compiler.misc.local.self.ref + +class LocalSelfRef { + void test() { + var x = m(x); + } + + String m(String s) { return s; } +} diff --git a/test/langtools/tools/javac/diags/examples/VarNotAllowed.java b/test/langtools/tools/javac/diags/examples/VarNotAllowed.java new file mode 100644 index 00000000000..eb5afc8dbd4 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/VarNotAllowed.java @@ -0,0 +1,28 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.var.not.allowed + +class var { } diff --git a/test/langtools/tools/javac/diags/examples/VarNotAllowedArray.java b/test/langtools/tools/javac/diags/examples/VarNotAllowedArray.java new file mode 100644 index 00000000000..036f9a23224 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/VarNotAllowedArray.java @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.var.not.allowed.array + +class VarNotAllowedArray { + void test(String[] s) { + var[] x = s; + } +} diff --git a/test/langtools/tools/javac/diags/examples/VarNotAllowedCompound.java b/test/langtools/tools/javac/diags/examples/VarNotAllowedCompound.java new file mode 100644 index 00000000000..78edae64ee4 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/VarNotAllowedCompound.java @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.var.not.allowed.compound + +class VarNotAllowedCompound { + void test() { + var x = 1, y = 2; + } +} diff --git a/test/langtools/tools/javac/diags/examples/VarNotAllowedHere.java b/test/langtools/tools/javac/diags/examples/VarNotAllowedHere.java new file mode 100644 index 00000000000..03aa3174400 --- /dev/null +++ b/test/langtools/tools/javac/diags/examples/VarNotAllowedHere.java @@ -0,0 +1,30 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +// key: compiler.err.var.not.allowed.here + +class VarNotAllowedField { + var s = ""; +} diff --git a/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.java b/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.java new file mode 100644 index 00000000000..fac9de159fa --- /dev/null +++ b/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.java @@ -0,0 +1,33 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @compile/fail/ref=BadLocalVarInferenceTest.out -XDrawDiagnostics BadLocalVarInferenceTest.java + */ + +class BadLocalVarInferenceTest { + + interface Foo { + void m(X x); + } + + interface Supplier { + void m(X x); + } + + void test() { + var x; + var f = () -> { }; + var m = this::l; + var g = null; + var d = d = 1; + var k = { 1 , 2 }; + var l = new Foo<>() { //LHS was Foo + @Override + void m(String s) { } + }; + var s = f(x -> { x.charAt(0); }); //LHS was String + } + + Z f(Supplier sz) { return null; } +} diff --git a/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.out b/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.out new file mode 100644 index 00000000000..7425bc1fc6b --- /dev/null +++ b/test/langtools/tools/javac/lvti/BadLocalVarInferenceTest.out @@ -0,0 +1,10 @@ +BadLocalVarInferenceTest.java:19:13: compiler.err.cant.infer.local.var.type: x, (compiler.misc.local.missing.init) +BadLocalVarInferenceTest.java:20:13: compiler.err.cant.infer.local.var.type: f, (compiler.misc.local.lambda.missing.target) +BadLocalVarInferenceTest.java:21:13: compiler.err.cant.infer.local.var.type: m, (compiler.misc.local.mref.missing.target) +BadLocalVarInferenceTest.java:22:13: compiler.err.cant.infer.local.var.type: g, (compiler.misc.local.cant.infer.null) +BadLocalVarInferenceTest.java:23:13: compiler.err.cant.infer.local.var.type: d, (compiler.misc.local.self.ref) +BadLocalVarInferenceTest.java:24:13: compiler.err.cant.infer.local.var.type: k, (compiler.misc.local.array.missing.target) +BadLocalVarInferenceTest.java:25:29: compiler.err.does.not.override.abstract: compiler.misc.anonymous.class: BadLocalVarInferenceTest$1, m(java.lang.Object), BadLocalVarInferenceTest.Foo +BadLocalVarInferenceTest.java:26:13: compiler.err.method.does.not.override.superclass +BadLocalVarInferenceTest.java:29:27: compiler.err.cant.resolve.location.args: kindname.method, charAt, , int, (compiler.misc.location.1: kindname.variable, x, java.lang.Object) +9 errors diff --git a/test/langtools/tools/javac/lvti/FoldingTest.java b/test/langtools/tools/javac/lvti/FoldingTest.java new file mode 100644 index 00000000000..b8fe0317643 --- /dev/null +++ b/test/langtools/tools/javac/lvti/FoldingTest.java @@ -0,0 +1,47 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @compile/fail/ref=FoldingTest.out -XDrawDiagnostics FoldingTest.java + */ +class FoldingTest { + + void testReachability() { + for(var i = 0; i < 3; i++) { + // ok + } + System.out.println("foo"); //this should be reachable + } + + void testCase(String s) { + var c = ""; + final String c2 = "" + c; + switch (s) { + case c: break; //error! + case c2: break; //error! + } + } + + void testAnno() { + @Anno1(s1) //error + var s1 = ""; + @Anno2(s2) //error + var s2 = ""; + @Anno3(s3) //error + var s3 = ""; + } + + @interface Anno1 { + String value(); + } + @interface Anno2 { + Class value(); + } + @interface Anno3 { + Foo value(); + } + + enum Foo { + A, B; + } +} diff --git a/test/langtools/tools/javac/lvti/FoldingTest.out b/test/langtools/tools/javac/lvti/FoldingTest.out new file mode 100644 index 00000000000..682a0494d2e --- /dev/null +++ b/test/langtools/tools/javac/lvti/FoldingTest.out @@ -0,0 +1,6 @@ +FoldingTest.java:20:18: compiler.err.string.const.req +FoldingTest.java:21:18: compiler.err.string.const.req +FoldingTest.java:26:16: compiler.err.attribute.value.must.be.constant +FoldingTest.java:28:16: compiler.err.annotation.value.must.be.class.literal +FoldingTest.java:30:16: compiler.err.enum.annotation.must.be.enum.constant +5 errors diff --git a/test/langtools/tools/javac/lvti/ParserTest.java b/test/langtools/tools/javac/lvti/ParserTest.java new file mode 100644 index 00000000000..6e6287393b1 --- /dev/null +++ b/test/langtools/tools/javac/lvti/ParserTest.java @@ -0,0 +1,71 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @compile -source 9 ParserTest.java + * @compile/fail/ref=ParserTest.out -XDrawDiagnostics ParserTest.java + */ + +import java.lang.annotation.ElementType; +import java.lang.annotation.Target; +import java.util.function.Function; +import java.util.List; + +class ParserTest { + static class TestClass { + static class var { } //illegal + } + + static class TestInterface { + interface var { } //illegal + } + + static class TestEnum { + enum var { } //illegal + } + + static class TestAnno { + @interface var { } //illegal + } + + @Target(ElementType.TYPE_USE) + @interface TA { } + + @interface DA { } + + static class var extends RuntimeException { } //illegal + + var x = null; //illegal + + void test() { + var[] x1 = null; //illegal + var x2[] = null; //illegal + var[][] x3 = null; //illegal + var x4[][] = null; //illegal + var[] x5 = null; //illegal + var x6[] = null; //illegal + var@TA[]@TA[] x7 = null; //illegal + var x8@TA[]@TA[] = null; //illegal + var x9 = null, y = null; //illegal + final @DA var x10 = m(); //ok + @DA final var x11 = m(); //ok + } + + var m() { //illegal + return null; + } + + void test2(var x) { //error + List l1; //error + List l2; //error + List l3; //error + try { + Function f = (var x2) -> ""; //error + } catch (var ex) { } //error + } + + void test3(Object o) { + boolean b1 = o instanceof var; //error + Object o2 = (var)o; //error + } +} diff --git a/test/langtools/tools/javac/lvti/ParserTest.out b/test/langtools/tools/javac/lvti/ParserTest.out new file mode 100644 index 00000000000..7b001acd4fb --- /dev/null +++ b/test/langtools/tools/javac/lvti/ParserTest.out @@ -0,0 +1,25 @@ +ParserTest.java:14:18: compiler.err.var.not.allowed: var +ParserTest.java:16:22: compiler.err.var.not.allowed: var +ParserTest.java:20:19: compiler.err.var.not.allowed: var +ParserTest.java:24:14: compiler.err.var.not.allowed: var +ParserTest.java:28:20: compiler.err.var.not.allowed: var +ParserTest.java:36:18: compiler.err.var.not.allowed: var +ParserTest.java:38:5: compiler.err.var.not.allowed.here +ParserTest.java:41:15: compiler.err.var.not.allowed.array +ParserTest.java:42:13: compiler.err.var.not.allowed.array +ParserTest.java:43:17: compiler.err.var.not.allowed.array +ParserTest.java:44:13: compiler.err.var.not.allowed.array +ParserTest.java:45:15: compiler.err.var.not.allowed.array +ParserTest.java:46:13: compiler.err.var.not.allowed.array +ParserTest.java:49:13: compiler.err.var.not.allowed.compound +ParserTest.java:54:5: compiler.err.var.not.allowed.here +ParserTest.java:58:16: compiler.err.var.not.allowed.here +ParserTest.java:59:14: compiler.err.var.not.allowed.here +ParserTest.java:60:24: compiler.err.var.not.allowed.here +ParserTest.java:61:22: compiler.err.var.not.allowed.here +ParserTest.java:63:22: compiler.err.var.not.allowed.here +ParserTest.java:63:40: compiler.err.var.not.allowed.here +ParserTest.java:64:18: compiler.err.var.not.allowed.here +ParserTest.java:68:35: compiler.err.var.not.allowed.here +ParserTest.java:69:22: compiler.err.var.not.allowed.here +24 errors diff --git a/test/langtools/tools/javac/lvti/SelfRefTest.java b/test/langtools/tools/javac/lvti/SelfRefTest.java new file mode 100644 index 00000000000..98f956d96b2 --- /dev/null +++ b/test/langtools/tools/javac/lvti/SelfRefTest.java @@ -0,0 +1,23 @@ +/* + * @test /nodynamiccopyright/ + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @compile/fail/ref=SelfRefTest.out -XDrawDiagnostics SelfRefTest.java + */ + +import java.util.function.Function; + +class SelfRefTest { + + int q() { return 42; } + int m(int t) { return t; } + + void test(boolean cond) { + var x = cond ? x : x; //error - self reference + var y = (Function)(Integer y) -> y; //error - bad shadowing + var z = (Runnable)() -> { int z2 = m(z); }; //error - self reference + var w = new Object() { int w = 42; void test() { int w2 = w; } }; //ok + int u = u; //ok + int q = q(); //ok + } +} diff --git a/test/langtools/tools/javac/lvti/SelfRefTest.out b/test/langtools/tools/javac/lvti/SelfRefTest.out new file mode 100644 index 00000000000..97a558c4543 --- /dev/null +++ b/test/langtools/tools/javac/lvti/SelfRefTest.out @@ -0,0 +1,4 @@ +SelfRefTest.java:16:12: compiler.err.cant.infer.local.var.type: x, (compiler.misc.local.self.ref) +SelfRefTest.java:17:53: compiler.err.already.defined: kindname.variable, y, kindname.method, test(boolean) +SelfRefTest.java:18:12: compiler.err.cant.infer.local.var.type: z, (compiler.misc.local.self.ref) +3 errors diff --git a/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.java b/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.java new file mode 100644 index 00000000000..0dfe7b6f08b --- /dev/null +++ b/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.java @@ -0,0 +1,43 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test /nodynamioccopyright/ + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @compile -source 8 pkg/var.java + * @compile pkg/nested/var/A.java + * @compile/fail/ref=BadTypeReference.out -XDrawDiagnostics BadTypeReference.java + */ + +import pkg.*; + +public class BadTypeReference { + void test(Object o) { + var vs = null; //error + Object o2 = var.x; //error + pkg.nested.var.A a = new pkg.nested.var.A(); //ok + } +} diff --git a/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.out b/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.out new file mode 100644 index 00000000000..28460500aff --- /dev/null +++ b/test/langtools/tools/javac/lvti/badTypeReference/BadTypeReference.out @@ -0,0 +1,3 @@ +BadTypeReference.java:39:9: compiler.err.illegal.ref.to.var.type: var +BadTypeReference.java:40:21: compiler.err.illegal.ref.to.var.type: var +2 errors diff --git a/test/langtools/tools/javac/lvti/badTypeReference/pkg/nested/var/A.java b/test/langtools/tools/javac/lvti/badTypeReference/pkg/nested/var/A.java new file mode 100644 index 00000000000..320e0dc859c --- /dev/null +++ b/test/langtools/tools/javac/lvti/badTypeReference/pkg/nested/var/A.java @@ -0,0 +1,29 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package pkg.nested.var; + +public class A { +} diff --git a/test/langtools/tools/javac/lvti/badTypeReference/pkg/var.java b/test/langtools/tools/javac/lvti/badTypeReference/pkg/var.java new file mode 100644 index 00000000000..39910388528 --- /dev/null +++ b/test/langtools/tools/javac/lvti/badTypeReference/pkg/var.java @@ -0,0 +1,30 @@ +/* + * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package pkg; + +public class var { + public static Object x = ""; +} diff --git a/test/langtools/tools/javac/lvti/harness/InferredType.java b/test/langtools/tools/javac/lvti/harness/InferredType.java new file mode 100644 index 00000000000..2492aa68a41 --- /dev/null +++ b/test/langtools/tools/javac/lvti/harness/InferredType.java @@ -0,0 +1,28 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +public @interface InferredType { + String value(); +} diff --git a/test/langtools/tools/javac/lvti/harness/LocalVariableInferenceTester.java b/test/langtools/tools/javac/lvti/harness/LocalVariableInferenceTester.java new file mode 100644 index 00000000000..226a32727ea --- /dev/null +++ b/test/langtools/tools/javac/lvti/harness/LocalVariableInferenceTester.java @@ -0,0 +1,162 @@ +/* + * Copyright (c) 2016, 2017 Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +import java.io.File; +import java.io.IOException; +import java.util.Arrays; +import java.util.HashMap; +import java.util.Locale; +import java.util.Map; + +import javax.lang.model.element.Element; +import javax.lang.model.element.ElementKind; +import javax.lang.model.type.TypeMirror; +import javax.tools.JavaCompiler; +import javax.tools.JavaFileObject; +import javax.tools.StandardJavaFileManager; +import javax.tools.ToolProvider; + +import com.sun.source.tree.CompilationUnitTree; +import com.sun.source.tree.Tree; +import com.sun.source.tree.VariableTree; +import com.sun.source.util.JavacTask; +import com.sun.source.util.TreePathScanner; +import com.sun.tools.javac.api.JavacTaskImpl; +import com.sun.tools.javac.api.JavacTrees; +import com.sun.tools.javac.code.Printer; +import com.sun.tools.javac.code.Type; +import com.sun.tools.javac.code.Type.CapturedType; +import com.sun.tools.javac.code.Type.ClassType; +import com.sun.tools.javac.code.Types; +import com.sun.tools.javac.util.Log; + +import static javax.tools.StandardLocation.SOURCE_PATH; + +public class LocalVariableInferenceTester { + + static final JavaCompiler comp = ToolProvider.getSystemJavaCompiler(); + static final StandardJavaFileManager fm = comp.getStandardFileManager(null, null, null); + + public static void main(String[] args) throws IOException { + try { + if (args.length != 1) { + System.err.println("Usage: LocalVariableInferenceTester "); + System.exit(1); + } + File path = new File(System.getProperty("test.src")); + fm.setLocation(SOURCE_PATH, Arrays.asList(path)); + File input = new File(path, args[0]); + JavaFileObject jfo = fm.getJavaFileObjects(input).iterator().next(); + new LocalVariableInferenceTester().compileAndCheck(jfo); + } finally { + fm.close(); + } + } + + int errors = 0; + int checks = 0; + + void compileAndCheck(JavaFileObject input) throws IOException { + JavaCompiler c = ToolProvider.getSystemJavaCompiler(); + JavacTask task = (JavacTask) c.getTask(null, fm, null, null, null, Arrays.asList(input)); + JavacTrees trees = JavacTrees.instance(task); + Types types = Types.instance(((JavacTaskImpl)task).getContext()); + Iterable roots = task.parse(); + task.analyze(); //force attribution + Log log = Log.instance(((JavacTaskImpl)task).getContext()); + errors += log.nerrors; + new LocalVarTypeChecker(trees, types).scan(roots, null); + System.err.println("Checks executed: " + checks); + if (errors != 0) { + throw new AssertionError("Errors were found"); + } + } + + void error(Tree node, String msg) { + System.err.println(node); + System.err.println("ERROR: " + msg); + errors++; + } + + class LocalVarTypeChecker extends TreePathScanner { + + JavacTrees trees; + Types types; + + LocalVarTypeChecker(JavacTrees trees, Types types) { + this.trees = trees; + this.types = types; + } + + @Override + public Void visitVariable(VariableTree node, Void aVoid) { + Element e = trees.getElement(getCurrentPath()); + if (e.getKind() == ElementKind.LOCAL_VARIABLE) { + TypeMirror type = e.asType(); + InferredType inferredAnno = e.getAnnotation(InferredType.class); + if (inferredAnno != null) { + checks++; + String req = inferredAnno.value(); + String found = new TypePrinter().visit((Type)type, null); + if (!req.equals(found)) { + error(node, "Inferred type mismatch; expected: " + req + " - found: " + found); + } + } + } + return super.visitVariable(node, null); + } + + class TypePrinter extends Printer { + + Map capturedIdMap = new HashMap<>(); + + @Override + protected String localize(Locale locale, String key, Object... args) { + throw new UnsupportedOperationException(); + } + + @Override + public String visitCapturedType(CapturedType t, Locale locale) { + return "CAP#" + capturedVarId(t, locale); + } + + @Override + protected String capturedVarId(CapturedType t, Locale locale) { + return String.valueOf(capturedIdMap.getOrDefault(t, capturedIdMap.size())); + } + + @Override + public String visitClassType(ClassType t, Locale locale) { + if (!t.isCompound() && t.tsym.name.isEmpty()) { + return "#ANON(" + types.directSupertypes(t) + ")"; + } else if (t.isCompound()) { + return "#INT(" + types.directSupertypes(t) + ")"; + } else { + return super.visitClassType(t, locale); + } + } + } + } +} diff --git a/test/langtools/tools/javac/lvti/harness/NonDenotableTest.java b/test/langtools/tools/javac/lvti/harness/NonDenotableTest.java new file mode 100644 index 00000000000..940da65043e --- /dev/null +++ b/test/langtools/tools/javac/lvti/harness/NonDenotableTest.java @@ -0,0 +1,148 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @modules jdk.compiler/com.sun.source.tree + * jdk.compiler/com.sun.source.util + * jdk.compiler/com.sun.tools.javac.api + * jdk.compiler/com.sun.tools.javac.code + * jdk.compiler/com.sun.tools.javac.util + * @build LocalVariableInferenceTester + * @run main LocalVariableInferenceTester NonDenotableTest.java + */ +import java.util.List; + +class NonDenotableTest { + + static final String OBJECT = "java.lang.Object"; + static final String STRING = "java.lang.String"; + static final String ANON_OBJECT = "#ANON(java.lang.Object)"; + static final String ANON_RUNNABLE = "#ANON(java.lang.Object,java.lang.Runnable)"; + static final String LIST_EXT = "java.util.List"; + static final String LIST_SUP = "java.util.List"; + static final String LIST_UNB = "java.util.List"; + static final String COMP_UNB = "java.lang.Comparable"; + static final String LIST_EXT_COMP_UNB = "java.util.List>"; + static final String LIST_SUP_COMP_UNB = "java.util.List>"; + static final String INT_INTEGER_DOUBLE = "#INT(java.lang.Number,java.lang.Comparable>)"; + + void testExtends() { + @InferredType(LIST_EXT) + var s = extString(); + for (@InferredType(LIST_EXT) var s2 = extString() ; ; ) { break; } + for (@InferredType(LIST_EXT) var s2 : extStringArr()) { break; } + for (@InferredType(LIST_EXT) var s2 : extStringIter()) { break; } + for (@InferredType(STRING) var s2 : extString()) { break; } + } + + void testExtendsFbound() { + @InferredType(LIST_EXT_COMP_UNB) + var s = extFbound(); + for (@InferredType(LIST_EXT_COMP_UNB) var s2 = extFbound() ; ; ) { break; } + for (@InferredType(LIST_EXT_COMP_UNB) var s2 : extFboundArr()) { break; } + for (@InferredType(LIST_EXT_COMP_UNB) var s2 : extFboundIter()) { break; } + for (@InferredType(COMP_UNB) var s2 : extFbound()) { break; } + } + + void testSuperFbound() { + @InferredType(LIST_UNB) + var s = supFbound(); + for (@InferredType(LIST_UNB) var s2 = supFbound() ; ; ) { break; } + for (@InferredType(LIST_UNB) var s2 : supFboundArr()) { break; } + for (@InferredType(LIST_UNB) var s2 : supFboundIter()) { break; } + for (@InferredType(OBJECT) var s2 : supFbound()) { break; } + } + + void testSuper() { + @InferredType(LIST_SUP) + var s = supString(); + for (@InferredType(LIST_SUP) var s2 = supString() ; ; ) { break; } + for (@InferredType(LIST_SUP) var s2 : supStringArr()) { break; } + for (@InferredType(LIST_SUP) var s2 : supStringIter()) { break; } + for (@InferredType(OBJECT) var s2 : supString()) { break; } + } + + void testUnbound() { + @InferredType(LIST_UNB) + var s = unbString(); + for (@InferredType(LIST_UNB) var s2 = unbString() ; ; ) { break; } + for (@InferredType(LIST_UNB) var s2 : unbStringArr()) { break; } + for (@InferredType(LIST_UNB) var s2 : unbStringIter()) { break; } + for (@InferredType(OBJECT) var s2 : unbString()) { break; } + } + + void testAnonymousClass() { + @InferredType(ANON_OBJECT) + var o = new Object() { }; + for (@InferredType(ANON_OBJECT) var s2 = new Object() { } ; ; ) { break; } + for (@InferredType(ANON_OBJECT) var s2 : arrayOf(new Object() { })) { break; } + for (@InferredType(ANON_OBJECT) var s2 : listOf(new Object() { })) { break; } + } + + void testAnonymousInterface() { + @InferredType(ANON_RUNNABLE) + var r = new Runnable() { public void run() { } }; + for (@InferredType(ANON_RUNNABLE) var s2 = new Runnable() { public void run() { } } ; ; ) { break; } + for (@InferredType(ANON_RUNNABLE) var s2 : arrayOf(new Runnable() { public void run() { } })) { break; } + for (@InferredType(ANON_RUNNABLE) var s2 : listOf(new Runnable() { public void run() { } })) { break; } + } + + void testIntersection() { + @InferredType(INT_INTEGER_DOUBLE) + var c = choose(1, 1L); + for (@InferredType(INT_INTEGER_DOUBLE) var s2 = choose(1, 1L) ; ;) { break; } + for (@InferredType(INT_INTEGER_DOUBLE) var s2 : arrayOf(choose(1, 1L))) { break; } + for (@InferredType(INT_INTEGER_DOUBLE) var s2 : listOf(choose(1, 1L))) { break; } + } + + List extString() { return null; } + List supString() { return null; } + List unbString() { return null; } + + List[] extStringArr() { return null; } + List[] supStringArr() { return null; } + List[] unbStringArr() { return null; } + + Iterable> extStringIter() { return null; } + Iterable> supStringIter() { return null; } + Iterable> unbStringIter() { return null; } + + > List extFbound() { return null; } + > List supFbound() { return null; } + + > List[] extFboundArr() { return null; } + > List[] supFboundArr() { return null; } + + > Iterable> extFboundIter() { return null; } + > Iterable> supFboundIter() { return null; } + + List listOf(Z z) { return null; } + Z[] arrayOf(Z z) { return null; } + + Z choose(Z z1, Z z2) { return z1; } +} diff --git a/test/langtools/tools/javac/lvti/harness/PrimitiveTypeTest.java b/test/langtools/tools/javac/lvti/harness/PrimitiveTypeTest.java new file mode 100644 index 00000000000..6f5a1c345f7 --- /dev/null +++ b/test/langtools/tools/javac/lvti/harness/PrimitiveTypeTest.java @@ -0,0 +1,97 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @modules jdk.compiler/com.sun.source.tree + * jdk.compiler/com.sun.source.util + * jdk.compiler/com.sun.tools.javac.api + * jdk.compiler/com.sun.tools.javac.code + * jdk.compiler/com.sun.tools.javac.util + * @build LocalVariableInferenceTester + * @run main LocalVariableInferenceTester PrimitiveTypeTest.java + */ +class PrimitiveTypeTest { + + byte[] b_arr = { 0 }; + short[] s_arr = { 0 }; + int[] i_arr = { 0 }; + long[] l_arr = { 0 }; + float[] f_arr = { 0 }; + double[] d_arr = { 0 }; + char[] c_arr = { 0 }; + boolean[] z_arr = { false }; + + void testPrimitive() { + @InferredType("byte") + var b = (byte)0; + @InferredType("short") + var s = (short)0; + @InferredType("int") + var i = 0; + @InferredType("long") + var l = 0L; + @InferredType("float") + var f = 0f; + @InferredType("double") + var d = 0d; + @InferredType("char") + var c = 'c'; + @InferredType("boolean") + var z = false; + } + + void testPrimitiveArray() { + @InferredType("byte[]") + var b = b_arr; + @InferredType("short[]") + var s = s_arr; + @InferredType("int[]") + var i = i_arr; + @InferredType("long[]") + var l = l_arr; + @InferredType("float[]") + var f = f_arr; + @InferredType("double[]") + var d = d_arr; + @InferredType("char[]") + var c = c_arr; + @InferredType("boolean[]") + var z = z_arr; + } + + void testForEachPrimitive() { + for (@InferredType("byte") var b : b_arr) { break; } + for (@InferredType("short") var s : s_arr) { break; } + for (@InferredType("int") var i : i_arr) { break; } + for (@InferredType("long") var l : l_arr) { break; } + for (@InferredType("float") var f : f_arr) { break; } + for (@InferredType("double") var d : d_arr) { break; } + for (@InferredType("char") var c : c_arr) { break; } + for (@InferredType("boolean") var z : z_arr) { break; } + } +} diff --git a/test/langtools/tools/javac/lvti/harness/ReferenceTypeTest.java b/test/langtools/tools/javac/lvti/harness/ReferenceTypeTest.java new file mode 100644 index 00000000000..26e434d5585 --- /dev/null +++ b/test/langtools/tools/javac/lvti/harness/ReferenceTypeTest.java @@ -0,0 +1,60 @@ +/* + * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* + * @test + * @bug 8177466 + * @summary Add compiler support for local variable type-inference + * @modules jdk.compiler/com.sun.source.tree + * jdk.compiler/com.sun.source.util + * jdk.compiler/com.sun.tools.javac.api + * jdk.compiler/com.sun.tools.javac.code + * jdk.compiler/com.sun.tools.javac.util + * @build LocalVariableInferenceTester + * @run main LocalVariableInferenceTester ReferenceTypeTest.java + */ +class ReferenceTypeTest { + + static final String STRING = "java.lang.String"; + static final String FOO = "ReferenceTypeTest.Foo"; + + void test() { + @InferredType(STRING) + var s = ""; + for (@InferredType(STRING) var s2 = "" ; ; ) { break; } + for (@InferredType(STRING) var s2 : stringArray()) { break; } + for (@InferredType(STRING) var s2 : stringIterable()) { break; } + try (@InferredType(FOO) var s2 = new Foo()) { } finally { } + try (@InferredType(FOO) var s2 = new Foo(); @InferredType(FOO) var s3 = new Foo()) { } finally { } + } + + String[] stringArray() { return null; } + Iterable stringIterable() { return null; } + + static class Foo implements AutoCloseable { + @Override + public void close() { } + } +} diff --git a/test/langtools/tools/javac/parser/extend/TrialParser.java b/test/langtools/tools/javac/parser/extend/TrialParser.java index 0ff5b0cccbf..8579590fa5a 100644 --- a/test/langtools/tools/javac/parser/extend/TrialParser.java +++ b/test/langtools/tools/javac/parser/extend/TrialParser.java @@ -224,7 +224,7 @@ class TrialParser extends JavacParser { //mods.flags |= Flags.STATIC; List defs = variableDeclaratorsRest(pos, mods, t, name, false, dc, - new ListBuffer()).toList(); + new ListBuffer(), true).toList(); accept(SEMI); storeEnd(defs.last(), S.prevToken().endPos); return defs; diff --git a/test/langtools/tools/lib/types/TypeHarness.java b/test/langtools/tools/lib/types/TypeHarness.java index 24cb434ef23..71be5c00c4a 100644 --- a/test/langtools/tools/lib/types/TypeHarness.java +++ b/test/langtools/tools/lib/types/TypeHarness.java @@ -354,7 +354,7 @@ public class TypeHarness { public TypeVar TypeVariable(Type bound) { TypeSymbol tvsym = new TypeVariableSymbol(0, syntheticName(), null, predef.noSymbol); - tvsym.type = new TypeVar(tvsym, bound, null); + tvsym.type = new TypeVar(tvsym, bound, Type.noType); return (TypeVar)tvsym.type; }