1 /* 2 * Copyright (c) 2002, 2025, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. Oracle designates this 8 * particular file as subject to the "Classpath" exception as provided 9 * by Oracle in the LICENSE file that accompanied this code. 10 * 11 * This code is distributed in the hope that it will be useful, but WITHOUT 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 14 * version 2 for more details (a copy is included in the LICENSE file that 15 * accompanied this code). 16 * 17 * You should have received a copy of the GNU General Public License version 18 * 2 along with this work; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 20 * 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 22 * or visit www.oracle.com if you need additional information or have any 23 * questions. 24 */ 25 26 package com.sun.tools.javac.code; 27 28 import java.util.*; 29 30 import javax.lang.model.SourceVersion; 31 import static javax.lang.model.SourceVersion.*; 32 33 import com.sun.tools.javac.jvm.Target; 34 import com.sun.tools.javac.resources.CompilerProperties.Errors; 35 import com.sun.tools.javac.resources.CompilerProperties.Fragments; 36 import com.sun.tools.javac.util.*; 37 import com.sun.tools.javac.util.JCDiagnostic.Error; 38 import com.sun.tools.javac.util.JCDiagnostic.Fragment; 39 40 import static com.sun.tools.javac.main.Option.*; 41 42 /** The source language version accepted. 43 * 44 * <p><b>This is NOT part of any supported API. 45 * If you write code that depends on this, you do so at your own risk. 46 * This code and its internal interfaces are subject to change or 47 * deletion without notice.</b> 48 */ 49 public enum Source { 50 /* 1.0 had no inner classes, and so could not pass the JCK. */ 51 // public static final Source JDK1_0 = new Source("1.0"); 52 53 /* 1.1 did not have strictfp, and so could not pass the JCK. */ 54 // public static final Source JDK1_1 = new Source("1.1"); 55 56 /** 1.2 introduced strictfp. */ 57 JDK1_2("1.2"), 58 59 /** 1.3 is the same language as 1.2. */ 60 JDK1_3("1.3"), 61 62 /** 1.4 introduced assert. */ 63 JDK1_4("1.4"), 64 65 /** 1.5 introduced generics, attributes, foreach, boxing, static import, 66 * covariant return, enums, varargs, et al. */ 67 JDK5("5"), 68 69 /** 1.6 reports encoding problems as errors instead of warnings. */ 70 JDK6("6"), 71 72 /** 1.7 introduced try-with-resources, multi-catch, string switch, etc. */ 73 JDK7("7"), 74 75 /** 1.8 lambda expressions and default methods. */ 76 JDK8("8"), 77 78 /** 1.9 modularity. */ 79 JDK9("9"), 80 81 /** 1.10 local-variable type inference (var). */ 82 JDK10("10"), 83 84 /** 1.11 local-variable syntax for lambda parameters */ 85 JDK11("11"), 86 87 /** 12, no language features; switch expression in preview */ 88 JDK12("12"), 89 90 /** 91 * 13, no language features; text blocks and revised switch 92 * expressions in preview 93 */ 94 JDK13("13"), 95 96 /** 97 * 14, switch expressions; pattern matching, records, and revised 98 * text blocks in preview 99 */ 100 JDK14("14"), 101 102 /** 103 * 15, text blocks 104 */ 105 JDK15("15"), 106 107 /** 108 * 16, records and pattern matching for instanceof 109 */ 110 JDK16("16"), 111 112 /** 113 * 17, sealed classes, restoration of always-strict floating-point 114 */ 115 JDK17("17"), 116 117 /** 118 * 18, no major changes 119 */ 120 JDK18("18"), 121 122 /** 123 * 19, no major changes 124 */ 125 JDK19("19"), 126 127 /** 128 * 20, no major changes 129 */ 130 JDK20("20"), 131 132 /** 133 * 21, tbd 134 */ 135 JDK21("21"), 136 137 /** 138 * 22, tbd 139 */ 140 JDK22("22"), 141 142 /** 143 * 23, tbd 144 */ 145 JDK23("23"), 146 147 /** 148 * 24, tbd 149 */ 150 JDK24("24"), 151 152 /** 153 * 25, tbd 154 */ 155 JDK25("25"), 156 157 /** 158 * 26, tbd 159 */ 160 JDK26("26"), 161 ; // Reduce code churn when appending new constants 162 163 private static final Context.Key<Source> sourceKey = new Context.Key<>(); 164 165 public static Source instance(Context context) { 166 Source instance = context.get(sourceKey); 167 if (instance == null) { 168 Options options = Options.instance(context); 169 String sourceString = options.get(SOURCE); 170 if (sourceString != null) instance = lookup(sourceString); 171 if (instance == null) instance = DEFAULT; 172 context.put(sourceKey, instance); 173 } 174 return instance; 175 } 176 177 public final String name; 178 179 private static final Map<String,Source> tab = new HashMap<>(); 180 static { 181 for (Source s : values()) { 182 tab.put(s.name, s); 183 } 184 tab.put("1.5", JDK5); // Make 5 an alias for 1.5 185 tab.put("1.6", JDK6); // Make 6 an alias for 1.6 186 tab.put("1.7", JDK7); // Make 7 an alias for 1.7 187 tab.put("1.8", JDK8); // Make 8 an alias for 1.8 188 tab.put("1.9", JDK9); // Make 9 an alias for 1.9 189 tab.put("1.10", JDK10); // Make 10 an alias for 1.10 190 // Decline to make 1.11 an alias for 11. 191 } 192 193 private Source(String name) { 194 this.name = name; 195 } 196 197 public static final Source MIN = Source.JDK8; 198 199 private static final Source MAX = values()[values().length - 1]; 200 201 public static final Source DEFAULT = MAX; 202 203 public static Source lookup(String name) { 204 return tab.get(name); 205 } 206 207 public boolean isSupported() { 208 return this.compareTo(MIN) >= 0; 209 } 210 211 public static boolean isSupported(Feature feature, int majorVersion) { 212 Source source = null; 213 for (Target target : Target.values()) { 214 if (majorVersion == target.majorVersion) { 215 source = lookup(target.name); 216 } 217 } 218 if (source != null) { 219 return feature.allowedInSource(source); 220 } 221 return false; 222 } 223 224 public Target requiredTarget() { 225 return switch(this) { 226 case JDK26 -> Target.JDK1_26; 227 case JDK25 -> Target.JDK1_25; 228 case JDK24 -> Target.JDK1_24; 229 case JDK23 -> Target.JDK1_23; 230 case JDK22 -> Target.JDK1_22; 231 case JDK21 -> Target.JDK1_21; 232 case JDK20 -> Target.JDK1_20; 233 case JDK19 -> Target.JDK1_19; 234 case JDK18 -> Target.JDK1_18; 235 case JDK17 -> Target.JDK1_17; 236 case JDK16 -> Target.JDK1_16; 237 case JDK15 -> Target.JDK1_15; 238 case JDK14 -> Target.JDK1_14; 239 case JDK13 -> Target.JDK1_13; 240 case JDK12 -> Target.JDK1_12; 241 case JDK11 -> Target.JDK1_11; 242 case JDK10 -> Target.JDK1_10; 243 case JDK9 -> Target.JDK1_9; 244 case JDK8 -> Target.JDK1_8; 245 case JDK7 -> Target.JDK1_7; 246 case JDK6 -> Target.JDK1_6; 247 case JDK5 -> Target.JDK1_5; 248 case JDK1_4 -> Target.JDK1_4; 249 default -> Target.JDK1_1; 250 }; 251 } 252 253 /** 254 * Models a feature of the Java programming language. Each feature can be associated with a 255 * minimum source level, a maximum source level and a diagnostic fragment describing the feature, 256 * which is used to generate error messages of the kind {@code feature XYZ not supported in source N}. 257 */ 258 public enum Feature { 259 260 MODULES(JDK9, Fragments.FeatureModules, DiagKind.PLURAL), 261 EFFECTIVELY_FINAL_VARIABLES_IN_TRY_WITH_RESOURCES(JDK9, Fragments.FeatureVarInTryWithResources, DiagKind.PLURAL), 262 DEPRECATION_ON_IMPORT(MIN, JDK8), 263 PRIVATE_SAFE_VARARGS(JDK9), 264 DIAMOND_WITH_ANONYMOUS_CLASS_CREATION(JDK9, Fragments.FeatureDiamondAndAnonClass, DiagKind.NORMAL), 265 UNDERSCORE_IDENTIFIER(MIN, JDK8), 266 PRIVATE_INTERFACE_METHODS(JDK9, Fragments.FeaturePrivateIntfMethods, DiagKind.PLURAL), 267 LOCAL_VARIABLE_TYPE_INFERENCE(JDK10), 268 VAR_SYNTAX_IMPLICIT_LAMBDAS(JDK11, Fragments.FeatureVarSyntaxInImplicitLambda, DiagKind.PLURAL), 269 IMPORT_ON_DEMAND_OBSERVABLE_PACKAGES(JDK1_2, JDK8), 270 SWITCH_MULTIPLE_CASE_LABELS(JDK14, Fragments.FeatureMultipleCaseLabels, DiagKind.PLURAL), 271 SWITCH_RULE(JDK14, Fragments.FeatureSwitchRules, DiagKind.PLURAL), 272 SWITCH_EXPRESSION(JDK14, Fragments.FeatureSwitchExpressions, DiagKind.PLURAL), 273 NO_TARGET_ANNOTATION_APPLICABILITY(JDK14), 274 TEXT_BLOCKS(JDK15, Fragments.FeatureTextBlocks, DiagKind.PLURAL), 275 PATTERN_MATCHING_IN_INSTANCEOF(JDK16, Fragments.FeaturePatternMatchingInstanceof, DiagKind.NORMAL), 276 REIFIABLE_TYPES_INSTANCEOF(JDK16, Fragments.FeatureReifiableTypesInstanceof, DiagKind.PLURAL), 277 RECORDS(JDK16, Fragments.FeatureRecords, DiagKind.PLURAL), 278 SEALED_CLASSES(JDK17, Fragments.FeatureSealedClasses, DiagKind.PLURAL), 279 CASE_NULL(JDK21, Fragments.FeatureCaseNull, DiagKind.NORMAL), 280 PATTERN_SWITCH(JDK21, Fragments.FeaturePatternSwitch, DiagKind.PLURAL), 281 REDUNDANT_STRICTFP(JDK17), 282 UNCONDITIONAL_PATTERN_IN_INSTANCEOF(JDK21, Fragments.FeatureUnconditionalPatternsInInstanceof, DiagKind.PLURAL), 283 RECORD_PATTERNS(JDK21, Fragments.FeatureDeconstructionPatterns, DiagKind.PLURAL), 284 IMPLICIT_CLASSES(JDK25, Fragments.FeatureImplicitClasses, DiagKind.PLURAL), 285 WARN_ON_ILLEGAL_UTF8(MIN, JDK21), 286 UNNAMED_VARIABLES(JDK22, Fragments.FeatureUnnamedVariables, DiagKind.PLURAL), 287 PRIMITIVE_PATTERNS(JDK23, Fragments.FeaturePrimitivePatterns, DiagKind.PLURAL), 288 VALUE_CLASSES(JDK22, Fragments.FeatureValueClasses, DiagKind.PLURAL), 289 FLEXIBLE_CONSTRUCTORS(JDK25, Fragments.FeatureFlexibleConstructors, DiagKind.NORMAL), 290 MODULE_IMPORTS(JDK25, Fragments.FeatureModuleImports, DiagKind.PLURAL), 291 JAVA_BASE_TRANSITIVE(JDK25, Fragments.FeatureJavaBaseTransitive, DiagKind.PLURAL), 292 PRIVATE_MEMBERS_IN_PERMITS_CLAUSE(JDK19), 293 ERASE_POLY_SIG_RETURN_TYPE(JDK24), 294 ; 295 296 enum DiagKind { 297 NORMAL, 298 PLURAL; 299 } 300 301 private final Source minLevel; 302 private final Source maxLevel; 303 private final Fragment optFragment; 304 private final DiagKind optKind; 305 306 Feature(Source minLevel) { 307 this(minLevel, null, null); 308 } 309 310 Feature(Source minLevel, Fragment optFragment, DiagKind optKind) { 311 this(minLevel, MAX, optFragment, optKind); 312 } 313 314 Feature(Source minLevel, Source maxLevel) { 315 this(minLevel, maxLevel, null, null); 316 } 317 318 Feature(Source minLevel, Source maxLevel, Fragment optFragment, DiagKind optKind) { 319 this.minLevel = minLevel; 320 this.maxLevel = maxLevel; 321 this.optFragment = optFragment; 322 this.optKind = optKind; 323 } 324 325 public boolean allowedInSource(Source source) { 326 return source.compareTo(minLevel) >= 0 && 327 source.compareTo(maxLevel) <= 0; 328 } 329 330 public boolean isPlural() { 331 Assert.checkNonNull(optKind); 332 return optKind == DiagKind.PLURAL; 333 } 334 335 public Fragment nameFragment() { 336 Assert.checkNonNull(optFragment); 337 return optFragment; 338 } 339 340 public Fragment fragment(String sourceName) { 341 Assert.checkNonNull(optFragment); 342 return optKind == DiagKind.NORMAL ? 343 Fragments.FeatureNotSupportedInSource(optFragment, sourceName, minLevel.name) : 344 Fragments.FeatureNotSupportedInSourcePlural(optFragment, sourceName, minLevel.name); 345 } 346 347 public Error error(String sourceName) { 348 Assert.checkNonNull(optFragment); 349 return optKind == DiagKind.NORMAL ? 350 Errors.FeatureNotSupportedInSource(optFragment, sourceName, minLevel.name) : 351 Errors.FeatureNotSupportedInSourcePlural(optFragment, sourceName, minLevel.name); 352 } 353 } 354 355 public static SourceVersion toSourceVersion(Source source) { 356 return switch(source) { 357 case JDK1_2 -> RELEASE_2; 358 case JDK1_3 -> RELEASE_3; 359 case JDK1_4 -> RELEASE_4; 360 case JDK5 -> RELEASE_5; 361 case JDK6 -> RELEASE_6; 362 case JDK7 -> RELEASE_7; 363 case JDK8 -> RELEASE_8; 364 case JDK9 -> RELEASE_9; 365 case JDK10 -> RELEASE_10; 366 case JDK11 -> RELEASE_11; 367 case JDK12 -> RELEASE_12; 368 case JDK13 -> RELEASE_13; 369 case JDK14 -> RELEASE_14; 370 case JDK15 -> RELEASE_15; 371 case JDK16 -> RELEASE_16; 372 case JDK17 -> RELEASE_17; 373 case JDK18 -> RELEASE_18; 374 case JDK19 -> RELEASE_19; 375 case JDK20 -> RELEASE_20; 376 case JDK21 -> RELEASE_21; 377 case JDK22 -> RELEASE_22; 378 case JDK23 -> RELEASE_23; 379 case JDK24 -> RELEASE_24; 380 case JDK25 -> RELEASE_25; 381 case JDK26 -> RELEASE_26; 382 default -> null; 383 }; 384 } 385 }