blob: a145a47635d6986c7f773684f08eeecd165c749c [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2016, 2019 IBM Corporation and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
*
*******************************************************************************/
package org.eclipse.jdt.internal.compiler.lookup;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.eclipse.jdt.core.compiler.CharOperation;
import org.eclipse.jdt.internal.compiler.classfmt.ClassFileConstants;
import org.eclipse.jdt.internal.compiler.env.IModuleAwareNameEnvironment;
import org.eclipse.jdt.internal.compiler.env.IUpdatableModule;
import org.eclipse.jdt.internal.compiler.util.HashtableOfPackage;
import org.eclipse.jdt.internal.compiler.util.SimpleLookupTable;
import org.eclipse.jdt.internal.compiler.util.SimpleSetOfCharArray;
/**
* This class serves a dual purpose
* <p>
* First, it canonically represents modules in the world of bindings.
* </p>
* <p>
* Secondly, it adds a graph layer on top of {@link LookupEnvironment}:
* ModuleBindins are linked through "read" edges as per JPMS (see java.lang.module.Configuration).
* Additionally, each ModuleBinding holds its own instance of LookupEnviroment,
* capturing all packages and types that are visible to the current module.
* As a subset of all visible types, the ModuleBinding knows the set of
* packages locally declared in this module.
* </p>
*/
public class ModuleBinding extends Binding implements IUpdatableModule {
/** Name of the unnamed module. */
public static final char[] UNNAMED = "".toCharArray(); //$NON-NLS-1$
/** Name to represent unnamed modules in --add-exports & --add-reads options. */
public static final char[] ALL_UNNAMED = "ALL-UNNAMED".toCharArray(); //$NON-NLS-1$
/** Module name for package/type lookup that doesn't care about modules. */
public static final char[] ANY = "".toCharArray(); //$NON-NLS-1$
/** Module name for package/type lookup that should look into all named modules. */
public static final char[] ANY_NAMED = "".toCharArray(); //$NON-NLS-1$
/** Module name for an unobservable module */
public static final char[] UNOBSERVABLE = "".toCharArray(); //$NON-NLS-1$
public static class UnNamedModule extends ModuleBinding {
private static final char[] UNNAMED_READABLE_NAME = "<unnamed>".toCharArray(); //$NON-NLS-1$
@SuppressWarnings("synthetic-access")
UnNamedModule(LookupEnvironment env) {
super(env);
}
@Override
public ModuleBinding[] getAllRequiredModules() {
return Binding.NO_MODULES;
}
@Override
public boolean canAccess(PackageBinding pkg) {
if (pkg instanceof SplitPackageBinding) {
for (PackageBinding p : ((SplitPackageBinding) pkg).incarnations) {
if (canAccess(p)) {
return true;
}
}
return false;
} else {
ModuleBinding mod = pkg.enclosingModule;
if (mod != null && mod != this)
return mod.isPackageExportedTo(pkg, this);
}
return true;
}
@Override
public boolean isPackageExportedTo(PackageBinding pkg, ModuleBinding client) {
// per JLS 7.7.5 an unnamed module exports all its packages
return pkg.isDeclaredIn(this) && pkg.hasCompilationUnit(false);
}
@Override
PlainPackageBinding getDeclaredPackage(char[] flatName) {
PlainPackageBinding declaredPackage = super.getDeclaredPackage(flatName);
if (declaredPackage == null && this.environment.useModuleSystem) {
IModuleAwareNameEnvironment moduleEnv = (IModuleAwareNameEnvironment) this.environment.nameEnvironment;
char[][] compoundName = CharOperation.splitOn('.', flatName);
char[][] declaringModuleNames = moduleEnv.getUniqueModulesDeclaringPackage(compoundName, nameForLookup());
if (declaringModuleNames != null && CharOperation.containsEqual(declaringModuleNames, this.moduleName)) {
declaredPackage = getOrCreateDeclaredPackage(compoundName);
}
}
return declaredPackage;
}
@Override
public boolean isUnnamed() {
return true;
}
@Override
public char[] nameForLookup() {
return ANY;
}
@Override
public char[] nameForCUCheck() {
return UNNAMED;
}
@Override
public char[] readableName() {
return UNNAMED_READABLE_NAME;
}
@Override
public String toString() {
return "The Unnamed Module"; //$NON-NLS-1$
}
}
public char[] moduleName;
protected ModuleBinding[] requires;
protected ModuleBinding[] requiresTransitive;
protected PlainPackageBinding[] exportedPackages;
private Map<PlainPackageBinding,SimpleSetOfCharArray> exportRestrictions; // RHS is unresolved names, because unresolvable names are legal in this position
protected PlainPackageBinding[] openedPackages;
private Map<PlainPackageBinding,SimpleSetOfCharArray> openRestrictions; // RHS is unresolved names, because unresolvable names are legal in this position
protected TypeBinding[] uses;
protected TypeBinding[] services;
public Map<TypeBinding,TypeBinding[]> implementations;
public char[] mainClassName;
private SimpleSetOfCharArray packageNames;
public int modifiers;
public LookupEnvironment environment;
public long tagBits;
public int defaultNullness = NO_NULL_DEFAULT;
ModuleBinding[] requiredModules = null;
boolean isAuto = false;
private boolean[] isComplete = new boolean[UpdateKind.values().length];
private Set<ModuleBinding> transitiveRequires;
SimpleLookupTable storedAnnotations = null;
/**
* Packages declared in this module (indexed by qualified name).
* We consider a package as declared in a module,
* if a compilation unit associated with the module
* declares the package or a subpackage thereof.
* <p>
* A package in this structures is always represented by a {@link PlainPackageBinding},
* as opposed to {@link SplitPackageBinding}, which are only maintained in the trees
* below {@link LookupEnvironment#knownPackages}.
* <p>
* This structure is populated early during compilation with all packages that
* are referenced in exports and opens directives, plus their parent packages.
* </p>
*/
public HashtableOfPackage<PlainPackageBinding> declaredPackages;
/** Constructor for the unnamed module. */
private ModuleBinding(LookupEnvironment env) {
this.moduleName = ModuleBinding.UNNAMED;
this.environment = env;
this.requires = Binding.NO_MODULES;
this.requiresTransitive = Binding.NO_MODULES;
this.exportedPackages = Binding.NO_PLAIN_PACKAGES;
this.openedPackages = Binding.NO_PLAIN_PACKAGES;
this.declaredPackages = new HashtableOfPackage<PlainPackageBinding>();
Arrays.fill(this.isComplete, true);
}
/* For error binding and sub class SourceModuleBinding. */
ModuleBinding(char[] moduleName) {
this.moduleName = moduleName;
this.requires = Binding.NO_MODULES;
this.requiresTransitive = Binding.NO_MODULES;
this.exportedPackages = Binding.NO_PLAIN_PACKAGES;
this.openedPackages = Binding.NO_PLAIN_PACKAGES;
this.uses = Binding.NO_TYPES;
this.services = Binding.NO_TYPES;
this.declaredPackages = new HashtableOfPackage<PlainPackageBinding>(5);
}
/* For sub class BinaryModuleBinding */
protected ModuleBinding(char[] moduleName, LookupEnvironment existingEnvironment) {
this.moduleName = moduleName;
this.requires = Binding.NO_MODULES;
this.requiresTransitive = Binding.NO_MODULES;
this.environment = new LookupEnvironment(existingEnvironment.root, this);
this.declaredPackages = new HashtableOfPackage<PlainPackageBinding>(5);
}
public PlainPackageBinding[] getExports() {
completeIfNeeded(UpdateKind.PACKAGE);
return this.exportedPackages;
}
public String[] getExportRestrictions(PackageBinding pack) {
completeIfNeeded(UpdateKind.PACKAGE);
if (this.exportRestrictions != null) {
SimpleSetOfCharArray set = this.exportRestrictions.get(pack);
if (set != null) {
char[][] names = new char[set.elementSize][];
set.asArray(names);
return CharOperation.charArrayToStringArray(names);
}
}
return CharOperation.NO_STRINGS;
}
public PlainPackageBinding[] getOpens() {
completeIfNeeded(UpdateKind.PACKAGE);
return this.openedPackages;
}
public String[] getOpenRestrictions(PackageBinding pack) {
completeIfNeeded(UpdateKind.PACKAGE);
if (this.openRestrictions != null) {
SimpleSetOfCharArray set = this.openRestrictions.get(pack);
if (set != null) {
char[][] names = new char[set.elementSize][];
set.asArray(names);
return CharOperation.charArrayToStringArray(names);
}
}
return CharOperation.NO_STRINGS;
}
public TypeBinding[] getImplementations(TypeBinding binding) {
if (this.implementations != null) {
return this.implementations.get(binding);
}
return null;
}
public ModuleBinding[] getRequires() {
completeIfNeeded(UpdateKind.MODULE);
return this.requires;
}
public ModuleBinding[] getRequiresTransitive() {
completeIfNeeded(UpdateKind.MODULE);
return this.requiresTransitive;
}
public TypeBinding[] getUses() {
return this.uses;
}
public TypeBinding[] getServices() {
return this.services;
}
void completeIfNeeded(IUpdatableModule.UpdateKind kind) {
if (!this.isComplete[kind.ordinal()]) {
this.isComplete[kind.ordinal()] = true;
if (this.environment.nameEnvironment instanceof IModuleAwareNameEnvironment) {
((IModuleAwareNameEnvironment) this.environment.nameEnvironment).applyModuleUpdates(this, kind);
}
}
}
// --- Implement IUpdatableModule: ---
@Override
public void addReads(char[] requiredModuleName) {
ModuleBinding requiredModule = this.environment.getModule(requiredModuleName);
if (requiredModule != null) {
int len = this.requires.length;
if (len == 0) {
this.requires = new ModuleBinding[] { requiredModule };
} else {
System.arraycopy(this.requires, 0, this.requires = new ModuleBinding[len+1], 0, len);
this.requires[len] = requiredModule;
}
} else {
this.environment.problemReporter.missingModuleAddReads(requiredModuleName);
return;
}
}
@Override
public void addExports(char[] packageName, char[][] targetModules) {
PlainPackageBinding declaredPackage = getOrCreateDeclaredPackage(CharOperation.splitOn('.', packageName));
if (declaredPackage != null && declaredPackage.isValidBinding())
addResolvedExport(declaredPackage, targetModules);
}
@Override
public void setMainClassName(char[] mainClassName) {
this.mainClassName = mainClassName;
}
@Override
public void setPackageNames(SimpleSetOfCharArray packageNames) {
this.packageNames = packageNames;
}
// for code gen:
/** @return array of names, which may contain nulls. */
public char[][] getPackageNamesForClassFile() {
if (this.packageNames == null)
return null;
for (PlainPackageBinding packageBinding : this.exportedPackages)
this.packageNames.add(packageBinding.readableName());
for (PlainPackageBinding packageBinding : this.openedPackages)
this.packageNames.add(packageBinding.readableName());
if (this.implementations != null)
for (TypeBinding[] types : this.implementations.values())
for (TypeBinding typeBinding : types)
this.packageNames.add(((ReferenceBinding)typeBinding).fPackage.readableName());
return this.packageNames.values;
}
// ---
PlainPackageBinding createDeclaredToplevelPackage(char[] name) {
PlainPackageBinding packageBinding = new PlainPackageBinding(name, this.environment, this);
this.declaredPackages.put(name, packageBinding);
return packageBinding;
}
PlainPackageBinding createDeclaredPackage(char[][] compoundName, PackageBinding parent) {
PlainPackageBinding packageBinding = new PlainPackageBinding(compoundName, parent, this.environment, this);
this.declaredPackages.put(CharOperation.concatWith(compoundName, '.'), packageBinding);
return packageBinding;
}
public PlainPackageBinding getOrCreateDeclaredPackage(char[][] compoundName) {
char[] flatName = CharOperation.concatWith(compoundName, '.');
PlainPackageBinding pkgBinding = this.declaredPackages.get(flatName);
if (pkgBinding != null)
return pkgBinding;
if (compoundName.length > 1) {
PlainPackageBinding parent = getOrCreateDeclaredPackage(CharOperation.subarray(compoundName, 0, compoundName.length-1));
pkgBinding = new PlainPackageBinding(compoundName, parent, this.environment, this);
parent.addPackage(pkgBinding, this);
} else {
pkgBinding = new PlainPackageBinding(compoundName[0], this.environment, this);
PackageBinding problemPackage = this.environment.getPackage0(compoundName[0]);
if (problemPackage == LookupEnvironment.TheNotFoundPackage)
this.environment.knownPackages.put(compoundName[0], null); // forget TheNotFoundPackage if package was detected late (e.g. with APT in the loop)
}
this.declaredPackages.put(flatName, pkgBinding);
return pkgBinding;
}
public void addResolvedExport(PlainPackageBinding declaredPackage, char[][] targetModules) {
if (declaredPackage == null || !declaredPackage.isValidBinding()) {
// TODO(SHMOD) use a problem binding (if needed by DOM clients)? See https://bugs.eclipse.org/518794#c13
return;
}
if (this.exportedPackages == null || this.exportedPackages.length == 0) {
this.exportedPackages = new PlainPackageBinding[] { declaredPackage };
} else {
int len = this.exportedPackages.length;
System.arraycopy(this.exportedPackages, 0, this.exportedPackages = new PlainPackageBinding[len+1], 0, len);
this.exportedPackages[len] = declaredPackage;
}
declaredPackage.isExported = Boolean.TRUE;
recordExportRestrictions(declaredPackage, targetModules);
}
public void addResolvedOpens(PlainPackageBinding declaredPackage, char[][] targetModules) {
int len = this.openedPackages.length;
if (declaredPackage == null || !declaredPackage.isValidBinding()) {
// TODO(SHMOD) use a problem binding (if needed by DOM clients)? See https://bugs.eclipse.org/518794#c13
return;
}
if (len == 0) {
this.openedPackages = new PlainPackageBinding[] { declaredPackage };
} else {
System.arraycopy(this.openedPackages, 0, this.openedPackages = new PlainPackageBinding[len+1], 0, len);
this.openedPackages[len] = declaredPackage;
}
recordOpensRestrictions(declaredPackage, targetModules);
}
protected void recordExportRestrictions(PlainPackageBinding exportedPackage, char[][] targetModules) {
if (targetModules != null && targetModules.length > 0) {
SimpleSetOfCharArray targetModuleSet = null;
if (this.exportRestrictions != null) {
targetModuleSet = this.exportRestrictions.get(exportedPackage);
} else {
this.exportRestrictions = new HashMap<>();
}
if (targetModuleSet == null) {
targetModuleSet = new SimpleSetOfCharArray(targetModules.length);
this.exportRestrictions.put(exportedPackage, targetModuleSet);
}
for (int i = 0; i < targetModules.length; i++) {
targetModuleSet.add(targetModules[i]);
}
}
}
protected void recordOpensRestrictions(PlainPackageBinding openedPackage, char[][] targetModules) {
if (targetModules != null && targetModules.length > 0) {
SimpleSetOfCharArray targetModuleSet = null;
if (this.openRestrictions != null) {
targetModuleSet = this.openRestrictions.get(openedPackage);
} else {
this.openRestrictions = new HashMap<>();
}
if (targetModuleSet == null) {
targetModuleSet = new SimpleSetOfCharArray(targetModules.length);
this.openRestrictions.put(openedPackage, targetModuleSet);
}
for (int i = 0; i < targetModules.length; i++) {
targetModuleSet.add(targetModules[i]);
}
}
}
Stream<ModuleBinding> getRequiredModules(boolean transitiveOnly) {
return Stream.of(transitiveOnly ? this.getRequiresTransitive() : this.getRequires());
}
private void collectAllDependencies(Set<ModuleBinding> deps) {
getRequiredModules(false).forEach(m -> {
if (deps.add(m)) {
m.collectAllDependencies(deps);
}
});
}
private void collectTransitiveDependencies(Set<ModuleBinding> deps) {
getRequiredModules(true).forEach(m -> {
if (deps.add(m)) {
m.collectTransitiveDependencies(deps);
}
});
}
// All modules required by this module, either directly or indirectly
public Supplier<Collection<ModuleBinding>> dependencyGraphCollector() {
return () -> getRequiredModules(false)
.collect(HashSet::new,
(set, mod) -> {
set.add(mod);
mod.collectAllDependencies(set);
},
HashSet::addAll);
}
// All direct and transitive dependencies of this module
public Supplier<Collection<ModuleBinding>> dependencyCollector() {
return () -> getRequiredModules(false)
.collect(HashSet::new,
(set, mod) -> {
set.add(mod);
mod.collectTransitiveDependencies(set);
},
HashSet::addAll);
}
/**
* Get all the modules required by this module
* All required modules include modules explicitly specified as required in the module declaration
* as well as implicit dependencies - those specified as ' requires transitive ' by one of the
* dependencies
*
* @return
* An array of all required modules
*/
public ModuleBinding[] getAllRequiredModules() {
if (this.requiredModules != null)
return this.requiredModules;
Collection<ModuleBinding> allRequires = dependencyCollector().get();
if (allRequires.contains(this)) {
return NO_MODULES; // avoid entering unbounded recursion due to cyclic requires
}
ModuleBinding javaBase = this.environment.javaBaseModule();
// add java.base?
if (!CharOperation.equals(this.moduleName, TypeConstants.JAVA_BASE) // ... not if this *is* java.base
&& javaBase != null // ... nor when java.base is absent
&& javaBase != this.environment.UnNamedModule) // ..... or faked by the unnamed module
{
allRequires.add(javaBase);
}
return this.requiredModules = allRequires.size() > 0 ? allRequires.toArray(new ModuleBinding[allRequires.size()]) : Binding.NO_MODULES;
}
/** Answer the name of this module. The unnamed module is identified by {@link #UNNAMED}. */
@Override
public char[] name() {
return this.moduleName;
}
/**
* Answer the name of this module as it should be used for package or type lookup.
* Unnamed and automatic modules answer {@link #ANY} or {@link #ANY_NAMED} resp.,
* to signal that lookup should search in all accessible (named) modules.
*/
public char[] nameForLookup() {
return this.moduleName;
}
/**
* Answer the name of this module as it should be used for hasCompilationUnit() checks.
*/
public char[] nameForCUCheck() {
return nameForLookup();
}
/**
* Check if the specified package is owned by the current module and exported to the client module.
* True if the package appears in the list of exported packages and when the export is targeted,
* the module appears in the targets of the exports statement.
* @param pkg - the package whose visibility is to be checked
* @param client - the module that wishes to use the package
* @return true if the package is visible to the client module, false otherwise
*/
public boolean isPackageExportedTo(PackageBinding pkg, ModuleBinding client) {
// TODO(SHMOD): cache the result?
PackageBinding resolved = pkg.getIncarnation(this);
if (resolved != null) {
if (this.isAuto) { // all packages are exported by an automatic module
return pkg.enclosingModule == this; // no transitive export
}
PackageBinding[] initializedExports = getExports();
for (int i = 0; i < initializedExports.length; i++) {
PackageBinding export = initializedExports[i];
if (export.subsumes(resolved)) {
if (this.exportRestrictions != null) {
SimpleSetOfCharArray restrictions = this.exportRestrictions.get(export);
if (restrictions != null) {
if (client.isUnnamed())
return restrictions.includes(ALL_UNNAMED);
else
return restrictions.includes(client.name());
}
}
return true;
}
}
}
return false;
}
/**
* Return a package binding if there exists a package named name in this module's context and it can be seen by this module.
* A package can be seen by this module if it is declared in this module or any other module read by this module
* (JLS 7.4.3 for packages based on JLS 7.3 for compilation units).
* Package exports are not considered for visibility check (only when checking "uniquely visible" (JLS 7.4.3)).
* <p>
* The returned package may be a {@link SplitPackageBinding}, if more than one package of the given name is visible.
* </p>
* <p>
* When asked via the unnamed module or an automatic module all other named modules are considered visible.
* </p>
*/
public PackageBinding getTopLevelPackage(char[] name) {
return getVisiblePackage(null, name);
}
PlainPackageBinding getDeclaredPackage(char[] flatName) {
return this.declaredPackages.get(flatName);
}
// Given parent is visible in this module, see if there is sub package named name visible in this module
PackageBinding getVisiblePackage(PackageBinding parent, char[] name) {
// check caches in PackageBinding/LookupEnvironment, which contain the full SplitPackageBinding if appropriate:
PackageBinding pkg;
if (parent != null)
pkg = parent.getPackage0(name);
else
pkg = this.environment.getPackage0(name);
if (pkg != null) {
if (pkg == LookupEnvironment.TheNotFoundPackage)
return null;
else
return pkg;
}
// check cached plain PackageBinding in declaredPackages (which may need combining with siblings):
char[][] parentName = parent == null ? CharOperation.NO_CHAR_CHAR : parent.compoundName;
char[][] subPkgCompoundName = CharOperation.arrayConcat(parentName, name);
char[] fullFlatName = CharOperation.concatWith(subPkgCompoundName, '.');
PackageBinding binding = this.declaredPackages.get(fullFlatName);
char[][] declaringModuleNames = null;
if (this.environment.useModuleSystem) {
IModuleAwareNameEnvironment moduleEnv = (IModuleAwareNameEnvironment) this.environment.nameEnvironment;
declaringModuleNames = moduleEnv.getUniqueModulesDeclaringPackage(subPkgCompoundName, nameForLookup());
if (binding == null) {
if (declaringModuleNames != null) {
if (CharOperation.containsEqual(declaringModuleNames, this.moduleName)) {
if (parent != null) {
PackageBinding singleParent = parent.getIncarnation(this);
if (singleParent != null && singleParent != parent) {
// parent.getPackage0() may have been too shy, so drill into the split:
binding = singleParent.getPackage0(name);
}
}
if (binding == null) {
// declared here, not yet known, so create it now:
binding = this.createDeclaredPackage(subPkgCompoundName, parent);
}
} else {
// visible but foreign (when current is unnamed or auto):
for (char[] declaringModuleName : declaringModuleNames) {
ModuleBinding declaringModule = this.environment.root.getModule(declaringModuleName);
if (declaringModule != null) {
PlainPackageBinding declaredPackage = declaringModule.getDeclaredPackage(fullFlatName);
binding = SplitPackageBinding.combine(declaredPackage, binding, this);
}
}
}
}
}
} else {
if (this.environment.nameEnvironment.isPackage(parentName, name))
binding = this.createDeclaredPackage(subPkgCompoundName, parent);
}
binding = combineWithPackagesFromOtherRelevantModules(binding, subPkgCompoundName, declaringModuleNames);
assert binding == null || binding instanceof PlainPackageBinding || binding.enclosingModule == this;
if (binding == null || !binding.isValidBinding()) {
if (parent != null) {
if (binding == null) {
parent.addNotFoundPackage(name);
} else {
parent.knownPackages.put(name, binding);
}
} else {
this.environment.knownPackages.put(name, LookupEnvironment.TheNotFoundPackage);
}
return null;
}
// remember
if (parentName.length == 0) {
this.environment.knownPackages.put(name, binding);
} else if (parent != null) {
binding = parent.addPackage(binding, this);
}
return binding;
}
/**
* Answer the package of the given qualified name and visible in this module,
* or {@code null} if no such package exists.
* Accessibility (based on package exports) is <strong>not</strong> checked.
* <p>
* May answer a {@link SplitPackageBinding}.
* </p>
*/
public PackageBinding getVisiblePackage(char[][] qualifiedPackageName) {
if (qualifiedPackageName == null || qualifiedPackageName.length == 0) {
return this.environment.defaultPackage;
}
PackageBinding parent = getTopLevelPackage(qualifiedPackageName[0]);
if (parent == null)
return null;
// check each sub package
for (int i = 1; i < qualifiedPackageName.length; i++) {
PackageBinding binding = getVisiblePackage(parent, qualifiedPackageName[i]);
if (binding == null) {
return null;
}
parent = binding;
}
return parent;
}
PackageBinding combineWithPackagesFromOtherRelevantModules(PackageBinding currentBinding, char[][] compoundName, char[][] declaringModuleNames) {
for (ModuleBinding moduleBinding : otherRelevantModules(declaringModuleNames)) {
PlainPackageBinding nextBinding = moduleBinding.getDeclaredPackage(CharOperation.concatWith(compoundName, '.'));
currentBinding = SplitPackageBinding.combine(nextBinding, currentBinding, this);
}
return currentBinding;
}
List<ModuleBinding> otherRelevantModules(char[][] declaringModuleNames) {
if (isUnnamed() && declaringModuleNames != null) {
// unnamed module reads all named modules,
// so all modules declaring the given package are relevant:
return Arrays.stream(declaringModuleNames)
.filter(modName -> modName != UNNAMED)
.map(modName -> this.environment.getModule(modName))
.filter(Objects::nonNull)
.collect(Collectors.toList());
} else {
return Arrays.asList(getAllRequiredModules());
}
}
/**
* Check if the given package is accessible by this module. True when the package is declared in
* this module or exported by some required module to this module.
* See {@link #isPackageExportedTo(PackageBinding, ModuleBinding)}
*
* @param pkg
*
* @return True, if the package is accessible by this module, false otherwise
*/
public boolean canAccess(PackageBinding pkg) {
if (pkg.isDeclaredIn(this))
return true;
for (ModuleBinding requiredModule : getAllRequiredModules()) {
// If pkg is a SplitPackageBinding, we actually ask the intersection of all required modules
// and modules declaring the package, if any of them exports the package to this module.
// The intersection is computed when inside isPackageExportedTo we ask for pkg's incarnation in requiredModule.
if (requiredModule.isPackageExportedTo(pkg, ModuleBinding.this))
return true;
// TODO(SHMOD): store export status in the PackageBinding?
}
return false;
}
@Override
public char[] computeUniqueKey(boolean isLeaf) {
return CharOperation.prepend('"', this.moduleName);
}
@Override
public int kind() {
//
return Binding.MODULE;
}
@Override
public char[] readableName() {
return this.moduleName;
}
@Override
public String toString() {
StringBuffer buffer = new StringBuffer(30);
if (isOpen())
buffer.append("open "); //$NON-NLS-1$
buffer.append("module " + new String(readableName())); //$NON-NLS-1$
if (this.requires.length > 0) {
buffer.append("\n/* requires */\n"); //$NON-NLS-1$
for (int i = 0; i < this.requires.length; i++) {
buffer.append("\n\t"); //$NON-NLS-1$
if (this.requiresTransitive != null) {
for (ModuleBinding reqTrans : this.requiresTransitive) {
if (reqTrans == this.requires[i]) {
buffer.append("transitive "); //$NON-NLS-1$
break;
}
}
}
buffer.append(this.requires[i].moduleName);
}
} else {
buffer.append("\nNo Requires"); //$NON-NLS-1$
}
if (this.exportedPackages != null && this.exportedPackages.length > 0) {
buffer.append("\n/* exports */\n"); //$NON-NLS-1$
for (int i = 0; i < this.exportedPackages.length; i++) {
PackageBinding export = this.exportedPackages[i];
buffer.append("\n\t"); //$NON-NLS-1$
if (export == null) {
buffer.append("<unresolved>"); //$NON-NLS-1$
continue;
}
buffer.append(export.readableName());
SimpleSetOfCharArray restrictions = this.exportRestrictions != null ? this.exportRestrictions.get(export) : null;
if (restrictions != null) {
buffer.append(" to "); //$NON-NLS-1$
String sep = ""; //$NON-NLS-1$
char[][] allNames = new char[restrictions.elementSize][];
restrictions.asArray(allNames);
for (char[] targetModule : allNames) {
buffer.append(sep);
buffer.append(targetModule);
sep = ", "; //$NON-NLS-1$
}
}
}
} else {
buffer.append("\nNo Exports"); //$NON-NLS-1$
}
if (this.openedPackages != null && this.openedPackages.length > 0) {
buffer.append("\n/* exports */\n"); //$NON-NLS-1$
for (int i = 0; i < this.openedPackages.length; i++) {
PackageBinding opens = this.openedPackages[i];
buffer.append("\n\t"); //$NON-NLS-1$
if (opens == null) {
buffer.append("<unresolved>"); //$NON-NLS-1$
continue;
}
buffer.append(opens.readableName());
SimpleSetOfCharArray restrictions = this.openRestrictions != null ? this.openRestrictions.get(opens) : null;
if (restrictions != null) {
buffer.append(" to "); //$NON-NLS-1$
String sep = ""; //$NON-NLS-1$
char[][] allNames = new char[restrictions.elementSize][];
restrictions.asArray(allNames);
for (char[] targetModule : allNames) {
buffer.append(sep);
buffer.append(targetModule);
sep = ", "; //$NON-NLS-1$
}
}
}
} else {
buffer.append("\nNo Opens"); //$NON-NLS-1$
}
if (this.uses != null && this.uses.length > 0) {
buffer.append("\n/* uses /*\n"); //$NON-NLS-1$
for (int i = 0; i < this.uses.length; i++) {
buffer.append("\n\t"); //$NON-NLS-1$
buffer.append(this.uses[i].debugName());
}
} else {
buffer.append("\nNo Uses"); //$NON-NLS-1$
}
if (this.services != null && this.services.length > 0) {
buffer.append("\n/* Services */\n"); //$NON-NLS-1$
for (int i = 0; i < this.services.length; i++) {
buffer.append("\n\t"); //$NON-NLS-1$
buffer.append("provides "); //$NON-NLS-1$
buffer.append(this.services[i].debugName());
buffer.append(" with "); //$NON-NLS-1$
if (this.implementations != null && this.implementations.containsKey(this.services[i])) {
String sep = ""; //$NON-NLS-1$
for (TypeBinding impl : this.implementations.get(this.services[i])) {
buffer.append(sep).append(impl.debugName());
sep = ", "; //$NON-NLS-1$
}
} else {
buffer.append("<missing implementations>"); //$NON-NLS-1$
}
}
} else {
buffer.append("\nNo Services"); //$NON-NLS-1$
}
return buffer.toString();
}
public boolean isUnnamed() {
return false;
}
public boolean isOpen() {
return (this.modifiers & ClassFileConstants.ACC_OPEN) != 0;
}
public boolean isDeprecated() {
return (this.tagBits & TagBits.AnnotationDeprecated) != 0;
}
public boolean hasUnstableAutoName() {
return false;
}
public boolean isTransitivelyRequired(ModuleBinding otherModule) {
if (this.transitiveRequires == null) {
Set<ModuleBinding> transitiveDeps = new HashSet<>();
collectTransitiveDependencies(transitiveDeps);
this.transitiveRequires = transitiveDeps;
}
return this.transitiveRequires.contains(otherModule);
}
public int getDefaultNullness() {
getAnnotationTagBits(); // ensure annotations are initialized
return this.defaultNullness;
}
SimpleLookupTable storedAnnotations(boolean forceInitialize, boolean forceStore) {
if (forceInitialize && this.storedAnnotations == null) {
if (!this.environment.globalOptions.storeAnnotations && !forceStore)
return null; // not supported during this compile
this.storedAnnotations = new SimpleLookupTable(3);
}
return this.storedAnnotations;
}
public AnnotationHolder retrieveAnnotationHolder(Binding binding, boolean forceInitialization) {
SimpleLookupTable store = storedAnnotations(forceInitialization, false);
return store == null ? null : (AnnotationHolder) store.get(binding);
}
AnnotationBinding[] retrieveAnnotations(Binding binding) {
AnnotationHolder holder = retrieveAnnotationHolder(binding, true);
return holder == null ? Binding.NO_ANNOTATIONS : holder.getAnnotations();
}
@Override
public void setAnnotations(AnnotationBinding[] annotations, boolean forceStore) {
storeAnnotations(this, annotations, forceStore);
}
void storeAnnotationHolder(Binding binding, AnnotationHolder holder) {
if (holder == null) {
SimpleLookupTable store = storedAnnotations(false, false);
if (store != null)
store.removeKey(binding);
} else {
SimpleLookupTable store = storedAnnotations(true, false);
if (store != null)
store.put(binding, holder);
}
}
void storeAnnotations(Binding binding, AnnotationBinding[] annotations, boolean forceStore) {
AnnotationHolder holder = null;
if (annotations == null || annotations.length == 0) {
SimpleLookupTable store = storedAnnotations(false, forceStore);
if (store != null)
holder = (AnnotationHolder) store.get(binding);
if (holder == null) return; // nothing to delete
} else {
SimpleLookupTable store = storedAnnotations(true, forceStore);
if (store == null) return; // not supported
holder = (AnnotationHolder) store.get(binding);
if (holder == null)
holder = new AnnotationHolder();
}
storeAnnotationHolder(binding, holder.setAnnotations(annotations));
}
}