Skip to content

Commit 1e97c1c

Browse files
committedNov 14, 2024
8335989: Implement JEP 494: Module Import Declarations (Second Preview)
Reviewed-by: vromero, abimpoudis, mcimadamore, alanb
1 parent e7d90b9 commit 1e97c1c

File tree

28 files changed

+716
-182
lines changed

28 files changed

+716
-182
lines changed
 

‎make/langtools/src/classes/build/tools/symbolgenerator/CreateSymbols.java

+5
Original file line numberDiff line numberDiff line change
@@ -1896,6 +1896,11 @@ private void finishClassLoading(ClassList classes, Map<String, ModuleDescription
18961896
continue;
18971897
}
18981898

1899+
if (ed.packageName.equals("jdk/internal/javac")) {
1900+
//keep jdk/internal/javac untouched. It is used to determine participates in preview:
1901+
continue;
1902+
}
1903+
18991904
Set<String> usingModules = package2ModulesUsingIt.getOrDefault(ed.packageName(), Set.of());
19001905

19011906
ed.to.retainAll(usingModules);

‎src/java.base/share/classes/jdk/internal/javac/PreviewFeature.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ public enum Feature {
7777
@JEP(number=466, title="ClassFile API", status="Second Preview")
7878
CLASSFILE_API,
7979
STREAM_GATHERERS,
80-
@JEP(number=476, title="Module Import Declarations", status="Preview")
80+
@JEP(number=494, title="Module Import Declarations", status="Second Preview")
8181
MODULE_IMPORTS,
8282
@JEP(number=478, title="Key Derivation Function API", status="Preview")
8383
KEY_DERIVATION,

‎src/java.base/share/classes/jdk/internal/module/ModuleInfo.java

+18-6
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
import java.io.IOException;
3232
import java.io.InputStream;
3333
import java.io.UncheckedIOException;
34+
import java.lang.classfile.ClassFile;
3435
import java.lang.module.InvalidModuleDescriptorException;
3536
import java.lang.module.ModuleDescriptor;
3637
import java.lang.module.ModuleDescriptor.Builder;
@@ -189,6 +190,7 @@ private Attributes doRead(DataInput input) throws IOException {
189190

190191
int minor_version = in.readUnsignedShort();
191192
int major_version = in.readUnsignedShort();
193+
boolean isPreview = minor_version == ClassFile.PREVIEW_MINOR_VERSION;
192194
if (!VM.isSupportedModuleDescriptorVersion(major_version, minor_version)) {
193195
throw invalidModuleDescriptor("Unsupported major.minor version "
194196
+ major_version + "." + minor_version);
@@ -248,7 +250,7 @@ private Attributes doRead(DataInput input) throws IOException {
248250

249251
switch (attribute_name) {
250252
case MODULE :
251-
builder = readModuleAttribute(in, cpool, major_version);
253+
builder = readModuleAttribute(in, cpool, major_version, isPreview);
252254
break;
253255

254256
case MODULE_PACKAGES :
@@ -344,7 +346,8 @@ private Attributes doRead(DataInput input) throws IOException {
344346
* Reads the Module attribute, returning the ModuleDescriptor.Builder to
345347
* build the corresponding ModuleDescriptor.
346348
*/
347-
private Builder readModuleAttribute(DataInput in, ConstantPool cpool, int major)
349+
private Builder readModuleAttribute(DataInput in, ConstantPool cpool, int major,
350+
boolean isPreview)
348351
throws IOException
349352
{
350353
// module_name
@@ -405,14 +408,23 @@ private Builder readModuleAttribute(DataInput in, ConstantPool cpool, int major)
405408
throw invalidModuleDescriptor("The requires entry for java.base"
406409
+ " has ACC_SYNTHETIC set");
407410
}
411+
// requires transitive java.base is illegal unless:
412+
// - the major version is 53 (JDK 9), or:
413+
// - the classfile is a preview classfile, or:
414+
// - the module is deemed to be participating in preview
415+
// (i.e. the module is a java.* module)
416+
// requires static java.base is illegal unless:
417+
// - the major version is 53 (JDK 9), or:
408418
if (major >= 54
409-
&& (mods.contains(Requires.Modifier.TRANSITIVE)
419+
&& ((mods.contains(Requires.Modifier.TRANSITIVE)
420+
&& !isPreview
421+
&& !"java.se".equals(mn))
410422
|| mods.contains(Requires.Modifier.STATIC))) {
411423
String flagName;
412-
if (mods.contains(Requires.Modifier.TRANSITIVE)) {
413-
flagName = "ACC_TRANSITIVE";
414-
} else {
424+
if (mods.contains(Requires.Modifier.STATIC)) {
415425
flagName = "ACC_STATIC_PHASE";
426+
} else {
427+
flagName = "ACC_TRANSITIVE";
416428
}
417429
throw invalidModuleDescriptor("The requires entry for java.base"
418430
+ " has " + flagName + " set");

‎src/java.base/share/classes/module-info.java

+1
Original file line numberDiff line numberDiff line change
@@ -154,6 +154,7 @@
154154
exports jdk.internal.javac to
155155
java.compiler,
156156
java.desktop, // for ScopedValue
157+
java.se, // for ParticipatesInPreview
157158
jdk.compiler,
158159
jdk.incubator.vector, // participates in preview features
159160
jdk.jartool, // participates in preview features

‎src/java.se/share/classes/module-info.java

+4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@
2323
* questions.
2424
*/
2525

26+
import jdk.internal.javac.ParticipatesInPreview;
27+
2628
/**
2729
* Defines the API of the Java SE Platform.
2830
*
@@ -38,7 +40,9 @@
3840
* @moduleGraph
3941
* @since 9
4042
*/
43+
@ParticipatesInPreview
4144
module java.se {
45+
requires transitive java.base;
4246
requires transitive java.compiler;
4347
requires transitive java.datatransfer;
4448
requires transitive java.desktop;

‎src/jdk.compiler/share/classes/com/sun/tools/javac/api/JavacScope.java

+28-8
Original file line numberDiff line numberDiff line change
@@ -87,12 +87,26 @@ public JavacScope getEnclosingScope() {
8787
} else {
8888
// synthesize an outermost "star-import" scope
8989
return new JavacScope(env) {
90-
public boolean isStarImportScope() {
91-
return true;
90+
@Override
91+
public ScopeType getScopeType() {
92+
return ScopeType.STAR_IMPORT;
9293
}
9394
@DefinedBy(Api.COMPILER_TREE)
9495
public JavacScope getEnclosingScope() {
95-
return null;
96+
return new JavacScope(env) {
97+
@Override
98+
public ScopeType getScopeType() {
99+
return ScopeType.MODULE_IMPORT;
100+
}
101+
@Override @DefinedBy(Api.COMPILER_TREE)
102+
public JavacScope getEnclosingScope() {
103+
return null;
104+
}
105+
@Override @DefinedBy(Api.COMPILER_TREE)
106+
public Iterable<? extends Element> getLocalElements() {
107+
return env.toplevel.moduleImportScope.getSymbols(VALIDATOR);
108+
}
109+
};
96110
}
97111
@DefinedBy(Api.COMPILER_TREE)
98112
public Iterable<? extends Element> getLocalElements() {
@@ -122,21 +136,27 @@ public Env<AttrContext> getEnv() {
122136
return env;
123137
}
124138

125-
public boolean isStarImportScope() {
126-
return false;
139+
public ScopeType getScopeType() {
140+
return ScopeType.ORDINARY;
127141
}
128142

129143
public boolean equals(Object other) {
130144
return other instanceof JavacScope javacScope
131145
&& env.equals(javacScope.env)
132-
&& isStarImportScope() == javacScope.isStarImportScope();
146+
&& getScopeType()== javacScope.getScopeType();
133147
}
134148

135149
public int hashCode() {
136-
return env.hashCode() + (isStarImportScope() ? 1 : 0);
150+
return env.hashCode() + getScopeType().hashCode();
137151
}
138152

139153
public String toString() {
140-
return "JavacScope[env=" + env + ",starImport=" + isStarImportScope() + "]";
154+
return "JavacScope[env=" + env + ", scope type=" + getScopeType() + "]";
155+
}
156+
157+
private enum ScopeType {
158+
ORDINARY,
159+
STAR_IMPORT,
160+
MODULE_IMPORT;
141161
}
142162
}

‎src/jdk.compiler/share/classes/com/sun/tools/javac/api/JavacTrees.java

+1
Original file line numberDiff line numberDiff line change
@@ -1402,6 +1402,7 @@ public void putComment(JCTree tree, Comment c) {
14021402
jcCompilationUnit.namedImportScope = new NamedImportScope(psym);
14031403
jcCompilationUnit.packge = psym;
14041404
jcCompilationUnit.starImportScope = new StarImportScope(psym);
1405+
jcCompilationUnit.moduleImportScope = new StarImportScope(psym);
14051406
jcCompilationUnit.toplevelScope = WriteableScope.create(psym);
14061407
return new TreePath(jcCompilationUnit);
14071408
}

‎src/jdk.compiler/share/classes/com/sun/tools/javac/code/Preview.java

+15-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727

2828
import com.sun.tools.javac.code.Lint.LintCategory;
2929
import com.sun.tools.javac.code.Source.Feature;
30+
import com.sun.tools.javac.code.Symbol.ModuleSymbol;
3031
import com.sun.tools.javac.jvm.Target;
3132
import com.sun.tools.javac.resources.CompilerProperties.Errors;
3233
import com.sun.tools.javac.resources.CompilerProperties.Warnings;
@@ -133,11 +134,23 @@ public boolean participatesInPreview(Symtab syms, Symbol s, Symbol previewSymbol
133134
return true;
134135
}
135136

137+
return participatesInPreview(syms, s.packge().modle);
138+
}
139+
140+
/**
141+
* Returns true if module {@code m} is deemed to participate in the preview, and
142+
* therefore no warnings or errors will be produced.
143+
*
144+
* @param syms the symbol table
145+
* @param m the module to check
146+
* @return true if {@code m} is participating in the preview of {@code previewSymbol}
147+
*/
148+
public boolean participatesInPreview(Symtab syms, ModuleSymbol m) {
136149
// If java.base's jdk.internal.javac package is exported to s's module then
137150
// s participates in the preview API
138151
return syms.java_base.exports.stream()
139152
.filter(ed -> ed.packge.fullname == names.jdk_internal_javac)
140-
.anyMatch(ed -> ed.modules.contains(s.packge().modle));
153+
.anyMatch(ed -> ed.modules.contains(m));
141154
}
142155

143156
/**
@@ -211,6 +224,7 @@ public boolean isPreview(Feature feature) {
211224
case FLEXIBLE_CONSTRUCTORS -> true;
212225
case PRIMITIVE_PATTERNS -> true;
213226
case MODULE_IMPORTS -> true;
227+
case JAVA_BASE_TRANSITIVE -> true;
214228
//Note: this is a backdoor which allows to optionally treat all features as 'preview' (for testing).
215229
//When real preview features will be added, this method can be implemented to return 'true'
216230
//for those selected features, and 'false' for all the others.

‎src/jdk.compiler/share/classes/com/sun/tools/javac/code/Source.java

+1
Original file line numberDiff line numberDiff line change
@@ -262,6 +262,7 @@ public enum Feature {
262262
PRIMITIVE_PATTERNS(JDK23, Fragments.FeaturePrimitivePatterns, DiagKind.PLURAL),
263263
FLEXIBLE_CONSTRUCTORS(JDK22, Fragments.FeatureFlexibleConstructors, DiagKind.NORMAL),
264264
MODULE_IMPORTS(JDK23, Fragments.FeatureModuleImports, DiagKind.PLURAL),
265+
JAVA_BASE_TRANSITIVE(JDK24, Fragments.FeatureJavaBaseTransitive, DiagKind.PLURAL),
265266
PRIVATE_MEMBERS_IN_PERMITS_CLAUSE(JDK19),
266267
ERASE_POLY_SIG_RETURN_TYPE(JDK24),
267268
;

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Enter.java

+1
Original file line numberDiff line numberDiff line change
@@ -223,6 +223,7 @@ Env<AttrContext> topLevelEnv(JCCompilationUnit tree) {
223223
tree.toplevelScope = WriteableScope.create(tree.packge);
224224
tree.namedImportScope = new NamedImportScope(tree.packge);
225225
tree.starImportScope = new StarImportScope(tree.packge);
226+
tree.moduleImportScope = new StarImportScope(tree.packge);
226227
localEnv.info.scope = tree.toplevelScope;
227228
localEnv.info.lint = lint;
228229
return localEnv;

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Modules.java

+30-11
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2009, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2009, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -64,8 +64,11 @@
6464
import com.sun.tools.javac.code.Directive.UsesDirective;
6565
import com.sun.tools.javac.code.Flags;
6666
import com.sun.tools.javac.code.Flags.Flag;
67+
import com.sun.tools.javac.code.Kinds;
68+
import com.sun.tools.javac.code.Lint;
6769
import com.sun.tools.javac.code.Lint.LintCategory;
6870
import com.sun.tools.javac.code.ModuleFinder;
71+
import com.sun.tools.javac.code.Preview;
6972
import com.sun.tools.javac.code.Source;
7073
import com.sun.tools.javac.code.Source.Feature;
7174
import com.sun.tools.javac.code.Symbol;
@@ -74,6 +77,7 @@
7477
import com.sun.tools.javac.code.Symbol.CompletionFailure;
7578
import com.sun.tools.javac.code.Symbol.MethodSymbol;
7679
import com.sun.tools.javac.code.Symbol.ModuleFlags;
80+
import com.sun.tools.javac.code.Symbol.ModuleResolutionFlags;
7781
import com.sun.tools.javac.code.Symbol.ModuleSymbol;
7882
import com.sun.tools.javac.code.Symbol.PackageSymbol;
7983
import com.sun.tools.javac.code.Symtab;
@@ -99,6 +103,7 @@
99103
import com.sun.tools.javac.tree.TreeInfo;
100104
import com.sun.tools.javac.util.Assert;
101105
import com.sun.tools.javac.util.Context;
106+
import com.sun.tools.javac.util.JCDiagnostic.DiagnosticFlag;
102107
import com.sun.tools.javac.util.JCDiagnostic.DiagnosticPosition;
103108
import com.sun.tools.javac.util.List;
104109
import com.sun.tools.javac.util.ListBuffer;
@@ -112,14 +117,9 @@
112117
import static com.sun.tools.javac.code.Flags.PUBLIC;
113118
import static com.sun.tools.javac.code.Flags.UNATTRIBUTED;
114119

115-
import com.sun.tools.javac.code.Kinds;
116-
117120
import static com.sun.tools.javac.code.Kinds.Kind.ERR;
118121
import static com.sun.tools.javac.code.Kinds.Kind.MDL;
119122
import static com.sun.tools.javac.code.Kinds.Kind.MTH;
120-
import com.sun.tools.javac.code.Lint;
121-
122-
import com.sun.tools.javac.code.Symbol.ModuleResolutionFlags;
123123

124124
import static com.sun.tools.javac.code.TypeTag.CLASS;
125125

@@ -141,6 +141,7 @@ public class Modules extends JCTree.Visitor {
141141
private final Symtab syms;
142142
private final Attr attr;
143143
private final Check chk;
144+
private final Preview preview;
144145
private final DeferredLintHandler deferredLintHandler;
145146
private final TypeEnvs typeEnvs;
146147
private final Types types;
@@ -150,6 +151,7 @@ public class Modules extends JCTree.Visitor {
150151
private final Target target;
151152
private final boolean allowModules;
152153
private final boolean allowAccessIntoSystem;
154+
private final boolean allowRequiresTransitiveJavaBase;
153155

154156
public final boolean multiModuleMode;
155157

@@ -192,6 +194,7 @@ protected Modules(Context context) {
192194
syms = Symtab.instance(context);
193195
attr = Attr.instance(context);
194196
chk = Check.instance(context);
197+
preview = Preview.instance(context);
195198
deferredLintHandler = DeferredLintHandler.instance(context);
196199
typeEnvs = TypeEnvs.instance(context);
197200
moduleFinder = ModuleFinder.instance(context);
@@ -203,6 +206,12 @@ protected Modules(Context context) {
203206
Options options = Options.instance(context);
204207

205208
allowAccessIntoSystem = options.isUnset(Option.RELEASE);
209+
210+
Preview preview = Preview.instance(context);
211+
212+
allowRequiresTransitiveJavaBase =
213+
Feature.JAVA_BASE_TRANSITIVE.allowedInSource(source) &&
214+
(!preview.isPreview(Feature.JAVA_BASE_TRANSITIVE) || preview.isEnabled());
206215
lintOptions = options.isUnset(Option.XLINT_CUSTOM, "-" + LintCategory.OPTIONS.option);
207216

208217
multiModuleMode = fileManager.hasLocation(StandardLocation.MODULE_SOURCE_PATH);
@@ -249,7 +258,12 @@ public void initModules(List<JCCompilationUnit> trees) {
249258

250259
public boolean enter(List<JCCompilationUnit> trees, ClassSymbol c) {
251260
Assert.check(rootModules != null || inInitModules || !allowModules);
252-
return enter(trees, modules -> {}, c);
261+
return enter(trees, modules -> {
262+
//make sure java.base is completed in all cases before continuing.
263+
//the next steps may query if the current module participates in preview,
264+
//and that requires a completed java.base:
265+
syms.java_base.complete();
266+
}, c);
253267
}
254268

255269
private boolean enter(List<JCCompilationUnit> trees, Consumer<Set<ModuleSymbol>> init, ClassSymbol c) {
@@ -807,11 +821,16 @@ public void visitRequires(JCRequires tree) {
807821
allRequires.add(msym);
808822
Set<RequiresFlag> flags = EnumSet.noneOf(RequiresFlag.class);
809823
if (tree.isTransitive) {
810-
if (msym == syms.java_base && source.compareTo(Source.JDK10) >= 0) {
811-
log.error(tree.pos(), Errors.ModifierNotAllowedHere(names.transitive));
812-
} else {
813-
flags.add(RequiresFlag.TRANSITIVE);
824+
if (msym == syms.java_base &&
825+
!allowRequiresTransitiveJavaBase &&
826+
!preview.participatesInPreview(syms, sym)) {
827+
if (source.compareTo(Source.JDK10) >= 0) {
828+
log.error(DiagnosticFlag.SOURCE_LEVEL,
829+
tree.pos(),
830+
Feature.JAVA_BASE_TRANSITIVE.error(source.name));
831+
}
814832
}
833+
flags.add(RequiresFlag.TRANSITIVE);
815834
}
816835
if (tree.isStaticPhase) {
817836
if (msym == syms.java_base && source.compareTo(Source.JDK10) >= 0) {

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/Resolve.java

+27-13
Original file line numberDiff line numberDiff line change
@@ -2149,23 +2149,32 @@ public interface RecoveryLoadClass {
21492149
return null;
21502150
};
21512151

2152-
private final RecoveryLoadClass starImportScopeRecovery = (env, name) -> {
2153-
Scope importScope = env.toplevel.starImportScope;
2154-
Symbol existing = importScope.findFirst(Convert.shortName(name),
2155-
sym -> sym.kind == TYP && sym.flatName() == name);
2152+
private final RecoveryLoadClass starImportScopeRecovery =
2153+
onDemandImportScopeRecovery(false);
21562154

2157-
if (existing != null) {
2158-
try {
2159-
existing = finder.loadClass(existing.packge().modle, name);
2155+
private final RecoveryLoadClass moduleImportScopeRecovery =
2156+
onDemandImportScopeRecovery(true);
2157+
2158+
private RecoveryLoadClass onDemandImportScopeRecovery(boolean moduleImportScope) {
2159+
return (env, name) -> {
2160+
Scope importScope = moduleImportScope ? env.toplevel.moduleImportScope
2161+
: env.toplevel.starImportScope;
2162+
Symbol existing = importScope.findFirst(Convert.shortName(name),
2163+
sym -> sym.kind == TYP && sym.flatName() == name);
21602164

2161-
return new InvisibleSymbolError(env, true, existing);
2162-
} catch (CompletionFailure cf) {
2163-
//ignore
2165+
if (existing != null) {
2166+
try {
2167+
existing = finder.loadClass(existing.packge().modle, name);
2168+
2169+
return new InvisibleSymbolError(env, true, existing);
2170+
} catch (CompletionFailure cf) {
2171+
//ignore
2172+
}
21642173
}
2165-
}
21662174

2167-
return null;
2168-
};
2175+
return null;
2176+
};
2177+
}
21692178

21702179
Symbol lookupPackage(Env<AttrContext> env, Name name) {
21712180
PackageSymbol pack = syms.lookupPackage(env.toplevel.modle, name);
@@ -2433,6 +2442,11 @@ Symbol findType(Env<AttrContext> env, Name name) {
24332442
sym = findGlobalType(env, env.toplevel.starImportScope, name, starImportScopeRecovery);
24342443
if (sym.exists()) return sym;
24352444
else bestSoFar = bestOf(bestSoFar, sym);
2445+
2446+
sym = findGlobalType(env, env.toplevel.moduleImportScope, name, moduleImportScopeRecovery);
2447+
if (sym.exists()) return sym;
2448+
2449+
else bestSoFar = bestOf(bestSoFar, sym);
24362450
}
24372451

24382452
return bestSoFar;

‎src/jdk.compiler/share/classes/com/sun/tools/javac/comp/TypeEnter.java

+20-10
Original file line numberDiff line numberDiff line change
@@ -220,6 +220,7 @@ void finishImports(JCCompilationUnit toplevel, Runnable resolve) {
220220
chk.checkImportedPackagesObservable(toplevel);
221221
toplevel.namedImportScope.finalizeScope();
222222
toplevel.starImportScope.finalizeScope();
223+
toplevel.moduleImportScope.finalizeScope();
223224
} catch (CompletionFailure cf) {
224225
chk.completionError(toplevel.pos(), cf);
225226
} finally {
@@ -331,7 +332,7 @@ private void implicitImports(JCCompilationUnit tree, Env<AttrContext> env) {
331332
throw new Abort();
332333
}
333334
importAll(make.at(tree.pos()).Import(make.Select(make.QualIdent(javaLang.owner), javaLang), false),
334-
javaLang, env);
335+
javaLang, env, false);
335336

336337
List<JCTree> defs = tree.getTypeDecls();
337338
boolean isImplicitClass = !defs.isEmpty() &&
@@ -341,7 +342,7 @@ private void implicitImports(JCCompilationUnit tree, Env<AttrContext> env) {
341342
doModuleImport(make.ModuleImport(make.QualIdent(syms.java_base)));
342343
if (peekTypeExists(syms.ioType.tsym)) {
343344
doImport(make.Import(make.Select(make.QualIdent(syms.ioType.tsym),
344-
names.asterisk), true));
345+
names.asterisk), true), false);
345346
}
346347
}
347348
}
@@ -413,7 +414,7 @@ private void handleImports(List<JCImportBase> imports) {
413414
if (imp instanceof JCModuleImport mimp) {
414415
doModuleImport(mimp);
415416
} else {
416-
doImport((JCImport) imp);
417+
doImport((JCImport) imp, false);
417418
}
418419
}
419420
}
@@ -438,7 +439,7 @@ private void checkClassPackageClash(JCPackageDecl tree) {
438439
annotate.annotateLater(tree.annotations, env, env.toplevel.packge, tree.pos());
439440
}
440441

441-
private void doImport(JCImport tree) {
442+
private void doImport(JCImport tree, boolean fromModuleImport) {
442443
JCFieldAccess imp = tree.qualid;
443444
Name name = TreeInfo.name(imp);
444445

@@ -450,16 +451,20 @@ private void doImport(JCImport tree) {
450451
if (name == names.asterisk) {
451452
// Import on demand.
452453
chk.checkCanonical(imp.selected);
453-
if (tree.staticImport)
454+
if (tree.staticImport) {
455+
Assert.check(!fromModuleImport);
454456
importStaticAll(tree, p, env);
455-
else
456-
importAll(tree, p, env);
457+
} else {
458+
importAll(tree, p, env, fromModuleImport);
459+
}
457460
} else {
458461
// Named type import.
459462
if (tree.staticImport) {
463+
Assert.check(!fromModuleImport);
460464
importNamedStatic(tree, p, name, localEnv);
461465
chk.checkCanonical(imp.selected);
462466
} else {
467+
Assert.check(!fromModuleImport);
463468
Type importedType = attribImportType(imp, localEnv);
464469
Type originalType = importedType.getOriginalType();
465470
TypeSymbol c = originalType.hasTag(CLASS) ? originalType.tsym : importedType.tsym;
@@ -506,7 +511,7 @@ private void doModuleImport(JCModuleImport tree) {
506511
JCImport nestedImport = make.at(tree.pos)
507512
.Import(make.Select(make.QualIdent(pkg), names.asterisk), false);
508513

509-
doImport(nestedImport);
514+
doImport(nestedImport, true);
510515
}
511516

512517
for (RequiresDirective requires : currentModule.requires) {
@@ -542,8 +547,13 @@ Type attribImportType(JCTree tree, Env<AttrContext> env) {
542547
*/
543548
private void importAll(JCImport imp,
544549
final TypeSymbol tsym,
545-
Env<AttrContext> env) {
546-
env.toplevel.starImportScope.importAll(types, tsym.members(), typeImportFilter, imp, cfHandler);
550+
Env<AttrContext> env,
551+
boolean fromModuleImport) {
552+
StarImportScope targetScope =
553+
fromModuleImport ? env.toplevel.moduleImportScope
554+
: env.toplevel.starImportScope;
555+
556+
targetScope.importAll(types, tsym.members(), typeImportFilter, imp, cfHandler);
547557
}
548558

549559
/** Import all static members of a class or package on demand.

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

+7-2
Original file line numberDiff line numberDiff line change
@@ -199,6 +199,9 @@ public class ClassReader {
199199
/** The minor version number of the class file being read. */
200200
int minorVersion;
201201

202+
/** true if the class file being read is a preview class file. */
203+
boolean previewClassFile;
204+
202205
/** UTF-8 validation level */
203206
Convert.Validation utf8validation;
204207

@@ -1200,7 +1203,9 @@ protected void read(Symbol sym, int attrLen) {
12001203
ModuleSymbol rsym = poolReader.getModule(nextChar());
12011204
Set<RequiresFlag> flags = readRequiresFlags(nextChar());
12021205
if (rsym == syms.java_base && majorVersion >= V54.major) {
1203-
if (flags.contains(RequiresFlag.TRANSITIVE)) {
1206+
if (flags.contains(RequiresFlag.TRANSITIVE) &&
1207+
(majorVersion != Version.MAX().major || !previewClassFile) &&
1208+
!preview.participatesInPreview(syms, msym)) {
12041209
throw badClassFile("bad.requires.flag", RequiresFlag.TRANSITIVE);
12051210
}
12061211
if (flags.contains(RequiresFlag.STATIC_PHASE)) {
@@ -3185,7 +3190,7 @@ private void readClassBuffer(ClassSymbol c) throws IOException {
31853190
majorVersion = nextChar();
31863191
int maxMajor = Version.MAX().major;
31873192
int maxMinor = Version.MAX().minor;
3188-
boolean previewClassFile =
3193+
previewClassFile =
31893194
minorVersion == ClassFile.PREVIEW_MINOR_VERSION;
31903195
if (majorVersion > maxMajor ||
31913196
majorVersion * 1000 + minorVersion <

‎src/jdk.compiler/share/classes/com/sun/tools/javac/resources/compiler.properties

+3-4
Original file line numberDiff line numberDiff line change
@@ -921,10 +921,6 @@ compiler.misc.unexpected.ret.val=\
921921
compiler.err.mod.not.allowed.here=\
922922
modifier {0} not allowed here
923923

924-
# 0: name
925-
compiler.err.modifier.not.allowed.here=\
926-
modifier {0} not allowed here
927-
928924
compiler.err.intf.not.allowed.here=\
929925
interface not allowed here
930926

@@ -3252,6 +3248,9 @@ compiler.misc.feature.flexible.constructors=\
32523248
compiler.misc.feature.module.imports=\
32533249
module imports
32543250

3251+
compiler.misc.feature.java.base.transitive=\
3252+
transitive modifier for java.base
3253+
32553254
compiler.warn.underscore.as.identifier=\
32563255
as of release 9, ''_'' is a keyword, and may not be used as an identifier
32573256

‎src/jdk.compiler/share/classes/com/sun/tools/javac/tree/JCTree.java

+2
Original file line numberDiff line numberDiff line change
@@ -538,6 +538,8 @@ public static class JCCompilationUnit extends JCTree implements CompilationUnitT
538538
public NamedImportScope namedImportScope;
539539
/** A scope for all import-on-demands. */
540540
public StarImportScope starImportScope;
541+
/** A scope for all single module imports. */
542+
public StarImportScope moduleImportScope;
541543
/** Line starting positions, defined only if option -g is set. */
542544
public Position.LineMap lineMap = null;
543545
/** A table that stores all documentation comments indexed by the tree

‎src/jdk.compiler/share/classes/com/sun/tools/javac/tree/TreeMaker.java

+11-20
Original file line numberDiff line numberDiff line change
@@ -1138,26 +1138,17 @@ boolean isUnqualifiable(Symbol sym) {
11381138
sym.owner.kind == MTH || sym.owner.kind == VAR) {
11391139
return true;
11401140
} else if (sym.kind == TYP && toplevel != null) {
1141-
Iterator<Symbol> it = toplevel.namedImportScope.getSymbolsByName(sym.name).iterator();
1142-
if (it.hasNext()) {
1143-
Symbol s = it.next();
1144-
return
1145-
s == sym &&
1146-
!it.hasNext();
1147-
}
1148-
it = toplevel.packge.members().getSymbolsByName(sym.name).iterator();
1149-
if (it.hasNext()) {
1150-
Symbol s = it.next();
1151-
return
1152-
s == sym &&
1153-
!it.hasNext();
1154-
}
1155-
it = toplevel.starImportScope.getSymbolsByName(sym.name).iterator();
1156-
if (it.hasNext()) {
1157-
Symbol s = it.next();
1158-
return
1159-
s == sym &&
1160-
!it.hasNext();
1141+
for (Scope scope : new Scope[] {toplevel.namedImportScope,
1142+
toplevel.packge.members(),
1143+
toplevel.starImportScope,
1144+
toplevel.moduleImportScope}) {
1145+
Iterator<Symbol> it = scope.getSymbolsByName(sym.name).iterator();
1146+
if (it.hasNext()) {
1147+
Symbol s = it.next();
1148+
return
1149+
s == sym &&
1150+
!it.hasNext();
1151+
}
11611152
}
11621153
}
11631154
return sym.kind == TYP && sym.isImplicit();

‎test/jdk/java/lang/module/ClassFileVersionsTest.java

+44-24
Original file line numberDiff line numberDiff line change
@@ -31,10 +31,13 @@
3131
* @summary Test parsing of module-info.class with different class file versions
3232
*/
3333

34+
import java.lang.classfile.ClassFile;
3435
import java.lang.module.InvalidModuleDescriptorException;
3536
import java.lang.module.ModuleDescriptor;
3637
import java.lang.module.ModuleDescriptor.Requires.Modifier;
3738
import java.nio.ByteBuffer;
39+
import java.util.ArrayList;
40+
import java.util.List;
3841
import java.util.Set;
3942

4043
import static java.lang.module.ModuleDescriptor.Requires.Modifier.*;
@@ -46,6 +49,8 @@
4649
import static org.testng.Assert.*;
4750

4851
public class ClassFileVersionsTest {
52+
private static final int PREVIEW_MINOR_VERSION =
53+
ClassFile.PREVIEW_MINOR_VERSION;
4954
private static final int FEATURE;
5055
static {
5156
FEATURE = Runtime.version().feature();
@@ -56,26 +61,34 @@ public class ClassFileVersionsTest {
5661
@DataProvider(name = "supported")
5762
public Object[][] supported() {
5863
/*
59-
* There are four test cases for JDK 9 and then one test case
64+
* There are four test cases for JDK 9, one test case
6065
* for each subsequent JDK version from JDK 10 to the current
61-
* feature release for a total of (4 + (FEATURE - 9) ) =>
62-
* (feature - 5) rows.
66+
* feature release, and two tests for the current release with
67+
* a preview flag set, for a total of (4 + (FEATURE - 9) + 2)
68+
* rows.
6369
*/
64-
Object[][] result = new Object[(FEATURE - 5)][];
70+
List<Object[]> result = new ArrayList<>(4 + (FEATURE - 9) + 2);
6571

6672
// Class file version of JDK 9 is 53.0
67-
result[0] = new Object[]{ 53, 0, Set.of()};
68-
result[1] = new Object[]{ 53, 0, Set.of(STATIC) };
69-
result[2] = new Object[]{ 53, 0, Set.of(TRANSITIVE) };
70-
result[3] = new Object[]{ 53, 0, Set.of(STATIC, TRANSITIVE) };
73+
result.add(new Object[]{ 53, 0, Set.of()});
74+
result.add(new Object[]{ 53, 0, Set.of(STATIC) });
75+
result.add(new Object[]{ 53, 0, Set.of(TRANSITIVE) });
76+
result.add(new Object[]{ 53, 0, Set.of(STATIC, TRANSITIVE) });
7177

7278
// Major class file version of JDK N is 44 + n. Create rows
7379
// for JDK 10 through FEATURE.
74-
for (int i = 4; i < (FEATURE - 5) ; i++) {
75-
result[i] = new Object[]{i + 50, 0, Set.of()};
80+
for (int i = 10; i <= FEATURE; i++) {
81+
result.add(new Object[]{ 44 + i, 0, Set.of()});
7682
}
7783

78-
return result;
84+
result.add(new Object[]{ 44 + FEATURE,
85+
PREVIEW_MINOR_VERSION,
86+
Set.of()});
87+
result.add(new Object[]{ 44 + FEATURE,
88+
PREVIEW_MINOR_VERSION,
89+
Set.of(TRANSITIVE) });
90+
91+
return result.toArray(s -> new Object[s][]);
7992
}
8093

8194
// major, minor, modifiers for requires java.base
@@ -84,27 +97,34 @@ public Object[][] unsupported() {
8497
/*
8598
* There are three test cases for releases prior to JDK 9,
8699
* three test cases for each JDK version from JDK 10 to the
87-
* current feature release, plus one addition test case for
88-
* the next release for a total of (3 + (FEATURE - 9) * 3 + 1)
100+
* current feature release, two tests for the current release with
101+
* the preview flag set, plus one addition test case for
102+
* the next release for a total of (3 + (FEATURE - 9) * 3 + 2 + 1)
89103
* rows.
90104
*/
91-
int unsupportedCount = 3 + (FEATURE - 9)*3 + 1;
92-
Object[][] result = new Object[unsupportedCount][];
105+
List<Object[]> result = new ArrayList<>(3 + (FEATURE - 9) * 3 + 2 + 1);
93106

94-
result[0] = new Object[]{50, 0, Set.of()}; // JDK 6
95-
result[1] = new Object[]{51, 0, Set.of()}; // JDK 7
96-
result[2] = new Object[]{52, 0, Set.of()}; // JDK 8
107+
result.add(new Object[]{50, 0, Set.of()}); // JDK 6
108+
result.add(new Object[]{51, 0, Set.of()}); // JDK 7
109+
result.add(new Object[]{52, 0, Set.of()}); // JDK 8
97110

98111
for (int i = 10; i <= FEATURE ; i++) {
99-
int base = 3 + (i-10)*3;
100112
// Major class file version of JDK N is 44+n
101-
result[base] = new Object[]{i + 44, 0, Set.of(STATIC)};
102-
result[base + 1] = new Object[]{i + 44, 0, Set.of(TRANSITIVE)};
103-
result[base + 2] = new Object[]{i + 44, 0, Set.of(STATIC, TRANSITIVE)};
113+
result.add(new Object[]{i + 44, 0, Set.of(STATIC)});
114+
result.add(new Object[]{i + 44, 0, Set.of(TRANSITIVE)});
115+
result.add(new Object[]{i + 44, 0, Set.of(STATIC, TRANSITIVE)});
104116
}
105117

106-
result[unsupportedCount - 1] = new Object[]{FEATURE+1+44, 0, Set.of()};
107-
return result;
118+
result.add(new Object[]{FEATURE + 44,
119+
PREVIEW_MINOR_VERSION,
120+
Set.of(STATIC)});
121+
result.add(new Object[]{FEATURE + 44,
122+
PREVIEW_MINOR_VERSION,
123+
Set.of(STATIC, TRANSITIVE)});
124+
125+
result.add(new Object[]{FEATURE+1+44, 0, Set.of()});
126+
127+
return result.toArray(s -> new Object[s][]);
108128
}
109129

110130
@Test(dataProvider = "supported")

‎test/jdk/java/lang/module/ModuleDescriptorTest.java

+66
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,8 @@
6161
import jdk.internal.access.JavaLangModuleAccess;
6262
import jdk.internal.access.SharedSecrets;
6363
import java.lang.classfile.ClassFile;
64+
import java.lang.classfile.ClassFileVersion;
65+
import java.lang.classfile.ClassTransform;
6466
import java.lang.classfile.attribute.ModuleAttribute;
6567
import java.lang.constant.PackageDesc;
6668
import java.lang.constant.ModuleDesc;
@@ -1522,4 +1524,68 @@ public void testToString() {
15221524
assertTrue(s.contains("p1"));
15231525
}
15241526

1527+
@Test(expectedExceptions = InvalidModuleDescriptorException.class)
1528+
public void testRequiresTransitiveJavaBaseNotPermitted1() throws Exception {
1529+
ModuleDescriptor descriptor = ModuleDescriptor.newModule("foo")
1530+
.requires(Set.of(Modifier.TRANSITIVE), "java.base")
1531+
.build();
1532+
1533+
ByteArrayOutputStream baos = new ByteArrayOutputStream();
1534+
ModuleInfoWriter.write(descriptor, baos);
1535+
ByteBuffer bb = ByteBuffer.wrap(baos.toByteArray());
1536+
1537+
ModuleDescriptor.read(bb, () -> Set.of("p", "q"));
1538+
}
1539+
1540+
@Test(expectedExceptions = InvalidModuleDescriptorException.class)
1541+
public void testRequiresTransitiveJavaBaseNotPermitted2() throws Exception {
1542+
ModuleDescriptor descriptor = ModuleDescriptor.newModule("foo")
1543+
.requires(Set.of(Modifier.TRANSITIVE), "java.base")
1544+
.build();
1545+
1546+
ByteArrayOutputStream baos = new ByteArrayOutputStream();
1547+
ModuleInfoWriter.write(descriptor, baos);
1548+
byte[] bytecode = baos.toByteArray();
1549+
ByteBuffer bb = ByteBuffer.wrap(bytecode);
1550+
setClassFileVersion(bb, ClassFile.JAVA_21_VERSION, -1);
1551+
1552+
ModuleDescriptor.read(bb, () -> Set.of("p", "q"));
1553+
}
1554+
1555+
public void testRequiresTransitiveJavaBasePermitted() throws Exception {
1556+
ModuleDescriptor descriptor = ModuleDescriptor.newModule("foo")
1557+
.requires(Set.of(Modifier.TRANSITIVE), "java.base")
1558+
.build();
1559+
1560+
ByteArrayOutputStream baos = new ByteArrayOutputStream();
1561+
ModuleInfoWriter.write(descriptor, baos);
1562+
byte[] bytecode = baos.toByteArray();
1563+
ByteBuffer bb = ByteBuffer.wrap(bytecode);
1564+
setClassFileVersion(bb, -1, ClassFile.PREVIEW_MINOR_VERSION);
1565+
1566+
descriptor = ModuleDescriptor.read(bb, () -> Set.of("p", "q"));
1567+
1568+
assertEquals(descriptor.requires().size(), 1);
1569+
Requires javaBase = descriptor.requires().iterator().next();
1570+
assertEquals(javaBase.name(), "java.base");
1571+
assertEquals(javaBase.modifiers(), Set.of(Modifier.TRANSITIVE));
1572+
}
1573+
1574+
/**Change the classfile versions of the provided classfile to the provided
1575+
* values.
1576+
*
1577+
* @param bytecode the classfile content to modify
1578+
* @param major the major classfile version to set,
1579+
* -1 if the existing version should be kept
1580+
* @param minor the minor classfile version to set,
1581+
* -1 if the existing version should be kept
1582+
*/
1583+
private void setClassFileVersion(ByteBuffer bb, int major, int minor) {
1584+
if (minor != (-1)) {
1585+
bb.putShort(4, (short) minor);
1586+
}
1587+
if (major != (-1)) {
1588+
bb.putShort(6, (short) major);
1589+
}
1590+
}
15251591
}

‎test/langtools/tools/javac/6402516/TestClass.java

+11-11
Original file line numberDiff line numberDiff line change
@@ -26,32 +26,32 @@
2626

2727
class Test {
2828
void m1(int m1_arg) {
29-
String x = "Test; 0; 0";
30-
String y = "Test; 0; 0";
31-
String z = "Test; 0; 0";
29+
String x = "Test; 0; 0; 0";
30+
String y = "Test; 0; 0; 0";
31+
String z = "Test; 0; 0; 0";
3232
Object o = new Object() {
3333
public boolean equals(Object other) {
34-
String p = "-; Test; 0; 0";
35-
String q = "-; Test; 0; 0";
36-
String r = "-; Test; 0; 0";
34+
String p = "-; Test; 0; 0; 0";
35+
String q = "-; Test; 0; 0; 0";
36+
String r = "-; Test; 0; 0; 0";
3737
return (this == other);
3838
}
3939
};
4040
}
4141

42-
String s = "Test; 0; 0";
42+
String s = "Test; 0; 0; 0";
4343

4444
boolean b = new Object() {
4545
public boolean equals(Object other) {
46-
String p = "-; Test; 0; 0";
47-
String q = "-; Test; 0; 0";
48-
String r = "-; Test; 0; 0";
46+
String p = "-; Test; 0; 0; 0";
47+
String q = "-; Test; 0; 0; 0";
48+
String r = "-; Test; 0; 0; 0";
4949
return (this == other);
5050
}
5151

5252
}.equals(null);
5353

5454
class Test2 {
55-
String s = "Test.Test2; Test; 0; 0";
55+
String s = "Test.Test2; Test; 0; 0; 0";
5656
}
5757
}

‎test/langtools/tools/javac/6402516/TestLocalElements.java

+11-11
Original file line numberDiff line numberDiff line change
@@ -23,29 +23,29 @@
2323

2424
import java.util.List;
2525
import java.io.*;
26-
//TOPLEVEL_SCOPE:List, Test2, Test; java.io.*, java.lang.*
26+
//TOPLEVEL_SCOPE:List, Test2, Test; java.io.*, java.lang.*;
2727
class Test {
2828
void m1(int m1_arg) {
29-
String x = "x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
30-
String y = "y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
31-
String z = "z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
29+
String x = "x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
30+
String y = "y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
31+
String z = "z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
3232
Object o = new Object() {
3333
public boolean equals(Object other) {
34-
String p = "p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
35-
String q = "q, p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
36-
String r = "r, q, p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*";
34+
String p = "p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
35+
String q = "q, p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
36+
String r = "r, q, p, other, super, this; -, o, z, y, x, m1_arg, super, this; List, Test2, Test; java.io.*, java.lang.*;";
3737
return (this == other);
3838
}
3939
};
4040
}
4141

42-
String s = "super, this; List, Test2, Test; java.io.*, java.lang.*";
42+
String s = "super, this; List, Test2, Test; java.io.*, java.lang.*;";
4343

4444
boolean b = new Object() {
4545
public boolean equals(Object other) {
46-
String p = "p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*";
47-
String q = "q, p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*";
48-
String r = "r, q, p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*";
46+
String p = "p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*;";
47+
String q = "q, p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*;";
48+
String r = "r, q, p, other, super, this; -, super, this; List, Test2, Test; java.io.*, java.lang.*;";
4949
return (this == other);
5050
}
5151

‎test/langtools/tools/javac/6402516/TestMethod.java

+11-11
Original file line numberDiff line numberDiff line change
@@ -25,32 +25,32 @@
2525

2626
class Test {
2727
void m1(int m1_arg) {
28-
String x = "m1; 0; 0";
29-
String y = "m1; 0; 0";
30-
String z = "m1; 0; 0";
28+
String x = "m1; 0; 0; 0";
29+
String y = "m1; 0; 0; 0";
30+
String z = "m1; 0; 0; 0";
3131
Object o = new Object() {
3232
public boolean equals(Object other) {
33-
String p = "equals; m1; 0; 0";
34-
String q = "equals; m1; 0; 0";
35-
String r = "equals; m1; 0; 0";
33+
String p = "equals; m1; 0; 0; 0";
34+
String q = "equals; m1; 0; 0; 0";
35+
String r = "equals; m1; 0; 0; 0";
3636
return (this == other);
3737
}
3838
};
3939
}
4040

41-
String s = "0; 0; 0";
41+
String s = "0; 0; 0; 0";
4242

4343
boolean b = new Object() {
4444
public boolean equals(Object other) {
45-
String p = "equals; 0; 0; 0";
46-
String q = "equals; 0; 0; 0";
47-
String r = "equals; 0; 0; 0";
45+
String p = "equals; 0; 0; 0; 0";
46+
String q = "equals; 0; 0; 0; 0";
47+
String r = "equals; 0; 0; 0; 0";
4848
return (this == other);
4949
}
5050

5151
}.equals(null);
5252

5353
class Test2 {
54-
String s = "0; 0; 0; 0";
54+
String s = "0; 0; 0; 0; 0";
5555
}
5656
}

‎test/langtools/tools/javac/ImportModule.java

+99-23
Original file line numberDiff line numberDiff line change
@@ -218,28 +218,14 @@ public class Test {
218218
List<String> actualErrors;
219219
List<String> expectedErrors;
220220

221-
actualErrors =
222-
new JavacTask(tb)
223-
.options("--enable-preview", "--release", SOURCE_VERSION,
224-
"-XDrawDiagnostics")
225-
.outdir(classes)
226-
.files(tb.findJavaFiles(src))
227-
.run(Task.Expect.FAIL)
228-
.writeAll()
229-
.getOutputLines(Task.OutputKind.DIRECT);
230-
231-
expectedErrors = List.of(
232-
"Test.java:5:5: compiler.err.ref.ambiguous: Logger, kindname.interface, java.lang.System.Logger, java.lang.System, kindname.class, java.util.logging.Logger, java.util.logging",
233-
"- compiler.note.preview.filename: Test.java, DEFAULT",
234-
"- compiler.note.preview.recompile",
235-
"1 error"
236-
);
237-
238-
if (!Objects.equals(expectedErrors, actualErrors)) {
239-
throw new AssertionError("Incorrect Output, expected: " + expectedErrors +
240-
", actual: " + out);
241-
242-
}
221+
new JavacTask(tb)
222+
.options("--enable-preview", "--release", SOURCE_VERSION,
223+
"-XDrawDiagnostics")
224+
.outdir(classes)
225+
.files(tb.findJavaFiles(src))
226+
.run(Task.Expect.SUCCESS)
227+
.writeAll()
228+
.getOutputLines(Task.OutputKind.DIRECT);
243229

244230
tb.writeJavaFiles(src,
245231
"""
@@ -793,7 +779,7 @@ public class C {}
793779

794780
if (!Objects.equals(expectedErrors, actualErrors)) {
795781
throw new AssertionError("Incorrect Output, expected: " + expectedErrors +
796-
", actual: " + out);
782+
", actual: " + actualErrors);
797783

798784
}
799785
}
@@ -843,4 +829,94 @@ public class Test {
843829
}
844830
}
845831

832+
public void testPackageImportDisambiguates(Path base) throws Exception {
833+
Path current = base.resolve(".");
834+
Path src = current.resolve("src");
835+
Path classes = current.resolve("classes");
836+
Path ma = src.resolve("ma");
837+
tb.writeJavaFiles(ma,
838+
"""
839+
module ma {
840+
exports ma.p1;
841+
}
842+
""",
843+
"""
844+
package ma.p1;
845+
public class A {}
846+
""");
847+
Path mb = src.resolve("mb");
848+
tb.writeJavaFiles(mb,
849+
"""
850+
module mb {
851+
exports mb.p1;
852+
}
853+
""",
854+
"""
855+
package mb.p1;
856+
public class A {}
857+
""");
858+
Path test = src.resolve("test");
859+
tb.writeJavaFiles(test,
860+
"""
861+
module test {
862+
requires ma;
863+
requires mb;
864+
}
865+
""",
866+
"""
867+
package test;
868+
import module ma;
869+
import module mb;
870+
public class Test {
871+
A a;
872+
}
873+
""");
874+
875+
Files.createDirectories(classes);
876+
877+
List<String> actualErrors = new JavacTask(tb)
878+
.options("-XDrawDiagnostics",
879+
"--enable-preview", "--release", SOURCE_VERSION,
880+
"--module-source-path", src.toString())
881+
.outdir(classes)
882+
.files(tb.findJavaFiles(src))
883+
.run(Task.Expect.FAIL)
884+
.writeAll()
885+
.getOutputLines(Task.OutputKind.DIRECT);
886+
887+
List<String> expectedErrors = List.of(
888+
"Test.java:5:5: compiler.err.ref.ambiguous: A, kindname.class, mb.p1.A, mb.p1, kindname.class, ma.p1.A, ma.p1",
889+
"- compiler.note.preview.filename: Test.java, DEFAULT",
890+
"- compiler.note.preview.recompile",
891+
"1 error"
892+
);
893+
894+
if (!Objects.equals(expectedErrors, actualErrors)) {
895+
throw new AssertionError("Incorrect Output, expected: " + expectedErrors +
896+
", actual: " + actualErrors);
897+
898+
}
899+
900+
tb.writeJavaFiles(test,
901+
"""
902+
package test;
903+
import module ma;
904+
import module mb;
905+
import mb.p1.*;
906+
public class Test {
907+
A a;
908+
}
909+
""");
910+
911+
Files.createDirectories(classes);
912+
913+
new JavacTask(tb)
914+
.options("-XDrawDiagnostics",
915+
"--enable-preview", "--release", SOURCE_VERSION,
916+
"--module-source-path", src.toString())
917+
.outdir(classes)
918+
.files(tb.findJavaFiles(src))
919+
.run(Task.Expect.SUCCESS)
920+
.writeAll();
921+
}
846922
}

‎test/langtools/tools/javac/api/TestGetScopeResult.java

+75
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,9 @@
7373
import com.sun.tools.javac.tree.JCTree.JCStatement;
7474
import com.sun.tools.javac.util.Context;
7575
import com.sun.tools.javac.util.Context.Factory;
76+
import javax.lang.model.element.TypeElement;
77+
import javax.lang.model.util.ElementFilter;
78+
import javax.tools.JavaFileObject;
7679

7780
import static javax.tools.JavaFileObject.Kind.SOURCE;
7881

@@ -89,6 +92,7 @@ public static void main(String... args) throws IOException {
8992
new TestGetScopeResult().testLocalRecordAnnotation();
9093
new TestGetScopeResult().testRuleCases();
9194
new TestGetScopeResult().testNestedSwitchExpression();
95+
new TestGetScopeResult().testModuleImportScope();
9296
}
9397

9498
public void run() throws IOException {
@@ -823,6 +827,64 @@ JCTree getCaseBody(Scope scope) {
823827
}
824828
}
825829

830+
void testModuleImportScope() throws IOException {
831+
JavacTool c = JavacTool.create();
832+
try (StandardJavaFileManager fm = c.getStandardFileManager(null, null, null)) {
833+
String code = """
834+
import module java.compiler;
835+
import java.util.*;
836+
import java.lang.System;
837+
class Test {
838+
}
839+
""";
840+
Context ctx = new Context();
841+
TestAnalyzer.preRegister(ctx);
842+
JavaFileObject input =
843+
SimpleJavaFileObject.forSource(URI.create("myfo:///Test.java"), code);
844+
JavacTask t = (JavacTask) c.getTask(null, fm, null, null, null,
845+
List.of(input),
846+
ctx);
847+
CompilationUnitTree cut = t.parse().iterator().next();
848+
t.analyze();
849+
850+
TreePath topLevelClass = new TreePath(new TreePath(cut), cut.getTypeDecls().get(0));
851+
Scope scope = Trees.instance(t).getScope(topLevelClass);
852+
853+
if (scope.getEnclosingClass() == null) {
854+
throw new AssertionError("Expected an enclosing class.");
855+
}
856+
857+
scope = scope.getEnclosingScope();
858+
859+
if (scope.getEnclosingClass() != null) {
860+
throw new AssertionError("Did not expect an enclosing class.");
861+
}
862+
863+
asssertScopeContainsTypeWithFQN(scope, "java.lang.System");
864+
asssertScopeContainsTypeWithFQN(scope, "Test");
865+
866+
scope = scope.getEnclosingScope();
867+
868+
if (scope.getEnclosingClass() != null) {
869+
throw new AssertionError("Did not expect an enclosing class.");
870+
}
871+
872+
asssertScopeContainsTypeWithFQN(scope, "java.util.List");
873+
874+
scope = scope.getEnclosingScope();
875+
876+
if (scope.getEnclosingClass() != null) {
877+
throw new AssertionError("Did not expect an enclosing class.");
878+
}
879+
880+
asssertScopeContainsTypeWithFQN(scope, "javax.tools.ToolProvider");
881+
882+
if (scope.getEnclosingScope() != null) {
883+
throw new AssertionError("Did not expect an enclosing scope.");
884+
}
885+
}
886+
}
887+
826888
private List<String> dumpScope(Scope scope) {
827889
List<String> content = new ArrayList<>();
828890
while (scope.getEnclosingClass() != null) {
@@ -833,4 +895,17 @@ private List<String> dumpScope(Scope scope) {
833895
}
834896
return content;
835897
}
898+
899+
private void asssertScopeContainsTypeWithFQN(Scope scope, String fqn) {
900+
for (TypeElement type : ElementFilter.typesIn(scope.getLocalElements())) {
901+
if (type.getQualifiedName().contentEquals(fqn)) {
902+
return ;
903+
}
904+
}
905+
906+
throw new AssertionError("Expected to find: " + fqn +
907+
" in: " + scope.getLocalElements() +
908+
", but it is missing.");
909+
}
910+
836911
}

‎test/langtools/tools/javac/diags/examples/ModifierNotAllowed/module-info.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,8 @@
2121
* questions.
2222
*/
2323

24-
// key: compiler.err.modifier.not.allowed.here
24+
// key: compiler.err.feature.not.supported.in.source.plural
25+
// key: compiler.misc.feature.java.base.transitive
2526

2627
module m {
2728
requires transitive java.base;

‎test/langtools/tools/javac/modules/ConvenientAccessErrorsTest.java

+34
Original file line numberDiff line numberDiff line change
@@ -432,6 +432,40 @@ public void testInImportOnDemand(Path base) throws Exception {
432432
throw new Exception("expected output not found; actual: " + log);
433433
}
434434

435+
@Test
436+
public void testInModuleImport(Path base) throws Exception {
437+
Path src = base.resolve("src");
438+
Path src_m1 = src.resolve("m1x");
439+
tb.writeJavaFiles(src_m1,
440+
"module m1x { }",
441+
"package api; public class Api { public String test() { return null; } }");
442+
Path src_m2 = src.resolve("m2x");
443+
tb.writeJavaFiles(src_m2,
444+
"module m2x { requires m1x; }",
445+
"package test; import module m1x; public class Test { Api api; { api.test().length(); } }");
446+
Path classes = base.resolve("classes");
447+
tb.createDirectories(classes);
448+
449+
List<String> log = new JavacTask(tb)
450+
.options("-XDrawDiagnostics",
451+
"--enable-preview", "--source", System.getProperty("java.specification.version"),
452+
"--module-source-path", src.toString())
453+
.outdir(classes)
454+
.files(findJavaFiles(src))
455+
.run(Task.Expect.FAIL)
456+
.writeAll()
457+
.getOutputLines(Task.OutputKind.DIRECT);
458+
459+
List<String> expected = Arrays.asList(
460+
"Test.java:1:54: compiler.err.cant.resolve.location: kindname.class, Api, , , (compiler.misc.location: kindname.class, test.Test, null)",
461+
"- compiler.note.preview.filename: Test.java, DEFAULT",
462+
"- compiler.note.preview.recompile",
463+
"1 error");
464+
465+
if (!expected.equals(log))
466+
throw new Exception("expected output not found; actual: " + log);
467+
}
468+
435469
@Test
436470
public void testUnusedImportOnDemand2(Path base) throws Exception {
437471
Path src = base.resolve("src");

‎test/langtools/tools/javac/modules/EdgeCases.java

+89
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,9 @@
7373
import com.sun.tools.javac.code.Symbol.ModuleSymbol;
7474
import com.sun.tools.javac.code.Symtab;
7575
import java.util.ArrayList;
76+
import java.util.concurrent.atomic.AtomicBoolean;
77+
import java.util.function.Consumer;
78+
import javax.lang.model.element.ModuleElement.DirectiveKind;
7679

7780
import toolbox.JarTask;
7881
import toolbox.JavacTask;
@@ -1153,4 +1156,90 @@ private void record(TaskEvent e, String phase) {
11531156
}
11541157
}
11551158

1159+
@Test
1160+
public void testJavaSEHasRequiresTransitiveJavaBase(Path base) throws Exception {
1161+
Path src = base.resolve("src");
1162+
Path a = src.resolve("a");
1163+
tb.writeJavaFiles(a,
1164+
"module a { requires java.se; }",
1165+
"""
1166+
package test;
1167+
import module java.se;
1168+
public class Test {
1169+
ArrayList<String> l;
1170+
}
1171+
""");
1172+
Path classes = base.resolve("classes");
1173+
tb.createDirectories(classes);
1174+
1175+
AtomicBoolean seenJavaSEDependency = new AtomicBoolean();
1176+
1177+
List<String> log;
1178+
1179+
log = new JavacTask(tb)
1180+
.outdir(classes)
1181+
.options("-XDrawDiagnostics", "-XDshould-stop.at=FLOW")
1182+
.callback(verifyJavaSEDependency(true, seenJavaSEDependency))
1183+
.files(findJavaFiles(src))
1184+
.run(Task.Expect.FAIL)
1185+
.writeAll()
1186+
.getOutputLines(Task.OutputKind.DIRECT);
1187+
1188+
List<String> expected = List.of(
1189+
"Test.java:2:8: compiler.err.preview.feature.disabled.plural: (compiler.misc.feature.module.imports)",
1190+
"1 error");
1191+
1192+
if (!expected.equals(log))
1193+
throw new Exception("expected output not found: " + log);
1194+
1195+
if (!seenJavaSEDependency.get()) {
1196+
throw new AssertionError("Didn't find the java.se dependency!");
1197+
}
1198+
1199+
seenJavaSEDependency.set(false);
1200+
1201+
new JavacTask(tb)
1202+
.outdir(classes)
1203+
.options("--enable-preview",
1204+
"--source", System.getProperty("java.specification.version"))
1205+
.callback(verifyJavaSEDependency(true, seenJavaSEDependency))
1206+
.files(findJavaFiles(src))
1207+
.run(Task.Expect.SUCCESS)
1208+
.writeAll()
1209+
.getOutputLines(Task.OutputKind.DIRECT);
1210+
1211+
if (!seenJavaSEDependency.get()) {
1212+
throw new AssertionError("Didn't find the java.se dependency!");
1213+
}
1214+
}
1215+
private Consumer<com.sun.source.util.JavacTask> verifyJavaSEDependency(
1216+
boolean expectedTransitive,
1217+
AtomicBoolean seenJavaSEDependency) {
1218+
return t -> {
1219+
t.addTaskListener(new TaskListener() {
1220+
@Override
1221+
public void finished(TaskEvent e) {
1222+
if (e.getKind() == TaskEvent.Kind.ANALYZE) {
1223+
ModuleElement javaBase =
1224+
t.getElements().getModuleElement("java.base");
1225+
ModuleElement javaSE =
1226+
t.getElements().getModuleElement("java.se");
1227+
RequiresDirective requiresJavaBase =
1228+
javaSE.getDirectives()
1229+
.stream()
1230+
.filter(d -> d.getKind() == DirectiveKind.REQUIRES)
1231+
.map(d -> (RequiresDirective) d)
1232+
.filter(d -> d.getDependency() == javaBase)
1233+
.findAny()
1234+
.orElseThrow();
1235+
if (requiresJavaBase.isTransitive() != expectedTransitive) {
1236+
throw new AssertionError("Expected: " + expectedTransitive + ", " +
1237+
"but got: " + requiresJavaBase.isTransitive());
1238+
}
1239+
seenJavaSEDependency.set(true);
1240+
}
1241+
}
1242+
});
1243+
};
1244+
}
11561245
}

‎test/langtools/tools/javac/modules/JavaBaseTest.java

+99-25
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
/**
2525
* @test
26-
* @bug 8193125 8196623
26+
* @bug 8193125 8196623 8335989
2727
* @summary test modifiers with java.base
2828
* @library /tools/lib
2929
* @enablePreview
@@ -49,13 +49,16 @@
4949

5050
import com.sun.tools.javac.jvm.Target;
5151
import com.sun.tools.javac.platform.JDKPlatformProvider;
52+
import java.util.function.Supplier;
5253

5354
import toolbox.JavacTask;
5455
import toolbox.Task;
5556
import toolbox.ToolBox;
5657

5758
public class JavaBaseTest {
5859

60+
private static final String CURRENT_VERSION = System.getProperty("java.specification.version");
61+
5962
public static void main(String... args) throws Exception {
6063
JavaBaseTest t = new JavaBaseTest();
6164
t.run();
@@ -75,13 +78,11 @@ enum Mode { SOURCE, CLASS };
7578

7679
void run() throws Exception {
7780
Set<String> targets = new LinkedHashSet<>();
78-
StreamSupport.stream(new JDKPlatformProvider().getSupportedPlatformNames()
79-
.spliterator(),
80-
false)
81-
.filter(p -> Integer.parseInt(p) >= 9)
82-
.forEach(targets::add);
83-
//run without --release:
81+
targets.add("9");
82+
targets.add("10");
83+
targets.add("21");
8484
targets.add("current");
85+
targets.add("current-preview");
8586
for (List<String> mods : modifiers) {
8687
for (String target : targets) {
8788
for (Mode mode : Mode.values()) {
@@ -119,15 +120,43 @@ void testSource(Path base, List<String> mods, String target) throws Exception {
119120
tb.writeJavaFiles(src,
120121
"module m { requires " + String.join(" ", mods) + " java.base; }");
121122
Path modules = Files.createDirectories(base.resolve("modules"));
122-
boolean expectOK = target.equals("9");
123+
boolean expectOK;
123124

124125
JavacTask jct = new JavacTask(tb)
125126
.outdir(modules);
126127

127-
if (target.equals("current"))
128-
jct.options("-XDrawDiagnostics");
129-
else
130-
jct.options("-XDrawDiagnostics", "--release", target);
128+
List<String> options = new ArrayList<>();
129+
130+
switch (target) {
131+
case "current":
132+
options.add("--release");
133+
options.add(CURRENT_VERSION);
134+
expectOK = false;
135+
break;
136+
case "current-preview":
137+
options.add("--enable-preview");
138+
options.add("--release");
139+
options.add(CURRENT_VERSION);
140+
expectOK = true;
141+
break;
142+
case "9":
143+
options.add("--release");
144+
options.add(target);
145+
expectOK = true;
146+
break;
147+
default:
148+
options.add("--release");
149+
options.add(target);
150+
expectOK = false;
151+
break;
152+
}
153+
154+
if (mods.contains("static") && !"9".equals(target)) {
155+
expectOK = false;
156+
}
157+
158+
options.add("-XDrawDiagnostics");
159+
jct.options(options);
131160

132161
String log = jct.files(tb.findJavaFiles(src))
133162
.run(expectOK ? Task.Expect.SUCCESS : Task.Expect.FAIL)
@@ -138,9 +167,9 @@ void testSource(Path base, List<String> mods, String target) throws Exception {
138167
boolean foundErrorMessage = false;
139168
for (String mod : mods) {
140169
String key = mod.equals("static")
141-
? "compiler.err.mod.not.allowed.here"
142-
: "compiler.err.modifier.not.allowed.here";
143-
String message = "module-info.java:1:12: " + key + ": " + mod;
170+
? "compiler.err.mod.not.allowed.here: " + mod
171+
: "compiler.err.feature.not.supported.in.source.plural: (compiler.misc.feature.java.base.transitive)";
172+
String message = "module-info.java:1:12: " + key;
144173
if (log.contains(message)) {
145174
foundErrorMessage = true;
146175
}
@@ -152,18 +181,57 @@ void testSource(Path base, List<String> mods, String target) throws Exception {
152181
}
153182

154183
void testClass(Path base, List<String> mods, String target) throws Exception {
155-
createClass(base, mods, target);
184+
boolean expectOK;
185+
List<String> options = new ArrayList<>();
186+
187+
switch (target) {
188+
case "current":
189+
options.add("--release");
190+
options.add(CURRENT_VERSION);
191+
expectOK = false;
192+
break;
193+
case "current-preview":
194+
options.add("--enable-preview");
195+
options.add("--release");
196+
options.add(CURRENT_VERSION);
197+
expectOK = true;
198+
break;
199+
case "9":
200+
options.add("--release");
201+
options.add(target);
202+
expectOK = true;
203+
break;
204+
default:
205+
options.add("--release");
206+
options.add(target);
207+
expectOK = false;
208+
break;
209+
}
210+
211+
if (mods.contains("static") && !"9".equals(target)) {
212+
expectOK = false;
213+
}
214+
215+
createClass(base, mods, options);
216+
217+
List<String> testOptions = new ArrayList<>();
218+
219+
testOptions.add("-XDrawDiagnostics");
220+
testOptions.add("--module-path"); testOptions.add(base.resolve("test-modules").toString());
221+
222+
if (options.contains("--enable-preview")) {
223+
testOptions.add("--enable-preview");
224+
testOptions.add("--source"); testOptions.add(CURRENT_VERSION);
225+
}
156226

157227
Path src = base.resolve("src");
158228
tb.writeJavaFiles(src,
159229
"module mx { requires m; }");
160230
Path modules = Files.createDirectories(base.resolve("modules"));
161231

162-
boolean expectOK = target.equals("9");
163232
String log = new JavacTask(tb)
164233
.outdir(modules)
165-
.options("-XDrawDiagnostics",
166-
"--module-path", base.resolve("test-modules").toString())
234+
.options(testOptions)
167235
.files(tb.findJavaFiles(src))
168236
.run(expectOK ? Task.Expect.SUCCESS : Task.Expect.FAIL)
169237
.writeAll()
@@ -188,7 +256,7 @@ void testClass(Path base, List<String> mods, String target) throws Exception {
188256
}
189257
}
190258

191-
void createClass(Path base, List<String> mods, String target) throws Exception {
259+
void createClass(Path base, List<String> mods, List<String> options) throws Exception {
192260
Path src1 = base.resolve("interim-src");
193261
tb.writeJavaFiles(src1,
194262
"module m { requires java.base; }");
@@ -197,9 +265,7 @@ void createClass(Path base, List<String> mods, String target) throws Exception {
197265
JavacTask jct = new JavacTask(tb)
198266
.outdir(modules1);
199267

200-
if (!target.equals("current")) {
201-
jct.options("--release", target);
202-
}
268+
jct.options(options);
203269

204270
jct.files(tb.findJavaFiles(src1))
205271
.run(Task.Expect.SUCCESS);
@@ -226,6 +292,8 @@ void createClass(Path base, List<String> mods, String target) throws Exception {
226292
requires.set(i, e2);
227293
}
228294

295+
boolean preview = options.contains("--enable-preview");
296+
229297
ModuleAttribute modAttr2 = ModuleAttribute.of(
230298
modAttr1.moduleName(),
231299
modAttr1.moduleFlagsMask(),
@@ -237,8 +305,14 @@ void createClass(Path base, List<String> mods, String target) throws Exception {
237305
modAttr1.provides());
238306
Path modInfo = base.resolve("test-modules").resolve("module-info.class");
239307
Files.createDirectories(modInfo.getParent());
240-
byte[] newBytes = ClassFile.of().transformClass(cm1, ClassTransform.dropping(ce -> ce instanceof ModuleAttribute).
241-
andThen(ClassTransform.endHandler(classBuilder -> classBuilder.with(modAttr2))));
308+
ClassTransform replace = (builder, element) -> {
309+
switch (element) {
310+
case ClassFileVersion cfv when preview -> builder.withVersion(cfv.majorVersion(), 0xFFFF);
311+
case ModuleAttribute _ -> builder.with(modAttr2);
312+
default -> builder.with(element);
313+
}
314+
};
315+
byte[] newBytes = ClassFile.of().transformClass(cm1, replace);
242316
try (OutputStream out = Files.newOutputStream(modInfo)) {
243317
out.write(newBytes);
244318
}

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Nov 14, 2024

@openjdk-notifier[bot]
Please sign in to comment.