8174823: Module system implementation refresh (3/2017)

Co-authored-by: Mandy Chung <mandy.chung@oracle.com>
Co-authored-by: Sundararajan Athijegannathan <sundararajan.athijegannathan@oracle.com>
Co-authored-by: Peter Levart <peter.levart@gmail.com>
Reviewed-by: chegar, mchung, alanb
This commit is contained in:
Alan Bateman 2017-03-22 16:26:27 +00:00
parent 10a42feace
commit b19eb2766f
101 changed files with 3708 additions and 1541 deletions

View File

@ -128,7 +128,6 @@ SUNWprivate_1.1 {
Java_java_lang_Class_isInstance; Java_java_lang_Class_isInstance;
Java_java_lang_Class_registerNatives; Java_java_lang_Class_registerNatives;
Java_java_lang_ClassLoader_findBootstrapClass; Java_java_lang_ClassLoader_findBootstrapClass;
Java_java_lang_ClassLoader_defineClass0;
Java_java_lang_ClassLoader_defineClass1; Java_java_lang_ClassLoader_defineClass1;
Java_java_lang_ClassLoader_defineClass2; Java_java_lang_ClassLoader_defineClass2;
Java_java_lang_ClassLoader_findBuiltinLib; Java_java_lang_ClassLoader_findBuiltinLib;

View File

@ -64,9 +64,9 @@ import java.util.StringJoiner;
import jdk.internal.HotSpotIntrinsicCandidate; import jdk.internal.HotSpotIntrinsicCandidate;
import jdk.internal.loader.BootLoader; import jdk.internal.loader.BootLoader;
import jdk.internal.loader.BuiltinClassLoader; import jdk.internal.loader.BuiltinClassLoader;
import jdk.internal.loader.ResourceHelper;
import jdk.internal.misc.Unsafe; import jdk.internal.misc.Unsafe;
import jdk.internal.misc.VM; import jdk.internal.misc.VM;
import jdk.internal.module.Resources;
import jdk.internal.reflect.CallerSensitive; import jdk.internal.reflect.CallerSensitive;
import jdk.internal.reflect.ConstantPool; import jdk.internal.reflect.ConstantPool;
import jdk.internal.reflect.Reflection; import jdk.internal.reflect.Reflection;
@ -2563,11 +2563,11 @@ public final class Class<T> implements java.io.Serializable,
Module module = getModule(); Module module = getModule();
if (module.isNamed()) { if (module.isNamed()) {
if (!ResourceHelper.isSimpleResource(name)) { if (Resources.canEncapsulate(name)) {
Module caller = Reflection.getCallerClass().getModule(); Module caller = Reflection.getCallerClass().getModule();
if (caller != module) { if (caller != module) {
Set<String> packages = module.getDescriptor().packages(); Set<String> packages = module.getDescriptor().packages();
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
if (packages.contains(pn) && !module.isOpen(pn, caller)) { if (packages.contains(pn) && !module.isOpen(pn, caller)) {
// resource is in package not open to caller // resource is in package not open to caller
return null; return null;
@ -2665,11 +2665,11 @@ public final class Class<T> implements java.io.Serializable,
Module module = getModule(); Module module = getModule();
if (module.isNamed()) { if (module.isNamed()) {
if (!ResourceHelper.isSimpleResource(name)) { if (Resources.canEncapsulate(name)) {
Module caller = Reflection.getCallerClass().getModule(); Module caller = Reflection.getCallerClass().getModule();
if (caller != module) { if (caller != module) {
Set<String> packages = module.getDescriptor().packages(); Set<String> packages = module.getDescriptor().packages();
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
if (packages.contains(pn) && !module.isOpen(pn, caller)) { if (packages.contains(pn) && !module.isOpen(pn, caller)) {
// resource is in package not open to caller // resource is in package not open to caller
return null; return null;

View File

@ -983,7 +983,7 @@ public abstract class ClassLoader {
{ {
protectionDomain = preDefineClass(name, protectionDomain); protectionDomain = preDefineClass(name, protectionDomain);
String source = defineClassSourceLocation(protectionDomain); String source = defineClassSourceLocation(protectionDomain);
Class<?> c = defineClass1(name, b, off, len, protectionDomain, source); Class<?> c = defineClass1(this, name, b, off, len, protectionDomain, source);
postDefineClass(c, protectionDomain); postDefineClass(c, protectionDomain);
return c; return c;
} }
@ -1075,15 +1075,15 @@ public abstract class ClassLoader {
protectionDomain = preDefineClass(name, protectionDomain); protectionDomain = preDefineClass(name, protectionDomain);
String source = defineClassSourceLocation(protectionDomain); String source = defineClassSourceLocation(protectionDomain);
Class<?> c = defineClass2(name, b, b.position(), len, protectionDomain, source); Class<?> c = defineClass2(this, name, b, b.position(), len, protectionDomain, source);
postDefineClass(c, protectionDomain); postDefineClass(c, protectionDomain);
return c; return c;
} }
private native Class<?> defineClass1(String name, byte[] b, int off, int len, static native Class<?> defineClass1(ClassLoader loader, String name, byte[] b, int off, int len,
ProtectionDomain pd, String source); ProtectionDomain pd, String source);
private native Class<?> defineClass2(String name, java.nio.ByteBuffer b, static native Class<?> defineClass2(ClassLoader loader, String name, java.nio.ByteBuffer b,
int off, int len, ProtectionDomain pd, int off, int len, ProtectionDomain pd,
String source); String source);

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1997, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -26,9 +26,7 @@
package java.lang; package java.lang;
import java.security.*; import java.security.*;
import java.util.Enumeration; import java.lang.module.ModuleFinder;
import java.util.Hashtable;
import java.util.StringTokenizer;
/** /**
* This class is for runtime permissions. A {@code RuntimePermission} * This class is for runtime permissions. A {@code RuntimePermission}
@ -265,6 +263,16 @@ import java.util.StringTokenizer;
* </tr> * </tr>
* *
* <tr> * <tr>
* <td>defineClass</td>
* <td>Define a class with
* {@link java.lang.invoke.MethodHandles.Lookup#defineClass(byte[])
* Lookup.defineClass}.</td>
* <td>This grants code with a suitably privileged {@code Lookup} object
* permission to define classes in the same package as the {@code Lookup}'s
* lookup class. </td>
* </tr>
*
* <tr>
* <td>accessDeclaredMembers</td> * <td>accessDeclaredMembers</td>
* <td>Access to the declared members of a class</td> * <td>Access to the declared members of a class</td>
* <td>This grants code permission to query a class for its public, * <td>This grants code permission to query a class for its public,
@ -359,6 +367,14 @@ import java.util.StringTokenizer;
* <td>See {@link java.lang.System.LoggerFinder java.lang.System.LoggerFinder} * <td>See {@link java.lang.System.LoggerFinder java.lang.System.LoggerFinder}
* for more information.</td> * for more information.</td>
* </tr> * </tr>
*
* <tr>
* <td>accessSystemModules</td>
* <td>Access system modules in the runtime image.</td>
* <td>This grants the permission to access resources in the
* {@linkplain ModuleFinder#ofSystem system modules} in the runtime image.</td>
* </tr>
*
* </table> * </table>
* *
* @implNote * @implNote

View File

@ -982,13 +982,6 @@ final class StackStreamFactory {
} }
private static boolean isReflectionFrame(Class<?> c) { private static boolean isReflectionFrame(Class<?> c) {
if (c.getName().startsWith("jdk.internal.reflect") &&
!MethodAccessor.class.isAssignableFrom(c) &&
!ConstructorAccessor.class.isAssignableFrom(c)) {
throw new InternalError("Not jdk.internal.reflect.MethodAccessor"
+ " or jdk.internal.reflect.ConstructorAccessor: "
+ c.toString());
}
// ## should filter all @Hidden frames? // ## should filter all @Hidden frames?
return c == Method.class || return c == Method.class ||
c == Constructor.class || c == Constructor.class ||

View File

@ -43,6 +43,7 @@ import java.lang.reflect.Modifier;
import java.lang.reflect.Module; import java.lang.reflect.Module;
import java.net.URL; import java.net.URL;
import java.security.AccessControlContext; import java.security.AccessControlContext;
import java.security.ProtectionDomain;
import java.util.Properties; import java.util.Properties;
import java.util.PropertyPermission; import java.util.PropertyPermission;
import java.util.Map; import java.util.Map;
@ -1844,6 +1845,39 @@ public final class System {
return new PrintStream(new BufferedOutputStream(fos, 128), true); return new PrintStream(new BufferedOutputStream(fos, 128), true);
} }
/**
* Logs an exception/error at initialization time to stdout or stderr.
*
* @param printToStderr to print to stderr rather than stdout
* @param printStackTrace to print the stack trace
* @param msg the message to print before the exception, can be {@code null}
* @param e the exception or error
*/
private static void logInitException(boolean printToStderr,
boolean printStackTrace,
String msg,
Throwable e) {
if (VM.initLevel() < 1) {
throw new InternalError("system classes not initialized");
}
PrintStream log = (printToStderr) ? err : out;
if (msg != null) {
log.println(msg);
}
if (printStackTrace) {
e.printStackTrace(log);
} else {
log.println(e);
for (Throwable suppressed : e.getSuppressed()) {
log.println("Suppressed: " + suppressed);
}
Throwable cause = e.getCause();
if (cause != null) {
log.println("Caused by: " + cause);
}
}
}
/** /**
* Initialize the system class. Called after thread initialization. * Initialize the system class. Called after thread initialization.
*/ */
@ -1922,13 +1956,25 @@ public final class System {
/* /*
* Invoked by VM. Phase 2 module system initialization. * Invoked by VM. Phase 2 module system initialization.
* Only classes in java.base can be loaded in this phase. * Only classes in java.base can be loaded in this phase.
*
* @param printToStderr print exceptions to stderr rather than stdout
* @param printStackTrace print stack trace when exception occurs
*
* @return JNI_OK for success, JNI_ERR for failure
*/ */
private static void initPhase2() { private static int initPhase2(boolean printToStderr, boolean printStackTrace) {
// initialize the module system try {
System.bootLayer = ModuleBootstrap.boot(); bootLayer = ModuleBootstrap.boot();
} catch (Exception | Error e) {
logInitException(printToStderr, printStackTrace,
"Error occurred during initialization of boot layer", e);
return -1; // JNI_ERR
}
// module system initialized // module system initialized
VM.initLevel(2); VM.initLevel(2);
return 0; // JNI_OK
} }
/* /*
@ -2034,6 +2080,9 @@ public final class System {
public ConcurrentHashMap<?, ?> createOrGetClassLoaderValueMap(ClassLoader cl) { public ConcurrentHashMap<?, ?> createOrGetClassLoaderValueMap(ClassLoader cl) {
return cl.createOrGetClassLoaderValueMap(); return cl.createOrGetClassLoaderValueMap();
} }
public Class<?> defineClass(ClassLoader loader, String name, byte[] b, ProtectionDomain pd, String source) {
return ClassLoader.defineClass1(loader, name, b, 0, b.length, pd, source);
}
public Class<?> findBootstrapClassOrNull(ClassLoader cl, String name) { public Class<?> findBootstrapClassOrNull(ClassLoader cl, String name) {
return cl.findBootstrapClassOrNull(name); return cl.findBootstrapClassOrNull(name);
} }

View File

@ -25,6 +25,9 @@
package java.lang.invoke; package java.lang.invoke;
import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.IllegalAccessLogger;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.reflect.CallerSensitive; import jdk.internal.reflect.CallerSensitive;
import jdk.internal.reflect.Reflection; import jdk.internal.reflect.Reflection;
import jdk.internal.vm.annotation.ForceInline; import jdk.internal.vm.annotation.ForceInline;
@ -43,6 +46,9 @@ import java.lang.reflect.Modifier;
import java.lang.reflect.Module; import java.lang.reflect.Module;
import java.lang.reflect.ReflectPermission; import java.lang.reflect.ReflectPermission;
import java.nio.ByteOrder; import java.nio.ByteOrder;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.security.ProtectionDomain;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.BitSet; import java.util.BitSet;
@ -191,6 +197,12 @@ public class MethodHandles {
} }
if ((lookup.lookupModes() & Lookup.MODULE) == 0) if ((lookup.lookupModes() & Lookup.MODULE) == 0)
throw new IllegalAccessException("lookup does not have MODULE lookup mode"); throw new IllegalAccessException("lookup does not have MODULE lookup mode");
if (!callerModule.isNamed() && targetModule.isNamed()) {
IllegalAccessLogger logger = IllegalAccessLogger.illegalAccessLogger();
if (logger != null) {
logger.logIfOpenedByBackdoor(lookup, targetClass);
}
}
return new Lookup(targetClass); return new Lookup(targetClass);
} }
@ -855,6 +867,112 @@ public class MethodHandles {
return new Lookup(lookupClass(), newModes); return new Lookup(lookupClass(), newModes);
} }
/**
* Defines a class to the same class loader and in the same runtime package and
* {@linkplain java.security.ProtectionDomain protection domain} as this lookup's
* {@linkplain #lookupClass() lookup class}.
*
* <p> The {@linkplain #lookupModes() lookup modes} for this lookup must include
* {@link #PACKAGE PACKAGE} access as default (package) members will be
* accessible to the class. The {@code PACKAGE} lookup mode serves to authenticate
* that the lookup object was created by a caller in the runtime package (or derived
* from a lookup originally created by suitably privileged code to a target class in
* the runtime package). The lookup modes cannot include {@link #PRIVATE PRIVATE}
* access. A lookup with {@code PRIVATE} access can be downgraded to drop this lookup
* mode with the {@linkplain #dropLookupMode(int) dropLookupMode} method. </p>
*
* <p> The {@code bytes} parameter is the class bytes of a valid class file (as defined
* by the <em>The Java Virtual Machine Specification</em>) with a class name in the
* same package as the lookup class. </p>
*
* <p> This method does not run the class initializer. The class initializer may
* run at a later time, as detailed in section 12.4 of the <em>The Java Language
* Specification</em>. </p>
*
* <p> If there is a security manager, its {@code checkPermission} method is first called
* to check {@code RuntimePermission("defineClass")}. </p>
*
* @param bytes the class bytes
* @return the {@code Class} object for the class
* @throws IllegalArgumentException the bytes are for a class in a different package
* to the lookup class
* @throws IllegalAccessException if this lookup does not have {@code PACKAGE} access
* @throws UnsupportedOperationException if the lookup class has {@code PRIVATE} access
* @throws LinkageError if the class is malformed ({@code ClassFormatError}), cannot be
* verified ({@code VerifyError}), is already defined, or another linkage error occurs
* @throws SecurityException if denied by the security manager
* @throws NullPointerException if {@code bytes} is {@code null}
* @since 9
* @spec JPMS
* @see Lookup#privateLookupIn
* @see Lookup#dropLookupMode
* @see ClassLoader#defineClass(String,byte[],int,int,ProtectionDomain)
*/
public Class<?> defineClass(byte[] bytes) throws IllegalAccessException {
SecurityManager sm = System.getSecurityManager();
if (sm != null)
sm.checkPermission(new RuntimePermission("defineClass"));
if (hasPrivateAccess())
throw new UnsupportedOperationException("PRIVATE access not supported");
if ((lookupModes() & PACKAGE) == 0)
throw new IllegalAccessException("Lookup does not have PACKAGE access");
assert (lookupModes() & (MODULE|PUBLIC)) != 0;
// parse class bytes to get class name (in internal form)
bytes = bytes.clone();
String name;
try {
ClassReader reader = new ClassReader(bytes);
name = reader.getClassName();
} catch (RuntimeException e) {
// ASM exceptions are poorly specified
ClassFormatError cfe = new ClassFormatError();
cfe.initCause(e);
throw cfe;
}
// get package and class name in binary form
String cn, pn;
int index = name.lastIndexOf('/');
if (index == -1) {
cn = name;
pn = "";
} else {
cn = name.replace('/', '.');
pn = cn.substring(0, index);
}
if (!pn.equals(lookupClass.getPackageName())) {
throw new IllegalArgumentException("Class not in same package as lookup class");
}
// invoke the class loader's defineClass method
ClassLoader loader = lookupClass.getClassLoader();
ProtectionDomain pd = (loader != null) ? lookupClassProtectionDomain() : null;
String source = "__Lookup_defineClass__";
Class<?> clazz = SharedSecrets.getJavaLangAccess().defineClass(loader, cn, bytes, pd, source);
assert clazz.getClassLoader() == lookupClass.getClassLoader()
&& clazz.getPackageName().equals(lookupClass.getPackageName())
&& protectionDomain(clazz) == lookupClassProtectionDomain();
return clazz;
}
private ProtectionDomain lookupClassProtectionDomain() {
ProtectionDomain pd = cachedProtectionDomain;
if (pd == null) {
cachedProtectionDomain = pd = protectionDomain(lookupClass);
}
return pd;
}
private ProtectionDomain protectionDomain(Class<?> clazz) {
PrivilegedAction<ProtectionDomain> pa = clazz::getProtectionDomain;
return AccessController.doPrivileged(pa);
}
// cached protection domain
private volatile ProtectionDomain cachedProtectionDomain;
// Make sure outer class is initialized first. // Make sure outer class is initialized first.
static { IMPL_NAMES.getClass(); } static { IMPL_NAMES.getClass(); }
@ -1948,7 +2066,7 @@ return mh1;
/** /**
* Returns {@code true} if this lookup has {@code PRIVATE} access. * Returns {@code true} if this lookup has {@code PRIVATE} access.
* @return {@code true} if this lookup has {@code PRIVATE} acesss. * @return {@code true} if this lookup has {@code PRIVATE} access.
* @since 9 * @since 9
*/ */
public boolean hasPrivateAccess() { public boolean hasPrivateAccess() {

View File

@ -112,11 +112,9 @@ public final class Configuration {
// module constraints on target // module constraints on target
private final String osName; private final String osName;
private final String osArch; private final String osArch;
private final String osVersion;
String osName() { return osName; } String osName() { return osName; }
String osArch() { return osArch; } String osArch() { return osArch; }
String osVersion() { return osVersion; }
private Configuration() { private Configuration() {
this.parents = Collections.emptyList(); this.parents = Collections.emptyList();
@ -125,7 +123,6 @@ public final class Configuration {
this.nameToModule = Collections.emptyMap(); this.nameToModule = Collections.emptyMap();
this.osName = null; this.osName = null;
this.osArch = null; this.osArch = null;
this.osVersion = null;
} }
private Configuration(List<Configuration> parents, private Configuration(List<Configuration> parents,
@ -152,7 +149,6 @@ public final class Configuration {
this.osName = resolver.osName(); this.osName = resolver.osName();
this.osArch = resolver.osArch(); this.osArch = resolver.osArch();
this.osVersion = resolver.osVersion();
} }
/** /**
@ -281,6 +277,7 @@ public final class Configuration {
* <em>observability-related</em> reasons: </p> * <em>observability-related</em> reasons: </p>
* *
* <ul> * <ul>
*
* <li><p> A root module, or a direct or transitive dependency, is not * <li><p> A root module, or a direct or transitive dependency, is not
* found. </p></li> * found. </p></li>
* *
@ -289,13 +286,6 @@ public final class Configuration {
* descriptor ({@code module-info.class}) or two versions of the same * descriptor ({@code module-info.class}) or two versions of the same
* module are found in the same directory. </p></li> * module are found in the same directory. </p></li>
* *
* <li><p> A module with the required name is found but the module
* requires a different {@link ModuleDescriptor#osName() operating
* system}, {@link ModuleDescriptor#osArch() architecture}, or {@link
* ModuleDescriptor#osVersion() version} to other modules that have
* been resolved for the new configuration or modules in the parent
* configurations. </p></li>
*
* </ul> * </ul>
* *
* <p> Post-resolution consistency checks may fail with {@code * <p> Post-resolution consistency checks may fail with {@code
@ -306,6 +296,10 @@ public final class Configuration {
* <li><p> A cycle is detected, say where module {@code m1} requires * <li><p> A cycle is detected, say where module {@code m1} requires
* module {@code m2} and {@code m2} requires {@code m1}. </p></li> * module {@code m2} and {@code m2} requires {@code m1}. </p></li>
* *
* <li><p> A module reads two or more modules with the same name. This
* includes the case where a module reads another with the same name as
* itself. </p></li>
*
* <li><p> Two or more modules in the configuration export the same * <li><p> Two or more modules in the configuration export the same
* package to a module that reads both. This includes the case where a * package to a module that reads both. This includes the case where a
* module {@code M} containing package {@code p} reads another module * module {@code M} containing package {@code p} reads another module
@ -319,8 +313,9 @@ public final class Configuration {
* </ul> * </ul>
* *
* @implNote In the implementation then observability of modules may depend * @implNote In the implementation then observability of modules may depend
* on referential integrity checks that ensure different builds of tightly * on referential integrity or other checks that ensure different builds of
* coupled modules are not combined in the same configuration. * tightly coupled modules or modules for specific operating systems or
* architectures are not combined in the same configuration.
* *
* @param before * @param before
* The <em>before</em> module finder to find modules * The <em>before</em> module finder to find modules

View File

@ -179,8 +179,10 @@ public class ModuleDescriptor
private final Set<Modifier> mods; private final Set<Modifier> mods;
private final String name; private final String name;
private final Version compiledVersion; private final Version compiledVersion;
private final String rawCompiledVersion;
private Requires(Set<Modifier> ms, String mn, Version v) { private Requires(Set<Modifier> ms, String mn, Version v, String vs) {
assert v == null || vs == null;
if (ms.isEmpty()) { if (ms.isEmpty()) {
ms = Collections.emptySet(); ms = Collections.emptySet();
} else { } else {
@ -189,12 +191,14 @@ public class ModuleDescriptor
this.mods = ms; this.mods = ms;
this.name = mn; this.name = mn;
this.compiledVersion = v; this.compiledVersion = v;
this.rawCompiledVersion = vs;
} }
private Requires(Set<Modifier> ms, String mn, Version v, boolean unused) { private Requires(Set<Modifier> ms, String mn, Version v, boolean unused) {
this.mods = ms; this.mods = ms;
this.name = mn; this.name = mn;
this.compiledVersion = v; this.compiledVersion = v;
this.rawCompiledVersion = null;
} }
/** /**
@ -218,12 +222,33 @@ public class ModuleDescriptor
/** /**
* Returns the version of the module if recorded at compile-time. * Returns the version of the module if recorded at compile-time.
* *
* @return The version of the module if recorded at compile-time * @return The version of the module if recorded at compile-time,
* or an empty {@code Optional} if no version was recorded or
* the version string recorded is {@linkplain Version#parse(String)
* unparseable}
*/ */
public Optional<Version> compiledVersion() { public Optional<Version> compiledVersion() {
return Optional.ofNullable(compiledVersion); return Optional.ofNullable(compiledVersion);
} }
/**
* Returns the string with the possibly-unparseable version of the module
* if recorded at compile-time.
*
* @return The string containing the version of the module if recorded
* at compile-time, or an empty {@code Optional} if no version
* was recorded
*
* @see #compiledVersion()
*/
public Optional<String> rawCompiledVersion() {
if (compiledVersion != null) {
return Optional.of(compiledVersion.toString());
} else {
return Optional.ofNullable(rawCompiledVersion);
}
}
/** /**
* Compares this module dependence to another. * Compares this module dependence to another.
* *
@ -236,7 +261,10 @@ public class ModuleDescriptor
* recorded at compile-time are compared. When comparing the versions * recorded at compile-time are compared. When comparing the versions
* recorded at compile-time then a dependence that has a recorded * recorded at compile-time then a dependence that has a recorded
* version is considered to succeed a dependence that does not have a * version is considered to succeed a dependence that does not have a
* recorded version. </p> * recorded version. If both recorded versions are {@linkplain
* Version#parse(String) unparseable} then the {@linkplain
* #rawCompiledVersion() raw version strings} are compared
* lexicographically. </p>
* *
* @param that * @param that
* The module dependence to compare * The module dependence to compare
@ -262,6 +290,10 @@ public class ModuleDescriptor
c = compare(this.compiledVersion, that.compiledVersion); c = compare(this.compiledVersion, that.compiledVersion);
if (c != 0) return c; if (c != 0) return c;
// rawCompiledVersion
c = compare(this.rawCompiledVersion, that.rawCompiledVersion);
if (c != 0) return c;
return 0; return 0;
} }
@ -289,7 +321,8 @@ public class ModuleDescriptor
return false; return false;
Requires that = (Requires)ob; Requires that = (Requires)ob;
return name.equals(that.name) && mods.equals(that.mods) return name.equals(that.name) && mods.equals(that.mods)
&& Objects.equals(compiledVersion, that.compiledVersion); && Objects.equals(compiledVersion, that.compiledVersion)
&& Objects.equals(rawCompiledVersion, that.rawCompiledVersion);
} }
/** /**
@ -306,6 +339,8 @@ public class ModuleDescriptor
int hash = name.hashCode() * 43 + mods.hashCode(); int hash = name.hashCode() * 43 + mods.hashCode();
if (compiledVersion != null) if (compiledVersion != null)
hash = hash * 43 + compiledVersion.hashCode(); hash = hash * 43 + compiledVersion.hashCode();
if (rawCompiledVersion != null)
hash = hash * 43 + rawCompiledVersion.hashCode();
return hash; return hash;
} }
@ -774,7 +809,7 @@ public class ModuleDescriptor
/** /**
* Returns the fully qualified class name of the service type. * Returns the fully qualified class name of the service type.
* *
* @return The fully qualified class name of the service type. * @return The fully qualified class name of the service type
*/ */
public String service() { return service; } public String service() { return service; }
@ -1199,6 +1234,7 @@ public class ModuleDescriptor
private final String name; private final String name;
private final Version version; private final Version version;
private final String rawVersionString;
private final Set<Modifier> modifiers; private final Set<Modifier> modifiers;
private final boolean open; // true if modifiers contains OPEN private final boolean open; // true if modifiers contains OPEN
private final boolean automatic; // true if modifiers contains AUTOMATIC private final boolean automatic; // true if modifiers contains AUTOMATIC
@ -1209,12 +1245,10 @@ public class ModuleDescriptor
private final Set<Provides> provides; private final Set<Provides> provides;
private final Set<String> packages; private final Set<String> packages;
private final String mainClass; private final String mainClass;
private final String osName;
private final String osArch;
private final String osVersion;
private ModuleDescriptor(String name, private ModuleDescriptor(String name,
Version version, Version version,
String rawVersionString,
Set<Modifier> modifiers, Set<Modifier> modifiers,
Set<Requires> requires, Set<Requires> requires,
Set<Exports> exports, Set<Exports> exports,
@ -1222,13 +1256,12 @@ public class ModuleDescriptor
Set<String> uses, Set<String> uses,
Set<Provides> provides, Set<Provides> provides,
Set<String> packages, Set<String> packages,
String mainClass, String mainClass)
String osName,
String osArch,
String osVersion)
{ {
assert version == null || rawVersionString == null;
this.name = name; this.name = name;
this.version = version; this.version = version;
this.rawVersionString = rawVersionString;
this.modifiers = emptyOrUnmodifiableSet(modifiers); this.modifiers = emptyOrUnmodifiableSet(modifiers);
this.open = modifiers.contains(Modifier.OPEN); this.open = modifiers.contains(Modifier.OPEN);
this.automatic = modifiers.contains(Modifier.AUTOMATIC); this.automatic = modifiers.contains(Modifier.AUTOMATIC);
@ -1242,9 +1275,6 @@ public class ModuleDescriptor
this.packages = emptyOrUnmodifiableSet(packages); this.packages = emptyOrUnmodifiableSet(packages);
this.mainClass = mainClass; this.mainClass = mainClass;
this.osName = osName;
this.osArch = osArch;
this.osVersion = osVersion;
} }
/** /**
@ -1261,13 +1291,11 @@ public class ModuleDescriptor
Set<Provides> provides, Set<Provides> provides,
Set<String> packages, Set<String> packages,
String mainClass, String mainClass,
String osName,
String osArch,
String osVersion,
int hashCode, int hashCode,
boolean unused) { boolean unused) {
this.name = name; this.name = name;
this.version = version; this.version = version;
this.rawVersionString = null;
this.modifiers = modifiers; this.modifiers = modifiers;
this.open = modifiers.contains(Modifier.OPEN); this.open = modifiers.contains(Modifier.OPEN);
this.automatic = modifiers.contains(Modifier.AUTOMATIC); this.automatic = modifiers.contains(Modifier.AUTOMATIC);
@ -1278,9 +1306,6 @@ public class ModuleDescriptor
this.provides = provides; this.provides = provides;
this.packages = packages; this.packages = packages;
this.mainClass = mainClass; this.mainClass = mainClass;
this.osName = osName;
this.osArch = osArch;
this.osVersion = osVersion;
this.hash = hashCode; this.hash = hashCode;
} }
@ -1394,18 +1419,37 @@ public class ModuleDescriptor
/** /**
* <p> Returns the module version. </p> * <p> Returns the module version. </p>
* *
* @return This module's version * @return This module's version, or an empty {@code Optional} if the
* module does not have a version or the version is
* {@linkplain Version#parse(String) unparseable}
*/ */
public Optional<Version> version() { public Optional<Version> version() {
return Optional.ofNullable(version); return Optional.ofNullable(version);
} }
/**
* <p> Returns the string with the possibly-unparseable version of the
* module </p>
*
* @return The string containing the version of the module or an empty
* {@code Optional} if the module does not have a version
*
* @see #version()
*/
public Optional<String> rawVersion() {
if (version != null) {
return Optional.of(version.toString());
} else {
return Optional.ofNullable(rawVersionString);
}
}
/** /**
* <p> Returns a string containing the module name and, if present, its * <p> Returns a string containing the module name and, if present, its
* version. </p> * version. </p>
* *
* @return A string containing the module name and, if present, its * @return A string containing the module name and, if present, its
* version. * version
*/ */
public String toNameAndVersion() { public String toNameAndVersion() {
if (version != null) { if (version != null) {
@ -1424,42 +1468,13 @@ public class ModuleDescriptor
return Optional.ofNullable(mainClass); return Optional.ofNullable(mainClass);
} }
/**
* Returns the operating system name if the module is operating system
* specific.
*
* @return The operating system name or an empty {@code Optional}
* if the module is not operating system specific
*/
public Optional<String> osName() {
return Optional.ofNullable(osName);
}
/**
* Returns the operating system architecture if the module is operating
* system architecture specific.
*
* @return The operating system architecture or an empty {@code Optional}
* if the module is not operating system architecture specific
*/
public Optional<String> osArch() {
return Optional.ofNullable(osArch);
}
/**
* Returns the operating system version if the module is operating
* system version specific.
*
* @return The operating system version or an empty {@code Optional}
* if the module is not operating system version specific
*/
public Optional<String> osVersion() {
return Optional.ofNullable(osVersion);
}
/** /**
* Returns the set of packages in the module. * Returns the set of packages in the module.
* *
* <p> The set of packages includes all exported and open packages, as well
* as the packages of any service providers, and the package for the main
* class. </p>
*
* @return A possibly-empty unmodifiable set of the packages in the module * @return A possibly-empty unmodifiable set of the packages in the module
*/ */
public Set<String> packages() { public Set<String> packages() {
@ -1518,9 +1533,7 @@ public class ModuleDescriptor
final Set<String> uses = new HashSet<>(); final Set<String> uses = new HashSet<>();
final Map<String, Provides> provides = new HashMap<>(); final Map<String, Provides> provides = new HashMap<>();
Version version; Version version;
String osName; String rawVersionString;
String osArch;
String osVersion;
String mainClass; String mainClass;
/** /**
@ -1604,24 +1617,21 @@ public class ModuleDescriptor
Objects.requireNonNull(compiledVersion); Objects.requireNonNull(compiledVersion);
if (strict) if (strict)
mn = requireModuleName(mn); mn = requireModuleName(mn);
return requires(new Requires(ms, mn, compiledVersion)); return requires(new Requires(ms, mn, compiledVersion, null));
} }
/* package */Builder requires(Set<Requires.Modifier> ms, /* package */Builder requires(Set<Requires.Modifier> ms,
String mn, String mn,
String compiledVersion) { String rawCompiledVersion) {
Version v = null; Requires r;
try { try {
v = Version.parse(compiledVersion); Version v = Version.parse(rawCompiledVersion);
r = new Requires(ms, mn, v, null);
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {
// for now, drop un-parsable version when non-strict
if (strict) throw e; if (strict) throw e;
r = new Requires(ms, mn, null, rawCompiledVersion);
} }
if (v == null) { return requires(r);
return requires(ms, mn);
} else {
return requires(ms, mn, v);
}
} }
/** /**
@ -1646,7 +1656,7 @@ public class ModuleDescriptor
public Builder requires(Set<Requires.Modifier> ms, String mn) { public Builder requires(Set<Requires.Modifier> ms, String mn) {
if (strict) if (strict)
mn = requireModuleName(mn); mn = requireModuleName(mn);
return requires(new Requires(ms, mn, null)); return requires(new Requires(ms, mn, null, null));
} }
/** /**
@ -1952,7 +1962,7 @@ public class ModuleDescriptor
* a class in a named package * a class in a named package
* @throws IllegalStateException * @throws IllegalStateException
* If a dependency on the service type has already been declared * If a dependency on the service type has already been declared
* or this is a builder for an an automatic module * or this is a builder for an automatic module
*/ */
public Builder uses(String service) { public Builder uses(String service) {
if (automatic) if (automatic)
@ -2068,6 +2078,7 @@ public class ModuleDescriptor
*/ */
public Builder version(Version v) { public Builder version(Version v) {
version = requireNonNull(v); version = requireNonNull(v);
rawVersionString = null;
return this; return this;
} }
@ -2086,19 +2097,16 @@ public class ModuleDescriptor
* @see Version#parse(String) * @see Version#parse(String)
*/ */
public Builder version(String vs) { public Builder version(String vs) {
Version v;
if (strict) {
v = Version.parse(vs);
} else {
try { try {
v = Version.parse(vs); version = Version.parse(vs);
} catch (IllegalArgumentException ignore) { rawVersionString = null;
// for now, ignore when non-strict } catch (IllegalArgumentException e) {
if (strict) throw e;
version = null;
rawVersionString = vs;
}
return this; return this;
} }
}
return version(v);
}
/** /**
* Sets the module main class. The package for the main class is added * Sets the module main class. The package for the main class is added
@ -2131,60 +2139,6 @@ public class ModuleDescriptor
return this; return this;
} }
/**
* Sets the operating system name.
*
* @param name
* The operating system name
*
* @return This builder
*
* @throws IllegalArgumentException
* If {@code name} is {@code null} or the empty String
*/
public Builder osName(String name) {
if (name == null || name.isEmpty())
throw new IllegalArgumentException("OS name is null or empty");
osName = name;
return this;
}
/**
* Sets the operating system architecture.
*
* @param arch
* The operating system architecture
*
* @return This builder
*
* @throws IllegalArgumentException
* If {@code name} is {@code null} or the empty String
*/
public Builder osArch(String arch) {
if (arch == null || arch.isEmpty())
throw new IllegalArgumentException("OS arch is null or empty");
osArch = arch;
return this;
}
/**
* Sets the operating system version.
*
* @param version
* The operating system version
*
* @return This builder
*
* @throws IllegalArgumentException
* If {@code name} is {@code null} or the empty String
*/
public Builder osVersion(String version) {
if (version == null || version.isEmpty())
throw new IllegalArgumentException("OS version is null or empty");
osVersion = version;
return this;
}
/** /**
* Builds and returns a {@code ModuleDescriptor} from its components. * Builds and returns a {@code ModuleDescriptor} from its components.
* *
@ -2208,6 +2162,7 @@ public class ModuleDescriptor
&& !this.requires.containsKey("java.base")) { && !this.requires.containsKey("java.base")) {
requires.add(new Requires(Set.of(Requires.Modifier.MANDATED), requires.add(new Requires(Set.of(Requires.Modifier.MANDATED),
"java.base", "java.base",
null,
null)); null));
} }
@ -2215,6 +2170,7 @@ public class ModuleDescriptor
return new ModuleDescriptor(name, return new ModuleDescriptor(name,
version, version,
rawVersionString,
modifiers, modifiers,
requires, requires,
exports, exports,
@ -2222,10 +2178,7 @@ public class ModuleDescriptor
uses, uses,
provides, provides,
packages, packages,
mainClass, mainClass);
osName,
osArch,
osVersion);
} }
} }
@ -2237,9 +2190,11 @@ public class ModuleDescriptor
* module names lexicographically. Where the module names are equal then the * module names lexicographically. Where the module names are equal then the
* module versions are compared. When comparing the module versions then a * module versions are compared. When comparing the module versions then a
* module descriptor with a version is considered to succeed a module * module descriptor with a version is considered to succeed a module
* descriptor that does not have a version. Where the module names are equal * descriptor that does not have a version. If both versions are {@linkplain
* and the versions are equal (or not present in both), then the set of * Version#parse(String) unparseable} then the {@linkplain #rawVersion()
* modifiers are compared. Sets of modifiers are compared by comparing * raw version strings} are compared lexicographically. Where the module names
* are equal and the versions are equal (or not present in both), then the
* set of modifiers are compared. Sets of modifiers are compared by comparing
* a <em>binary value</em> computed for each set. If a modifier is present * a <em>binary value</em> computed for each set. If a modifier is present
* in the set then the bit at the position of its ordinal is {@code 1} * in the set then the bit at the position of its ordinal is {@code 1}
* in the binary value, otherwise {@code 0}. If the two set of modifiers * in the binary value, otherwise {@code 0}. If the two set of modifiers
@ -2263,6 +2218,9 @@ public class ModuleDescriptor
c = compare(this.version, that.version); c = compare(this.version, that.version);
if (c != 0) return c; if (c != 0) return c;
c = compare(this.rawVersionString, that.rawVersionString);
if (c != 0) return c;
long v1 = modsValue(this.modifiers()); long v1 = modsValue(this.modifiers());
long v2 = modsValue(that.modifiers()); long v2 = modsValue(that.modifiers());
c = Long.compare(v1, v2); c = Long.compare(v1, v2);
@ -2289,15 +2247,6 @@ public class ModuleDescriptor
c = compare(this.mainClass, that.mainClass); c = compare(this.mainClass, that.mainClass);
if (c != 0) return c; if (c != 0) return c;
c = compare(this.osName, that.osName);
if (c != 0) return c;
c = compare(this.osArch, that.osArch);
if (c != 0) return c;
c = compare(this.osVersion, that.osVersion);
if (c != 0) return c;
return 0; return 0;
} }
@ -2333,10 +2282,8 @@ public class ModuleDescriptor
&& uses.equals(that.uses) && uses.equals(that.uses)
&& provides.equals(that.provides) && provides.equals(that.provides)
&& Objects.equals(version, that.version) && Objects.equals(version, that.version)
&& Objects.equals(mainClass, that.mainClass) && Objects.equals(rawVersionString, that.rawVersionString)
&& Objects.equals(osName, that.osName) && Objects.equals(mainClass, that.mainClass));
&& Objects.equals(osArch, that.osArch)
&& Objects.equals(osVersion, that.osVersion));
} }
/** /**
@ -2361,10 +2308,8 @@ public class ModuleDescriptor
hc = hc * 43 + uses.hashCode(); hc = hc * 43 + uses.hashCode();
hc = hc * 43 + provides.hashCode(); hc = hc * 43 + provides.hashCode();
hc = hc * 43 + Objects.hashCode(version); hc = hc * 43 + Objects.hashCode(version);
hc = hc * 43 + Objects.hashCode(rawVersionString);
hc = hc * 43 + Objects.hashCode(mainClass); hc = hc * 43 + Objects.hashCode(mainClass);
hc = hc * 43 + Objects.hashCode(osName);
hc = hc * 43 + Objects.hashCode(osArch);
hc = hc * 43 + Objects.hashCode(osVersion);
if (hc == 0) if (hc == 0)
hc = -1; hc = -1;
hash = hc; hash = hc;
@ -2713,8 +2658,8 @@ public class ModuleDescriptor
public void requires(ModuleDescriptor.Builder builder, public void requires(ModuleDescriptor.Builder builder,
Set<Requires.Modifier> ms, Set<Requires.Modifier> ms,
String mn, String mn,
String compiledVersion) { String rawCompiledVersion) {
builder.requires(ms, mn, compiledVersion); builder.requires(ms, mn, rawCompiledVersion);
} }
@Override @Override
@ -2762,9 +2707,6 @@ public class ModuleDescriptor
Set<Provides> provides, Set<Provides> provides,
Set<String> packages, Set<String> packages,
String mainClass, String mainClass,
String osName,
String osArch,
String osVersion,
int hashCode) { int hashCode) {
return new ModuleDescriptor(name, return new ModuleDescriptor(name,
version, version,
@ -2776,9 +2718,6 @@ public class ModuleDescriptor
provides, provides,
packages, packages,
mainClass, mainClass,
osName,
osArch,
osVersion,
hashCode, hashCode,
false); false);
} }

View File

@ -25,8 +25,6 @@
package java.lang.module; package java.lang.module;
import java.io.File;
import java.io.FilePermission;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.Paths; import java.nio.file.Paths;
@ -43,9 +41,9 @@ import java.util.Optional;
import java.util.Set; import java.util.Set;
import jdk.internal.module.ModuleBootstrap; import jdk.internal.module.ModuleBootstrap;
import jdk.internal.module.ModulePatcher;
import jdk.internal.module.ModulePath; import jdk.internal.module.ModulePath;
import jdk.internal.module.SystemModuleFinder; import jdk.internal.module.SystemModuleFinder;
import sun.security.action.GetPropertyAction;
/** /**
* A finder of modules. A {@code ModuleFinder} is used to find modules during * A finder of modules. A {@code ModuleFinder} is used to find modules during
@ -146,9 +144,9 @@ public interface ModuleFinder {
* *
* <p> If there is a security manager set then its {@link * <p> If there is a security manager set then its {@link
* SecurityManager#checkPermission(Permission) checkPermission} method is * SecurityManager#checkPermission(Permission) checkPermission} method is
* invoked to check that the caller has been granted {@link FilePermission} * invoked to check that the caller has been granted
* to recursively read the directory that is the value of the system * {@link RuntimePermission RuntimePermission("accessSystemModules")}
* property {@code java.home}. </p> * to access the system modules. </p>
* *
* @return A {@code ModuleFinder} that locates the system modules * @return A {@code ModuleFinder} that locates the system modules
* *
@ -156,32 +154,55 @@ public interface ModuleFinder {
* If denied by the security manager * If denied by the security manager
*/ */
static ModuleFinder ofSystem() { static ModuleFinder ofSystem() {
String home;
SecurityManager sm = System.getSecurityManager(); SecurityManager sm = System.getSecurityManager();
if (sm != null) { if (sm != null) {
PrivilegedAction<String> pa = new GetPropertyAction("java.home"); sm.checkPermission(new RuntimePermission("accessSystemModules"));
home = AccessController.doPrivileged(pa); PrivilegedAction<ModuleFinder> pa = ModuleFinder::privilegedOfSystem;
Permission p = new FilePermission(home + File.separator + "-", "read"); return AccessController.doPrivileged(pa);
sm.checkPermission(p);
} else { } else {
home = System.getProperty("java.home"); return privilegedOfSystem();
}
} }
/**
* Returns a module finder that locates the system modules. This method
* assumes it has permissions to access the runtime image.
*/
private static ModuleFinder privilegedOfSystem() {
String home = System.getProperty("java.home");
Path modules = Paths.get(home, "lib", "modules"); Path modules = Paths.get(home, "lib", "modules");
if (Files.isRegularFile(modules)) { if (Files.isRegularFile(modules)) {
return SystemModuleFinder.getInstance(); return SystemModuleFinder.getInstance();
} else { } else {
Path mlib = Paths.get(home, "modules"); Path dir = Paths.get(home, "modules");
if (Files.isDirectory(mlib)) { if (Files.isDirectory(dir)) {
// exploded build may be patched return privilegedOf(ModuleBootstrap.patcher(), dir);
return ModulePath.of(ModuleBootstrap.patcher(), mlib);
} else { } else {
throw new InternalError("Unable to detect the run-time image"); throw new InternalError("Unable to detect the run-time image");
} }
} }
} }
/**
* Returns a module finder that locates the system modules in an exploded
* image. The image may be patched.
*/
private static ModuleFinder privilegedOf(ModulePatcher patcher, Path dir) {
ModuleFinder finder = ModulePath.of(patcher, dir);
return new ModuleFinder() {
@Override
public Optional<ModuleReference> find(String name) {
PrivilegedAction<Optional<ModuleReference>> pa = () -> finder.find(name);
return AccessController.doPrivileged(pa);
}
@Override
public Set<ModuleReference> findAll() {
PrivilegedAction<Set<ModuleReference>> pa = finder::findAll;
return AccessController.doPrivileged(pa);
}
};
}
/** /**
* Returns a module finder that locates modules on the file system by * Returns a module finder that locates modules on the file system by
* searching a sequence of directories and/or packaged modules. * searching a sequence of directories and/or packaged modules.
@ -201,7 +222,7 @@ public interface ModuleFinder {
* *
* <p> If an element is a path to a directory of modules then each entry in * <p> If an element is a path to a directory of modules then each entry in
* the directory is a packaged module or the top-level directory of an * the directory is a packaged module or the top-level directory of an
* exploded module. It it an error if a directory contains more than one * exploded module. It is an error if a directory contains more than one
* module with the same name. If an element is a path to a directory, and * module with the same name. If an element is a path to a directory, and
* that directory contains a file named {@code module-info.class}, then the * that directory contains a file named {@code module-info.class}, then the
* directory is treated as an exploded module rather than a directory of * directory is treated as an exploded module rather than a directory of

View File

@ -48,7 +48,11 @@ import java.util.stream.Stream;
* <p> A resource in a module is identified by an abstract name that is a * <p> A resource in a module is identified by an abstract name that is a
* '{@code /}'-separated path string. For example, module {@code java.base} may * '{@code /}'-separated path string. For example, module {@code java.base} may
* have a resource "{@code java/lang/Object.class}" that, by convention, is the * have a resource "{@code java/lang/Object.class}" that, by convention, is the
* class file for {@code java.lang.Object}. </p> * class file for {@code java.lang.Object}. A module reader may treat
* directories in the module content as resources (whether it does or not is
* module reader specific). Where the module content contains a directory
* that can be located as a resource then its name ends with a slash ('/'). The
* directory can also be located with a name that drops the trailing slash. </p>
* *
* <p> A {@code ModuleReader} is {@linkplain ModuleReference#open open} upon * <p> A {@code ModuleReader} is {@linkplain ModuleReference#open open} upon
* creation and is closed by invoking the {@link #close close} method. Failure * creation and is closed by invoking the {@link #close close} method. Failure
@ -80,6 +84,9 @@ public interface ModuleReader extends Closeable {
/** /**
* Finds a resource, returning a URI to the resource in the module. * Finds a resource, returning a URI to the resource in the module.
* *
* <p> If the module reader can determine that the name locates a directory
* then the resulting URI will end with a slash ('/'). </p>
*
* @param name * @param name
* The name of the resource to open for reading * The name of the resource to open for reading
* *
@ -140,7 +147,7 @@ public interface ModuleReader extends Closeable {
* *
* @apiNote This method is intended for high-performance class loading. It * @apiNote This method is intended for high-performance class loading. It
* is not capable (or intended) to read arbitrary large resources that * is not capable (or intended) to read arbitrary large resources that
* could potentially be 2GB or larger. The rational for using this method * could potentially be 2GB or larger. The rationale for using this method
* in conjunction with the {@code release} method is to allow module reader * in conjunction with the {@code release} method is to allow module reader
* implementations manage buffers in an efficient manner. * implementations manage buffers in an efficient manner.
* *
@ -195,7 +202,9 @@ public interface ModuleReader extends Closeable {
/** /**
* Lists the contents of the module, returning a stream of elements that * Lists the contents of the module, returning a stream of elements that
* are the names of all resources in the module. * are the names of all resources in the module. Whether the stream of
* elements includes names corresponding to directories in the module is
* module reader specific.
* *
* <p> In lazy implementations then an {@code IOException} may be thrown * <p> In lazy implementations then an {@code IOException} may be thrown
* when using the stream to list the module contents. If this occurs then * when using the stream to list the module contents. If this occurs then

View File

@ -47,6 +47,7 @@ import java.util.stream.Collectors;
import jdk.internal.module.ModuleHashes; import jdk.internal.module.ModuleHashes;
import jdk.internal.module.ModuleReferenceImpl; import jdk.internal.module.ModuleReferenceImpl;
import jdk.internal.module.ModuleTarget;
/** /**
* The resolver used by {@link Configuration#resolve} and {@link * The resolver used by {@link Configuration#resolve} and {@link
@ -69,11 +70,9 @@ final class Resolver {
// module constraints on target platform // module constraints on target platform
private String osName; private String osName;
private String osArch; private String osArch;
private String osVersion;
String osName() { return osName; } String osName() { return osName; }
String osArch() { return osArch; } String osArch() { return osArch; }
String osVersion() { return osVersion; }
/** /**
* @throws IllegalArgumentException if there are more than one parent and * @throws IllegalArgumentException if there are more than one parent and
@ -110,16 +109,6 @@ final class Resolver {
} }
} }
} }
value = parent.osVersion();
if (value != null) {
if (osVersion == null) {
osVersion = value;
} else {
if (!value.equals(osVersion)) {
failParentConflict("OS version", osVersion, value);
}
}
}
} }
} }
@ -318,13 +307,15 @@ final class Resolver {
* the target platform with the constraints of other modules. * the target platform with the constraints of other modules.
*/ */
private void addFoundModule(ModuleReference mref) { private void addFoundModule(ModuleReference mref) {
ModuleDescriptor descriptor = mref.descriptor(); String mn = mref.descriptor().name();
nameToReference.put(descriptor.name(), mref);
if (descriptor.osName().isPresent() if (mref instanceof ModuleReferenceImpl) {
|| descriptor.osArch().isPresent() ModuleTarget target = ((ModuleReferenceImpl)mref).moduleTarget();
|| descriptor.osVersion().isPresent()) if (target != null)
checkTargetConstraints(descriptor); checkTargetConstraints(mn, target);
}
nameToReference.put(mn, mref);
} }
/** /**
@ -332,58 +323,44 @@ final class Resolver {
* conflict with the constraints of other modules resolved so far or * conflict with the constraints of other modules resolved so far or
* modules in parent configurations. * modules in parent configurations.
*/ */
private void checkTargetConstraints(ModuleDescriptor descriptor) { private void checkTargetConstraints(String mn, ModuleTarget target) {
String value = descriptor.osName().orElse(null); String value = target.osName();
if (value != null) { if (value != null) {
if (osName == null) { if (osName == null) {
osName = value; osName = value;
} else { } else {
if (!value.equals(osName)) { if (!value.equals(osName)) {
failTargetConstraint(descriptor); failTargetConstraint(mn, target);
} }
} }
} }
value = descriptor.osArch().orElse(null); value = target.osArch();
if (value != null) { if (value != null) {
if (osArch == null) { if (osArch == null) {
osArch = value; osArch = value;
} else { } else {
if (!value.equals(osArch)) { if (!value.equals(osArch)) {
failTargetConstraint(descriptor); failTargetConstraint(mn, target);
}
}
}
value = descriptor.osVersion().orElse(null);
if (value != null) {
if (osVersion == null) {
osVersion = value;
} else {
if (!value.equals(osVersion)) {
failTargetConstraint(descriptor);
} }
} }
} }
} }
private void failTargetConstraint(ModuleDescriptor md) { private void failTargetConstraint(String mn, ModuleTarget target) {
String s1 = targetAsString(osName, osArch, osVersion); String s1 = targetAsString(osName, osArch);
String s2 = targetAsString(md); String s2 = targetAsString(target.osName(), target.osArch());
findFail("Module %s has constraints on target platform that conflict" + findFail("Module %s has constraints on target platform (%s) that"
" with other modules: %s, %s", md.name(), s1, s2); + " conflict with other modules: %s", mn, s1, s2);
} }
private String targetAsString(ModuleDescriptor descriptor) { private String targetAsString(ModuleTarget target) {
String osName = descriptor.osName().orElse(null); return targetAsString(target.osName(), target.osArch());
String osArch = descriptor.osArch().orElse(null);
String osVersion = descriptor.osVersion().orElse(null);
return targetAsString(osName, osArch, osVersion);
} }
private String targetAsString(String osName, String osArch, String osVersion) { private String targetAsString(String osName, String osArch) {
return new StringJoiner("-") return new StringJoiner("-")
.add(Objects.toString(osName, "*")) .add(Objects.toString(osName, "*"))
.add(Objects.toString(osArch, "*")) .add(Objects.toString(osArch, "*"))
.add(Objects.toString(osVersion, "*"))
.toString(); .toString();
} }
@ -712,16 +689,30 @@ final class Resolver {
/** /**
* Checks the readability graph to ensure that no two modules export the * Checks the readability graph to ensure that
* same package to a module. This includes the case where module M has * <ol>
* a local package P and M reads another module that exports P to M. * <li><p> A module does not read two or more modules with the same name.
* Also checks the uses/provides of module M to ensure that it reads a * This includes the case where a module reads another another with the
* module that exports the package of the service type to M. * same name as itself. </p></li>
* <li><p> Two or more modules in the configuration don't export the same
* package to a module that reads both. This includes the case where a
* module {@code M} containing package {@code p} reads another module
* that exports {@code p} to {@code M}. </p></li>
* <li><p> A module {@code M} doesn't declare that it "{@code uses p.S}"
* or "{@code provides p.S with ...}" but package {@code p} is neither
* in module {@code M} nor exported to {@code M} by any module that
* {@code M} reads. </p></li>
* </ol>
*/ */
private void checkExportSuppliers(Map<ResolvedModule, Set<ResolvedModule>> graph) { private void checkExportSuppliers(Map<ResolvedModule, Set<ResolvedModule>> graph) {
for (Map.Entry<ResolvedModule, Set<ResolvedModule>> e : graph.entrySet()) { for (Map.Entry<ResolvedModule, Set<ResolvedModule>> e : graph.entrySet()) {
ModuleDescriptor descriptor1 = e.getKey().descriptor(); ModuleDescriptor descriptor1 = e.getKey().descriptor();
String name1 = descriptor1.name();
// the names of the modules that are read (including self)
Set<String> names = new HashSet<>();
names.add(name1);
// the map of packages that are local or exported to descriptor1 // the map of packages that are local or exported to descriptor1
Map<String, ModuleDescriptor> packageToExporter = new HashMap<>(); Map<String, ModuleDescriptor> packageToExporter = new HashMap<>();
@ -737,6 +728,17 @@ final class Resolver {
for (ResolvedModule endpoint : reads) { for (ResolvedModule endpoint : reads) {
ModuleDescriptor descriptor2 = endpoint.descriptor(); ModuleDescriptor descriptor2 = endpoint.descriptor();
String name2 = descriptor2.name();
if (descriptor2 != descriptor1 && !names.add(name2)) {
if (name2.equals(name1)) {
resolveFail("Module %s reads another module named %s",
name1, name1);
} else{
resolveFail("Module %s reads more than one module named %s",
name1, name2);
}
}
if (descriptor2.isAutomatic()) { if (descriptor2.isAutomatic()) {
// automatic modules read self and export all packages // automatic modules read self and export all packages
if (descriptor2 != descriptor1) { if (descriptor2 != descriptor1) {

View File

@ -28,9 +28,12 @@ package java.lang.reflect;
import java.lang.annotation.Annotation; import java.lang.annotation.Annotation;
import java.security.AccessController; import java.security.AccessController;
import jdk.internal.misc.VM;
import jdk.internal.module.IllegalAccessLogger;
import jdk.internal.reflect.CallerSensitive; import jdk.internal.reflect.CallerSensitive;
import jdk.internal.reflect.Reflection; import jdk.internal.reflect.Reflection;
import jdk.internal.reflect.ReflectionFactory; import jdk.internal.reflect.ReflectionFactory;
import sun.security.action.GetPropertyAction;
/** /**
* The {@code AccessibleObject} class is the base class for {@code Field}, * The {@code AccessibleObject} class is the base class for {@code Field},
@ -288,27 +291,20 @@ public class AccessibleObject implements AnnotatedElement {
if (callerModule == Object.class.getModule()) return true; if (callerModule == Object.class.getModule()) return true;
if (!declaringModule.isNamed()) return true; if (!declaringModule.isNamed()) return true;
// package is open to caller String pn = declaringClass.getPackageName();
String pn = packageName(declaringClass);
if (declaringModule.isOpen(pn, callerModule)) {
dumpStackIfOpenedReflectively(declaringModule, pn, callerModule);
return true;
}
// package is exported to caller
boolean isExported = declaringModule.isExported(pn, callerModule);
boolean isClassPublic = Modifier.isPublic(declaringClass.getModifiers());
int modifiers; int modifiers;
if (this instanceof Executable) { if (this instanceof Executable) {
modifiers = ((Executable) this).getModifiers(); modifiers = ((Executable) this).getModifiers();
} else { } else {
modifiers = ((Field) this).getModifiers(); modifiers = ((Field) this).getModifiers();
} }
if (isExported && isClassPublic) {
// class is public and package is exported to caller
boolean isClassPublic = Modifier.isPublic(declaringClass.getModifiers());
if (isClassPublic && declaringModule.isExported(pn, callerModule)) {
// member is public // member is public
if (Modifier.isPublic(modifiers)) { if (Modifier.isPublic(modifiers)) {
dumpStackIfExportedReflectively(declaringModule, pn, callerModule); logIfExportedByBackdoor(caller, declaringClass);
return true; return true;
} }
@ -316,11 +312,17 @@ public class AccessibleObject implements AnnotatedElement {
if (Modifier.isProtected(modifiers) if (Modifier.isProtected(modifiers)
&& Modifier.isStatic(modifiers) && Modifier.isStatic(modifiers)
&& isSubclassOf(caller, declaringClass)) { && isSubclassOf(caller, declaringClass)) {
dumpStackIfExportedReflectively(declaringModule, pn, callerModule); logIfExportedByBackdoor(caller, declaringClass);
return true; return true;
} }
} }
// package is open to caller
if (declaringModule.isOpen(pn, callerModule)) {
logIfOpenedByBackdoor(caller, declaringClass);
return true;
}
if (throwExceptionIfDenied) { if (throwExceptionIfDenied) {
// not accessible // not accessible
String msg = "Unable to make "; String msg = "Unable to make ";
@ -333,7 +335,7 @@ public class AccessibleObject implements AnnotatedElement {
msg += "opens"; msg += "opens";
msg += " " + pn + "\" to " + callerModule; msg += " " + pn + "\" to " + callerModule;
InaccessibleObjectException e = new InaccessibleObjectException(msg); InaccessibleObjectException e = new InaccessibleObjectException(msg);
if (Reflection.printStackTraceWhenAccessFails()) { if (printStackTraceWhenAccessFails()) {
e.printStackTrace(System.err); e.printStackTrace(System.err);
} }
throw e; throw e;
@ -351,48 +353,35 @@ public class AccessibleObject implements AnnotatedElement {
return false; return false;
} }
private void dumpStackIfOpenedReflectively(Module module, private void logIfOpenedByBackdoor(Class<?> caller, Class<?> declaringClass) {
String pn, Module callerModule = caller.getModule();
Module other) { Module targetModule = declaringClass.getModule();
dumpStackIfExposedReflectively(module, pn, other, true); // callerModule is null during early startup
if (callerModule != null && !callerModule.isNamed() && targetModule.isNamed()) {
IllegalAccessLogger logger = IllegalAccessLogger.illegalAccessLogger();
if (logger != null) {
logger.logIfOpenedByBackdoor(caller, declaringClass, this::toShortString);
}
}
} }
private void dumpStackIfExportedReflectively(Module module, private void logIfExportedByBackdoor(Class<?> caller, Class<?> declaringClass) {
String pn, Module callerModule = caller.getModule();
Module other) { Module targetModule = declaringClass.getModule();
dumpStackIfExposedReflectively(module, pn, other, false); // callerModule is null during early startup
if (callerModule != null && !callerModule.isNamed() && targetModule.isNamed()) {
IllegalAccessLogger logger = IllegalAccessLogger.illegalAccessLogger();
if (logger != null) {
logger.logIfExportedByBackdoor(caller, declaringClass, this::toShortString);
} }
private void dumpStackIfExposedReflectively(Module module,
String pn,
Module other,
boolean open)
{
if (Reflection.printStackTraceWhenAccessSucceeds()
&& !module.isStaticallyExportedOrOpen(pn, other, open))
{
String msg = other + " allowed to invoke setAccessible on ";
if (this instanceof Field)
msg += "field ";
msg += this;
new Exception(msg) {
private static final long serialVersionUID = 42L;
public String toString() {
return "WARNING: " + getMessage();
}
}.printStackTrace(System.err);
} }
} }
/** /**
* Returns the package name of the given class. * Returns a short descriptive string to describe this object in log messages.
*/ */
private static String packageName(Class<?> c) { String toShortString() {
while (c.isArray()) { return toString();
c = c.getComponentType();
}
String pn = c.getPackageName();
return (pn != null) ? pn : "";
} }
/** /**
@ -409,6 +398,7 @@ public class AccessibleObject implements AnnotatedElement {
* it should use {@link #canAccess(Object)}. * it should use {@link #canAccess(Object)}.
* *
* @revised 9 * @revised 9
* @spec JPMS
*/ */
@Deprecated(since="9") @Deprecated(since="9")
public boolean isAccessible() { public boolean isAccessible() {
@ -483,10 +473,7 @@ public class AccessibleObject implements AnnotatedElement {
} else { } else {
targetClass = Modifier.isStatic(modifiers) ? null : obj.getClass(); targetClass = Modifier.isStatic(modifiers) ? null : obj.getClass();
} }
return Reflection.verifyMemberAccess(caller, return verifyAccess(caller, declaringClass, targetClass, modifiers);
declaringClass,
targetClass,
modifiers);
} }
/** /**
@ -597,9 +584,22 @@ public class AccessibleObject implements AnnotatedElement {
final void checkAccess(Class<?> caller, Class<?> memberClass, final void checkAccess(Class<?> caller, Class<?> memberClass,
Class<?> targetClass, int modifiers) Class<?> targetClass, int modifiers)
throws IllegalAccessException throws IllegalAccessException
{
if (!verifyAccess(caller, memberClass, targetClass, modifiers)) {
IllegalAccessException e = Reflection.newIllegalAccessException(
caller, memberClass, targetClass, modifiers);
if (printStackTraceWhenAccessFails()) {
e.printStackTrace(System.err);
}
throw e;
}
}
final boolean verifyAccess(Class<?> caller, Class<?> memberClass,
Class<?> targetClass, int modifiers)
{ {
if (caller == memberClass) { // quick check if (caller == memberClass) { // quick check
return; // ACCESS IS OK return true; // ACCESS IS OK
} }
Object cache = securityCheckCache; // read volatile Object cache = securityCheckCache; // read volatile
if (targetClass != null // instance member or constructor if (targetClass != null // instance member or constructor
@ -610,26 +610,31 @@ public class AccessibleObject implements AnnotatedElement {
Class<?>[] cache2 = (Class<?>[]) cache; Class<?>[] cache2 = (Class<?>[]) cache;
if (cache2[1] == targetClass && if (cache2[1] == targetClass &&
cache2[0] == caller) { cache2[0] == caller) {
return; // ACCESS IS OK return true; // ACCESS IS OK
} }
// (Test cache[1] first since range check for [1] // (Test cache[1] first since range check for [1]
// subsumes range check for [0].) // subsumes range check for [0].)
} }
} else if (cache == caller) { } else if (cache == caller) {
// Non-protected case (or targetClass == memberClass or static member). // Non-protected case (or targetClass == memberClass or static member).
return; // ACCESS IS OK return true; // ACCESS IS OK
} }
// If no return, fall through to the slow path. // If no return, fall through to the slow path.
slowCheckMemberAccess(caller, memberClass, targetClass, modifiers); return slowVerifyAccess(caller, memberClass, targetClass, modifiers);
} }
// Keep all this slow stuff out of line: // Keep all this slow stuff out of line:
void slowCheckMemberAccess(Class<?> caller, Class<?> memberClass, private boolean slowVerifyAccess(Class<?> caller, Class<?> memberClass,
Class<?> targetClass, int modifiers) Class<?> targetClass, int modifiers)
throws IllegalAccessException
{ {
Reflection.ensureMemberAccess(caller, memberClass, targetClass, modifiers); if (!Reflection.verifyMemberAccess(caller, memberClass, targetClass, modifiers)) {
// access denied
return false;
}
// access okay
logIfExportedByBackdoor(caller, memberClass);
// Success: Update the cache. // Success: Update the cache.
Object cache = (targetClass != null Object cache = (targetClass != null
@ -643,5 +648,27 @@ public class AccessibleObject implements AnnotatedElement {
// guarantees that the initializing stores for the cache // guarantees that the initializing stores for the cache
// elements will occur before the volatile write. // elements will occur before the volatile write.
securityCheckCache = cache; // write volatile securityCheckCache = cache; // write volatile
return true;
}
// true to print a stack trace when access fails
private static volatile boolean printStackWhenAccessFails;
// true if printStack* values are initialized
private static volatile boolean printStackPropertiesSet;
/**
* Returns true if a stack trace should be printed when access fails.
*/
private static boolean printStackTraceWhenAccessFails() {
if (!printStackPropertiesSet && VM.initLevel() >= 1) {
String s = GetPropertyAction.privilegedGetProperty(
"sun.reflect.debugModuleAccessChecks");
if (s != null) {
printStackWhenAccessFails = !s.equalsIgnoreCase("false");
}
printStackPropertiesSet = true;
}
return printStackWhenAccessFails;
} }
} }

View File

@ -38,6 +38,7 @@ import sun.reflect.generics.factory.GenericsFactory;
import sun.reflect.generics.scope.ConstructorScope; import sun.reflect.generics.scope.ConstructorScope;
import java.lang.annotation.Annotation; import java.lang.annotation.Annotation;
import java.lang.annotation.AnnotationFormatError; import java.lang.annotation.AnnotationFormatError;
import java.util.StringJoiner;
/** /**
* {@code Constructor} provides information about, and access to, a single * {@code Constructor} provides information about, and access to, a single
@ -360,6 +361,20 @@ public final class Constructor<T> extends Executable {
sb.append(getDeclaringClass().getTypeName()); sb.append(getDeclaringClass().getTypeName());
} }
@Override
String toShortString() {
StringBuilder sb = new StringBuilder("constructor ");
sb.append(getDeclaringClass().getTypeName());
sb.append('(');
StringJoiner sj = new StringJoiner(",");
for (Class<?> parameterType : getParameterTypes()) {
sj.add(parameterType.getTypeName());
}
sb.append(sj);
sb.append(')');
return sb.toString();
}
/** /**
* Returns a string describing this {@code Constructor}, * Returns a string describing this {@code Constructor},
* including type parameters. The string is formatted as the * including type parameters. The string is formatted as the

View File

@ -324,6 +324,11 @@ class Field extends AccessibleObject implements Member {
+ getName()); + getName());
} }
@Override
String toShortString() {
return "field " + getDeclaringClass().getTypeName() + "." + getName();
}
/** /**
* Returns a string describing this {@code Field}, including * Returns a string describing this {@code Field}, including
* its generic type. The format is the access modifiers for the * its generic type. The format is the access modifiers for the

View File

@ -66,9 +66,7 @@ import sun.security.util.SecurityConstants;
* ResolvedModule} in the configuration. For each resolved module that is * ResolvedModule} in the configuration. For each resolved module that is
* {@link ResolvedModule#reads() read}, the {@code Module} {@link * {@link ResolvedModule#reads() read}, the {@code Module} {@link
* Module#canRead reads} the corresponding run-time {@code Module}, which may * Module#canRead reads} the corresponding run-time {@code Module}, which may
* be in the same layer or a {@link #parents() parent} layer. The {@code Module} * be in the same layer or a {@link #parents() parent} layer. </p>
* {@link Module#isExported(String) exports} and {@link Module#isOpen(String)
* opens} the packages described by its {@link ModuleDescriptor}. </p>
* *
* <p> The {@link #defineModulesWithOneLoader defineModulesWithOneLoader} and * <p> The {@link #defineModulesWithOneLoader defineModulesWithOneLoader} and
* {@link #defineModulesWithManyLoaders defineModulesWithManyLoaders} methods * {@link #defineModulesWithManyLoaders defineModulesWithManyLoaders} methods
@ -91,6 +89,28 @@ import sun.security.util.SecurityConstants;
* built-in</a> into the Java virtual machine. The boot layer will often be * built-in</a> into the Java virtual machine. The boot layer will often be
* the {@link #parents() parent} when creating additional layers. </p> * the {@link #parents() parent} when creating additional layers. </p>
* *
* <p> Each {@code Module} in a layer is created so that it {@link
* Module#isExported(String) exports} and {@link Module#isOpen(String) opens}
* the packages described by its {@link ModuleDescriptor}. Qualified exports
* (where a package is exported to a set of target modules rather than all
* modules) are reified when creating the layer as follows: </p>
* <ul>
* <li> If module {@code X} exports a package to {@code Y}, and if the
* runtime {@code Module} {@code X} reads {@code Module} {@code Y}, then
* the package is exported to {@code Module} {@code Y} (which may be in
* the same layer as {@code X} or a parent layer). </li>
*
* <li> If module {@code X} exports a package to {@code Y}, and if the
* runtime {@code Module} {@code X} does not read {@code Y} then target
* {@code Y} is located as if by invoking {@link #findModule(String)
* findModule} to find the module in the layer or its parent layers. If
* {@code Y} is found then the package is exported to the instance of
* {@code Y} that was found. If {@code Y} is not found then the qualified
* export is ignored. </li>
* </ul>
*
* <p> Qualified opens are handled in same way as qualified exports. </p>
*
* <p> As when creating a {@code Configuration}, * <p> As when creating a {@code Configuration},
* {@link ModuleDescriptor#isAutomatic() automatic} modules receive special * {@link ModuleDescriptor#isAutomatic() automatic} modules receive special
* treatment when creating a layer. An automatic module is created in the * treatment when creating a layer. An automatic module is created in the
@ -193,7 +213,7 @@ public final class Layer {
} }
private void ensureInLayer(Module source) { private void ensureInLayer(Module source) {
if (!layer.modules().contains(source)) if (source.getLayer() != layer)
throw new IllegalArgumentException(source + " not in layer"); throw new IllegalArgumentException(source + " not in layer");
} }
@ -220,9 +240,8 @@ public final class Layer {
* @see Module#addReads * @see Module#addReads
*/ */
public Controller addReads(Module source, Module target) { public Controller addReads(Module source, Module target) {
Objects.requireNonNull(source);
Objects.requireNonNull(target);
ensureInLayer(source); ensureInLayer(source);
Objects.requireNonNull(target);
Modules.addReads(source, target); Modules.addReads(source, target);
return this; return this;
} }
@ -248,9 +267,9 @@ public final class Layer {
* @see Module#addOpens * @see Module#addOpens
*/ */
public Controller addOpens(Module source, String pn, Module target) { public Controller addOpens(Module source, String pn, Module target) {
Objects.requireNonNull(source);
Objects.requireNonNull(target);
ensureInLayer(source); ensureInLayer(source);
Objects.requireNonNull(pn);
Objects.requireNonNull(target);
Modules.addOpens(source, pn, target); Modules.addOpens(source, pn, target);
return this; return this;
} }
@ -408,8 +427,8 @@ public final class Layer {
* </ul> * </ul>
* *
* <p> In addition, a layer cannot be created if the configuration contains * <p> In addition, a layer cannot be created if the configuration contains
* a module named "{@code java.base}" or a module with a package name * a module named "{@code java.base}", or a module contains a package named
* starting with "{@code java.}". </p> * "{@code java}" or a package with a name starting with "{@code java.}". </p>
* *
* <p> If there is a security manager then the class loader created by * <p> If there is a security manager then the class loader created by
* this method will load classes and resources with privileges that are * this method will load classes and resources with privileges that are
@ -418,7 +437,7 @@ public final class Layer {
* @param cf * @param cf
* The configuration for the layer * The configuration for the layer
* @param parentLayers * @param parentLayers
* The list parent layers in search order * The list of parent layers in search order
* @param parentLoader * @param parentLoader
* The parent class loader for the class loader created by this * The parent class loader for the class loader created by this
* method; may be {@code null} for the bootstrap class loader * method; may be {@code null} for the bootstrap class loader
@ -485,7 +504,7 @@ public final class Layer {
* @param cf * @param cf
* The configuration for the layer * The configuration for the layer
* @param parentLayers * @param parentLayers
* The list parent layers in search order * The list of parent layers in search order
* @param parentLoader * @param parentLoader
* The parent class loader for each of the class loaders created by * The parent class loader for each of the class loaders created by
* this method; may be {@code null} for the bootstrap class loader * this method; may be {@code null} for the bootstrap class loader
@ -497,8 +516,10 @@ public final class Layer {
* the parent layers, including order * the parent layers, including order
* @throws LayerInstantiationException * @throws LayerInstantiationException
* If the layer cannot be created because the configuration contains * If the layer cannot be created because the configuration contains
* a module named "{@code java.base}" or a module with a package * a module named "{@code java.base}" or a module contains a package
* name starting with "{@code java.}" * named "{@code java}" or a package with a name starting with
* "{@code java.}"
*
* @throws SecurityException * @throws SecurityException
* If {@code RuntimePermission("createClassLoader")} or * If {@code RuntimePermission("createClassLoader")} or
* {@code RuntimePermission("getClassLoader")} is denied by * {@code RuntimePermission("getClassLoader")} is denied by
@ -558,10 +579,11 @@ public final class Layer {
* *
* <p> In addition, a layer cannot be created if the configuration contains * <p> In addition, a layer cannot be created if the configuration contains
* a module named "{@code java.base}", a configuration contains a module * a module named "{@code java.base}", a configuration contains a module
* with a package name starting with "{@code java.}" is mapped to a class * with a package named "{@code java}" or a package name starting with
* loader other than the {@link ClassLoader#getPlatformClassLoader() * "{@code java.}" and the module is mapped to a class loader other than
* platform class loader}, or the function to map a module name to a class * the {@link ClassLoader#getPlatformClassLoader() platform class loader},
* loader returns {@code null}. </p> * or the function to map a module name to a class loader returns
* {@code null}. </p>
* *
* <p> If the function to map a module name to class loader throws an error * <p> If the function to map a module name to class loader throws an error
* or runtime exception then it is propagated to the caller of this method. * or runtime exception then it is propagated to the caller of this method.
@ -575,7 +597,7 @@ public final class Layer {
* @param cf * @param cf
* The configuration for the layer * The configuration for the layer
* @param parentLayers * @param parentLayers
* The list parent layers in search order * The list of parent layers in search order
* @param clf * @param clf
* The function to map a module name to a class loader * The function to map a module name to a class loader
* *
@ -754,9 +776,15 @@ public final class Layer {
* @return A possibly-empty unmodifiable set of the modules in this layer * @return A possibly-empty unmodifiable set of the modules in this layer
*/ */
public Set<Module> modules() { public Set<Module> modules() {
return Collections.unmodifiableSet( Set<Module> modules = this.modules;
nameToModule.values().stream().collect(Collectors.toSet())); if (modules == null) {
this.modules = modules =
Collections.unmodifiableSet(new HashSet<>(nameToModule.values()));
} }
return modules;
}
private volatile Set<Module> modules;
/** /**
@ -776,6 +804,8 @@ public final class Layer {
*/ */
public Optional<Module> findModule(String name) { public Optional<Module> findModule(String name) {
Objects.requireNonNull(name); Objects.requireNonNull(name);
if (this == EMPTY_LAYER)
return Optional.empty();
Module m = nameToModule.get(name); Module m = nameToModule.get(name);
if (m != null) if (m != null)
return Optional.of(m); return Optional.of(m);

View File

@ -42,6 +42,7 @@ import sun.reflect.annotation.AnnotationParser;
import java.lang.annotation.Annotation; import java.lang.annotation.Annotation;
import java.lang.annotation.AnnotationFormatError; import java.lang.annotation.AnnotationFormatError;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.util.StringJoiner;
/** /**
* A {@code Method} provides information about, and access to, a single method * A {@code Method} provides information about, and access to, a single method
@ -416,6 +417,21 @@ public final class Method extends Executable {
sb.append(getName()); sb.append(getName());
} }
@Override
String toShortString() {
StringBuilder sb = new StringBuilder("method ");
sb.append(getDeclaringClass().getTypeName()).append('.');
sb.append(getName());
sb.append('(');
StringJoiner sj = new StringJoiner(",");
for (Class<?> parameterType : getParameterTypes()) {
sj.add(parameterType.getTypeName());
}
sb.append(sj);
sb.append(')');
return sb.toString();
}
/** /**
* Returns a string describing this {@code Method}, including * Returns a string describing this {@code Method}, including
* type parameters. The string is formatted as the method access * type parameters. The string is formatted as the method access

View File

@ -39,8 +39,10 @@ import java.net.URI;
import java.net.URL; import java.net.URL;
import java.security.AccessController; import java.security.AccessController;
import java.security.PrivilegedAction; import java.security.PrivilegedAction;
import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.Optional; import java.util.Optional;
@ -51,11 +53,11 @@ import java.util.stream.Stream;
import jdk.internal.loader.BuiltinClassLoader; import jdk.internal.loader.BuiltinClassLoader;
import jdk.internal.loader.BootLoader; import jdk.internal.loader.BootLoader;
import jdk.internal.loader.ResourceHelper;
import jdk.internal.misc.JavaLangAccess; import jdk.internal.misc.JavaLangAccess;
import jdk.internal.misc.JavaLangReflectModuleAccess; import jdk.internal.misc.JavaLangReflectModuleAccess;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.ServicesCatalog; import jdk.internal.module.ServicesCatalog;
import jdk.internal.module.Resources;
import jdk.internal.org.objectweb.asm.AnnotationVisitor; import jdk.internal.org.objectweb.asm.AnnotationVisitor;
import jdk.internal.org.objectweb.asm.Attribute; import jdk.internal.org.objectweb.asm.Attribute;
import jdk.internal.org.objectweb.asm.ClassReader; import jdk.internal.org.objectweb.asm.ClassReader;
@ -369,17 +371,7 @@ public final class Module implements AnnotatedElement {
* If {@code syncVM} is {@code true} then the VM is notified. * If {@code syncVM} is {@code true} then the VM is notified.
*/ */
private void implAddReads(Module other, boolean syncVM) { private void implAddReads(Module other, boolean syncVM) {
Objects.requireNonNull(other); if (!canRead(other)) {
// nothing to do
if (other == this || !this.isNamed())
return;
// check if we already read this module
Set<Module> reads = this.reads;
if (reads != null && reads.contains(other))
return;
// update VM first, just in case it fails // update VM first, just in case it fails
if (syncVM) { if (syncVM) {
if (other == ALL_UNNAMED_MODULE) { if (other == ALL_UNNAMED_MODULE) {
@ -392,6 +384,7 @@ public final class Module implements AnnotatedElement {
// add reflective read // add reflective read
reflectivelyReads.putIfAbsent(this, other, Boolean.TRUE); reflectivelyReads.putIfAbsent(this, other, Boolean.TRUE);
} }
}
// -- exported and open packages -- // -- exported and open packages --
@ -553,7 +546,7 @@ public final class Module implements AnnotatedElement {
* Returns {@code true} if this module exports or opens a package to * Returns {@code true} if this module exports or opens a package to
* the given module via its module declaration. * the given module via its module declaration.
*/ */
boolean isStaticallyExportedOrOpen(String pn, Module other, boolean open) { private boolean isStaticallyExportedOrOpen(String pn, Module other, boolean open) {
// package is open to everyone or <other> // package is open to everyone or <other>
Map<String, Set<Module>> openPackages = this.openPackages; Map<String, Set<Module>> openPackages = this.openPackages;
if (openPackages != null) { if (openPackages != null) {
@ -909,9 +902,7 @@ public final class Module implements AnnotatedElement {
* Returns an array of the package names of the packages in this module. * Returns an array of the package names of the packages in this module.
* *
* <p> For named modules, the returned array contains an element for each * <p> For named modules, the returned array contains an element for each
* package in the module. It may contain elements corresponding to packages * package in the module. </p>
* added to the module, <a href="Proxy.html#dynamicmodule">dynamic modules</a>
* for example, after it was loaded.
* *
* <p> For unnamed modules, this method is the equivalent to invoking the * <p> For unnamed modules, this method is the equivalent to invoking the
* {@link ClassLoader#getDefinedPackages() getDefinedPackages} method of * {@link ClassLoader#getDefinedPackages() getDefinedPackages} method of
@ -949,15 +940,6 @@ public final class Module implements AnnotatedElement {
} }
} }
/**
* Add a package to this module.
*
* @apiNote This method is for Proxy use.
*/
void addPackage(String pn) {
implAddPackage(pn, true);
}
/** /**
* Add a package to this module without notifying the VM. * Add a package to this module without notifying the VM.
* *
@ -1080,20 +1062,28 @@ public final class Module implements AnnotatedElement {
// reads // reads
Set<Module> reads = new HashSet<>(); Set<Module> reads = new HashSet<>();
// name -> source Module when in parent layer
Map<String, Module> nameToSource = Collections.emptyMap();
for (ResolvedModule other : resolvedModule.reads()) { for (ResolvedModule other : resolvedModule.reads()) {
Module m2 = null; Module m2 = null;
if (other.configuration() == cf) { if (other.configuration() == cf) {
String dn = other.reference().descriptor().name(); // this configuration
m2 = nameToModule.get(dn); m2 = nameToModule.get(other.name());
assert m2 != null;
} else { } else {
// parent layer
for (Layer parent: layer.parents()) { for (Layer parent: layer.parents()) {
m2 = findModule(parent, other); m2 = findModule(parent, other);
if (m2 != null) if (m2 != null)
break; break;
} }
}
assert m2 != null; assert m2 != null;
if (nameToSource.isEmpty())
nameToSource = new HashMap<>();
nameToSource.put(other.name(), m2);
}
reads.add(m2); reads.add(m2);
// update VM view // update VM view
@ -1107,7 +1097,7 @@ public final class Module implements AnnotatedElement {
} }
// exports and opens // exports and opens
initExportsAndOpens(descriptor, nameToModule, m); initExportsAndOpens(m, nameToSource, nameToModule, layer.parents());
} }
// register the modules in the boot layer // register the modules in the boot layer
@ -1159,15 +1149,17 @@ public final class Module implements AnnotatedElement {
.orElse(null); .orElse(null);
} }
/** /**
* Initialize the maps of exported and open packages for module m. * Initialize the maps of exported and open packages for module m.
*/ */
private static void initExportsAndOpens(ModuleDescriptor descriptor, private static void initExportsAndOpens(Module m,
Map<String, Module> nameToSource,
Map<String, Module> nameToModule, Map<String, Module> nameToModule,
Module m) List<Layer> parents) {
{
// The VM doesn't special case open or automatic modules so need to // The VM doesn't special case open or automatic modules so need to
// export all packages // export all packages
ModuleDescriptor descriptor = m.getDescriptor();
if (descriptor.isOpen() || descriptor.isAutomatic()) { if (descriptor.isOpen() || descriptor.isAutomatic()) {
assert descriptor.opens().isEmpty(); assert descriptor.opens().isEmpty();
for (String source : descriptor.packages()) { for (String source : descriptor.packages()) {
@ -1187,8 +1179,7 @@ public final class Module implements AnnotatedElement {
// qualified opens // qualified opens
Set<Module> targets = new HashSet<>(); Set<Module> targets = new HashSet<>();
for (String target : opens.targets()) { for (String target : opens.targets()) {
// only open to modules that are in this configuration Module m2 = findModule(target, nameToSource, nameToModule, parents);
Module m2 = nameToModule.get(target);
if (m2 != null) { if (m2 != null) {
addExports0(m, source, m2); addExports0(m, source, m2);
targets.add(m2); targets.add(m2);
@ -1217,8 +1208,7 @@ public final class Module implements AnnotatedElement {
// qualified exports // qualified exports
Set<Module> targets = new HashSet<>(); Set<Module> targets = new HashSet<>();
for (String target : exports.targets()) { for (String target : exports.targets()) {
// only export to modules that are in this configuration Module m2 = findModule(target, nameToSource, nameToModule, parents);
Module m2 = nameToModule.get(target);
if (m2 != null) { if (m2 != null) {
// skip qualified export if already open to m2 // skip qualified export if already open to m2
if (openToTargets == null || !openToTargets.contains(m2)) { if (openToTargets == null || !openToTargets.contains(m2)) {
@ -1244,6 +1234,32 @@ public final class Module implements AnnotatedElement {
m.exportedPackages = exportedPackages; m.exportedPackages = exportedPackages;
} }
/**
* Find the runtime Module with the given name. The module name is the
* name of a target module in a qualified exports or opens directive.
*
* @param target The target module to find
* @param nameToSource The modules in parent layers that are read
* @param nameToModule The modules in the layer under construction
* @param parents The parent layers
*/
private static Module findModule(String target,
Map<String, Module> nameToSource,
Map<String, Module> nameToModule,
List<Layer> parents) {
Module m = nameToSource.get(target);
if (m == null) {
m = nameToModule.get(target);
if (m == null) {
for (Layer parent : parents) {
m = parent.findModule(target).orElse(null);
if (m != null) break;
}
}
}
return m;
}
// -- annotations -- // -- annotations --
@ -1428,12 +1444,12 @@ public final class Module implements AnnotatedElement {
name = name.substring(1); name = name.substring(1);
} }
if (isNamed() && !ResourceHelper.isSimpleResource(name)) { if (isNamed() && Resources.canEncapsulate(name)) {
Module caller = Reflection.getCallerClass().getModule(); Module caller = Reflection.getCallerClass().getModule();
if (caller != this && caller != Object.class.getModule()) { if (caller != this && caller != Object.class.getModule()) {
// ignore packages added for proxies via addPackage // ignore packages added for proxies via addPackage
Set<String> packages = getDescriptor().packages(); Set<String> packages = getDescriptor().packages();
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
if (packages.contains(pn) && !isOpen(pn, caller)) { if (packages.contains(pn) && !isOpen(pn, caller)) {
// resource is in package not open to caller // resource is in package not open to caller
return null; return null;
@ -1531,24 +1547,24 @@ public final class Module implements AnnotatedElement {
m.implAddReads(Module.ALL_UNNAMED_MODULE); m.implAddReads(Module.ALL_UNNAMED_MODULE);
} }
@Override @Override
public void addExports(Module m, String pn) {
m.implAddExportsOrOpens(pn, Module.EVERYONE_MODULE, false, true);
}
@Override
public void addExports(Module m, String pn, Module other) { public void addExports(Module m, String pn, Module other) {
m.implAddExportsOrOpens(pn, other, false, true); m.implAddExportsOrOpens(pn, other, false, true);
} }
@Override @Override
public void addOpens(Module m, String pn, Module other) { public void addExportsToAllUnnamed(Module m, String pn) {
m.implAddExportsOrOpens(pn, other, true, true); m.implAddExportsOrOpens(pn, Module.ALL_UNNAMED_MODULE, false, true);
} }
@Override @Override
public void addExportsToAll(Module m, String pn) { public void addOpens(Module m, String pn) {
m.implAddExportsOrOpens(pn, Module.EVERYONE_MODULE, false, true);
}
@Override
public void addOpensToAll(Module m, String pn) {
m.implAddExportsOrOpens(pn, Module.EVERYONE_MODULE, true, true); m.implAddExportsOrOpens(pn, Module.EVERYONE_MODULE, true, true);
} }
@Override @Override
public void addExportsToAllUnnamed(Module m, String pn) { public void addOpens(Module m, String pn, Module other) {
m.implAddExportsOrOpens(pn, Module.ALL_UNNAMED_MODULE, false, true); m.implAddExportsOrOpens(pn, other, true, true);
} }
@Override @Override
public void addOpensToAllUnnamed(Module m, String pn) { public void addOpensToAllUnnamed(Module m, String pn) {
@ -1559,10 +1575,6 @@ public final class Module implements AnnotatedElement {
m.implAddUses(service); m.implAddUses(service);
} }
@Override @Override
public void addPackage(Module m, String pn) {
m.implAddPackage(pn, true);
}
@Override
public ServicesCatalog getServicesCatalog(Layer layer) { public ServicesCatalog getServicesCatalog(Layer layer) {
return layer.getServicesCatalog(); return layer.getServicesCatalog();
} }
@ -1574,10 +1586,6 @@ public final class Module implements AnnotatedElement {
public Stream<Layer> layers(ClassLoader loader) { public Stream<Layer> layers(ClassLoader loader) {
return Layer.layers(loader); return Layer.layers(loader);
} }
@Override
public boolean isStaticallyExported(Module module, String pn, Module other) {
return module.isStaticallyExportedOrOpen(pn, other, false);
}
}); });
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1999, 2013, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -359,10 +359,11 @@ public class Proxy implements java.io.Serializable {
* @throws NullPointerException if the {@code interfaces} array * @throws NullPointerException if the {@code interfaces} array
* argument or any of its elements are {@code null} * argument or any of its elements are {@code null}
* *
* @deprecated Proxy classes generated in a named module are encapsulated and not * @deprecated Proxy classes generated in a named module are encapsulated
* accessible to code outside its module. * and not accessible to code outside its module.
* {@link Constructor#newInstance(Object...) Constructor.newInstance} will throw * {@link Constructor#newInstance(Object...) Constructor.newInstance}
* {@code IllegalAccessException} when it is called on an inaccessible proxy class. * will throw {@code IllegalAccessException} when it is called on
* an inaccessible proxy class.
* Use {@link #newProxyInstance(ClassLoader, Class[], InvocationHandler)} * Use {@link #newProxyInstance(ClassLoader, Class[], InvocationHandler)}
* to create a proxy instance instead. * to create a proxy instance instead.
* *
@ -511,16 +512,18 @@ public class Proxy implements java.io.Serializable {
"Unnamed package cannot be added to " + m); "Unnamed package cannot be added to " + m);
} }
// add the package to the runtime module if not exists
if (m.isNamed()) { if (m.isNamed()) {
m.addPackage(proxyPkg); if (!m.getDescriptor().packages().contains(proxyPkg)) {
throw new InternalError(proxyPkg + " not exist in " + m.getName());
}
} }
/* /*
* Choose a name for the proxy class to generate. * Choose a name for the proxy class to generate.
*/ */
long num = nextUniqueNumber.getAndIncrement(); long num = nextUniqueNumber.getAndIncrement();
String proxyName = proxyPkg.isEmpty() ? proxyClassNamePrefix + num String proxyName = proxyPkg.isEmpty()
? proxyClassNamePrefix + num
: proxyPkg + "." + proxyClassNamePrefix + num; : proxyPkg + "." + proxyClassNamePrefix + num;
ClassLoader loader = getLoader(m); ClassLoader loader = getLoader(m);
@ -581,9 +584,13 @@ public class Proxy implements java.io.Serializable {
c.getModule().getName(), c.getName(), access, ld); c.getModule().getName(), c.getName(), access, ld);
} }
static void trace(String cn, Module module, ClassLoader loader, List<Class<?>> interfaces) { static void trace(String cn,
Module module,
ClassLoader loader,
List<Class<?>> interfaces) {
if (isDebug()) { if (isDebug()) {
System.out.format("PROXY: %s/%s defined by %s%n", module.getName(), cn, loader); System.err.format("PROXY: %s/%s defined by %s%n",
module.getName(), cn, loader);
} }
if (isDebug("debug")) { if (isDebug("debug")) {
interfaces.stream() interfaces.stream()
@ -603,15 +610,16 @@ public class Proxy implements java.io.Serializable {
// ProxyBuilder instance members start here.... // ProxyBuilder instance members start here....
private final ClassLoader loader;
private final List<Class<?>> interfaces; private final List<Class<?>> interfaces;
private final Module module; private final Module module;
ProxyBuilder(ClassLoader loader, List<Class<?>> interfaces) { ProxyBuilder(ClassLoader loader, List<Class<?>> interfaces) {
if (!VM.isModuleSystemInited()) { if (!VM.isModuleSystemInited()) {
throw new InternalError("Proxy is not supported until module system is fully initialized"); throw new InternalError("Proxy is not supported until "
+ "module system is fully initialized");
} }
if (interfaces.size() > 65535) { if (interfaces.size() > 65535) {
throw new IllegalArgumentException("interface limit exceeded: " + interfaces.size()); throw new IllegalArgumentException("interface limit exceeded: "
+ interfaces.size());
} }
Set<Class<?>> refTypes = referencedTypes(loader, interfaces); Set<Class<?>> refTypes = referencedTypes(loader, interfaces);
@ -619,7 +627,6 @@ public class Proxy implements java.io.Serializable {
// IAE if violates any restrictions specified in newProxyInstance // IAE if violates any restrictions specified in newProxyInstance
validateProxyInterfaces(loader, interfaces, refTypes); validateProxyInterfaces(loader, interfaces, refTypes);
this.loader = loader;
this.interfaces = interfaces; this.interfaces = interfaces;
this.module = mapToModule(loader, interfaces, refTypes); this.module = mapToModule(loader, interfaces, refTypes);
assert getLoader(module) == loader; assert getLoader(module) == loader;
@ -659,8 +666,8 @@ public class Proxy implements java.io.Serializable {
* Validate the given proxy interfaces and the given referenced types * Validate the given proxy interfaces and the given referenced types
* are visible to the defining loader. * are visible to the defining loader.
* *
* @throws IllegalArgumentException if it violates the restrictions specified * @throws IllegalArgumentException if it violates the restrictions
* in {@link Proxy#newProxyInstance} * specified in {@link Proxy#newProxyInstance}
*/ */
private static void validateProxyInterfaces(ClassLoader loader, private static void validateProxyInterfaces(ClassLoader loader,
List<Class<?>> interfaces, List<Class<?>> interfaces,
@ -731,9 +738,9 @@ public class Proxy implements java.io.Serializable {
* is in the same module of the package-private interface. * is in the same module of the package-private interface.
* *
* If all proxy interfaces are public and at least one in a non-exported * If all proxy interfaces are public and at least one in a non-exported
* package, then the proxy class is in a dynamic module in a non-exported * package, then the proxy class is in a dynamic module in a
* package. Reads edge and qualified exports are added for * non-exported package. Reads edge and qualified exports are added
* dynamic module to access. * for dynamic module to access.
*/ */
private static Module mapToModule(ClassLoader loader, private static Module mapToModule(ClassLoader loader,
List<Class<?>> interfaces, List<Class<?>> interfaces,
@ -752,11 +759,12 @@ public class Proxy implements java.io.Serializable {
} }
} }
// all proxy interfaces are public and exported, the proxy class is in unnamed module // all proxy interfaces are public and exported, the proxy class
// Such proxy class is accessible to any unnamed module and named module that // is in unnamed module. Such proxy class is accessible to
// can read unnamed module // any unnamed module and named module that can read unnamed module
if (packagePrivateTypes.isEmpty() && modulePrivateTypes.isEmpty()) { if (packagePrivateTypes.isEmpty() && modulePrivateTypes.isEmpty()) {
return loader != null ? loader.getUnnamedModule() : BootLoader.getUnnamedModule(); return loader != null ? loader.getUnnamedModule()
: BootLoader.getUnnamedModule();
} }
if (packagePrivateTypes.size() > 0) { if (packagePrivateTypes.size() > 0) {
@ -778,7 +786,8 @@ public class Proxy implements java.io.Serializable {
Module target = null; Module target = null;
for (Module m : packagePrivateTypes.values()) { for (Module m : packagePrivateTypes.values()) {
if (getLoader(m) != loader) { if (getLoader(m) != loader) {
// the specified loader is not the same class loader of the non-public interface // the specified loader is not the same class loader
// of the non-public interface
throw new IllegalArgumentException( throw new IllegalArgumentException(
"non-public interface is not defined by the given loader"); "non-public interface is not defined by the given loader");
} }
@ -799,8 +808,9 @@ public class Proxy implements java.io.Serializable {
return target; return target;
} }
// all proxy interfaces are public and at least one in a non-exported package // All proxy interfaces are public and at least one in a non-exported
// map to dynamic proxy module and add reads edge and qualified exports, if necessary // package. So maps to a dynamic proxy module and add reads edge
// and qualified exports, if necessary
Module target = getDynamicModule(loader); Module target = getDynamicModule(loader);
// set up proxy class access to proxy interfaces and types // set up proxy class access to proxy interfaces and types
@ -856,8 +866,8 @@ public class Proxy implements java.io.Serializable {
private static final AtomicInteger counter = new AtomicInteger(); private static final AtomicInteger counter = new AtomicInteger();
/* /*
* Define a dynamic module for the generated proxy classes in a non-exported package * Define a dynamic module for the generated proxy classes in
* named com.sun.proxy.$MODULE. * a non-exported package named com.sun.proxy.$MODULE.
* *
* Each class loader will have one dynamic module. * Each class loader will have one dynamic module.
*/ */

View File

@ -1007,6 +1007,7 @@ public final class ServiceLoader<S>
{ {
static final String PREFIX = "META-INF/services/"; static final String PREFIX = "META-INF/services/";
Set<String> providerNames = new HashSet<>(); // to avoid duplicates
Enumeration<URL> configs; Enumeration<URL> configs;
Iterator<String> pending; Iterator<String> pending;
Class<?> nextClass; Class<?> nextClass;
@ -1016,7 +1017,7 @@ public final class ServiceLoader<S>
/** /**
* Parse a single line from the given configuration file, adding the * Parse a single line from the given configuration file, adding the
* name on the line to the names list. * name on the line to set of names if not already seen.
*/ */
private int parseLine(URL u, BufferedReader r, int lc, Set<String> names) private int parseLine(URL u, BufferedReader r, int lc, Set<String> names)
throws IOException throws IOException
@ -1041,8 +1042,10 @@ public final class ServiceLoader<S>
if (!Character.isJavaIdentifierPart(cp) && (cp != '.')) if (!Character.isJavaIdentifierPart(cp) && (cp != '.'))
fail(service, u, lc, "Illegal provider-class name: " + ln); fail(service, u, lc, "Illegal provider-class name: " + ln);
} }
if (providerNames.add(ln)) {
names.add(ln); names.add(ln);
} }
}
return lc + 1; return lc + 1;
} }
@ -1072,7 +1075,7 @@ public final class ServiceLoader<S>
return true; return true;
} }
Class<?> clazz = null; Class<?> clazz;
do { do {
if (configs == null) { if (configs == null) {
try { try {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -59,7 +59,7 @@ public class JmodFile implements AutoCloseable {
bis.read(magic); bis.read(magic);
if (magic[0] != JMOD_MAGIC_NUMBER[0] || if (magic[0] != JMOD_MAGIC_NUMBER[0] ||
magic[1] != JMOD_MAGIC_NUMBER[1]) { magic[1] != JMOD_MAGIC_NUMBER[1]) {
throw new IOException("Invalid jmod file: " + file.toString()); throw new IOException("Invalid JMOD file: " + file.toString());
} }
if (magic[2] > JMOD_MAJOR_VERSION || if (magic[2] > JMOD_MAJOR_VERSION ||
(magic[2] == JMOD_MAJOR_VERSION && magic[3] > JMOD_MINOR_VERSION)) { (magic[2] == JMOD_MAJOR_VERSION && magic[3] > JMOD_MINOR_VERSION)) {
@ -130,6 +130,13 @@ public class JmodFile implements AutoCloseable {
return name; return name;
} }
/**
* Returns true if the entry is a directory in the JMOD file.
*/
public boolean isDirectory() {
return zipEntry.isDirectory();
}
/** /**
* Returns the size of this entry. * Returns the size of this entry.
*/ */
@ -186,12 +193,12 @@ public class JmodFile implements AutoCloseable {
public Entry getEntry(Section section, String name) { public Entry getEntry(Section section, String name) {
String entry = section.jmodDir() + "/" + name; String entry = section.jmodDir() + "/" + name;
ZipEntry ze = zipfile.getEntry(entry); ZipEntry ze = zipfile.getEntry(entry);
return (ze == null || ze.isDirectory()) ? null : new Entry(ze); return (ze != null) ? new Entry(ze) : null;
} }
/** /**
* Opens an {@code InputStream} for reading the named entry of the given * Opens an {@code InputStream} for reading the named entry of the given
* section in this jmod file. * section in this JMOD file.
* *
* @throws IOException if the named entry is not found, or I/O error * @throws IOException if the named entry is not found, or I/O error
* occurs when reading it * occurs when reading it
@ -201,7 +208,7 @@ public class JmodFile implements AutoCloseable {
{ {
String entry = section.jmodDir() + "/" + name; String entry = section.jmodDir() + "/" + name;
ZipEntry e = zipfile.getEntry(entry); ZipEntry e = zipfile.getEntry(entry);
if (e == null || e.isDirectory()) { if (e == null) {
throw new IOException(name + " not found: " + file); throw new IOException(name + " not found: " + file);
} }
return zipfile.getInputStream(e); return zipfile.getInputStream(e);
@ -217,11 +224,10 @@ public class JmodFile implements AutoCloseable {
} }
/** /**
* Returns a stream of non-directory entries in this jmod file. * Returns a stream of entries in this JMOD file.
*/ */
public Stream<Entry> stream() { public Stream<Entry> stream() {
return zipfile.stream() return zipfile.stream()
.filter(e -> !e.isDirectory())
.map(Entry::new); .map(Entry::new);
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -65,14 +65,12 @@ public final class JrtFileSystemProvider extends FileSystemProvider {
} }
/** /**
* Need FilePermission ${java.home}/-", "read" to create or get jrt:/ * Need RuntimePermission "accessSystemModules" to create or get jrt:/
*/ */
private void checkPermission() { private void checkPermission() {
SecurityManager sm = System.getSecurityManager(); SecurityManager sm = System.getSecurityManager();
if (sm != null) { if (sm != null) {
String home = SystemImage.RUNTIME_HOME; RuntimePermission perm = new RuntimePermission("accessSystemModules");
FilePermission perm
= new FilePermission(home + File.separator + "-", "read");
sm.checkPermission(perm); sm.checkPermission(perm);
} }
} }

View File

@ -95,6 +95,14 @@ public class BootLoader {
return CLASS_LOADER_VALUE_MAP; return CLASS_LOADER_VALUE_MAP;
} }
/**
* Returns {@code true} if there is a class path associated with the
* BootLoader.
*/
public static boolean hasClassPath() {
return ClassLoaders.bootLoader().hasClassPath();
}
/** /**
* Register a module with this class loader so that its classes (and * Register a module with this class loader so that its classes (and
* resources) become visible via this class loader. * resources) become visible via this class loader.
@ -187,14 +195,6 @@ public class BootLoader {
.map(name -> getDefinedPackage(name.replace('/', '.'))); .map(name -> getDefinedPackage(name.replace('/', '.')));
} }
/**
* Returns {@code true} if there is a class path associated with the
* BootLoader.
*/
public static boolean hasClassPath() {
return ClassLoaders.bootLoader().hasClassPath();
}
/** /**
* Helper class to define {@code Package} objects for packages in modules * Helper class to define {@code Package} objects for packages in modules
* defined to the boot loader. * defined to the boot loader.

View File

@ -60,6 +60,7 @@ import java.util.stream.Stream;
import jdk.internal.misc.VM; import jdk.internal.misc.VM;
import jdk.internal.module.ModulePatcher.PatchedModuleReader; import jdk.internal.module.ModulePatcher.PatchedModuleReader;
import jdk.internal.module.SystemModules; import jdk.internal.module.SystemModules;
import jdk.internal.module.Resources;
/** /**
@ -162,6 +163,14 @@ public class BuiltinClassLoader
this.moduleToReader = new ConcurrentHashMap<>(); this.moduleToReader = new ConcurrentHashMap<>();
} }
/**
* Returns {@code true} if there is a class path associated with this
* class loader.
*/
boolean hasClassPath() {
return ucp != null;
}
/** /**
* Register a module this this class loader. This has the effect of making * Register a module this this class loader. This has the effect of making
* the types in the module visible. * the types in the module visible.
@ -248,18 +257,24 @@ public class BuiltinClassLoader
*/ */
@Override @Override
public URL findResource(String name) { public URL findResource(String name) {
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
LoadedModule module = packageToModule.get(pn); LoadedModule module = packageToModule.get(pn);
if (module != null) { if (module != null) {
// resource is in a package of a module defined to this loader // resource is in a package of a module defined to this loader
if (module.loader() == this if (module.loader() == this) {
&& (name.endsWith(".class") || isOpen(module.mref(), pn))) { URL url;
try { try {
return findResource(module.name(), name); // checks URL url = findResource(module.name(), name); // checks URL
} catch (IOException ioe) { } catch (IOException ioe) {
return null; return null;
} }
if (url != null
&& (name.endsWith(".class")
|| url.toString().endsWith("/")
|| isOpen(module.mref(), pn))) {
return url;
}
} }
} else { } else {
@ -293,15 +308,17 @@ public class BuiltinClassLoader
public Enumeration<URL> findResources(String name) throws IOException { public Enumeration<URL> findResources(String name) throws IOException {
List<URL> checked = new ArrayList<>(); // list of checked URLs List<URL> checked = new ArrayList<>(); // list of checked URLs
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
LoadedModule module = packageToModule.get(pn); LoadedModule module = packageToModule.get(pn);
if (module != null) { if (module != null) {
// resource is in a package of a module defined to this loader // resource is in a package of a module defined to this loader
if (module.loader() == this if (module.loader() == this) {
&& (name.endsWith(".class") || isOpen(module.mref(), pn))) {
URL url = findResource(module.name(), name); // checks URL URL url = findResource(module.name(), name); // checks URL
if (url != null) { if (url != null
&& (name.endsWith(".class")
|| url.toString().endsWith("/")
|| isOpen(module.mref(), pn))) {
checked.add(url); checked.add(url);
} }
} }
@ -351,11 +368,13 @@ public class BuiltinClassLoader
new PrivilegedExceptionAction<>() { new PrivilegedExceptionAction<>() {
@Override @Override
public List<URL> run() throws IOException { public List<URL> run() throws IOException {
List<URL> result = new ArrayList<>(); List<URL> result = null;
for (ModuleReference mref : nameToModule.values()) { for (ModuleReference mref : nameToModule.values()) {
URI u = moduleReaderFor(mref).find(name).orElse(null); URI u = moduleReaderFor(mref).find(name).orElse(null);
if (u != null) { if (u != null) {
try { try {
if (result == null)
result = new ArrayList<>();
result.add(u.toURL()); result.add(u.toURL());
} catch (MalformedURLException | } catch (MalformedURLException |
IllegalArgumentException e) { IllegalArgumentException e) {
@ -375,7 +394,7 @@ public class BuiltinClassLoader
map = new ConcurrentHashMap<>(); map = new ConcurrentHashMap<>();
this.resourceCache = new SoftReference<>(map); this.resourceCache = new SoftReference<>(map);
} }
if (urls.isEmpty()) if (urls == null)
urls = Collections.emptyList(); urls = Collections.emptyList();
map.putIfAbsent(name, urls); map.putIfAbsent(name, urls);
} }
@ -869,14 +888,6 @@ public class BuiltinClassLoader
sealBase); sealBase);
} }
/**
* Returns {@code true} if there is a class path associated with this
* class loader.
*/
boolean hasClassPath() {
return ucp != null;
}
/** /**
* Returns {@code true} if the specified package name is sealed according to * Returns {@code true} if the specified package name is sealed according to
* the given manifest. * the given manifest.
@ -975,7 +986,7 @@ public class BuiltinClassLoader
*/ */
private boolean isOpen(ModuleReference mref, String pn) { private boolean isOpen(ModuleReference mref, String pn) {
ModuleDescriptor descriptor = mref.descriptor(); ModuleDescriptor descriptor = mref.descriptor();
if (descriptor.isOpen()) if (descriptor.isOpen() || descriptor.isAutomatic())
return true; return true;
for (ModuleDescriptor.Opens opens : descriptor.opens()) { for (ModuleDescriptor.Opens opens : descriptor.opens()) {
String source = opens.source(); String source = opens.source();

View File

@ -60,6 +60,7 @@ import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.Resources;
/** /**
@ -356,46 +357,53 @@ public final class Loader extends SecureClassLoader {
@Override @Override
public URL findResource(String name) { public URL findResource(String name) {
URL url = null; String pn = Resources.toPackageName(name);
String pn = ResourceHelper.getPackageName(name);
LoadedModule module = localPackageToModule.get(pn); LoadedModule module = localPackageToModule.get(pn);
if (module != null) { if (module != null) {
if (name.endsWith(".class") || isOpen(module.mref(), pn)) {
try { try {
url = findResource(module.name(), name); URL url = findResource(module.name(), name);
if (url != null
&& (name.endsWith(".class")
|| url.toString().endsWith("/")
|| isOpen(module.mref(), pn))) {
return url;
}
} catch (IOException ioe) { } catch (IOException ioe) {
// ignore // ignore
} }
}
} else { } else {
for (ModuleReference mref : nameToModule.values()) { for (ModuleReference mref : nameToModule.values()) {
try { try {
url = findResource(mref.descriptor().name(), name); URL url = findResource(mref.descriptor().name(), name);
if (url != null) if (url != null) return url;
break;
} catch (IOException ioe) { } catch (IOException ioe) {
// ignore // ignore
} }
} }
} }
return url;
return null;
} }
@Override @Override
public Enumeration<URL> findResources(String name) throws IOException { public Enumeration<URL> findResources(String name) throws IOException {
List<URL> urls = new ArrayList<>(); List<URL> urls = new ArrayList<>();
String pn = ResourceHelper.getPackageName(name); String pn = Resources.toPackageName(name);
LoadedModule module = localPackageToModule.get(pn); LoadedModule module = localPackageToModule.get(pn);
if (module != null) { if (module != null) {
if (name.endsWith(".class") || isOpen(module.mref(), pn)) {
try { try {
URL url = findResource(module.name(), name); URL url = findResource(module.name(), name);
if (url != null) if (url != null
&& (name.endsWith(".class")
|| url.toString().endsWith("/")
|| isOpen(module.mref(), pn))) {
urls.add(url); urls.add(url);
}
} catch (IOException ioe) { } catch (IOException ioe) {
// ignore // ignore
} }
}
} else { } else {
for (ModuleReference mref : nameToModule.values()) { for (ModuleReference mref : nameToModule.values()) {
try { try {
@ -643,7 +651,7 @@ public final class Loader extends SecureClassLoader {
*/ */
private boolean isOpen(ModuleReference mref, String pn) { private boolean isOpen(ModuleReference mref, String pn) {
ModuleDescriptor descriptor = mref.descriptor(); ModuleDescriptor descriptor = mref.descriptor();
if (descriptor.isOpen()) if (descriptor.isOpen() || descriptor.isAutomatic())
return true; return true;
for (ModuleDescriptor.Opens opens : descriptor.opens()) { for (ModuleDescriptor.Opens opens : descriptor.opens()) {
String source = opens.source(); String source = opens.source();

View File

@ -1,125 +0,0 @@
/*
* Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package jdk.internal.loader;
import java.io.File;
import java.nio.file.Path;
import java.nio.file.Paths;
import jdk.internal.module.Checks;
/**
* Helper class for Class#getResource, Module#getResourceAsStream, and other
* methods that locate a resource in a module.
*/
public final class ResourceHelper {
private ResourceHelper() { }
/**
* Returns the <em>package name</em> for a resource or the empty package if
* the resource name does not contain a slash.
*/
public static String getPackageName(String name) {
int index = name.lastIndexOf('/');
if (index != -1) {
return name.substring(0, index).replace("/", ".");
} else {
return "";
}
}
/**
* Returns true if the resource is a <em>simple resource</em>. Simple
* resources can never be encapsulated. Resources ending in "{@code .class}"
* or where the package name is not a legal package name can not be
* encapsulated.
*/
public static boolean isSimpleResource(String name) {
int len = name.length();
if (len > 6 && name.endsWith(".class")) {
return true;
}
if (!Checks.isPackageName(getPackageName(name))) {
return true;
}
return false;
}
/**
* Converts a resource name to a file path. Returns {@code null} if the
* resource name cannot be converted into a file path. Resource names
* with empty elements, or elements that are "." or ".." are rejected,
* as is a resource name that translates to a file path with a root
* component.
*/
public static Path toFilePath(String name) {
// scan the resource name to eagerly reject obviously invalid names
int next;
int off = 0;
while ((next = name.indexOf('/', off)) != -1) {
int len = next - off;
if (!mayTranslate(name, off, len)) {
return null;
}
off = next + 1;
}
int rem = name.length() - off;
if (!mayTranslate(name, off, rem)) {
return null;
}
// convert to file path
Path path;
if (File.separatorChar == '/') {
path = Paths.get(name);
} else {
// not allowed to embed file separators
if (name.contains(File.separator))
return null;
path = Paths.get(name.replace('/', File.separatorChar));
}
// file path not allowed to have root component
return (path.getRoot() == null) ? path : null;
}
/**
* Returns {@code true} if the element in a resource name is a candidate
* to translate to the element of a file path.
*/
private static boolean mayTranslate(String name, int off, int len) {
if (len <= 2) {
if (len == 0)
return false;
boolean starsWithDot = (name.charAt(off) == '.');
if (len == 1 && starsWithDot)
return false;
if (len == 2 && starsWithDot && (name.charAt(off+1) == '.'))
return false;
}
return true;
}
}

View File

@ -33,6 +33,7 @@ import java.lang.reflect.Method;
import java.lang.reflect.Module; import java.lang.reflect.Module;
import java.net.URL; import java.net.URL;
import java.security.AccessControlContext; import java.security.AccessControlContext;
import java.security.ProtectionDomain;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Stream; import java.util.stream.Stream;
@ -149,6 +150,11 @@ public interface JavaLangAccess {
*/ */
ConcurrentHashMap<?, ?> createOrGetClassLoaderValueMap(ClassLoader cl); ConcurrentHashMap<?, ?> createOrGetClassLoaderValueMap(ClassLoader cl);
/**
* Defines a class with the given name to a class loader.
*/
Class<?> defineClass(ClassLoader cl, String name, byte[] b, ProtectionDomain pd, String source);
/** /**
* Returns a class loaded by the bootstrap class loader. * Returns a class loaded by the bootstrap class loader.
*/ */

View File

@ -73,7 +73,7 @@ public interface JavaLangModuleAccess {
void requires(ModuleDescriptor.Builder builder, void requires(ModuleDescriptor.Builder builder,
Set<Requires.Modifier> ms, Set<Requires.Modifier> ms,
String mn, String mn,
String compiledVersion); String rawCompiledVersion);
/** /**
* Returns a {@code ModuleDescriptor.Requires} of the given modifiers * Returns a {@code ModuleDescriptor.Requires} of the given modifiers
@ -127,9 +127,6 @@ public interface JavaLangModuleAccess {
Set<Provides> provides, Set<Provides> provides,
Set<String> packages, Set<String> packages,
String mainClass, String mainClass,
String osName,
String osArch,
String osVersion,
int hashCode); int hashCode);
/** /**

View File

@ -65,6 +65,11 @@ public interface JavaLangReflectModuleAccess {
*/ */
void addReadsAllUnnamed(Module m); void addReadsAllUnnamed(Module m);
/**
* Update module m to export a package to all modules.
*/
void addExports(Module m, String pn);
/** /**
* Updates module m1 to export a package to module m2. The export does * Updates module m1 to export a package to module m2. The export does
* not result in a strong reference to m2 (m2 can be GC'ed). * not result in a strong reference to m2 (m2 can be GC'ed).
@ -72,25 +77,20 @@ public interface JavaLangReflectModuleAccess {
void addExports(Module m1, String pkg, Module m2); void addExports(Module m1, String pkg, Module m2);
/** /**
* Updates module m1 to open a package to module m2. Opening the * Updates a module m to export a package to all unnamed modules.
* package does not result in a strong reference to m2 (m2 can be GC'ed).
*/ */
void addOpens(Module m1, String pkg, Module m2); void addExportsToAllUnnamed(Module m, String pkg);
/**
* Updates a module m to export a package to all modules.
*/
void addExportsToAll(Module m, String pkg);
/** /**
* Updates a module m to open a package to all modules. * Updates a module m to open a package to all modules.
*/ */
void addOpensToAll(Module m, String pkg); void addOpens(Module m, String pkg);
/** /**
* Updates a module m to export a package to all unnamed modules. * Updates module m1 to open a package to module m2. Opening the
* package does not result in a strong reference to m2 (m2 can be GC'ed).
*/ */
void addExportsToAllUnnamed(Module m, String pkg); void addOpens(Module m1, String pkg, Module m2);
/** /**
* Updates a module m to open a package to all unnamed modules. * Updates a module m to open a package to all unnamed modules.
@ -102,11 +102,6 @@ public interface JavaLangReflectModuleAccess {
*/ */
void addUses(Module m, Class<?> service); void addUses(Module m, Class<?> service);
/**
* Add a package to the given module.
*/
void addPackage(Module m, String pkg);
/** /**
* Returns the ServicesCatalog for the given Layer. * Returns the ServicesCatalog for the given Layer.
*/ */
@ -123,12 +118,4 @@ public interface JavaLangReflectModuleAccess {
* given class loader. * given class loader.
*/ */
Stream<Layer> layers(ClassLoader loader); Stream<Layer> layers(ClassLoader loader);
/**
* Tests if a module exports a package at least {@code other} via its
* module declaration.
*
* @apiNote This is a temporary method for debugging features.
*/
boolean isStaticallyExported(Module module, String pn, Module other);
} }

View File

@ -145,9 +145,6 @@ final class Builder {
Set<Provides> provides; Set<Provides> provides;
Version version; Version version;
String mainClass; String mainClass;
String osName;
String osArch;
String osVersion;
Builder(String name) { Builder(String name) {
this.name = name; this.name = name;
@ -247,30 +244,6 @@ final class Builder {
return this; return this;
} }
/**
* Sets the OS name.
*/
public Builder osName(String name) {
this.osName = name;
return this;
}
/**
* Sets the OS arch.
*/
public Builder osArch(String arch) {
this.osArch = arch;
return this;
}
/**
* Sets the OS version.
*/
public Builder osVersion(String version) {
this.osVersion = version;
return this;
}
/** /**
* Returns an immutable set of the module modifiers derived from the flags. * Returns an immutable set of the module modifiers derived from the flags.
*/ */
@ -305,9 +278,6 @@ final class Builder {
provides, provides,
packages, packages,
mainClass, mainClass,
osName,
osArch,
osVersion,
hashCode); hashCode);
} }
} }

View File

@ -180,41 +180,38 @@ public final class Checks {
} }
/** /**
* Returns {@code true} if the last character of the given name is legal * Returns {@code true} if a given legal module name contains an identifier
* as the last character of a module name. * that doesn't end with a Java letter.
*
* @throws IllegalArgumentException if name is empty
*/ */
public static boolean hasLegalModuleNameLastCharacter(String name) { public static boolean hasJavaIdentifierWithTrailingDigit(String name) {
if (name.isEmpty()) // quick scan to allow names that are just ASCII without digits
throw new IllegalArgumentException("name is empty"); boolean needToParse = false;
int len = name.length();
if (isASCIIString(name)) {
char c = name.charAt(len-1);
return Character.isJavaIdentifierStart(c);
} else {
int i = 0; int i = 0;
int cp = -1; while (i < name.length()) {
while (i < len) { int c = name.charAt(i);
cp = name.codePointAt(i); if (c > 0x7F || (c >= '0' && c <= '9')) {
i += Character.charCount(cp); needToParse = true;
break;
} }
return Character.isJavaIdentifierStart(cp);
}
}
/**
* Returns true if the given string only contains ASCII characters.
*/
private static boolean isASCIIString(String s) {
int i = 0;
while (i < s.length()) {
int c = s.charAt(i);
if (c > 0x7F)
return false;
i++; i++;
} }
if (!needToParse)
return false;
// slow path
int next;
int off = 0;
while ((next = name.indexOf('.', off)) != -1) {
int last = isJavaIdentifier(name, off, (next - off));
if (!Character.isJavaIdentifierStart(last))
return true; return true;
off = next+1;
}
int last = isJavaIdentifier(name, off, name.length() - off);
if (!Character.isJavaIdentifierStart(last))
return true;
return false;
} }
/** /**

View File

@ -292,11 +292,11 @@ public final class ClassFileAttributes {
attr.putShort(module_flags); attr.putShort(module_flags);
// module_version // module_version
Version v = descriptor.version().orElse(null); String vs = descriptor.rawVersion().orElse(null);
if (v == null) { if (vs == null) {
attr.putShort(0); attr.putShort(0);
} else { } else {
int module_version_index = cw.newUTF8(v.toString()); int module_version_index = cw.newUTF8(vs);
attr.putShort(module_version_index); attr.putShort(module_version_index);
} }
@ -320,11 +320,11 @@ public final class ClassFileAttributes {
attr.putShort(requires_flags); attr.putShort(requires_flags);
int requires_version_index; int requires_version_index;
v = r.compiledVersion().orElse(null); vs = r.rawCompiledVersion().orElse(null);
if (v == null) { if (vs == null) {
requires_version_index = 0; requires_version_index = 0;
} else { } else {
requires_version_index = cw.newUTF8(v.toString()); requires_version_index = cw.newUTF8(vs);
} }
attr.putShort(requires_version_index); attr.putShort(requires_version_index);
} }
@ -553,8 +553,6 @@ public final class ClassFileAttributes {
* u2 os_name_index; * u2 os_name_index;
* // index to CONSTANT_utf8_info structure with the OS arch * // index to CONSTANT_utf8_info structure with the OS arch
* u2 os_arch_index * u2 os_arch_index
* // index to CONSTANT_utf8_info structure with the OS version
* u2 os_version_index;
* } * }
* *
* } </pre> * } </pre>
@ -562,17 +560,23 @@ public final class ClassFileAttributes {
public static class ModuleTargetAttribute extends Attribute { public static class ModuleTargetAttribute extends Attribute {
private final String osName; private final String osName;
private final String osArch; private final String osArch;
private final String osVersion;
public ModuleTargetAttribute(String osName, String osArch, String osVersion) { public ModuleTargetAttribute(String osName, String osArch) {
super(MODULE_TARGET); super(MODULE_TARGET);
this.osName = osName; this.osName = osName;
this.osArch = osArch; this.osArch = osArch;
this.osVersion = osVersion;
} }
public ModuleTargetAttribute() { public ModuleTargetAttribute() {
this(null, null, null); this(null, null);
}
public String osName() {
return osName;
}
public String osArch() {
return osArch;
} }
@Override @Override
@ -586,7 +590,6 @@ public final class ClassFileAttributes {
String osName = null; String osName = null;
String osArch = null; String osArch = null;
String osVersion = null;
int name_index = cr.readUnsignedShort(off); int name_index = cr.readUnsignedShort(off);
if (name_index != 0) if (name_index != 0)
@ -598,12 +601,7 @@ public final class ClassFileAttributes {
osArch = cr.readUTF8(off, buf); osArch = cr.readUTF8(off, buf);
off += 2; off += 2;
int version_index = cr.readUnsignedShort(off); return new ModuleTargetAttribute(osName, osArch);
if (version_index != 0)
osVersion = cr.readUTF8(off, buf);
off += 2;
return new ModuleTargetAttribute(osName, osArch, osVersion);
} }
@Override @Override
@ -625,11 +623,6 @@ public final class ClassFileAttributes {
arch_index = cw.newUTF8(osArch); arch_index = cw.newUTF8(osArch);
attr.putShort(arch_index); attr.putShort(arch_index);
int version_index = 0;
if (osVersion != null && osVersion.length() > 0)
version_index = cw.newUTF8(osVersion);
attr.putShort(version_index);
return attr; return attr;
} }
} }

View File

@ -0,0 +1,318 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package jdk.internal.module;
import java.lang.invoke.MethodHandles;
import java.lang.reflect.Module;
import java.net.URL;
import java.security.AccessController;
import java.security.CodeSource;
import java.security.PrivilegedAction;
import java.security.ProtectionDomain;
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.WeakHashMap;
import java.util.function.Supplier;
import java.util.stream.Collectors;
/**
* Supports logging of access to members of API packages that are exported or
* opened via backdoor mechanisms to code in unnamed modules.
*/
public final class IllegalAccessLogger {
// true to print stack trace
private static final boolean PRINT_STACK_TRACE;
static {
String s = System.getProperty("sun.reflect.debugModuleAccessChecks");
PRINT_STACK_TRACE = "access".equals(s);
}
private static final StackWalker STACK_WALKER
= StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
// the maximum number of frames to capture
private static final int MAX_STACK_FRAMES = 32;
// lock to avoid interference when printing stack traces
private static final Object OUTPUT_LOCK = new Object();
// caller -> usages
private final Map<Class<?>, Set<Usage>> callerToUsages = new WeakHashMap<>();
// module -> (package name -> CLI option)
private final Map<Module, Map<String, String>> exported;
private final Map<Module, Map<String, String>> opened;
private IllegalAccessLogger(Map<Module, Map<String, String>> exported,
Map<Module, Map<String, String>> opened) {
this.exported = deepCopy(exported);
this.opened = deepCopy(opened);
}
/**
* Returns that a Builder that is seeded with the packages known to this logger.
*/
public Builder toBuilder() {
return new Builder(exported, opened);
}
/**
* Logs access to the member of a target class by a caller class if the class
* is in a package that is exported via a backdoor mechanism.
*
* The {@code whatSupplier} supplies the message that describes the member.
*/
public void logIfExportedByBackdoor(Class<?> caller,
Class<?> target,
Supplier<String> whatSupplier) {
Map<String, String> packages = exported.get(target.getModule());
if (packages != null) {
String how = packages.get(target.getPackageName());
if (how != null) {
log(caller, whatSupplier.get(), how);
}
}
}
/**
* Logs access to the member of a target class by a caller class if the class
* is in a package that is opened via a backdoor mechanism.
*
* The {@code what} parameter supplies the message that describes the member.
*/
public void logIfOpenedByBackdoor(Class<?> caller,
Class<?> target,
Supplier<String> whatSupplier) {
Map<String, String> packages = opened.get(target.getModule());
if (packages != null) {
String how = packages.get(target.getPackageName());
if (how != null) {
log(caller, whatSupplier.get(), how);
}
}
}
/**
* Logs access by a caller class. The {@code what} parameter describes
* the member is accessed, the {@code how} parameter is the means by which
* access is allocated (CLI option for example).
*/
private void log(Class<?> caller, String what, String how) {
log(caller, what, () -> {
PrivilegedAction<ProtectionDomain> pa = caller::getProtectionDomain;
CodeSource cs = AccessController.doPrivileged(pa).getCodeSource();
URL url = (cs != null) ? cs.getLocation() : null;
String source = caller.getName();
if (url != null)
source += " (" + url + ")";
return "WARNING: Illegal access by " + source + " to " + what
+ " (permitted by " + how + ")";
});
}
/**
* Logs access to caller class if the class is in a package that is opened via
* a backdoor mechanism.
*/
public void logIfOpenedByBackdoor(MethodHandles.Lookup caller, Class<?> target) {
Map<String, String> packages = opened.get(target.getModule());
if (packages != null) {
String how = packages.get(target.getPackageName());
if (how != null) {
log(caller.lookupClass(), target.getName(), () ->
"WARNING: Illegal access using Lookup on " + caller.lookupClass()
+ " to " + target + " (permitted by " + how + ")");
}
}
}
/**
* Log access by a caller. The {@code what} parameter describes the class or
* member that is being accessed. The {@code msgSupplier} supplies the log
* message.
*
* To reduce output, this method only logs the access if it hasn't been seen
* previously. "Seen previously" is implemented as a map of caller class -> Usage,
* where a Usage is the "what" and a hash of the stack trace. The map has weak
* keys so it can be expunged when the caller is GC'ed/unloaded.
*/
private void log(Class<?> caller, String what, Supplier<String> msgSupplier) {
// stack trace without the top-most frames in java.base
List<StackWalker.StackFrame> stack = STACK_WALKER.walk(s ->
s.dropWhile(this::isJavaBase)
.limit(MAX_STACK_FRAMES)
.collect(Collectors.toList())
);
// check if the access has already been recorded
Usage u = new Usage(what, hash(stack));
boolean firstUsage;
synchronized (this) {
firstUsage = callerToUsages.computeIfAbsent(caller, k -> new HashSet<>()).add(u);
}
// log message if first usage
if (firstUsage) {
String msg = msgSupplier.get();
if (PRINT_STACK_TRACE) {
synchronized (OUTPUT_LOCK) {
System.err.println(msg);
stack.forEach(f -> System.err.println("\tat " + f));
}
} else {
System.err.println(msg);
}
}
}
private static class Usage {
private final String what;
private final int stack;
Usage(String what, int stack) {
this.what = what;
this.stack = stack;
}
@Override
public int hashCode() {
return what.hashCode() ^ stack;
}
@Override
public boolean equals(Object ob) {
if (ob instanceof Usage) {
Usage that = (Usage)ob;
return what.equals(that.what) && stack == (that.stack);
} else {
return false;
}
}
}
/**
* Returns true if the stack frame is for a class in java.base.
*/
private boolean isJavaBase(StackWalker.StackFrame frame) {
Module caller = frame.getDeclaringClass().getModule();
return "java.base".equals(caller.getName());
}
/**
* Computes a hash code for the give stack frames. The hash code is based
* on the class, method name, and BCI.
*/
private int hash(List<StackWalker.StackFrame> stack) {
int hash = 0;
for (StackWalker.StackFrame frame : stack) {
hash = (31 * hash) + Objects.hash(frame.getDeclaringClass(),
frame.getMethodName(),
frame.getByteCodeIndex());
}
return hash;
}
// system-wide IllegalAccessLogger
private static volatile IllegalAccessLogger logger;
/**
* Sets the system-wide IllegalAccessLogger
*/
public static void setIllegalAccessLogger(IllegalAccessLogger l) {
if (l.exported.isEmpty() && l.opened.isEmpty()) {
logger = null;
} else {
logger = l;
}
}
/**
* Returns the system-wide IllegalAccessLogger or {@code null} if there is
* no logger.
*/
public static IllegalAccessLogger illegalAccessLogger() {
return logger;
}
/**
* A builder for IllegalAccessLogger objects.
*/
public static class Builder {
private Map<Module, Map<String, String>> exported;
private Map<Module, Map<String, String>> opened;
public Builder() { }
public Builder(Map<Module, Map<String, String>> exported,
Map<Module, Map<String, String>> opened) {
this.exported = deepCopy(exported);
this.opened = deepCopy(opened);
}
public void logAccessToExportedPackage(Module m, String pn, String how) {
if (!m.isExported(pn)) {
if (exported == null)
exported = new HashMap<>();
exported.computeIfAbsent(m, k -> new HashMap<>()).putIfAbsent(pn, how);
}
}
public void logAccessToOpenPackage(Module m, String pn, String how) {
// opens implies exported at run-time.
logAccessToExportedPackage(m, pn, how);
if (!m.isOpen(pn)) {
if (opened == null)
opened = new HashMap<>();
opened.computeIfAbsent(m, k -> new HashMap<>()).putIfAbsent(pn, how);
}
}
/**
* Builds the logger.
*/
public IllegalAccessLogger build() {
return new IllegalAccessLogger(exported, opened);
}
}
static Map<Module, Map<String, String>> deepCopy(Map<Module, Map<String, String>> map) {
if (map == null || map.isEmpty()) {
return new HashMap<>();
} else {
Map<Module, Map<String, String>> newMap = new HashMap<>();
for (Map.Entry<Module, Map<String, String>> e : map.entrySet()) {
newMap.put(e.getKey(), new HashMap<>(e.getValue()));
}
return newMap;
}
}
}

View File

@ -33,6 +33,7 @@ import java.lang.module.ModuleFinder;
import java.lang.module.ModuleReference; import java.lang.module.ModuleReference;
import java.lang.module.ResolvedModule; import java.lang.module.ResolvedModule;
import java.lang.reflect.Layer; import java.lang.reflect.Layer;
import java.lang.reflect.LayerInstantiationException;
import java.lang.reflect.Module; import java.lang.reflect.Module;
import java.net.URI; import java.net.URI;
import java.nio.file.Path; import java.nio.file.Path;
@ -46,7 +47,6 @@ import java.util.Map;
import java.util.Optional; import java.util.Optional;
import java.util.Set; import java.util.Set;
import java.util.function.Function; import java.util.function.Function;
import java.util.stream.Stream;
import jdk.internal.loader.BootLoader; import jdk.internal.loader.BootLoader;
import jdk.internal.loader.BuiltinClassLoader; import jdk.internal.loader.BuiltinClassLoader;
@ -327,8 +327,9 @@ public final class ModuleBootstrap {
for (String p : descriptor.packages()) { for (String p : descriptor.packages()) {
String other = packageToModule.putIfAbsent(p, name); String other = packageToModule.putIfAbsent(p, name);
if (other != null) { if (other != null) {
fail("Package " + p + " in both module " String msg = "Package " + p + " in both module "
+ name + " and module " + other); + name + " and module " + other;
throw new LayerInstantiationException(msg);
} }
} }
} }
@ -359,7 +360,7 @@ public final class ModuleBootstrap {
PerfCounters.loadModulesTime.addElapsedTimeFrom(t5); PerfCounters.loadModulesTime.addElapsedTimeFrom(t5);
// --add-reads, -add-exports/-add-opens // --add-reads, --add-exports/--add-opens
addExtraReads(bootLayer); addExtraReads(bootLayer);
addExtraExportsAndOpens(bootLayer); addExtraExportsAndOpens(bootLayer);
@ -514,26 +515,44 @@ public final class ModuleBootstrap {
* additional packages specified on the command-line. * additional packages specified on the command-line.
*/ */
private static void addExtraExportsAndOpens(Layer bootLayer) { private static void addExtraExportsAndOpens(Layer bootLayer) {
IllegalAccessLogger.Builder builder = new IllegalAccessLogger.Builder();
// --add-exports // --add-exports
String prefix = "jdk.module.addexports."; String prefix = "jdk.module.addexports.";
Map<String, List<String>> extraExports = decode(prefix); Map<String, List<String>> extraExports = decode(prefix);
if (!extraExports.isEmpty()) { if (!extraExports.isEmpty()) {
addExtraExportsOrOpens(bootLayer, extraExports, false); addExtraExportsOrOpens(bootLayer, extraExports, false, builder);
} }
// --add-opens // --add-opens
prefix = "jdk.module.addopens."; prefix = "jdk.module.addopens.";
Map<String, List<String>> extraOpens = decode(prefix); Map<String, List<String>> extraOpens = decode(prefix);
if (!extraOpens.isEmpty()) { if (!extraOpens.isEmpty()) {
addExtraExportsOrOpens(bootLayer, extraOpens, true); addExtraExportsOrOpens(bootLayer, extraOpens, true, builder);
} }
// --permit-illegal-access
if (getAndRemoveProperty("jdk.module.permitIllegalAccess") != null) {
warn("--permit-illegal-access will be removed in the next major release");
bootLayer.modules().stream().forEach(m -> {
m.getDescriptor()
.packages()
.stream()
.filter(pn -> !m.isOpen(pn))
.forEach(pn -> {
builder.logAccessToOpenPackage(m, pn, "--permit-illegal-access");
Modules.addOpensToAllUnnamed(m, pn);
});
});
}
IllegalAccessLogger.setIllegalAccessLogger(builder.build());
} }
private static void addExtraExportsOrOpens(Layer bootLayer, private static void addExtraExportsOrOpens(Layer bootLayer,
Map<String, List<String>> map, Map<String, List<String>> map,
boolean opens) boolean opens,
IllegalAccessLogger.Builder builder)
{ {
String option = opens ? ADD_OPENS : ADD_EXPORTS; String option = opens ? ADD_OPENS : ADD_EXPORTS;
for (Map.Entry<String, List<String>> e : map.entrySet()) { for (Map.Entry<String, List<String>> e : map.entrySet()) {
@ -581,8 +600,10 @@ public final class ModuleBootstrap {
} }
if (allUnnamed) { if (allUnnamed) {
if (opens) { if (opens) {
builder.logAccessToOpenPackage(m, pn, option);
Modules.addOpensToAllUnnamed(m, pn); Modules.addOpensToAllUnnamed(m, pn);
} else { } else {
builder.logAccessToExportedPackage(m, pn, option);
Modules.addExportsToAllUnnamed(m, pn); Modules.addExportsToAllUnnamed(m, pn);
} }
} else { } else {
@ -632,7 +653,7 @@ public final class ModuleBootstrap {
// value is <module>(,<module>)* or <file>(<pathsep><file>)* // value is <module>(,<module>)* or <file>(<pathsep><file>)*
if (!allowDuplicates && map.containsKey(key)) if (!allowDuplicates && map.containsKey(key))
fail(key + " specified more than once in " + option(prefix)); fail(key + " specified more than once to " + option(prefix));
List<String> values = map.computeIfAbsent(key, k -> new ArrayList<>()); List<String> values = map.computeIfAbsent(key, k -> new ArrayList<>());
int ntargets = 0; int ntargets = 0;
for (String s : rhs.split(regex)) { for (String s : rhs.split(regex)) {
@ -676,10 +697,6 @@ public final class ModuleBootstrap {
ModuleReference mref = rm.reference(); ModuleReference mref = rm.reference();
String mn = mref.descriptor().name(); String mn = mref.descriptor().name();
// emit warning if module name ends with a non-Java letter
if (!Checks.hasLegalModuleNameLastCharacter(mn))
warn("Module name \"" + mn + "\" may soon be illegal");
// emit warning if the WARN_INCUBATING module resolution bit set // emit warning if the WARN_INCUBATING module resolution bit set
if (ModuleResolution.hasIncubatingWarning(mref)) { if (ModuleResolution.hasIncubatingWarning(mref)) {
if (incubating == null) { if (incubating == null) {
@ -705,7 +722,7 @@ public final class ModuleBootstrap {
} }
static void warnUnknownModule(String option, String mn) { static void warnUnknownModule(String option, String mn) {
warn("Unknown module: " + mn + " specified in " + option); warn("Unknown module: " + mn + " specified to " + option);
} }
static String unableToParse(String option, String text, String value) { static String unableToParse(String option, String text, String value) {

View File

@ -89,18 +89,24 @@ public final class ModuleInfo {
*/ */
public static final class Attributes { public static final class Attributes {
private final ModuleDescriptor descriptor; private final ModuleDescriptor descriptor;
private final ModuleTarget target;
private final ModuleHashes recordedHashes; private final ModuleHashes recordedHashes;
private final ModuleResolution moduleResolution; private final ModuleResolution moduleResolution;
Attributes(ModuleDescriptor descriptor, Attributes(ModuleDescriptor descriptor,
ModuleTarget target,
ModuleHashes recordedHashes, ModuleHashes recordedHashes,
ModuleResolution moduleResolution) { ModuleResolution moduleResolution) {
this.descriptor = descriptor; this.descriptor = descriptor;
this.target = target;
this.recordedHashes = recordedHashes; this.recordedHashes = recordedHashes;
this.moduleResolution = moduleResolution; this.moduleResolution = moduleResolution;
} }
public ModuleDescriptor descriptor() { public ModuleDescriptor descriptor() {
return descriptor; return descriptor;
} }
public ModuleTarget target() {
return target;
}
public ModuleHashes recordedHashes() { public ModuleHashes recordedHashes() {
return recordedHashes; return recordedHashes;
} }
@ -221,8 +227,8 @@ public final class ModuleInfo {
Builder builder = null; Builder builder = null;
Set<String> allPackages = null; Set<String> allPackages = null;
String mainClass = null; String mainClass = null;
String[] osValues = null; ModuleTarget moduleTarget = null;
ModuleHashes hashes = null; ModuleHashes moduelHashes = null;
ModuleResolution moduleResolution = null; ModuleResolution moduleResolution = null;
for (int i = 0; i < attributes_count ; i++) { for (int i = 0; i < attributes_count ; i++) {
@ -251,12 +257,12 @@ public final class ModuleInfo {
break; break;
case MODULE_TARGET : case MODULE_TARGET :
osValues = readModuleTargetAttribute(in, cpool); moduleTarget = readModuleTargetAttribute(in, cpool);
break; break;
case MODULE_HASHES : case MODULE_HASHES :
if (parseHashes) { if (parseHashes) {
hashes = readModuleHashesAttribute(in, cpool); moduelHashes = readModuleHashesAttribute(in, cpool);
} else { } else {
in.skipBytes(length); in.skipBytes(length);
} }
@ -282,15 +288,10 @@ public final class ModuleInfo {
throw invalidModuleDescriptor(MODULE + " attribute not found"); throw invalidModuleDescriptor(MODULE + " attribute not found");
} }
// ModuleMainClass and ModuleTarget attributes // ModuleMainClass attribute
if (mainClass != null) { if (mainClass != null) {
builder.mainClass(mainClass); builder.mainClass(mainClass);
} }
if (osValues != null) {
if (osValues[0] != null) builder.osName(osValues[0]);
if (osValues[1] != null) builder.osArch(osValues[1]);
if (osValues[2] != null) builder.osVersion(osValues[2]);
}
// If the ModulePackages attribute is not present then the packageFinder // If the ModulePackages attribute is not present then the packageFinder
// is used to find the set of packages // is used to find the set of packages
@ -323,7 +324,10 @@ public final class ModuleInfo {
} }
ModuleDescriptor descriptor = builder.build(); ModuleDescriptor descriptor = builder.build();
return new Attributes(descriptor, hashes, moduleResolution); return new Attributes(descriptor,
moduleTarget,
moduelHashes,
moduleResolution);
} }
/** /**
@ -422,7 +426,11 @@ public final class ModuleInfo {
Set<String> targets = new HashSet<>(exports_to_count); Set<String> targets = new HashSet<>(exports_to_count);
for (int j=0; j<exports_to_count; j++) { for (int j=0; j<exports_to_count; j++) {
int exports_to_index = in.readUnsignedShort(); int exports_to_index = in.readUnsignedShort();
targets.add(cpool.getModuleName(exports_to_index)); String target = cpool.getModuleName(exports_to_index);
if (!targets.add(target)) {
throw invalidModuleDescriptor(pkg + " exported to "
+ target + " more than once");
}
} }
builder.exports(mods, pkg, targets); builder.exports(mods, pkg, targets);
} else { } else {
@ -458,7 +466,11 @@ public final class ModuleInfo {
Set<String> targets = new HashSet<>(open_to_count); Set<String> targets = new HashSet<>(open_to_count);
for (int j=0; j<open_to_count; j++) { for (int j=0; j<open_to_count; j++) {
int opens_to_index = in.readUnsignedShort(); int opens_to_index = in.readUnsignedShort();
targets.add(cpool.getModuleName(opens_to_index)); String target = cpool.getModuleName(opens_to_index);
if (!targets.add(target)) {
throw invalidModuleDescriptor(pkg + " opened to "
+ target + " more than once");
}
} }
builder.opens(mods, pkg, targets); builder.opens(mods, pkg, targets);
} else { } else {
@ -486,7 +498,10 @@ public final class ModuleInfo {
for (int j=0; j<with_count; j++) { for (int j=0; j<with_count; j++) {
index = in.readUnsignedShort(); index = in.readUnsignedShort();
String pn = cpool.getClassName(index); String pn = cpool.getClassName(index);
providers.add(pn); if (!providers.add(pn)) {
throw invalidModuleDescriptor(sn + " provides " + pn
+ " more than once");
}
} }
builder.provides(sn, providers); builder.provides(sn, providers);
} }
@ -528,24 +543,21 @@ public final class ModuleInfo {
/** /**
* Reads the ModuleTarget attribute * Reads the ModuleTarget attribute
*/ */
private String[] readModuleTargetAttribute(DataInput in, ConstantPool cpool) private ModuleTarget readModuleTargetAttribute(DataInput in, ConstantPool cpool)
throws IOException throws IOException
{ {
String[] values = new String[3]; String osName = null;
String osArch = null;
int name_index = in.readUnsignedShort(); int name_index = in.readUnsignedShort();
if (name_index != 0) if (name_index != 0)
values[0] = cpool.getUtf8(name_index); osName = cpool.getUtf8(name_index);
int arch_index = in.readUnsignedShort(); int arch_index = in.readUnsignedShort();
if (arch_index != 0) if (arch_index != 0)
values[1] = cpool.getUtf8(arch_index); osArch = cpool.getUtf8(arch_index);
int version_index = in.readUnsignedShort(); return new ModuleTarget(osName, osArch);
if (version_index != 0)
values[2] = cpool.getUtf8(version_index);
return values;
} }

View File

@ -65,7 +65,6 @@ public final class ModuleInfoExtender {
// the values for the ModuleTarget attribute // the values for the ModuleTarget attribute
private String osName; private String osName;
private String osArch; private String osArch;
private String osVersion;
// the hashes for the ModuleHashes attribute // the hashes for the ModuleHashes attribute
private ModuleHashes hashes; private ModuleHashes hashes;
@ -111,12 +110,9 @@ public final class ModuleInfoExtender {
/** /**
* Sets the values for the ModuleTarget attribute. * Sets the values for the ModuleTarget attribute.
*/ */
public ModuleInfoExtender targetPlatform(String osName, public ModuleInfoExtender targetPlatform(String osName, String osArch) {
String osArch,
String osVersion) {
this.osName = osName; this.osName = osName;
this.osArch = osArch; this.osArch = osArch;
this.osVersion = osVersion;
return this; return this;
} }
@ -203,8 +199,8 @@ public final class ModuleInfoExtender {
cv.addAttribute(new ModulePackagesAttribute(packages)); cv.addAttribute(new ModulePackagesAttribute(packages));
if (mainClass != null) if (mainClass != null)
cv.addAttribute(new ModuleMainClassAttribute(mainClass)); cv.addAttribute(new ModuleMainClassAttribute(mainClass));
if (osName != null || osArch != null || osVersion != null) if (osName != null || osArch != null)
cv.addAttribute(new ModuleTargetAttribute(osName, osArch, osVersion)); cv.addAttribute(new ModuleTargetAttribute(osName, osArch));
if (hashes != null) if (hashes != null)
cv.addAttribute(new ModuleHashesAttribute(hashes)); cv.addAttribute(new ModuleHashesAttribute(hashes));
if (moduleResolution != null) if (moduleResolution != null)

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -48,7 +48,7 @@ public final class ModuleInfoWriter {
* Writes the given module descriptor to a module-info.class file, * Writes the given module descriptor to a module-info.class file,
* returning it in a byte array. * returning it in a byte array.
*/ */
private static byte[] toModuleInfo(ModuleDescriptor md) { private static byte[] toModuleInfo(ModuleDescriptor md, ModuleTarget target) {
ClassWriter cw = new ClassWriter(0); ClassWriter cw = new ClassWriter(0);
cw.visit(Opcodes.V1_9, ACC_MODULE, "module-info", null, null, null); cw.visit(Opcodes.V1_9, ACC_MODULE, "module-info", null, null, null);
cw.visitAttribute(new ModuleAttribute(md)); cw.visitAttribute(new ModuleAttribute(md));
@ -66,18 +66,29 @@ public final class ModuleInfoWriter {
// write ModuleMainClass if the module has a main class // write ModuleMainClass if the module has a main class
md.mainClass().ifPresent(mc -> cw.visitAttribute(new ModuleMainClassAttribute(mc))); md.mainClass().ifPresent(mc -> cw.visitAttribute(new ModuleMainClassAttribute(mc)));
// write ModuleTarget attribute if have any of OS name/arch/version // write ModuleTarget if there is a platform OS/arch
String osName = md.osName().orElse(null); if (target != null) {
String osArch = md.osArch().orElse(null); cw.visitAttribute(new ModuleTargetAttribute(target.osName(),
String osVersion = md.osVersion().orElse(null); target.osArch()));
if (osName != null || osArch != null || osVersion != null) {
cw.visitAttribute(new ModuleTargetAttribute(osName, osArch, osVersion));
} }
cw.visitEnd(); cw.visitEnd();
return cw.toByteArray(); return cw.toByteArray();
} }
/**
* Writes a module descriptor to the given output stream as a
* module-info.class.
*/
public static void write(ModuleDescriptor descriptor,
ModuleTarget target,
OutputStream out)
throws IOException
{
byte[] bytes = toModuleInfo(descriptor, target);
out.write(bytes);
}
/** /**
* Writes a module descriptor to the given output stream as a * Writes a module descriptor to the given output stream as a
* module-info.class. * module-info.class.
@ -85,8 +96,7 @@ public final class ModuleInfoWriter {
public static void write(ModuleDescriptor descriptor, OutputStream out) public static void write(ModuleDescriptor descriptor, OutputStream out)
throws IOException throws IOException
{ {
byte[] bytes = toModuleInfo(descriptor); write(descriptor, null, out);
out.write(bytes);
} }
/** /**
@ -94,8 +104,7 @@ public final class ModuleInfoWriter {
* in module-info.class format. * in module-info.class format.
*/ */
public static ByteBuffer toByteBuffer(ModuleDescriptor descriptor) { public static ByteBuffer toByteBuffer(ModuleDescriptor descriptor) {
byte[] bytes = toModuleInfo(descriptor); byte[] bytes = toModuleInfo(descriptor, null);
return ByteBuffer.wrap(bytes); return ByteBuffer.wrap(bytes);
} }
} }

View File

@ -55,7 +55,6 @@ import java.util.stream.Collectors;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.loader.Resource; import jdk.internal.loader.Resource;
import jdk.internal.loader.ResourceHelper;
import jdk.internal.misc.JavaLangModuleAccess; import jdk.internal.misc.JavaLangModuleAccess;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import sun.net.www.ParseUtil; import sun.net.www.ParseUtil;
@ -165,9 +164,6 @@ public final class ModulePatcher {
descriptor.version().ifPresent(builder::version); descriptor.version().ifPresent(builder::version);
descriptor.mainClass().ifPresent(builder::mainClass); descriptor.mainClass().ifPresent(builder::mainClass);
descriptor.osName().ifPresent(builder::osName);
descriptor.osArch().ifPresent(builder::osArch);
descriptor.osVersion().ifPresent(builder::osVersion);
// original + new packages // original + new packages
builder.packages(descriptor.packages()); builder.packages(descriptor.packages());
@ -179,10 +175,12 @@ public final class ModulePatcher {
// return a module reference to the patched module // return a module reference to the patched module
URI location = mref.location().orElse(null); URI location = mref.location().orElse(null);
ModuleTarget target = null;
ModuleHashes recordedHashes = null; ModuleHashes recordedHashes = null;
ModuleResolution mres = null; ModuleResolution mres = null;
if (mref instanceof ModuleReferenceImpl) { if (mref instanceof ModuleReferenceImpl) {
ModuleReferenceImpl impl = (ModuleReferenceImpl)mref; ModuleReferenceImpl impl = (ModuleReferenceImpl)mref;
target = impl.moduleTarget();
recordedHashes = impl.recordedHashes(); recordedHashes = impl.recordedHashes();
mres = impl.moduleResolution(); mres = impl.moduleResolution();
} }
@ -191,6 +189,7 @@ public final class ModulePatcher {
location, location,
() -> new PatchedModuleReader(paths, mref), () -> new PatchedModuleReader(paths, mref),
this, this,
target,
recordedHashes, recordedHashes,
null, null,
mres); mres);
@ -226,7 +225,7 @@ public final class ModulePatcher {
private volatile ModuleReader delegate; private volatile ModuleReader delegate;
/** /**
* Creates the ModuleReader to reads resources a patched module. * Creates the ModuleReader to reads resources in a patched module.
*/ */
PatchedModuleReader(List<Path> patches, ModuleReference mref) { PatchedModuleReader(List<Path> patches, ModuleReference mref) {
List<ResourceFinder> finders = new ArrayList<>(); List<ResourceFinder> finders = new ArrayList<>();
@ -291,14 +290,17 @@ public final class ModulePatcher {
} }
/** /**
* Finds a resources in the patch locations. Returns null if not found. * Finds a resources in the patch locations. Returns null if not found
* or the name is "module-info.class" as that cannot be overridden.
*/ */
private Resource findResourceInPatch(String name) throws IOException { private Resource findResourceInPatch(String name) throws IOException {
if (!name.equals("module-info.class")) {
for (ResourceFinder finder : finders) { for (ResourceFinder finder : finders) {
Resource r = finder.find(name); Resource r = finder.find(name);
if (r != null) if (r != null)
return r; return r;
} }
}
return null; return null;
} }
@ -478,9 +480,7 @@ public final class ModulePatcher {
@Override @Override
public Stream<String> list() throws IOException { public Stream<String> list() throws IOException {
return jf.stream() return jf.stream().map(JarEntry::getName);
.filter(e -> !e.isDirectory())
.map(JarEntry::getName);
} }
} }
@ -500,15 +500,13 @@ public final class ModulePatcher {
@Override @Override
public Resource find(String name) throws IOException { public Resource find(String name) throws IOException {
Path path = ResourceHelper.toFilePath(name); Path file = Resources.toFilePath(dir, name);
if (path != null) { if (file != null) {
Path file = dir.resolve(path);
if (Files.isRegularFile(file)) {
return newResource(name, dir, file); return newResource(name, dir, file);
} } else {
}
return null; return null;
} }
}
private Resource newResource(String name, Path top, Path file) { private Resource newResource(String name, Path top, Path file) {
return new Resource() { return new Resource() {
@ -550,11 +548,9 @@ public final class ModulePatcher {
@Override @Override
public Stream<String> list() throws IOException { public Stream<String> list() throws IOException {
return Files.find(dir, Integer.MAX_VALUE, return Files.walk(dir, Integer.MAX_VALUE)
(path, attrs) -> attrs.isRegularFile()) .map(f -> Resources.toResourceName(dir, f))
.map(f -> dir.relativize(f) .filter(s -> s.length() > 0);
.toString()
.replace(File.separatorChar, '/'));
} }
} }

View File

@ -513,7 +513,7 @@ public class ModulePath implements ModuleFinder {
String pn = packageName(cn); String pn = packageName(cn);
if (!packages.contains(pn)) { if (!packages.contains(pn)) {
String msg = "Provider class " + cn + " not in module"; String msg = "Provider class " + cn + " not in module";
throw new IOException(msg); throw new InvalidModuleDescriptorException(msg);
} }
providerClasses.add(cn); providerClasses.add(cn);
} }
@ -533,7 +533,7 @@ public class ModulePath implements ModuleFinder {
String pn = packageName(mainClass); String pn = packageName(mainClass);
if (!packages.contains(pn)) { if (!packages.contains(pn)) {
String msg = "Main-Class " + mainClass + " not in module"; String msg = "Main-Class " + mainClass + " not in module";
throw new IOException(msg); throw new InvalidModuleDescriptorException(msg);
} }
builder.mainClass(mainClass); builder.mainClass(mainClass);
} }
@ -609,10 +609,9 @@ public class ModulePath implements ModuleFinder {
// no module-info.class so treat it as automatic module // no module-info.class so treat it as automatic module
try { try {
ModuleDescriptor md = deriveModuleDescriptor(jf); ModuleDescriptor md = deriveModuleDescriptor(jf);
attrs = new ModuleInfo.Attributes(md, null, null); attrs = new ModuleInfo.Attributes(md, null, null, null);
} catch (IllegalArgumentException e) { } catch (RuntimeException e) {
throw new FindException( throw new FindException("Unable to derive module descriptor for "
"Unable to derive module descriptor for: "
+ jf.getName(), e); + jf.getName(), e);
} }
@ -672,18 +671,18 @@ public class ModulePath implements ModuleFinder {
/** /**
* Maps the name of an entry in a JAR or ZIP file to a package name. * Maps the name of an entry in a JAR or ZIP file to a package name.
* *
* @throws IllegalArgumentException if the name is a class file in * @throws InvalidModuleDescriptorException if the name is a class file in
* the top-level directory of the JAR/ZIP file (and it's * the top-level directory of the JAR/ZIP file (and it's not
* not module-info.class) * module-info.class)
*/ */
private Optional<String> toPackageName(String name) { private Optional<String> toPackageName(String name) {
assert !name.endsWith("/"); assert !name.endsWith("/");
int index = name.lastIndexOf("/"); int index = name.lastIndexOf("/");
if (index == -1) { if (index == -1) {
if (name.endsWith(".class") && !name.equals(MODULE_INFO)) { if (name.endsWith(".class") && !name.equals(MODULE_INFO)) {
throw new IllegalArgumentException(name String msg = name + " found in top-level directory"
+ " found in top-level directory" + " (unnamed package not allowed in module)";
+ " (unnamed package not allowed in module)"); throw new InvalidModuleDescriptorException(msg);
} }
return Optional.empty(); return Optional.empty();
} }
@ -701,7 +700,7 @@ public class ModulePath implements ModuleFinder {
* Maps the relative path of an entry in an exploded module to a package * Maps the relative path of an entry in an exploded module to a package
* name. * name.
* *
* @throws IllegalArgumentException if the name is a class file in * @throws InvalidModuleDescriptorException if the name is a class file in
* the top-level directory (and it's not module-info.class) * the top-level directory (and it's not module-info.class)
*/ */
private Optional<String> toPackageName(Path file) { private Optional<String> toPackageName(Path file) {
@ -711,9 +710,9 @@ public class ModulePath implements ModuleFinder {
if (parent == null) { if (parent == null) {
String name = file.toString(); String name = file.toString();
if (name.endsWith(".class") && !name.equals(MODULE_INFO)) { if (name.endsWith(".class") && !name.equals(MODULE_INFO)) {
throw new IllegalArgumentException(name String msg = name + " found in top-level directory"
+ " found in top-level directory" + " (unnamed package not allowed in module)";
+ " (unnamed package not allowed in module)"); throw new InvalidModuleDescriptorException(msg);
} }
return Optional.empty(); return Optional.empty();
} }

View File

@ -46,6 +46,9 @@ public class ModuleReferenceImpl extends ModuleReference {
// non-null if the module is patched // non-null if the module is patched
private final ModulePatcher patcher; private final ModulePatcher patcher;
// ModuleTarget if the module is OS/architecture specific
private final ModuleTarget target;
// the hashes of other modules recorded in this module // the hashes of other modules recorded in this module
private final ModuleHashes recordedHashes; private final ModuleHashes recordedHashes;
@ -65,6 +68,7 @@ public class ModuleReferenceImpl extends ModuleReference {
URI location, URI location,
Supplier<ModuleReader> readerSupplier, Supplier<ModuleReader> readerSupplier,
ModulePatcher patcher, ModulePatcher patcher,
ModuleTarget target,
ModuleHashes recordedHashes, ModuleHashes recordedHashes,
ModuleHashes.HashSupplier hasher, ModuleHashes.HashSupplier hasher,
ModuleResolution moduleResolution) ModuleResolution moduleResolution)
@ -72,6 +76,7 @@ public class ModuleReferenceImpl extends ModuleReference {
super(descriptor, Objects.requireNonNull(location)); super(descriptor, Objects.requireNonNull(location));
this.readerSupplier = readerSupplier; this.readerSupplier = readerSupplier;
this.patcher = patcher; this.patcher = patcher;
this.target = target;
this.recordedHashes = recordedHashes; this.recordedHashes = recordedHashes;
this.hasher = hasher; this.hasher = hasher;
this.moduleResolution = moduleResolution; this.moduleResolution = moduleResolution;
@ -93,6 +98,13 @@ public class ModuleReferenceImpl extends ModuleReference {
return (patcher != null); return (patcher != null);
} }
/**
* Returns the ModuleTarget or {@code null} if the no target platform.
*/
public ModuleTarget moduleTarget() {
return target;
}
/** /**
* Returns the hashes recorded in this module or {@code null} if there * Returns the hashes recorded in this module or {@code null} if there
* are no hashes recorded. * are no hashes recorded.

View File

@ -25,7 +25,6 @@
package jdk.internal.module; package jdk.internal.module;
import java.io.File;
import java.io.IOError; import java.io.IOError;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
@ -50,7 +49,6 @@ import java.util.stream.Stream;
import java.util.zip.ZipFile; import java.util.zip.ZipFile;
import jdk.internal.jmod.JmodFile; import jdk.internal.jmod.JmodFile;
import jdk.internal.loader.ResourceHelper;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.ModuleHashes.HashSupplier; import jdk.internal.module.ModuleHashes.HashSupplier;
import jdk.internal.util.jar.VersionedStream; import jdk.internal.util.jar.VersionedStream;
@ -78,6 +76,7 @@ class ModuleReferences {
uri, uri,
supplier, supplier,
null, null,
attrs.target(),
attrs.recordedHashes(), attrs.recordedHashes(),
hasher, hasher,
attrs.moduleResolution()); attrs.moduleResolution());
@ -242,8 +241,7 @@ class ModuleReferences {
} }
private JarEntry getEntry(String name) { private JarEntry getEntry(String name) {
JarEntry entry = jf.getJarEntry(Objects.requireNonNull(name)); return jf.getJarEntry(Objects.requireNonNull(name));
return (entry == null || entry.isDirectory()) ? null : entry;
} }
@Override @Override
@ -252,6 +250,8 @@ class ModuleReferences {
if (je != null) { if (je != null) {
if (jf.isMultiRelease()) if (jf.isMultiRelease())
name = SharedSecrets.javaUtilJarAccess().getRealName(jf, je); name = SharedSecrets.javaUtilJarAccess().getRealName(jf, je);
if (je.isDirectory() && !name.endsWith("/"))
name += "/";
String encodedPath = ParseUtil.encodePath(name, false); String encodedPath = ParseUtil.encodePath(name, false);
String uris = "jar:" + uri + "!/" + encodedPath; String uris = "jar:" + uri + "!/" + encodedPath;
return Optional.of(URI.create(uris)); return Optional.of(URI.create(uris));
@ -274,7 +274,6 @@ class ModuleReferences {
Stream<String> implList() throws IOException { Stream<String> implList() throws IOException {
// take snapshot to avoid async close // take snapshot to avoid async close
List<String> names = VersionedStream.stream(jf) List<String> names = VersionedStream.stream(jf)
.filter(e -> !e.isDirectory())
.map(JarEntry::getName) .map(JarEntry::getName)
.collect(Collectors.toList()); .collect(Collectors.toList());
return names.stream(); return names.stream();
@ -316,6 +315,8 @@ class ModuleReferences {
Optional<URI> implFind(String name) { Optional<URI> implFind(String name) {
JmodFile.Entry je = getEntry(name); JmodFile.Entry je = getEntry(name);
if (je != null) { if (je != null) {
if (je.isDirectory() && !name.endsWith("/"))
name += "/";
String encodedPath = ParseUtil.encodePath(name, false); String encodedPath = ParseUtil.encodePath(name, false);
String uris = "jmod:" + uri + "!/" + encodedPath; String uris = "jmod:" + uri + "!/" + encodedPath;
return Optional.of(URI.create(uris)); return Optional.of(URI.create(uris));
@ -376,26 +377,10 @@ class ModuleReferences {
if (closed) throw new IOException("ModuleReader is closed"); if (closed) throw new IOException("ModuleReader is closed");
} }
/**
* Returns a Path to access the given resource. Returns null if the
* resource name does not convert to a file path that locates a regular
* file in the module.
*/
private Path toFilePath(String name) {
Path path = ResourceHelper.toFilePath(name);
if (path != null) {
Path file = dir.resolve(path);
if (Files.isRegularFile(file)) {
return file;
}
}
return null;
}
@Override @Override
public Optional<URI> find(String name) throws IOException { public Optional<URI> find(String name) throws IOException {
ensureOpen(); ensureOpen();
Path path = toFilePath(name); Path path = Resources.toFilePath(dir, name);
if (path != null) { if (path != null) {
try { try {
return Optional.of(path.toUri()); return Optional.of(path.toUri());
@ -410,7 +395,7 @@ class ModuleReferences {
@Override @Override
public Optional<InputStream> open(String name) throws IOException { public Optional<InputStream> open(String name) throws IOException {
ensureOpen(); ensureOpen();
Path path = toFilePath(name); Path path = Resources.toFilePath(dir, name);
if (path != null) { if (path != null) {
return Optional.of(Files.newInputStream(path)); return Optional.of(Files.newInputStream(path));
} else { } else {
@ -421,7 +406,7 @@ class ModuleReferences {
@Override @Override
public Optional<ByteBuffer> read(String name) throws IOException { public Optional<ByteBuffer> read(String name) throws IOException {
ensureOpen(); ensureOpen();
Path path = toFilePath(name); Path path = Resources.toFilePath(dir, name);
if (path != null) { if (path != null) {
return Optional.of(ByteBuffer.wrap(Files.readAllBytes(path))); return Optional.of(ByteBuffer.wrap(Files.readAllBytes(path)));
} else { } else {
@ -432,12 +417,9 @@ class ModuleReferences {
@Override @Override
public Stream<String> list() throws IOException { public Stream<String> list() throws IOException {
ensureOpen(); ensureOpen();
// sym links not followed return Files.walk(dir, Integer.MAX_VALUE)
return Files.find(dir, Integer.MAX_VALUE, .map(f -> Resources.toResourceName(dir, f))
(path, attrs) -> attrs.isRegularFile()) .filter(s -> s.length() > 0);
.map(f -> dir.relativize(f)
.toString()
.replace(File.separatorChar, '/'));
} }
@Override @Override

View File

@ -0,0 +1,46 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package jdk.internal.module;
public final class ModuleTarget {
private final String osName;
private final String osArch;
public ModuleTarget(String osName, String osArch) {
this.osName = osName;
this.osArch = osArch;
}
public String osName() {
return osName;
}
public String osArch() {
return osArch;
}
}

View File

@ -31,7 +31,6 @@ import java.lang.reflect.Module;
import java.net.URI; import java.net.URI;
import java.security.AccessController; import java.security.AccessController;
import java.security.PrivilegedAction; import java.security.PrivilegedAction;
import java.util.Set;
import jdk.internal.loader.BootLoader; import jdk.internal.loader.BootLoader;
import jdk.internal.loader.ClassLoaders; import jdk.internal.loader.ClassLoaders;
@ -39,10 +38,10 @@ import jdk.internal.misc.JavaLangReflectModuleAccess;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
/** /**
* A helper class to allow JDK classes create dynamic modules and to update * A helper class for creating and updating modules. This class is intended to
* modules, exports and the readability graph. It is also invoked by the VM * support command-line options, tests, and the instrumentation API. It is also
* to add read edges when agents are instrumenting code that need to link * used by the VM to add read edges when agents are instrumenting code that
* to supporting classes. * need to link to supporting classes.
* *
* The parameters that are package names in this API are the fully-qualified * The parameters that are package names in this API are the fully-qualified
* names of the packages as defined in section 6.5.3 of <cite>The Java&trade; * names of the packages as defined in section 6.5.3 of <cite>The Java&trade;
@ -72,25 +71,7 @@ public class Modules {
} }
/** /**
* Define a new module to the VM. The module has the given set of * Updates m1 to read m2.
* packages and is defined to the given class loader.
*
* The resulting Module is in a larval state in that it does not not read
* any other module and does not have any exports.
*/
public static Module defineModule(ClassLoader loader,
String name,
Set<String> packages)
{
ModuleDescriptor descriptor = ModuleDescriptor.newModule(name)
.packages(packages)
.build();
return JLRMA.defineModule(loader, descriptor, null);
}
/**
* Adds a read-edge so that module {@code m1} reads module {@code m1}.
* Same as m1.addReads(m2) but without a caller check. * Same as m1.addReads(m2) but without a caller check.
*/ */
public static void addReads(Module m1, Module m2) { public static void addReads(Module m1, Module m2) {
@ -98,20 +79,45 @@ public class Modules {
} }
/** /**
* Update module {@code m} to read all unnamed modules. * Update module m to read all unnamed modules.
*/ */
public static void addReadsAllUnnamed(Module m) { public static void addReadsAllUnnamed(Module m) {
JLRMA.addReadsAllUnnamed(m); JLRMA.addReadsAllUnnamed(m);
} }
/**
* Update module m to export a package to all modules.
*
* This method is for intended for use by tests only.
*/
public static void addExports(Module m, String pn) {
JLRMA.addExports(m, pn);
}
/** /**
* Updates module m1 to export a package to module m2. * Updates module m1 to export a package to module m2.
* Same as m1.addExports(pn, m2) but without a caller check. * Same as m1.addExports(pn, m2) but without a caller check
*/ */
public static void addExports(Module m1, String pn, Module m2) { public static void addExports(Module m1, String pn, Module m2) {
JLRMA.addExports(m1, pn, m2); JLRMA.addExports(m1, pn, m2);
} }
/**
* Updates module m to export a package to all unnamed modules.
*/
public static void addExportsToAllUnnamed(Module m, String pn) {
JLRMA.addExportsToAllUnnamed(m, pn);
}
/**
* Update module m to open a package to all modules.
*
* This method is for intended for use by tests only.
*/
public static void addOpens(Module m, String pn) {
JLRMA.addOpens(m, pn);
}
/** /**
* Updates module m1 to open a package to module m2. * Updates module m1 to open a package to module m2.
* Same as m1.addOpens(pn, m2) but without a caller check. * Same as m1.addOpens(pn, m2) but without a caller check.
@ -120,27 +126,6 @@ public class Modules {
JLRMA.addOpens(m1, pn, m2); JLRMA.addOpens(m1, pn, m2);
} }
/**
* Updates a module m to export a package to all modules.
*/
public static void addExportsToAll(Module m, String pn) {
JLRMA.addExportsToAll(m, pn);
}
/**
* Updates a module m to open a package to all modules.
*/
public static void addOpensToAll(Module m, String pn) {
JLRMA.addOpensToAll(m, pn);
}
/**
* Updates module m to export a package to all unnamed modules.
*/
public static void addExportsToAllUnnamed(Module m, String pn) {
JLRMA.addExportsToAllUnnamed(m, pn);
}
/** /**
* Updates module m to open a package to all unnamed modules. * Updates module m to open a package to all unnamed modules.
*/ */
@ -149,7 +134,8 @@ public class Modules {
} }
/** /**
* Updates module m to use a service * Updates module m to use a service.
* Same as m2.addUses(service) but without a caller check.
*/ */
public static void addUses(Module m, Class<?> service) { public static void addUses(Module m, Class<?> service) {
JLRMA.addUses(m, service); JLRMA.addUses(m, service);
@ -182,16 +168,6 @@ public class Modules {
} }
} }
/**
* Adds a package to a module's content.
*
* This method is a no-op if the module already contains the package or the
* module is an unnamed module.
*/
public static void addPackage(Module m, String pn) {
JLRMA.addPackage(m, pn);
}
/** /**
* Called by the VM when code in the given Module has been transformed by * Called by the VM when code in the given Module has been transformed by
* an agent and so may have been instrumented to call into supporting * an agent and so may have been instrumented to call into supporting

View File

@ -0,0 +1,167 @@
/*
* Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package jdk.internal.module;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.attribute.BasicFileAttributes;
/**
* A helper class to support working with resources in modules. Also provides
* support for translating resource names to file paths.
*/
public final class Resources {
private Resources() { }
/**
* Return true if a resource can be encapsulated. Resource with names
* ending in ".class" or "/" cannot be encapsulated. Resource names
* that map to a legal package name can be encapsulated.
*/
public static boolean canEncapsulate(String name) {
int len = name.length();
if (len > 6 && name.endsWith(".class")) {
return false;
} else {
return Checks.isPackageName(toPackageName(name));
}
}
/**
* Derive a <em>package name</em> for a resource. The package name
* returned by this method may not be a legal package name. This method
* returns null if the the resource name ends with a "/" (a directory)
* or the resource name does not contain a "/".
*/
public static String toPackageName(String name) {
int index = name.lastIndexOf('/');
if (index == -1 || index == name.length()-1) {
return "";
} else {
return name.substring(0, index).replace("/", ".");
}
}
/**
* Returns a resource name corresponding to the relative file path
* between {@code dir} and {@code file}. If the file is a directory
* then the name will end with a "/", except the top-level directory
* where the empty string is returned.
*/
public static String toResourceName(Path dir, Path file) {
String s = dir.relativize(file)
.toString()
.replace(File.separatorChar, '/');
if (s.length() > 0 && Files.isDirectory(file))
s += "/";
return s;
}
/**
* Returns a file path to a resource in a file tree. If the resource
* name has a trailing "/" then the file path will locate a directory.
* Returns {@code null} if the resource does not map to a file in the
* tree file.
*/
public static Path toFilePath(Path dir, String name) throws IOException {
boolean expectDirectory = name.endsWith("/");
if (expectDirectory) {
name = name.substring(0, name.length() - 1); // drop trailing "/"
}
Path path = toSafeFilePath(name);
if (path != null) {
Path file = dir.resolve(path);
try {
BasicFileAttributes attrs;
attrs = Files.readAttributes(file, BasicFileAttributes.class);
if (attrs.isDirectory()
|| (!attrs.isDirectory() && !expectDirectory))
return file;
} catch (NoSuchFileException ignore) { }
}
return null;
}
/**
* Map a resource name to a "safe" file path. Returns {@code null} if
* the resource name cannot be converted into a "safe" file path.
*
* Resource names with empty elements, or elements that are "." or ".."
* are rejected, as are resource names that translates to a file path
* with a root component.
*/
private static Path toSafeFilePath(String name) {
// scan elements of resource name
int next;
int off = 0;
while ((next = name.indexOf('/', off)) != -1) {
int len = next - off;
if (!mayTranslate(name, off, len)) {
return null;
}
off = next + 1;
}
int rem = name.length() - off;
if (!mayTranslate(name, off, rem)) {
return null;
}
// convert to file path
Path path;
if (File.separatorChar == '/') {
path = Paths.get(name);
} else {
// not allowed to embed file separators
if (name.contains(File.separator))
return null;
path = Paths.get(name.replace('/', File.separatorChar));
}
// file path not allowed to have root component
return (path.getRoot() == null) ? path : null;
}
/**
* Returns {@code true} if the element in a resource name is a candidate
* to translate to the element of a file path.
*/
private static boolean mayTranslate(String name, int off, int len) {
if (len <= 2) {
if (len == 0)
return false;
boolean starsWithDot = (name.charAt(off) == '.');
if (len == 1 && starsWithDot)
return false;
if (len == 2 && starsWithDot && (name.charAt(off+1) == '.'))
return false;
}
return true;
}
}

View File

@ -150,18 +150,21 @@ public class SystemModuleFinder implements ModuleFinder {
System.getProperty("jdk.system.module.finder.disabledFastPath") != null; System.getProperty("jdk.system.module.finder.disabledFastPath") != null;
ModuleDescriptor[] descriptors; ModuleDescriptor[] descriptors;
ModuleTarget[] targets;
ModuleHashes[] recordedHashes; ModuleHashes[] recordedHashes;
ModuleResolution[] moduleResolutions; ModuleResolution[] moduleResolutions;
// fast loading of ModuleDescriptor of system modules // fast loading of ModuleDescriptor of system modules
if (isFastPathSupported() && !disabled) { if (isFastPathSupported() && !disabled) {
descriptors = SystemModules.descriptors(); descriptors = SystemModules.descriptors();
targets = SystemModules.targets();
recordedHashes = SystemModules.hashes(); recordedHashes = SystemModules.hashes();
moduleResolutions = SystemModules.moduleResolutions(); moduleResolutions = SystemModules.moduleResolutions();
} else { } else {
// if fast loading of ModuleDescriptors is disabled // if fast loading of ModuleDescriptors is disabled
// fallback to read module-info.class // fallback to read module-info.class
descriptors = new ModuleDescriptor[n]; descriptors = new ModuleDescriptor[n];
targets = new ModuleTarget[n];
recordedHashes = new ModuleHashes[n]; recordedHashes = new ModuleHashes[n];
moduleResolutions = new ModuleResolution[n]; moduleResolutions = new ModuleResolution[n];
ImageReader imageReader = SystemImage.reader(); ImageReader imageReader = SystemImage.reader();
@ -171,6 +174,7 @@ public class SystemModuleFinder implements ModuleFinder {
ModuleInfo.Attributes attrs = ModuleInfo.Attributes attrs =
ModuleInfo.read(imageReader.getResourceBuffer(loc), null); ModuleInfo.read(imageReader.getResourceBuffer(loc), null);
descriptors[i] = attrs.descriptor(); descriptors[i] = attrs.descriptor();
targets[i] = attrs.target();
recordedHashes[i] = attrs.recordedHashes(); recordedHashes[i] = attrs.recordedHashes();
moduleResolutions[i] = attrs.moduleResolution(); moduleResolutions[i] = attrs.moduleResolution();
} }
@ -206,6 +210,7 @@ public class SystemModuleFinder implements ModuleFinder {
// create the ModuleReference // create the ModuleReference
ModuleReference mref = toModuleReference(md, ModuleReference mref = toModuleReference(md,
targets[i],
recordedHashes[i], recordedHashes[i],
hashSupplier(names[i]), hashSupplier(names[i]),
moduleResolutions[i]); moduleResolutions[i]);
@ -233,6 +238,7 @@ public class SystemModuleFinder implements ModuleFinder {
} }
private ModuleReference toModuleReference(ModuleDescriptor md, private ModuleReference toModuleReference(ModuleDescriptor md,
ModuleTarget target,
ModuleHashes recordedHashes, ModuleHashes recordedHashes,
HashSupplier hasher, HashSupplier hasher,
ModuleResolution mres) { ModuleResolution mres) {
@ -246,9 +252,14 @@ public class SystemModuleFinder implements ModuleFinder {
} }
}; };
ModuleReference mref = ModuleReference mref = new ModuleReferenceImpl(md,
new ModuleReferenceImpl(md, uri, readerSupplier, null, uri,
recordedHashes, hasher, mres); readerSupplier,
null,
target,
recordedHashes,
hasher,
mres);
// may need a reference to a patched module if --patch-module specified // may need a reference to a patched module if --patch-module specified
mref = ModuleBootstrap.patcher().patchIfNeeded(mref); mref = ModuleBootstrap.patcher().patchIfNeeded(mref);

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2015, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -27,7 +27,7 @@ package jdk.internal.module;
import java.lang.module.ModuleDescriptor; import java.lang.module.ModuleDescriptor;
/* /**
* SystemModules class will be generated at link time to create * SystemModules class will be generated at link time to create
* ModuleDescriptor for the system modules directly to improve * ModuleDescriptor for the system modules directly to improve
* the module descriptor reconstitution time. * the module descriptor reconstitution time.
@ -65,7 +65,7 @@ public final class SystemModules {
} }
/** /**
* Returns a non-empty array of ModuleDescriptors in the run-time image. * Returns a non-empty array of ModuleDescriptor objects in the run-time image.
* *
* When running an exploded image it returns an empty array. * When running an exploded image it returns an empty array.
*/ */
@ -73,6 +73,15 @@ public final class SystemModules {
throw new InternalError("expected to be overridden at link time"); throw new InternalError("expected to be overridden at link time");
} }
/**
* Returns a non-empty array of ModuleTarget objects in the run-time image.
*
* When running an exploded image it returns an empty array.
*/
public static ModuleTarget[] targets() {
throw new InternalError("expected to be overridden at link time");
}
/** /**
* Returns a non-empty array of ModuleHashes recorded in each module * Returns a non-empty array of ModuleHashes recorded in each module
* in the run-time image. * in the run-time image.

View File

@ -2508,7 +2508,7 @@ public class ClassReader {
} }
/** /**
* Reads a CONSTANT_Pakcage_info item in {@code b}. This method is * Reads a CONSTANT_Package_info item in {@code b}. This method is
* intended for {@link Attribute} sub slasses, and is normally not needed * intended for {@link Attribute} sub slasses, and is normally not needed
* by class generators or adapters.</i> * by class generators or adapters.</i>
* *

View File

@ -31,9 +31,7 @@ import java.util.HashMap;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import jdk.internal.HotSpotIntrinsicCandidate; import jdk.internal.HotSpotIntrinsicCandidate;
import jdk.internal.misc.SharedSecrets;
import jdk.internal.misc.VM; import jdk.internal.misc.VM;
import sun.security.action.GetPropertyAction;
/** Common utility routines used by both java.lang and /** Common utility routines used by both java.lang and
java.lang.reflect */ java.lang.reflect */
@ -104,39 +102,40 @@ public class Reflection {
int modifiers) int modifiers)
throws IllegalAccessException throws IllegalAccessException
{ {
if (currentClass == null || memberClass == null) {
throw new InternalError();
}
if (!verifyMemberAccess(currentClass, memberClass, targetClass, modifiers)) { if (!verifyMemberAccess(currentClass, memberClass, targetClass, modifiers)) {
throwIllegalAccessException(currentClass, memberClass, targetClass, modifiers); throw newIllegalAccessException(currentClass, memberClass, targetClass, modifiers);
} }
} }
/** /**
* Verify access to a member, returning {@code false} if no access * Verify access to a member and return {@code true} if it is granted.
*
* @param currentClass the class performing the access
* @param memberClass the declaring class of the member being accessed
* @param targetClass the class of target object if accessing instance
* field or method;
* or the declaring class if accessing constructor;
* or null if accessing static field or method
* @param modifiers the member's access modifiers
* @return {@code true} if access to member is granted
*/ */
public static boolean verifyMemberAccess(Class<?> currentClass, public static boolean verifyMemberAccess(Class<?> currentClass,
Class<?> memberClass, Class<?> memberClass,
Class<?> targetClass, Class<?> targetClass,
int modifiers) int modifiers)
{ {
// Verify that currentClass can access a field, method, or
// constructor of memberClass, where that member's access bits are
// "modifiers".
boolean gotIsSameClassPackage = false;
boolean isSameClassPackage = false;
if (currentClass == memberClass) { if (currentClass == memberClass) {
// Always succeeds // Always succeeds
return true; return true;
} }
if (!verifyModuleAccess(currentClass, memberClass)) { if (!verifyModuleAccess(currentClass.getModule(), memberClass)) {
return false; return false;
} }
boolean gotIsSameClassPackage = false;
boolean isSameClassPackage = false;
if (!Modifier.isPublic(getClassAccessFlags(memberClass))) { if (!Modifier.isPublic(getClassAccessFlags(memberClass))) {
isSameClassPackage = isSameClassPackage(currentClass, memberClass); isSameClassPackage = isSameClassPackage(currentClass, memberClass);
gotIsSameClassPackage = true; gotIsSameClassPackage = true;
@ -196,32 +195,21 @@ public class Reflection {
} }
/** /**
* Returns {@code true} if memberClass's's module exports memberClass's * Returns {@code true} if memberClass's module exports memberClass's
* package to currentClass's module. * package to currentModule.
*/ */
public static boolean verifyModuleAccess(Class<?> currentClass,
Class<?> memberClass) {
return verifyModuleAccess(currentClass.getModule(), memberClass);
}
public static boolean verifyModuleAccess(Module currentModule, Class<?> memberClass) { public static boolean verifyModuleAccess(Module currentModule, Class<?> memberClass) {
Module memberModule = memberClass.getModule(); Module memberModule = memberClass.getModule();
if (currentModule == memberModule) {
// module may be null during startup (initLevel 0) // same module (named or unnamed) or both null if called
if (currentModule == memberModule) // before module system is initialized, which means we are
return true; // same module (named or unnamed) // dealing with java.base only.
return true;
} else {
String pkg = memberClass.getPackageName(); String pkg = memberClass.getPackageName();
boolean allowed = memberModule.isExported(pkg, currentModule); return memberModule.isExported(pkg, currentModule);
if (allowed && memberModule.isNamed() && printStackTraceWhenAccessSucceeds()) {
if (!SharedSecrets.getJavaLangReflectModuleAccess()
.isStaticallyExported(memberModule, pkg, currentModule)) {
String msg = currentModule + " allowed access to member of " + memberClass;
new Exception(msg).printStackTrace(System.err);
} }
} }
return allowed;
}
/** /**
* Returns true if two classes in the same package. * Returns true if two classes in the same package.
@ -344,45 +332,13 @@ public class Reflection {
return false; return false;
} }
// true to print a stack trace when access fails
private static volatile boolean printStackWhenAccessFails;
// true to print a stack trace when access succeeds
private static volatile boolean printStackWhenAccessSucceeds;
// true if printStack* values are initialized
private static volatile boolean printStackPropertiesSet;
private static void ensurePrintStackPropertiesSet() {
if (!printStackPropertiesSet && VM.initLevel() >= 1) {
String s = GetPropertyAction.privilegedGetProperty(
"sun.reflect.debugModuleAccessChecks");
if (s != null) {
printStackWhenAccessFails = !s.equalsIgnoreCase("false");
printStackWhenAccessSucceeds = s.equalsIgnoreCase("access");
}
printStackPropertiesSet = true;
}
}
public static boolean printStackTraceWhenAccessFails() {
ensurePrintStackPropertiesSet();
return printStackWhenAccessFails;
}
public static boolean printStackTraceWhenAccessSucceeds() {
ensurePrintStackPropertiesSet();
return printStackWhenAccessSucceeds;
}
/** /**
* Throws IllegalAccessException with the an exception message based on * Returns an IllegalAccessException with an exception message based on
* the access that is denied. * the access that is denied.
*/ */
private static void throwIllegalAccessException(Class<?> currentClass, public static IllegalAccessException newIllegalAccessException(Class<?> currentClass,
Class<?> memberClass, Class<?> memberClass,
Object target, Class<?> targetClass,
int modifiers) int modifiers)
throws IllegalAccessException throws IllegalAccessException
{ {
@ -411,20 +367,6 @@ public class Reflection {
if (m2.isNamed()) msg += " to " + m1; if (m2.isNamed()) msg += " to " + m1;
} }
throwIllegalAccessException(msg); return new IllegalAccessException(msg);
}
/**
* Throws IllegalAccessException with the given exception message.
*/
public static void throwIllegalAccessException(String msg)
throws IllegalAccessException
{
IllegalAccessException e = new IllegalAccessException(msg);
ensurePrintStackPropertiesSet();
if (printStackWhenAccessFails) {
e.printStackTrace(System.err);
}
throw e;
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2001, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -44,6 +44,7 @@ import java.security.PrivilegedAction;
import java.util.Objects; import java.util.Objects;
import java.util.Properties; import java.util.Properties;
import jdk.internal.misc.VM;
import sun.reflect.misc.ReflectUtil; import sun.reflect.misc.ReflectUtil;
import sun.security.action.GetPropertyAction; import sun.security.action.GetPropertyAction;
@ -585,17 +586,10 @@ public class ReflectionFactory {
private static void checkInitted() { private static void checkInitted() {
if (initted) return; if (initted) return;
// Tests to ensure the system properties table is fully // Defer initialization until module system is initialized so as
// initialized. This is needed because reflection code is // to avoid inflation and spinning bytecode in unnamed modules
// called very early in the initialization process (before // during early startup.
// command-line arguments have been parsed and therefore if (!VM.isModuleSystemInited()) {
// these user-settable properties installed.) We assume that
// if System.out is non-null then the System class has been
// fully initialized and that the bulk of the startup code
// has been run.
if (System.out == null) {
// java.lang.System not yet fully initialized
return; return;
} }

View File

@ -125,10 +125,9 @@ module java.base {
jdk.jlink; jdk.jlink;
exports jdk.internal.loader to exports jdk.internal.loader to
java.instrument, java.instrument,
java.logging, java.logging;
jdk.jlink;
exports jdk.internal.jmod to exports jdk.internal.jmod to
jdk.compiler, jdk.compiler, // reflective dependency
jdk.jlink; jdk.jlink;
exports jdk.internal.logger to exports jdk.internal.logger to
java.logging; java.logging;
@ -140,10 +139,7 @@ module java.base {
exports jdk.internal.org.objectweb.asm.tree to exports jdk.internal.org.objectweb.asm.tree to
jdk.jlink; jdk.jlink;
exports jdk.internal.org.objectweb.asm.util to exports jdk.internal.org.objectweb.asm.util to
jdk.jlink,
jdk.scripting.nashorn; jdk.scripting.nashorn;
exports jdk.internal.org.objectweb.asm.tree.analysis to
jdk.jlink;
exports jdk.internal.org.objectweb.asm.commons to exports jdk.internal.org.objectweb.asm.commons to
jdk.scripting.nashorn; jdk.scripting.nashorn;
exports jdk.internal.org.objectweb.asm.signature to exports jdk.internal.org.objectweb.asm.signature to
@ -157,7 +153,6 @@ module java.base {
jdk.jlink; jdk.jlink;
exports jdk.internal.misc to exports jdk.internal.misc to
java.desktop, java.desktop,
jdk.incubator.httpclient,
java.logging, java.logging,
java.management, java.management,
java.naming, java.naming,
@ -166,8 +161,8 @@ module java.base {
java.sql, java.sql,
java.xml, java.xml,
jdk.charsets, jdk.charsets,
jdk.compiler, jdk.compiler, // reflective dependency
jdk.jartool, jdk.incubator.httpclient,
jdk.jdeps, jdk.jdeps,
jdk.jlink, jdk.jlink,
jdk.jshell, jdk.jshell,
@ -210,11 +205,10 @@ module java.base {
jdk.naming.dns; jdk.naming.dns;
exports sun.net.util to exports sun.net.util to
java.desktop, java.desktop,
jdk.jconsole, jdk.jconsole;
jdk.naming.dns;
exports sun.net.www to exports sun.net.www to
jdk.incubator.httpclient,
java.desktop, java.desktop,
jdk.incubator.httpclient,
jdk.jartool; jdk.jartool;
exports sun.net.www.protocol.http to exports sun.net.www.protocol.http to
java.security.jgss; java.security.jgss;

View File

@ -85,6 +85,7 @@ import java.util.stream.Collectors;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.misc.VM; import jdk.internal.misc.VM;
import jdk.internal.module.IllegalAccessLogger;
import jdk.internal.module.Modules; import jdk.internal.module.Modules;
@ -428,14 +429,20 @@ public final class LauncherHelper {
abort(null, "java.launcher.jar.error3", jarname); abort(null, "java.launcher.jar.error3", jarname);
} }
// Add-Exports and Add-Opens to break encapsulation // Add-Exports and Add-Opens to allow illegal access
String exports = mainAttrs.getValue(ADD_EXPORTS); String exports = mainAttrs.getValue(ADD_EXPORTS);
if (exports != null) { if (exports != null) {
addExportsOrOpens(exports, false); String warn = getLocalizedMessage("java.launcher.permitaccess.warning",
jarname, ADD_EXPORTS);
System.err.println(warn);
addExportsOrOpens(exports, false, ADD_EXPORTS);
} }
String opens = mainAttrs.getValue(ADD_OPENS); String opens = mainAttrs.getValue(ADD_OPENS);
if (opens != null) { if (opens != null) {
addExportsOrOpens(opens, true); String warn = getLocalizedMessage("java.launcher.permitaccess.warning",
jarname, ADD_OPENS);
System.err.println(warn);
addExportsOrOpens(opens, true, ADD_OPENS);
} }
/* /*
@ -460,23 +467,36 @@ public final class LauncherHelper {
* Process the Add-Exports or Add-Opens value. The value is * Process the Add-Exports or Add-Opens value. The value is
* {@code <module>/<package> ( <module>/<package>)*}. * {@code <module>/<package> ( <module>/<package>)*}.
*/ */
static void addExportsOrOpens(String value, boolean open) { static void addExportsOrOpens(String value, boolean open, String how) {
IllegalAccessLogger.Builder builder;
IllegalAccessLogger logger = IllegalAccessLogger.illegalAccessLogger();
if (logger == null) {
builder = new IllegalAccessLogger.Builder();
} else {
builder = logger.toBuilder();
}
for (String moduleAndPackage : value.split(" ")) { for (String moduleAndPackage : value.split(" ")) {
String[] s = moduleAndPackage.trim().split("/"); String[] s = moduleAndPackage.trim().split("/");
if (s.length == 2) { if (s.length == 2) {
String mn = s[0]; String mn = s[0];
String pn = s[1]; String pn = s[1];
Layer.boot().findModule(mn).ifPresent(m -> { Layer.boot().findModule(mn).ifPresent(m -> {
if (m.getDescriptor().packages().contains(pn)) { if (m.getDescriptor().packages().contains(pn)) {
if (open) { if (open) {
builder.logAccessToOpenPackage(m, pn, how);
Modules.addOpensToAllUnnamed(m, pn); Modules.addOpensToAllUnnamed(m, pn);
} else { } else {
builder.logAccessToExportedPackage(m, pn, how);
Modules.addExportsToAllUnnamed(m, pn); Modules.addExportsToAllUnnamed(m, pn);
} }
} }
}); });
} }
} }
IllegalAccessLogger.setIllegalAccessLogger(builder.build());
} }
// From src/share/bin/java.c: // From src/share/bin/java.c:

View File

@ -24,12 +24,15 @@
# #
# Translators please note do not translate the options themselves # Translators please note do not translate the options themselves
java.launcher.opt.header = Usage: {0} [options] class [args...]\n\ java.launcher.opt.header = Usage: {0} [options] <mainclass> [args...]\n\
\ (to execute a class)\n or {0} [options] -jar jarfile [args...]\n\ \ (to execute a class)\n or {0} [options] -jar <jarfile> [args...]\n\
\ (to execute a jar file)\n\ \ (to execute a jar file)\n\
\ or {0} [options] -p <modulepath> -m <modulename>[/<mainclass>] [args...]\n\ \ or {0} [options] -m <module>[/<mainclass>] [args...]\n\
\ {0} [options] --module <module>[/<mainclass>] [args...]\n\
\ (to execute the main class in a module)\n\n\ \ (to execute the main class in a module)\n\n\
where options include:\n\n \ Arguments following the main class, -jar <jarfile>, -m or --module\n\
\ <module>/<mainclass> are passed as the arguments to main class.\n\n\
\ where options include:\n\n
java.launcher.opt.datamodel =\ -d{0}\t Deprecated, will be removed in a future release\n java.launcher.opt.datamodel =\ -d{0}\t Deprecated, will be removed in a future release\n
java.launcher.opt.vmselect =\ {0}\t to select the "{1}" VM\n java.launcher.opt.vmselect =\ {0}\t to select the "{1}" VM\n
@ -49,10 +52,6 @@ java.launcher.opt.footer =\ -cp <class search path of directories and zip
\ A {0} separated list of directories, each directory\n\ \ A {0} separated list of directories, each directory\n\
\ is a directory of modules that replace upgradeable\n\ \ is a directory of modules that replace upgradeable\n\
\ modules in the runtime image\n\ \ modules in the runtime image\n\
\ -m <module>[/<mainclass>]\n\
\ --module <modulename>[/<mainclass>]\n\
\ the initial module to resolve, and the name of the main class\n\
\ to execute if not specified by the module\n\
\ --add-modules <modulename>[,<modulename>...]\n\ \ --add-modules <modulename>[,<modulename>...]\n\
\ root modules to resolve in addition to the initial module.\n\ \ root modules to resolve in addition to the initial module.\n\
\ <modulename> can also be ALL-DEFAULT, ALL-SYSTEM,\n\ \ <modulename> can also be ALL-DEFAULT, ALL-SYSTEM,\n\
@ -157,6 +156,10 @@ java.launcher.X.usage=\n\
\ --add-opens <module>/<package>=<target-module>(,<target-module>)*\n\ \ --add-opens <module>/<package>=<target-module>(,<target-module>)*\n\
\ updates <module> to open <package> to\n\ \ updates <module> to open <package> to\n\
\ <target-module>, regardless of module declaration.\n\ \ <target-module>, regardless of module declaration.\n\
\ --permit-illegal-access\n\
\ permit illegal access to members of types in named modules\n\
\ by code in unnamed modules. This compatibility option will\n\
\ be removed in the next release.\n\
\ --disable-@files disable further argument file expansion\n\ \ --disable-@files disable further argument file expansion\n\
\ --patch-module <module>=<file>({0}<file>)*\n\ \ --patch-module <module>=<file>({0}<file>)*\n\
\ Override or augment a module with classes and resources\n\ \ Override or augment a module with classes and resources\n\
@ -208,4 +211,6 @@ java.launcher.module.error2=\
java.launcher.module.error3=\ java.launcher.module.error3=\
Error: Unable to load main class {0} from module {1}\n\ Error: Unable to load main class {0} from module {1}\n\
\t{2} \t{2}
java.launcher.permitaccess.warning=\
WARNING: Main manifest of {0} contains {1} attribute to permit illegal access

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -26,8 +26,6 @@
package sun.net.www.protocol.jrt; package sun.net.www.protocol.jrt;
import java.io.ByteArrayInputStream; import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FilePermission;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.net.MalformedURLException; import java.net.MalformedURLException;
@ -44,7 +42,6 @@ import jdk.internal.loader.URLClassPath;
import jdk.internal.loader.Resource; import jdk.internal.loader.Resource;
import sun.net.www.ParseUtil; import sun.net.www.ParseUtil;
import sun.net.www.URLConnection; import sun.net.www.URLConnection;
import sun.security.action.GetPropertyAction;
/** /**
* URLConnection implementation that can be used to connect to resources * URLConnection implementation that can be used to connect to resources
@ -66,9 +63,6 @@ public class JavaRuntimeURLConnection extends URLConnection {
// the Resource when connected // the Resource when connected
private volatile Resource resource; private volatile Resource resource;
// the permission to access resources in the runtime image, created lazily
private static volatile Permission permission;
JavaRuntimeURLConnection(URL url) throws IOException { JavaRuntimeURLConnection(URL url) throws IOException {
super(url); super(url);
String path = url.getPath(); String path = url.getPath();
@ -164,14 +158,8 @@ public class JavaRuntimeURLConnection extends URLConnection {
} }
@Override @Override
public Permission getPermission() throws IOException { public Permission getPermission() {
Permission p = permission; return new RuntimePermission("accessSystemModules");
if (p == null) {
String home = GetPropertyAction.privilegedGetProperty("java.home");
p = new FilePermission(home + File.separator + "-", "read");
permission = p;
}
return p;
} }
/** /**

View File

@ -72,23 +72,9 @@ getUTF(JNIEnv *env, jstring str, char* localBuf, int bufSize)
return utfStr; return utfStr;
} }
// The existence or signature of this method is not guaranteed since it
// supports a private method. This method will be changed in 1.7.
JNIEXPORT jclass JNICALL
Java_java_lang_ClassLoader_defineClass0(JNIEnv *env,
jobject loader,
jstring name,
jbyteArray data,
jint offset,
jint length,
jobject pd)
{
return Java_java_lang_ClassLoader_defineClass1(env, loader, name, data, offset,
length, pd, NULL);
}
JNIEXPORT jclass JNICALL JNIEXPORT jclass JNICALL
Java_java_lang_ClassLoader_defineClass1(JNIEnv *env, Java_java_lang_ClassLoader_defineClass1(JNIEnv *env,
jclass cls,
jobject loader, jobject loader,
jstring name, jstring name,
jbyteArray data, jbyteArray data,
@ -163,6 +149,7 @@ Java_java_lang_ClassLoader_defineClass1(JNIEnv *env,
JNIEXPORT jclass JNICALL JNIEXPORT jclass JNICALL
Java_java_lang_ClassLoader_defineClass2(JNIEnv *env, Java_java_lang_ClassLoader_defineClass2(JNIEnv *env,
jclass cls,
jobject loader, jobject loader,
jstring name, jstring name,
jobject data, jobject data,

View File

@ -73,30 +73,32 @@ Java_java_lang_reflect_Module_defineModule0(JNIEnv *env, jclass cls, jobject mod
jstring location, jobjectArray packages) jstring location, jobjectArray packages)
{ {
char** pkgs = NULL; char** pkgs = NULL;
jsize idx;
jsize num_packages = (*env)->GetArrayLength(env, packages); jsize num_packages = (*env)->GetArrayLength(env, packages);
if (num_packages != 0 && (pkgs = calloc(num_packages, sizeof(char*))) == NULL) { if (num_packages != 0 && (pkgs = calloc(num_packages, sizeof(char*))) == NULL) {
JNU_ThrowOutOfMemoryError(env, NULL); JNU_ThrowOutOfMemoryError(env, NULL);
return; return;
} else { } else if ((*env)->EnsureLocalCapacity(env, (jint)num_packages) == 0) {
int valid = 1; jboolean failed = JNI_FALSE;
int idx;
for (idx = 0; idx < num_packages; idx++) { for (idx = 0; idx < num_packages; idx++) {
jstring pkg = (*env)->GetObjectArrayElement(env, packages, idx); jstring pkg = (*env)->GetObjectArrayElement(env, packages, idx);
pkgs[idx] = GetInternalPackageName(env, pkg, NULL, 0); char* name = GetInternalPackageName(env, pkg, NULL, 0);
if (pkgs[idx] == NULL) { if (name != NULL) {
valid = 0; pkgs[idx] = name;
} else {
failed = JNI_TRUE;
break; break;
} }
} }
if (!failed) {
if (valid != 0) {
JVM_DefineModule(env, module, is_open, version, location, JVM_DefineModule(env, module, is_open, version, location,
(const char* const*)pkgs, num_packages); (const char* const*)pkgs, num_packages);
} }
} }
if (num_packages > 0) { if (num_packages > 0) {
int idx;
for (idx = 0; idx < num_packages; idx++) { for (idx = 0; idx < num_packages; idx++) {
if (pkgs[idx] != NULL) { if (pkgs[idx] != NULL) {
free(pkgs[idx]); free(pkgs[idx]);

View File

@ -191,6 +191,9 @@ public interface ClassFileTransformer {
* if the input does not represent a well-formed class file * if the input does not represent a well-formed class file
* @return a well-formed class file buffer (the result of the transform), * @return a well-formed class file buffer (the result of the transform),
* or {@code null} if no transform is performed * or {@code null} if no transform is performed
*
* @revised 9
* @spec JPMS
*/ */
default byte[] default byte[]
transform( ClassLoader loader, transform( ClassLoader loader,

View File

@ -63,6 +63,7 @@ import jdk.internal.module.ModuleHashesBuilder;
import jdk.internal.module.ModuleInfo; import jdk.internal.module.ModuleInfo;
import jdk.internal.module.ModuleInfoExtender; import jdk.internal.module.ModuleInfoExtender;
import jdk.internal.module.ModuleResolution; import jdk.internal.module.ModuleResolution;
import jdk.internal.module.ModuleTarget;
import jdk.internal.util.jar.JarIndex; import jdk.internal.util.jar.JarIndex;
import static jdk.internal.util.jar.JarIndex.INDEX_NAME; import static jdk.internal.util.jar.JarIndex.INDEX_NAME;
@ -1780,6 +1781,7 @@ public class Main {
{ {
ModuleInfo.Attributes attrs = ModuleInfo.read(entryInputStream, null); ModuleInfo.Attributes attrs = ModuleInfo.read(entryInputStream, null);
ModuleDescriptor md = attrs.descriptor(); ModuleDescriptor md = attrs.descriptor();
ModuleTarget target = attrs.target();
ModuleHashes hashes = attrs.recordedHashes(); ModuleHashes hashes = attrs.recordedHashes();
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
@ -1824,11 +1826,14 @@ public class Main {
md.mainClass().ifPresent(v -> sb.append("\n main-class " + v)); md.mainClass().ifPresent(v -> sb.append("\n main-class " + v));
md.osName().ifPresent(v -> sb.append("\n operating-system-name " + v)); if (target != null) {
String osName = target.osName();
md.osArch().ifPresent(v -> sb.append("\n operating-system-architecture " + v)); if (osName != null)
sb.append("\n operating-system-name " + osName);
md.osVersion().ifPresent(v -> sb.append("\n operating-system-version " + v)); String osArch = target.osArch();
if (osArch != null)
sb.append("\n operating-system-architecture " + osArch);
}
if (hashes != null) { if (hashes != null) {
hashes.names().stream().sorted().forEach( hashes.names().stream().sorted().forEach(

View File

@ -151,16 +151,8 @@ public final class DefaultImageBuilder implements ImageBuilder {
@Override @Override
public void storeFiles(ResourcePool files) { public void storeFiles(ResourcePool files) {
try { try {
// populate targetOsName field up-front because it's used elsewhere. this.targetOsName = files.moduleView().
Optional<ResourcePoolModule> javaBase = files.moduleView().findModule("java.base"); findModule("java.base").get().osName();
javaBase.ifPresent(mod -> {
// fill release information available from transformed "java.base" module!
ModuleDescriptor desc = mod.descriptor();
desc.osName().ifPresent(s -> {
this.targetOsName = s;
});
});
if (this.targetOsName == null) { if (this.targetOsName == null) {
throw new PluginException("ModuleTarget attribute is missing for java.base module"); throw new PluginException("ModuleTarget attribute is missing for java.base module");
} }

View File

@ -43,6 +43,8 @@ import java.util.stream.Collectors;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.jimage.decompressor.Decompressor; import jdk.internal.jimage.decompressor.Decompressor;
import jdk.internal.module.ModuleInfo.Attributes;
import jdk.internal.module.ModuleTarget;
import jdk.tools.jlink.plugin.Plugin; import jdk.tools.jlink.plugin.Plugin;
import jdk.tools.jlink.builder.ImageBuilder; import jdk.tools.jlink.builder.ImageBuilder;
import jdk.tools.jlink.plugin.PluginException; import jdk.tools.jlink.plugin.PluginException;
@ -298,6 +300,7 @@ public final class ImagePluginStack {
final ResourcePoolModule module; final ResourcePoolModule module;
// lazily initialized // lazily initialized
ModuleDescriptor descriptor; ModuleDescriptor descriptor;
ModuleTarget target;
LastModule(ResourcePoolModule module) { LastModule(ResourcePoolModule module) {
this.module = module; this.module = module;
@ -316,12 +319,30 @@ public final class ImagePluginStack {
@Override @Override
public ModuleDescriptor descriptor() { public ModuleDescriptor descriptor() {
if (descriptor == null) { initModuleAttributes();
descriptor = ResourcePoolManager.readModuleDescriptor(this);
}
return descriptor; return descriptor;
} }
@Override
public String osName() {
initModuleAttributes();
return target != null? target.osName() : null;
}
@Override
public String osArch() {
initModuleAttributes();
return target != null? target.osArch() : null;
}
private void initModuleAttributes() {
if (this.descriptor == null) {
Attributes attr = ResourcePoolManager.readModuleAttributes(this);
this.descriptor = attr.descriptor();
this.target = attr.target();
}
}
@Override @Override
public Set<String> packages() { public Set<String> packages() {
return module.packages(); return module.packages();

View File

@ -54,7 +54,6 @@ import jdk.tools.jlink.internal.ImagePluginStack.ImageProvider;
import jdk.tools.jlink.plugin.PluginException; import jdk.tools.jlink.plugin.PluginException;
import jdk.tools.jlink.builder.DefaultImageBuilder; import jdk.tools.jlink.builder.DefaultImageBuilder;
import jdk.tools.jlink.plugin.Plugin; import jdk.tools.jlink.plugin.Plugin;
import jdk.internal.module.Checks;
import jdk.internal.module.ModulePath; import jdk.internal.module.ModulePath;
import jdk.internal.module.ModuleResolution; import jdk.internal.module.ModuleResolution;
@ -423,13 +422,6 @@ public class JlinkTask {
ModuleFinder.of(), ModuleFinder.of(),
roots); roots);
// emit warning for modules that end with a digit
cf.modules().stream()
.map(ResolvedModule::name)
.filter(mn -> !Checks.hasLegalModuleNameLastCharacter(mn))
.forEach(mn -> System.err.println("WARNING: Module name \""
+ mn + "\" may soon be illegal"));
// emit a warning for any incubating modules in the configuration // emit a warning for any incubating modules in the configuration
if (log != null) { if (log != null) {
String im = cf.modules() String im = cf.modules()

View File

@ -66,9 +66,6 @@ final class ResourcePoolConfiguration {
md.version().ifPresent(builder::version); md.version().ifPresent(builder::version);
md.mainClass().ifPresent(builder::mainClass); md.mainClass().ifPresent(builder::mainClass);
md.osName().ifPresent(builder::osName);
md.osArch().ifPresent(builder::osArch);
md.osVersion().ifPresent(builder::osVersion);
return builder.build(); return builder.build();
} }

View File

@ -35,7 +35,10 @@ import java.util.Optional;
import java.util.Set; import java.util.Set;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.jimage.decompressor.CompressedResourceHeader; import jdk.internal.jimage.decompressor.CompressedResourceHeader;
import jdk.internal.loader.ResourceHelper; import jdk.internal.module.Resources;
import jdk.internal.module.ModuleInfo;
import jdk.internal.module.ModuleInfo.Attributes;
import jdk.internal.module.ModuleTarget;
import jdk.tools.jlink.plugin.ResourcePool; import jdk.tools.jlink.plugin.ResourcePool;
import jdk.tools.jlink.plugin.ResourcePoolBuilder; import jdk.tools.jlink.plugin.ResourcePoolBuilder;
import jdk.tools.jlink.plugin.ResourcePoolEntry; import jdk.tools.jlink.plugin.ResourcePoolEntry;
@ -47,8 +50,8 @@ import jdk.tools.jlink.plugin.PluginException;
* A manager for pool of resources. * A manager for pool of resources.
*/ */
public class ResourcePoolManager { public class ResourcePoolManager {
// utility to read ModuleDescriptor of the given ResourcePoolModule // utility to read Module Attributes of the given ResourcePoolModule
static ModuleDescriptor readModuleDescriptor(ResourcePoolModule mod) { static Attributes readModuleAttributes(ResourcePoolModule mod) {
String p = "/" + mod.name() + "/module-info.class"; String p = "/" + mod.name() + "/module-info.class";
Optional<ResourcePoolEntry> content = mod.findEntry(p); Optional<ResourcePoolEntry> content = mod.findEntry(p);
if (!content.isPresent()) { if (!content.isPresent()) {
@ -57,9 +60,9 @@ public class ResourcePoolManager {
} }
ByteBuffer bb = ByteBuffer.wrap(content.get().contentBytes()); ByteBuffer bb = ByteBuffer.wrap(content.get().contentBytes());
try { try {
return ModuleDescriptor.read(bb); return ModuleInfo.read(bb, null);
} catch (RuntimeException re) { } catch (RuntimeException re) {
throw new RuntimeException("module descriptor cannot be read for " + mod.name(), re); throw new RuntimeException("module info cannot be read for " + mod.name(), re);
} }
} }
@ -68,7 +71,7 @@ public class ResourcePoolManager {
*/ */
public static boolean isNamedPackageResource(String path) { public static boolean isNamedPackageResource(String path) {
return (path.endsWith(".class") && !path.endsWith("module-info.class")) || return (path.endsWith(".class") && !path.endsWith("module-info.class")) ||
!ResourceHelper.isSimpleResource(path); Resources.canEncapsulate(path);
} }
class ResourcePoolModuleImpl implements ResourcePoolModule { class ResourcePoolModuleImpl implements ResourcePoolModule {
@ -76,6 +79,8 @@ public class ResourcePoolManager {
final Map<String, ResourcePoolEntry> moduleContent = new LinkedHashMap<>(); final Map<String, ResourcePoolEntry> moduleContent = new LinkedHashMap<>();
// lazily initialized // lazily initialized
private ModuleDescriptor descriptor; private ModuleDescriptor descriptor;
private ModuleTarget target;
final String name; final String name;
private ResourcePoolModuleImpl(String name) { private ResourcePoolModuleImpl(String name) {
@ -100,12 +105,30 @@ public class ResourcePoolManager {
@Override @Override
public ModuleDescriptor descriptor() { public ModuleDescriptor descriptor() {
if (descriptor == null) { initModuleAttributes();
descriptor = readModuleDescriptor(this);
}
return descriptor; return descriptor;
} }
@Override
public String osName() {
initModuleAttributes();
return target != null? target.osName() : null;
}
@Override
public String osArch() {
initModuleAttributes();
return target != null? target.osArch() : null;
}
private void initModuleAttributes() {
if (this.descriptor == null) {
Attributes attr = readModuleAttributes(this);
this.descriptor = attr.descriptor();
this.target = attr.target();
}
}
@Override @Override
public Set<String> packages() { public Set<String> packages() {
Set<String> pkgs = new HashSet<>(); Set<String> pkgs = new HashSet<>();

View File

@ -115,7 +115,7 @@ public final class ExcludeVMPlugin implements Plugin {
@Override @Override
public ResourcePool transform(ResourcePool in, ResourcePoolBuilder out) { public ResourcePool transform(ResourcePool in, ResourcePoolBuilder out) {
ResourcePoolModule javaBase = in.moduleView().findModule("java.base").get(); ResourcePoolModule javaBase = in.moduleView().findModule("java.base").get();
String[] jvmlibs = jvmlibs(javaBase.descriptor().osName().get()); String[] jvmlibs = jvmlibs(javaBase.osName());
TreeSet<Jvm> existing = new TreeSet<>(new JvmComparator()); TreeSet<Jvm> existing = new TreeSet<>(new JvmComparator());
TreeSet<Jvm> removed = new TreeSet<>(new JvmComparator()); TreeSet<Jvm> removed = new TreeSet<>(new JvmComparator());
if (!keepAll) { if (!keepAll) {

View File

@ -137,15 +137,13 @@ public final class ReleaseInfoPlugin implements Plugin {
javaBase.ifPresent(mod -> { javaBase.ifPresent(mod -> {
// fill release information available from transformed "java.base" module! // fill release information available from transformed "java.base" module!
ModuleDescriptor desc = mod.descriptor(); ModuleDescriptor desc = mod.descriptor();
desc.osName().ifPresent(s -> {
release.put("OS_NAME", quote(s));
});
desc.osVersion().ifPresent(s -> release.put("OS_VERSION", quote(s)));
desc.osArch().ifPresent(s -> release.put("OS_ARCH", quote(s)));
desc.version().ifPresent(s -> release.put("JAVA_VERSION", desc.version().ifPresent(s -> release.put("JAVA_VERSION",
quote(parseVersion(s.toString())))); quote(parseVersion(s.toString()))));
desc.version().ifPresent(s -> release.put("JAVA_FULL_VERSION", desc.version().ifPresent(s -> release.put("JAVA_FULL_VERSION",
quote(s.toString()))); quote(s.toString())));
release.put("OS_NAME", quote(mod.osName()));
release.put("OS_ARCH", quote(mod.osArch()));
}); });
// put topological sorted module names separated by space // put topological sorted module names separated by space

View File

@ -52,6 +52,7 @@ import jdk.internal.module.ModuleHashes;
import jdk.internal.module.ModuleInfo.Attributes; import jdk.internal.module.ModuleInfo.Attributes;
import jdk.internal.module.ModuleInfoExtender; import jdk.internal.module.ModuleInfoExtender;
import jdk.internal.module.ModuleResolution; import jdk.internal.module.ModuleResolution;
import jdk.internal.module.ModuleTarget;
import jdk.internal.module.SystemModules; import jdk.internal.module.SystemModules;
import jdk.internal.org.objectweb.asm.Attribute; import jdk.internal.org.objectweb.asm.Attribute;
import jdk.internal.org.objectweb.asm.ClassReader; import jdk.internal.org.objectweb.asm.ClassReader;
@ -206,12 +207,12 @@ public final class SystemModulesPlugin implements Plugin {
// add ModulePackages attribute if this module contains some packages // add ModulePackages attribute if this module contains some packages
// and ModulePackages is not present // and ModulePackages is not present
this.addModulePackages = packages.size() > 0 && !hasModulePackages(); this.addModulePackages = packages.size() > 0 && !hasModulePackages();
// drop target attribute only if any OS property is present // drop target attribute only if any OS property is present
if (dropModuleTarget) { ModuleTarget target = attrs.target();
this.dropModuleTarget = if (dropModuleTarget && target != null) {
descriptor.osName().isPresent() || this.dropModuleTarget = (target.osName() != null)
descriptor.osArch().isPresent() || || (target.osArch() != null);
descriptor.osVersion().isPresent();
} else { } else {
this.dropModuleTarget = false; this.dropModuleTarget = false;
} }
@ -230,6 +231,10 @@ public final class SystemModulesPlugin implements Plugin {
return packages; return packages;
} }
ModuleTarget target() {
return attrs.target();
}
ModuleHashes recordedHashes() { ModuleHashes recordedHashes() {
return attrs.recordedHashes(); return attrs.recordedHashes();
} }
@ -372,7 +377,7 @@ public final class SystemModulesPlugin implements Plugin {
} }
void dropModuleTarget() { void dropModuleTarget() {
extender.targetPlatform("", "", ""); extender.targetPlatform("", "");
} }
byte[] getBytes() throws IOException { byte[] getBytes() throws IOException {
@ -399,6 +404,10 @@ public final class SystemModulesPlugin implements Plugin {
"java/lang/module/ModuleDescriptor$Exports$Modifier"; "java/lang/module/ModuleDescriptor$Exports$Modifier";
private static final String OPENS_MODIFIER_CLASSNAME = private static final String OPENS_MODIFIER_CLASSNAME =
"java/lang/module/ModuleDescriptor$Opens$Modifier"; "java/lang/module/ModuleDescriptor$Opens$Modifier";
private static final String MODULE_TARGET_CLASSNAME =
"jdk/internal/module/ModuleTarget";
private static final String MODULE_TARGET_ARRAY_SIGNATURE =
"[Ljdk/internal/module/ModuleTarget;";
private static final String MODULE_HASHES_ARRAY_SIGNATURE = private static final String MODULE_HASHES_ARRAY_SIGNATURE =
"[Ljdk/internal/module/ModuleHashes;"; "[Ljdk/internal/module/ModuleHashes;";
private static final String MODULE_RESOLUTION_CLASSNAME = private static final String MODULE_RESOLUTION_CLASSNAME =
@ -414,6 +423,7 @@ public final class SystemModulesPlugin implements Plugin {
private final int BUILDER_VAR = 0; private final int BUILDER_VAR = 0;
private final int MD_VAR = 1; // variable for ModuleDescriptor private final int MD_VAR = 1; // variable for ModuleDescriptor
private final int MT_VAR = 1; // variable for ModuleTarget
private final int MH_VAR = 1; // variable for ModuleHashes private final int MH_VAR = 1; // variable for ModuleHashes
private int nextLocalVar = 2; // index to next local variable private int nextLocalVar = 2; // index to next local variable
@ -515,11 +525,10 @@ public final class SystemModulesPlugin implements Plugin {
if (entry.moduleName().equals("java.base")) { if (entry.moduleName().equals("java.base")) {
moduleInfo = new ModuleInfo(entry.contentBytes(), packages, false); moduleInfo = new ModuleInfo(entry.contentBytes(), packages, false);
ModuleDescriptor md = moduleInfo.descriptor; ModuleDescriptor md = moduleInfo.descriptor;
// drop Moduletarget attribute only if java.base has all OS properties // drop ModuleTarget attribute if java.base has all OS properties
// otherwise, retain it ModuleTarget target = moduleInfo.target();
if (dropModuleTarget && if (dropModuleTarget
md.osName().isPresent() && md.osArch().isPresent() && && (target.osName() != null) && (target.osArch() != null)) {
md.osVersion().isPresent()) {
dropModuleTarget = true; dropModuleTarget = true;
} else { } else {
dropModuleTarget = false; dropModuleTarget = false;
@ -584,15 +593,20 @@ public final class SystemModulesPlugin implements Plugin {
// generate SystemModules::descriptors // generate SystemModules::descriptors
genDescriptorsMethod(); genDescriptorsMethod();
// generate SystemModules::targets
genTargetsMethod();
// generate SystemModules::hashes // generate SystemModules::hashes
genHashesMethod(); genHashesMethod();
// generate SystemModules::moduleResolutions // generate SystemModules::moduleResolutions
genModuleResolutionsMethod(); genModuleResolutionsMethod();
return cw; return cw;
} }
/* /**
* Generate bytecode for SystemModules::descriptors method * Generate bytecode for SystemModules::descriptors method
*/ */
private void genDescriptorsMethod() { private void genDescriptorsMethod() {
@ -616,10 +630,47 @@ public final class SystemModulesPlugin implements Plugin {
mv.visitInsn(ARETURN); mv.visitInsn(ARETURN);
mv.visitMaxs(0, 0); mv.visitMaxs(0, 0);
mv.visitEnd(); mv.visitEnd();
} }
/* /**
* Generate bytecode for SystemModules::targets method
*/
private void genTargetsMethod() {
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC+ACC_STATIC,
"targets",
"()" + MODULE_TARGET_ARRAY_SIGNATURE,
"()" + MODULE_TARGET_ARRAY_SIGNATURE,
null);
mv.visitCode();
pushInt(mv, moduleInfos.size());
mv.visitTypeInsn(ANEWARRAY, MODULE_TARGET_CLASSNAME);
mv.visitVarInsn(ASTORE, MT_VAR);
for (int index=0; index < moduleInfos.size(); index++) {
ModuleInfo minfo = moduleInfos.get(index);
if (minfo.target() != null && !minfo.dropModuleTarget) {
mv.visitVarInsn(ALOAD, MT_VAR);
pushInt(mv, index);
// new ModuleTarget(String, String)
mv.visitTypeInsn(NEW, MODULE_TARGET_CLASSNAME);
mv.visitInsn(DUP);
mv.visitLdcInsn(minfo.target().osName());
mv.visitLdcInsn(minfo.target().osArch());
mv.visitMethodInsn(INVOKESPECIAL, MODULE_TARGET_CLASSNAME,
"<init>", "(Ljava/lang/String;Ljava/lang/String;)V", false);
mv.visitInsn(AASTORE);
}
}
mv.visitVarInsn(ALOAD, MT_VAR);
mv.visitInsn(ARETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
}
/**
* Generate bytecode for SystemModules::hashes method * Generate bytecode for SystemModules::hashes method
*/ */
private void genHashesMethod() { private void genHashesMethod() {
@ -647,10 +698,9 @@ public final class SystemModulesPlugin implements Plugin {
hmv.visitInsn(ARETURN); hmv.visitInsn(ARETURN);
hmv.visitMaxs(0, 0); hmv.visitMaxs(0, 0);
hmv.visitEnd(); hmv.visitEnd();
} }
/* /**
* Generate bytecode for SystemModules::methodResoultions method * Generate bytecode for SystemModules::methodResoultions method
*/ */
private void genModuleResolutionsMethod() { private void genModuleResolutionsMethod() {
@ -749,6 +799,7 @@ public final class SystemModulesPlugin implements Plugin {
final ModuleDescriptor md; final ModuleDescriptor md;
final Set<String> packages; final Set<String> packages;
final int index; final int index;
ModuleDescriptorBuilder(ModuleDescriptor md, Set<String> packages, int index) { ModuleDescriptorBuilder(ModuleDescriptor md, Set<String> packages, int index) {
if (md.isAutomatic()) { if (md.isAutomatic()) {
throw new InternalError("linking automatic module is not supported"); throw new InternalError("linking automatic module is not supported");
@ -786,11 +837,6 @@ public final class SystemModulesPlugin implements Plugin {
// main class // main class
md.mainClass().ifPresent(this::mainClass); md.mainClass().ifPresent(this::mainClass);
// os name, arch, version
targetPlatform(md.osName().orElse(null),
md.osArch().orElse(null),
md.osVersion().orElse(null));
putModuleDescriptor(); putModuleDescriptor();
} }
@ -1088,25 +1134,6 @@ public final class SystemModulesPlugin implements Plugin {
mv.visitInsn(POP); mv.visitInsn(POP);
} }
/*
* Invoke Builder.osName(String name)
* Builder.osArch(String arch)
* Builder.osVersion(String version)
*/
void targetPlatform(String osName, String osArch, String osVersion) {
if (osName != null) {
invokeBuilderMethod("osName", osName);
}
if (osArch != null) {
invokeBuilderMethod("osArch", osArch);
}
if (osVersion != null) {
invokeBuilderMethod("osVersion", osVersion);
}
}
void invokeBuilderMethod(String methodName, String value) { void invokeBuilderMethod(String methodName, String value) {
mv.visitVarInsn(ALOAD, BUILDER_VAR); mv.visitVarInsn(ALOAD, BUILDER_VAR);
mv.visitLdcInsn(value); mv.visitLdcInsn(value);

View File

@ -28,6 +28,7 @@ import java.lang.module.ModuleDescriptor;
import java.util.Optional; import java.util.Optional;
import java.util.Set; import java.util.Set;
import java.util.stream.Stream; import java.util.stream.Stream;
import jdk.internal.module.ModuleTarget;
/** /**
* Link-time representation of a module. * Link-time representation of a module.
@ -56,6 +57,20 @@ public interface ResourcePoolModule {
*/ */
public ModuleDescriptor descriptor(); public ModuleDescriptor descriptor();
/**
* The module target OS name for this module.
*
* @return The module target OS name
*/
public String osName();
/**
* The module target OS arch for this module.
*
* @return The module target OS arch
*/
public String osArch();
/** /**
* Retrieves all the packages located in this module. * Retrieves all the packages located in this module.
* *

View File

@ -96,13 +96,14 @@ import jdk.internal.joptsimple.OptionParser;
import jdk.internal.joptsimple.OptionSet; import jdk.internal.joptsimple.OptionSet;
import jdk.internal.joptsimple.OptionSpec; import jdk.internal.joptsimple.OptionSpec;
import jdk.internal.joptsimple.ValueConverter; import jdk.internal.joptsimple.ValueConverter;
import jdk.internal.loader.ResourceHelper;
import jdk.internal.module.ModuleHashes; import jdk.internal.module.ModuleHashes;
import jdk.internal.module.ModuleHashesBuilder; import jdk.internal.module.ModuleHashesBuilder;
import jdk.internal.module.ModuleInfo; import jdk.internal.module.ModuleInfo;
import jdk.internal.module.ModuleInfoExtender; import jdk.internal.module.ModuleInfoExtender;
import jdk.internal.module.ModulePath; import jdk.internal.module.ModulePath;
import jdk.internal.module.ModuleResolution; import jdk.internal.module.ModuleResolution;
import jdk.internal.module.ModuleTarget;
import jdk.internal.module.Resources;
import jdk.tools.jlink.internal.Utils; import jdk.tools.jlink.internal.Utils;
import static java.util.stream.Collectors.joining; import static java.util.stream.Collectors.joining;
@ -178,7 +179,6 @@ public class JmodTask {
String mainClass; String mainClass;
String osName; String osName;
String osArch; String osArch;
String osVersion;
Pattern modulesToHash; Pattern modulesToHash;
ModuleResolution moduleResolution; ModuleResolution moduleResolution;
boolean dryrun; boolean dryrun;
@ -311,7 +311,9 @@ public class JmodTask {
try (JmodFile jf = new JmodFile(options.jmodFile)) { try (JmodFile jf = new JmodFile(options.jmodFile)) {
try (InputStream in = jf.getInputStream(Section.CLASSES, MODULE_INFO)) { try (InputStream in = jf.getInputStream(Section.CLASSES, MODULE_INFO)) {
ModuleInfo.Attributes attrs = ModuleInfo.read(in, null); ModuleInfo.Attributes attrs = ModuleInfo.read(in, null);
printModuleDescriptor(attrs.descriptor(), attrs.recordedHashes()); printModuleDescriptor(attrs.descriptor(),
attrs.target(),
attrs.recordedHashes());
return true; return true;
} catch (IOException e) { } catch (IOException e) {
throw new CommandException("err.module.descriptor.not.found"); throw new CommandException("err.module.descriptor.not.found");
@ -325,7 +327,9 @@ public class JmodTask {
.collect(joining(" ")); .collect(joining(" "));
} }
private void printModuleDescriptor(ModuleDescriptor md, ModuleHashes hashes) private void printModuleDescriptor(ModuleDescriptor md,
ModuleTarget target,
ModuleHashes hashes)
throws IOException throws IOException
{ {
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
@ -365,11 +369,14 @@ public class JmodTask {
md.mainClass().ifPresent(v -> sb.append("\n main-class " + v)); md.mainClass().ifPresent(v -> sb.append("\n main-class " + v));
md.osName().ifPresent(v -> sb.append("\n operating-system-name " + v)); if (target != null) {
String osName = target.osName();
md.osArch().ifPresent(v -> sb.append("\n operating-system-architecture " + v)); if (osName != null)
sb.append("\n operating-system-name " + osName);
md.osVersion().ifPresent(v -> sb.append("\n operating-system-version " + v)); String osArch = target.osArch();
if (osArch != null)
sb.append("\n operating-system-architecture " + osArch);
}
if (hashes != null) { if (hashes != null) {
hashes.names().stream().sorted().forEach( hashes.names().stream().sorted().forEach(
@ -432,7 +439,6 @@ public class JmodTask {
final String mainClass = options.mainClass; final String mainClass = options.mainClass;
final String osName = options.osName; final String osName = options.osName;
final String osArch = options.osArch; final String osArch = options.osArch;
final String osVersion = options.osVersion;
final List<PathMatcher> excludes = options.excludes; final List<PathMatcher> excludes = options.excludes;
final ModuleResolution moduleResolution = options.moduleResolution; final ModuleResolution moduleResolution = options.moduleResolution;
@ -528,9 +534,9 @@ public class JmodTask {
if (mainClass != null) if (mainClass != null)
extender.mainClass(mainClass); extender.mainClass(mainClass);
// --os-name, --os-arch, --os-version // --os-name, --os-arch
if (osName != null || osArch != null || osVersion != null) if (osName != null || osArch != null)
extender.targetPlatform(osName, osArch, osVersion); extender.targetPlatform(osName, osArch);
// --module-version // --module-version
if (moduleVersion != null) if (moduleVersion != null)
@ -675,7 +681,7 @@ public class JmodTask {
*/ */
boolean isResource(String name) { boolean isResource(String name) {
name = name.replace(File.separatorChar, '/'); name = name.replace(File.separatorChar, '/');
return name.endsWith(".class") || !ResourceHelper.isSimpleResource(name); return name.endsWith(".class") || Resources.canEncapsulate(name);
} }
@ -1331,11 +1337,6 @@ public class JmodTask {
.withRequiredArg() .withRequiredArg()
.describedAs(getMessage("main.opt.os-arch.arg")); .describedAs(getMessage("main.opt.os-arch.arg"));
OptionSpec<String> osVersion
= parser.accepts("os-version", getMessage("main.opt.os-version"))
.withRequiredArg()
.describedAs(getMessage("main.opt.os-version.arg"));
OptionSpec<Void> doNotResolveByDefault OptionSpec<Void> doNotResolveByDefault
= parser.accepts("do-not-resolve-by-default", = parser.accepts("do-not-resolve-by-default",
getMessage("main.opt.do-not-resolve-by-default")); getMessage("main.opt.do-not-resolve-by-default"));
@ -1403,8 +1404,6 @@ public class JmodTask {
options.osName = getLastElement(opts.valuesOf(osName)); options.osName = getLastElement(opts.valuesOf(osName));
if (opts.has(osArch)) if (opts.has(osArch))
options.osArch = getLastElement(opts.valuesOf(osArch)); options.osArch = getLastElement(opts.valuesOf(osArch));
if (opts.has(osVersion))
options.osVersion = getLastElement(opts.valuesOf(osVersion));
if (opts.has(warnIfResolved)) if (opts.has(warnIfResolved))
options.moduleResolution = getLastElement(opts.valuesOf(warnIfResolved)); options.moduleResolution = getLastElement(opts.valuesOf(warnIfResolved));
if (opts.has(doNotResolveByDefault)) { if (opts.has(doNotResolveByDefault)) {

View File

@ -68,8 +68,6 @@ main.opt.os-name=Operating system name
main.opt.os-name.arg=os-name main.opt.os-name.arg=os-name
main.opt.os-arch=Operating system architecture main.opt.os-arch=Operating system architecture
main.opt.os-arch.arg=os-arch main.opt.os-arch.arg=os-arch
main.opt.os-version=Operating system version
main.opt.os-version.arg=os-version
main.opt.module-path=Module path main.opt.module-path=Module path
main.opt.hash-modules=Compute and record hashes to tie a packaged module\ main.opt.hash-modules=Compute and record hashes to tie a packaged module\
\ with modules matching the given <regex-pattern> and depending upon it directly\ \ with modules matching the given <regex-pattern> and depending upon it directly\

View File

@ -813,6 +813,7 @@ public final class Unsafe {
/** /**
* Tells the VM to define a class, without security checks. By default, the * Tells the VM to define a class, without security checks. By default, the
* class loader and protection domain come from the caller's class. * class loader and protection domain come from the caller's class.
* @see java.lang.invoke.MethodHandles.Lookup#defineClass(byte[])
*/ */
@ForceInline @ForceInline
public Class<?> defineClass(String name, byte[] b, int off, int len, public Class<?> defineClass(String name, byte[] b, int off, int len,

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -380,6 +380,7 @@ public class FieldSetAccessibleTest {
permissions.add(new RuntimePermission("closeClassLoader")); permissions.add(new RuntimePermission("closeClassLoader"));
permissions.add(new RuntimePermission("getClassLoader")); permissions.add(new RuntimePermission("getClassLoader"));
permissions.add(new RuntimePermission("accessDeclaredMembers")); permissions.add(new RuntimePermission("accessDeclaredMembers"));
permissions.add(new RuntimePermission("accessSystemModules"));
permissions.add(new ReflectPermission("suppressAccessChecks")); permissions.add(new ReflectPermission("suppressAccessChecks"));
permissions.add(new PropertyPermission("*", "read")); permissions.add(new PropertyPermission("*", "read"));
permissions.add(new FilePermission("<<ALL FILES>>", "read")); permissions.add(new FilePermission("<<ALL FILES>>", "read"));

View File

@ -0,0 +1,88 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @library /lib/testlibrary
* @build Driver Main JarUtils jdk.testlibrary.ProcessTools
* @run main Driver
* @summary Test ClassLoader.getResourceXXX to locate resources in an automatic
* module
*/
import java.lang.module.ModuleFinder;
import java.lang.module.ModuleReference;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.jar.Attributes;
import java.util.jar.Manifest;
import jdk.testlibrary.ProcessTools;
/**
* The driver creates a JAR file containing p/Foo.class, p/foo.properties,
* and p/resources/bar.properties. This ensures there are is a resource in
* a module package and a resource that is not in the module package. The
* test is then launched to locate every resource in the JAR file.
*/
public class Driver {
private static final String TEST_CLASSES = System.getProperty("test.classes");
public static void main(String[] args) throws Exception {
// create content for JAR file
Path dir = Files.createTempDirectory("classes");
Path p = Files.createDirectory(dir.resolve("p"));
Files.createFile(p.resolve("Foo.class"));
Files.createFile(p.resolve("foo.properties"));
Path resources = Files.createDirectory(p.resolve("resources"));
Files.createFile(resources.resolve("bar.properties"));
// create the JAR file, including a manifest
Path jarFile = Paths.get("library-1.0.jar");
Manifest man = new Manifest();
Attributes attrs = man.getMainAttributes();
attrs.put(Attributes.Name.MANIFEST_VERSION, "1.0.0");
JarUtils.createJarFile(jarFile, man, dir, p);
// get the module name
ModuleFinder finder = ModuleFinder.of(jarFile);
ModuleReference mref = finder.findAll().stream().findAny().orElse(null);
if (mref == null)
throw new RuntimeException("Module not found!!!");
String name = mref.descriptor().name();
// launch the test with the JAR file on the module path
if (ProcessTools.executeTestJava("-p", jarFile.toString(),
"--add-modules", name,
"-cp", TEST_CLASSES,
"Main", name)
.outputTo(System.out)
.errorTo(System.out)
.getExitValue() != 0)
throw new RuntimeException("Test failed - see output");
}
}

View File

@ -0,0 +1,94 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import java.io.IOException;
import java.io.InputStream;
import java.lang.module.ModuleReader;
import java.lang.module.ModuleReference;
import java.lang.module.ResolvedModule;
import java.lang.reflect.Layer;
import java.net.URL;
import java.util.Enumeration;
/**
* Usage: Main $MODULE
*
* Finds $MODULE in the boot layer and then tests that it can locate every
* resource in the module content (JAR file).
*/
public class Main {
static void testFind(String name) throws Exception {
// getResource
URL url = Main.class.getClassLoader().getResource(name);
if (url == null)
throw new RuntimeException("Unable to locate: " + name);
System.out.println(name + " => " + url);
// getResources
Enumeration<URL> urls = Main.class.getClassLoader().getResources(name);
if (!urls.hasMoreElements())
throw new RuntimeException("Unable to locate: " + name);
URL first = urls.nextElement();
if (!first.toURI().equals(url.toURI()))
throw new RuntimeException("found " + first + " ???");
// getResourceAsStream
if (!url.toString().endsWith("/")) {
InputStream in = Main.class.getClassLoader().getResourceAsStream(name);
if (in == null)
throw new RuntimeException("Unable to locate: " + name);
in.close();
}
}
static void testFindUnchecked(String name) {
try {
testFind(name);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
public static void main(String[] args) throws Exception {
String mn = args[0];
ModuleReference mref = Layer.boot()
.configuration()
.findModule(mn)
.map(ResolvedModule::reference)
.orElseThrow(() -> new RuntimeException(mn + " not resolved!!"));
try (ModuleReader reader = mref.open()) {
reader.list().forEach(name -> {
testFindUnchecked(name);
// if the resource is a directory then find without trailing slash
if (name.endsWith("/")) {
testFindUnchecked(name.substring(0, name.length() - 1));
}
});
}
}
}

View File

@ -0,0 +1,352 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/* @test
* @modules java.base/java.lang:open
* java.base/jdk.internal.org.objectweb.asm
* @run testng/othervm test.DefineClassTest
* @summary Basic test for java.lang.invoke.MethodHandles.Lookup.defineClass
*/
package test;
import java.lang.invoke.MethodHandles.Lookup;
import static java.lang.invoke.MethodHandles.*;
import static java.lang.invoke.MethodHandles.Lookup.*;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files;
import java.nio.file.Path;
import jdk.internal.org.objectweb.asm.ClassWriter;
import jdk.internal.org.objectweb.asm.MethodVisitor;
import static jdk.internal.org.objectweb.asm.Opcodes.*;
import org.testng.annotations.Test;
import static org.testng.Assert.*;
public class DefineClassTest {
private static final String THIS_PACKAGE = DefineClassTest.class.getPackageName();
/**
* Test that a class has the same class loader, and is in the same package and
* protection domain, as a lookup class.
*/
void testSameAbode(Class<?> clazz, Class<?> lc) {
assertTrue(clazz.getClassLoader() == lc.getClassLoader());
assertEquals(clazz.getPackageName(), lc.getPackageName());
assertTrue(clazz.getProtectionDomain() == lc.getProtectionDomain());
}
/**
* Tests that a class is discoverable by name using Class.forName and
* lookup.findClass
*/
void testDiscoverable(Class<?> clazz, Lookup lookup) throws Exception {
String cn = clazz.getName();
ClassLoader loader = clazz.getClassLoader();
assertTrue(Class.forName(cn, false, loader) == clazz);
assertTrue(lookup.findClass(cn) == clazz);
}
/**
* Basic test of defineClass to define a class in the same package as test.
*/
@Test
public void testDefineClass() throws Exception {
final String CLASS_NAME = THIS_PACKAGE + ".Foo";
Lookup lookup = lookup().dropLookupMode(PRIVATE);
Class<?> clazz = lookup.defineClass(generateClass(CLASS_NAME));
// test name
assertEquals(clazz.getName(), CLASS_NAME);
// test loader/package/protection-domain
testSameAbode(clazz, lookup.lookupClass());
// test discoverable
testDiscoverable(clazz, lookup);
// attempt defineClass again
try {
lookup.defineClass(generateClass(CLASS_NAME));
assertTrue(false);
} catch (LinkageError expected) { }
}
/**
* Test public/package/protected/private access from class defined with defineClass.
*/
@Test
public void testAccess() throws Exception {
final String THIS_CLASS = this.getClass().getName();
final String CLASS_NAME = THIS_PACKAGE + ".Runner";
Lookup lookup = lookup().dropLookupMode(PRIVATE);
// public
byte[] classBytes = generateRunner(CLASS_NAME + nextNumber(), THIS_CLASS, "method1");
testInvoke(lookup.defineClass(classBytes));
// package
classBytes = generateRunner(CLASS_NAME + nextNumber(), THIS_CLASS, "method2");
testInvoke(lookup.defineClass(classBytes));
// protected (same package)
classBytes = generateRunner(CLASS_NAME + nextNumber(), THIS_CLASS, "method3");
testInvoke(lookup.defineClass(classBytes));
// private
classBytes = generateRunner(CLASS_NAME + nextNumber(), THIS_CLASS, "method4");
Class<?> clazz = lookup.defineClass(classBytes);
Runnable r = (Runnable) clazz.newInstance();
try {
r.run();
assertTrue(false);
} catch (IllegalAccessError expected) { }
}
public static void method1() { }
static void method2() { }
protected static void method3() { }
private static void method4() { }
void testInvoke(Class<?> clazz) throws Exception {
Object obj = clazz.newInstance();
((Runnable) obj).run();
}
/**
* Test that defineClass does not run the class initializer
*/
@Test
public void testInitializerNotRun() throws Exception {
final String THIS_CLASS = this.getClass().getName();
final String CLASS_NAME = THIS_PACKAGE + ".ClassWithClinit";
byte[] classBytes = generateClassWithInitializer(CLASS_NAME, THIS_CLASS, "fail");
Lookup lookup = lookup().dropLookupMode(PRIVATE);
Class<?> clazz = lookup.defineClass(classBytes);
// trigger initializer to run
try {
clazz.newInstance();
assertTrue(false);
} catch (ExceptionInInitializerError e) {
assertTrue(e.getCause() instanceof IllegalCallerException);
}
}
static void fail() { throw new IllegalCallerException(); }
/**
* Test defineClass to define classes in a package containing classes with
* different protection domains.
*/
@Test
public void testTwoProtectionDomains() throws Exception {
// p.C1 in one exploded directory
Path dir1 = Files.createTempDirectory("classes");
Path p = Files.createDirectory(dir1.resolve("p"));
Files.write(p.resolve("C1.class"), generateClass("p.C1"));
URL url1 = dir1.toUri().toURL();
// p.C2 in another exploded directory
Path dir2 = Files.createTempDirectory("classes");
p = Files.createDirectory(dir2.resolve("p"));
Files.write(p.resolve("C2.class"), generateClass("p.C2"));
URL url2 = dir2.toUri().toURL();
// load p.C1 and p.C2
ClassLoader loader = new URLClassLoader(new URL[] { url1, url2 });
Class<?> target1 = Class.forName("p.C1", false, loader);
Class<?> target2 = Class.forName("p.C2", false, loader);
assertTrue(target1.getClassLoader() == loader);
assertTrue(target1.getClassLoader() == loader);
assertNotEquals(target1.getProtectionDomain(), target2.getProtectionDomain());
// protection domain 1
Lookup lookup1 = privateLookupIn(target1, lookup()).dropLookupMode(PRIVATE);
Class<?> clazz = lookup1.defineClass(generateClass("p.Foo"));
testSameAbode(clazz, lookup1.lookupClass());
testDiscoverable(clazz, lookup1);
// protection domain 2
Lookup lookup2 = privateLookupIn(target2, lookup()).dropLookupMode(PRIVATE);
clazz = lookup2.defineClass(generateClass("p.Bar"));
testSameAbode(clazz, lookup2.lookupClass());
testDiscoverable(clazz, lookup2);
}
/**
* Test defineClass defining a class to the boot loader
*/
@Test
public void testBootLoader() throws Exception {
Lookup lookup = privateLookupIn(Thread.class, lookup()).dropLookupMode(PRIVATE);
assertTrue(lookup.getClass().getClassLoader() == null);
Class<?> clazz = lookup.defineClass(generateClass("java.lang.Foo"));
assertEquals(clazz.getName(), "java.lang.Foo");
testSameAbode(clazz, Thread.class);
testDiscoverable(clazz, lookup);
}
@Test(expectedExceptions = { IllegalArgumentException.class })
public void testWrongPackage() throws Exception {
Lookup lookup = lookup().dropLookupMode(PRIVATE);
lookup.defineClass(generateClass("other.C"));
}
@Test(expectedExceptions = { IllegalAccessException.class })
public void testNoPackageAccess() throws Exception {
Lookup lookup = lookup().dropLookupMode(PACKAGE);
lookup.defineClass(generateClass(THIS_PACKAGE + ".C"));
}
@Test(expectedExceptions = { UnsupportedOperationException.class })
public void testHasPrivateAccess() throws Exception {
Lookup lookup = lookup();
assertTrue(lookup.hasPrivateAccess());
lookup.defineClass(generateClass(THIS_PACKAGE + ".C"));
}
@Test(expectedExceptions = { ClassFormatError.class })
public void testTruncatedClassFile() throws Exception {
Lookup lookup = lookup().dropLookupMode(PRIVATE);
lookup.defineClass(new byte[0]);
}
@Test(expectedExceptions = { NullPointerException.class })
public void testNull() throws Exception {
Lookup lookup = lookup().dropLookupMode(PRIVATE);
lookup.defineClass(null);
}
/**
* Generates a class file with the given class name
*/
byte[] generateClass(String className) {
ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS
+ ClassWriter.COMPUTE_FRAMES);
cw.visit(V1_9,
ACC_PUBLIC + ACC_SUPER,
className.replace(".", "/"),
null,
"java/lang/Object",
null);
// <init>
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
mv.visitVarInsn(ALOAD, 0);
mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
cw.visitEnd();
return cw.toByteArray();
}
/**
* Generate a class file with the given class name. The class implements Runnable
* with a run method to invokestatic the given targetClass/targetMethod.
*/
byte[] generateRunner(String className,
String targetClass,
String targetMethod) throws Exception {
ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS
+ ClassWriter.COMPUTE_FRAMES);
cw.visit(V1_9,
ACC_PUBLIC + ACC_SUPER,
className.replace(".", "/"),
null,
"java/lang/Object",
new String[] { "java/lang/Runnable" });
// <init>
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
mv.visitVarInsn(ALOAD, 0);
mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
// run()
String tc = targetClass.replace(".", "/");
mv = cw.visitMethod(ACC_PUBLIC, "run", "()V", null, null);
mv.visitMethodInsn(INVOKESTATIC, tc, targetMethod, "()V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
cw.visitEnd();
return cw.toByteArray();
}
/**
* Generate a class file with the given class name. The class will initializer
* to invokestatic the given targetClass/targetMethod.
*/
byte[] generateClassWithInitializer(String className,
String targetClass,
String targetMethod) throws Exception {
ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS
+ ClassWriter.COMPUTE_FRAMES);
cw.visit(V1_9,
ACC_PUBLIC + ACC_SUPER,
className.replace(".", "/"),
null,
"java/lang/Object",
null);
// <init>
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
mv.visitVarInsn(ALOAD, 0);
mv.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
// <clinit>
String tc = targetClass.replace(".", "/");
mv = cw.visitMethod(ACC_STATIC, "<clinit>", "()V", null, null);
mv.visitMethodInsn(INVOKESTATIC, tc, targetMethod, "()V", false);
mv.visitInsn(RETURN);
mv.visitMaxs(0, 0);
mv.visitEnd();
cw.visitEnd();
return cw.toByteArray();
}
private int nextNumber() {
return ++nextNumber;
}
private int nextNumber;
}

View File

@ -773,9 +773,6 @@ public class AutomaticModulesTest {
// test miscellaneous methods // test miscellaneous methods
assertTrue(m.isAutomatic()); assertTrue(m.isAutomatic());
assertFalse(m.modifiers().contains(ModuleDescriptor.Modifier.SYNTHETIC)); assertFalse(m.modifiers().contains(ModuleDescriptor.Modifier.SYNTHETIC));
assertFalse(m.osName().isPresent());
assertFalse(m.osArch().isPresent());
assertFalse(m.osVersion().isPresent());
} }

View File

@ -25,25 +25,31 @@
* @test * @test
* @library /lib/testlibrary * @library /lib/testlibrary
* @modules java.base/jdk.internal.misc * @modules java.base/jdk.internal.misc
* java.base/jdk.internal.module
* @build ConfigurationTest ModuleUtils * @build ConfigurationTest ModuleUtils
* @run testng ConfigurationTest * @run testng ConfigurationTest
* @summary Basic tests for java.lang.module.Configuration * @summary Basic tests for java.lang.module.Configuration
*/ */
import java.io.IOException;
import java.io.OutputStream;
import java.lang.module.Configuration; import java.lang.module.Configuration;
import java.lang.module.FindException; import java.lang.module.FindException;
import java.lang.module.ModuleDescriptor; import java.lang.module.ModuleDescriptor;
import java.lang.module.ModuleDescriptor.Builder;
import java.lang.module.ModuleDescriptor.Requires; import java.lang.module.ModuleDescriptor.Requires;
import java.lang.module.ModuleFinder; import java.lang.module.ModuleFinder;
import java.lang.module.ResolutionException; import java.lang.module.ResolutionException;
import java.lang.module.ResolvedModule; import java.lang.module.ResolvedModule;
import java.lang.reflect.Layer; import java.lang.reflect.Layer;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.List; import java.util.List;
import java.util.Optional; import java.util.Optional;
import java.util.Set; import java.util.Set;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.ModuleInfoWriter;
import jdk.internal.module.ModuleTarget;
import org.testng.annotations.DataProvider; import org.testng.annotations.DataProvider;
import org.testng.annotations.Test; import org.testng.annotations.Test;
import static org.testng.Assert.*; import static org.testng.Assert.*;
@ -437,70 +443,6 @@ public class ConfigurationTest {
} }
/**
* Basic test of "requires transitive" with configurations.
*
* The test consists of three configurations:
* - Configuration cf1: m1, m2 requires transitive m1
* - Configuration cf2: m1, m3 requires transitive m1
* - Configuration cf3(cf1,cf2): m4 requires m2, m3
*/
public void testRequiresTransitive6() {
ModuleDescriptor descriptor1 = newBuilder("m1")
.build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleDescriptor descriptor3 = newBuilder("m3")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleDescriptor descriptor4 = newBuilder("m4")
.requires("m2")
.requires("m3")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf1 = resolve(finder1, "m2");
assertTrue(cf1.modules().size() == 2);
assertTrue(cf1.findModule("m1").isPresent());
assertTrue(cf1.findModule("m2").isPresent());
assertTrue(cf1.parents().size() == 1);
assertTrue(cf1.parents().get(0) == Configuration.empty());
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor1, descriptor3);
Configuration cf2 = resolve(finder2, "m3");
assertTrue(cf2.modules().size() == 2);
assertTrue(cf2.findModule("m3").isPresent());
assertTrue(cf2.findModule("m1").isPresent());
assertTrue(cf2.parents().size() == 1);
assertTrue(cf2.parents().get(0) == Configuration.empty());
ModuleFinder finder3 = ModuleUtils.finderOf(descriptor4);
Configuration cf3 = Configuration.resolve(finder3,
List.of(cf1, cf2),
ModuleFinder.of(),
Set.of("m4"));
assertTrue(cf3.modules().size() == 1);
assertTrue(cf3.findModule("m4").isPresent());
ResolvedModule m1_l = cf1.findModule("m1").get();
ResolvedModule m1_r = cf2.findModule("m1").get();
ResolvedModule m2 = cf1.findModule("m2").get();
ResolvedModule m3 = cf2.findModule("m3").get();
ResolvedModule m4 = cf3.findModule("m4").get();
assertTrue(m4.configuration() == cf3);
assertTrue(m4.reads().size() == 4);
assertTrue(m4.reads().contains(m1_l));
assertTrue(m4.reads().contains(m1_r));
assertTrue(m4.reads().contains(m2));
assertTrue(m4.reads().contains(m3));
}
/** /**
* Basic test of "requires static": * Basic test of "requires static":
* m1 requires static m2 * m1 requires static m2
@ -1602,6 +1544,76 @@ public class ConfigurationTest {
} }
/**
* Basic test to detect reading a module with the same name as itself
*
* The test consists of three configurations:
* - Configuration cf1: m1, m2 requires transitive m1
* - Configuration cf2: m1 requires m2
*/
@Test(expectedExceptions = { ResolutionException.class })
public void testReadModuleWithSameNameAsSelf() {
ModuleDescriptor descriptor1_v1 = newBuilder("m1")
.build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleDescriptor descriptor1_v2 = newBuilder("m1")
.requires("m2")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1_v1, descriptor2);
Configuration cf1 = resolve(finder1, "m2");
assertTrue(cf1.modules().size() == 2);
// resolve should throw ResolutionException
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor1_v2);
resolve(cf1, finder2, "m1");
}
/**
* Basic test to detect reading two modules with the same name
*
* The test consists of three configurations:
* - Configuration cf1: m1, m2 requires transitive m1
* - Configuration cf2: m1, m3 requires transitive m1
* - Configuration cf3(cf1,cf2): m4 requires m2, m3
*/
@Test(expectedExceptions = { ResolutionException.class })
public void testReadTwoModuleWithSameName() {
ModuleDescriptor descriptor1 = newBuilder("m1")
.build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleDescriptor descriptor3 = newBuilder("m3")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleDescriptor descriptor4 = newBuilder("m4")
.requires("m2")
.requires("m3")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf1 = resolve(finder1, "m2");
assertTrue(cf1.modules().size() == 2);
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor1, descriptor3);
Configuration cf2 = resolve(finder2, "m3");
assertTrue(cf2.modules().size() == 2);
// should throw ResolutionException as m4 will read modules named "m1".
ModuleFinder finder3 = ModuleUtils.finderOf(descriptor4);
Configuration.resolve(finder3, List.of(cf1, cf2), ModuleFinder.of(), Set.of("m4"));
}
/** /**
* Test two modules exporting package p to a module that reads both. * Test two modules exporting package p to a module that reads both.
*/ */
@ -1832,26 +1844,17 @@ public class ConfigurationTest {
public Object[][] createPlatformMatches() { public Object[][] createPlatformMatches() {
return new Object[][]{ return new Object[][]{
{ "linux-*-*", "*-*-*" }, { "linux-arm", "*-*" },
{ "*-arm-*", "*-*-*" }, { "linux-*", "*-*" },
{ "*-*-2.6", "*-*-*" }, { "*-arm", "*-*" },
{ "linux-arm-*", "*-*-*" }, { "linux-*", "linux-*" },
{ "linux-*-2.6", "*-*-*" }, { "linux-arm", "linux-*" },
{ "*-arm-2.6", "*-*-*" },
{ "linux-arm-2.6", "*-*-*" }, { "*-arm", "*-arm" },
{ "linux-arm", "*-arm" },
{ "linux-*-*", "linux-*-*" }, { "linux-arm", "linux-arm" },
{ "*-arm-*", "*-arm-*" },
{ "*-*-2.6", "*-*-2.6" },
{ "linux-arm-*", "linux-arm-*" },
{ "linux-arm-*", "linux-*-*" },
{ "linux-*-2.6", "linux-*-2.6" },
{ "linux-*-2.6", "linux-arm-*" },
{ "linux-arm-2.6", "linux-arm-2.6" },
}; };
@ -1861,9 +1864,10 @@ public class ConfigurationTest {
public Object[][] createBad() { public Object[][] createBad() {
return new Object[][] { return new Object[][] {
{ "linux-*-*", "solaris-*-*" }, { "linux-*", "solaris-*" },
{ "linux-x86-*", "linux-arm-*" }, { "*-arm", "*-sparc" },
{ "linux-*-2.4", "linux-x86-2.6" }, { "linux-x86", "solaris-sparc" },
}; };
} }
@ -1871,21 +1875,25 @@ public class ConfigurationTest {
* Test creating a configuration containing platform specific modules. * Test creating a configuration containing platform specific modules.
*/ */
@Test(dataProvider = "platformmatch") @Test(dataProvider = "platformmatch")
public void testPlatformMatch(String s1, String s2) { public void testPlatformMatch(String s1, String s2) throws IOException {
Builder builder = newBuilder("m1").requires("m2"); ModuleDescriptor base = ModuleDescriptor.newModule("java.base").build();
addPlatformConstraints(builder, s1); Path system = writeModule(base, "*-*");
ModuleDescriptor descriptor1 = builder.build();
builder = newBuilder("m2"); ModuleDescriptor descriptor1 = ModuleDescriptor.newModule("m1")
addPlatformConstraints(builder, s2); .requires("m2")
ModuleDescriptor descriptor2 = builder.build(); .build();
Path dir1 = writeModule(descriptor1, s1);
ModuleFinder finder = ModuleUtils.finderOf(descriptor1, descriptor2); ModuleDescriptor descriptor2 = ModuleDescriptor.newModule("m2").build();
Path dir2 = writeModule(descriptor2, s2);
ModuleFinder finder = ModuleFinder.of(system, dir1, dir2);
Configuration cf = resolve(finder, "m1"); Configuration cf = resolve(finder, "m1");
assertTrue(cf.modules().size() == 2); assertTrue(cf.modules().size() == 3);
assertTrue(cf.findModule("java.base").isPresent());
assertTrue(cf.findModule("m1").isPresent()); assertTrue(cf.findModule("m1").isPresent());
assertTrue(cf.findModule("m2").isPresent()); assertTrue(cf.findModule("m2").isPresent());
} }
@ -1896,11 +1904,10 @@ public class ConfigurationTest {
*/ */
@Test(dataProvider = "platformmismatch", @Test(dataProvider = "platformmismatch",
expectedExceptions = FindException.class ) expectedExceptions = FindException.class )
public void testPlatformMisMatch(String s1, String s2) { public void testPlatformMisMatch(String s1, String s2) throws IOException {
testPlatformMatch(s1, s2); testPlatformMatch(s1, s2);
} }
// no parents // no parents
@Test(expectedExceptions = { IllegalArgumentException.class }) @Test(expectedExceptions = { IllegalArgumentException.class })
@ -1917,21 +1924,23 @@ public class ConfigurationTest {
// parents with modules for specific platforms // parents with modules for specific platforms
@Test(dataProvider = "platformmatch") @Test(dataProvider = "platformmatch")
public void testResolveRequiresWithCompatibleParents(String s1, String s2) { public void testResolveRequiresWithCompatibleParents(String s1, String s2)
Builder builder = newBuilder("m1"); throws IOException
addPlatformConstraints(builder, s1); {
ModuleDescriptor descriptor1 = builder.build(); ModuleDescriptor base = ModuleDescriptor.newModule("java.base").build();
Path system = writeModule(base, "*-*");
builder = newBuilder("m2"); ModuleDescriptor descriptor1 = ModuleDescriptor.newModule("m1").build();
addPlatformConstraints(builder, s2); Path dir1 = writeModule(descriptor1, s1);
ModuleDescriptor descriptor2 = builder.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1); ModuleDescriptor descriptor2 = ModuleDescriptor.newModule("m2").build();
Path dir2 = writeModule(descriptor2, s2);
ModuleFinder finder1 = ModuleFinder.of(system, dir1);
Configuration cf1 = resolve(finder1, "m1"); Configuration cf1 = resolve(finder1, "m1");
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2); ModuleFinder finder2 = ModuleFinder.of(system, dir2);
Configuration cf2 = resolve(finder2, "m2"); Configuration cf2 = resolve(finder2, "m2");
Configuration cf3 = Configuration.resolve(ModuleFinder.of(), Configuration cf3 = Configuration.resolve(ModuleFinder.of(),
@ -1941,32 +1950,16 @@ public class ConfigurationTest {
assertTrue(cf3.parents().size() == 2); assertTrue(cf3.parents().size() == 2);
} }
@Test(dataProvider = "platformmismatch", @Test(dataProvider = "platformmismatch",
expectedExceptions = IllegalArgumentException.class ) expectedExceptions = IllegalArgumentException.class )
public void testResolveRequiresWithConflictingParents(String s1, String s2) { public void testResolveRequiresWithConflictingParents(String s1, String s2)
Builder builder = newBuilder("m1"); throws IOException
addPlatformConstraints(builder, s1); {
ModuleDescriptor descriptor1 = builder.build(); testResolveRequiresWithCompatibleParents(s1, s2);
builder = newBuilder("m2");
addPlatformConstraints(builder, s2);
ModuleDescriptor descriptor2 = builder.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolve(finder1, "m1");
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2);
Configuration cf2 = resolve(finder2, "m2");
// should throw IAE
Configuration.resolve(ModuleFinder.of(),
List.of(cf1, cf2),
ModuleFinder.of(),
Set.of());
} }
// null handling // null handling
// finder1, finder2, roots // finder1, finder2, roots
@ -2120,31 +2113,24 @@ public class ConfigurationTest {
} }
/**
* Returns {@code true} if the configuration contains module mn1
* that reads module mn2.
*/
static boolean reads(Configuration cf, String mn1, String mn2) {
Optional<ResolvedModule> om1 = cf.findModule(mn1);
if (!om1.isPresent())
return false;
return om1.get().reads().stream()
.map(ResolvedModule::name)
.anyMatch(mn2::equals);
}
/** /**
* Decodes the platform string and calls the builder osName/osArch/osVersion * Decodes the platform string and calls the builder osName/osArch/osVersion
* methods to set the platform constraints. * methods to set the platform constraints.
*/ */
static void addPlatformConstraints(Builder builder, String platformString) { static Path writeModule(ModuleDescriptor descriptor, String platformString)
throws IOException
{
String[] s = platformString.split("-"); String[] s = platformString.split("-");
if (!s[0].equals("*")) String osName = !s[0].equals("*") ? s[0] : null;
builder.osName(s[0]); String osArch = !s[1].equals("*") ? s[1] : null;
if (!s[1].equals("*")) ModuleTarget target = new ModuleTarget(osName, osArch);
builder.osArch(s[1]);
if (!s[2].equals("*")) String name = descriptor.name();
builder.osVersion(s[2]); Path dir = Files.createTempDirectory(name);
Path mi = dir.resolve("module-info.class");
try (OutputStream out = Files.newOutputStream(mi)) {
ModuleInfoWriter.write(descriptor, target, out);
}
return dir;
} }
} }

View File

@ -50,11 +50,13 @@ import java.util.HashSet;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
import java.util.Objects; import java.util.Objects;
import java.util.Optional;
import java.util.Set; import java.util.Set;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import static java.lang.module.ModuleDescriptor.Requires.Modifier.*; import static java.lang.module.ModuleDescriptor.Requires.Modifier.*;
import jdk.internal.misc.JavaLangModuleAccess;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import jdk.internal.module.ModuleInfoWriter; import jdk.internal.module.ModuleInfoWriter;
import org.testng.annotations.DataProvider; import org.testng.annotations.DataProvider;
@ -988,6 +990,107 @@ public class ModuleDescriptorTest {
} }
@DataProvider(name = "unparseableVersions")
public Object[][] unparseableVersions() {
return new Object[][]{
{ null, "A1" }, // no version < unparseable
{ "A1", "A2" }, // unparseable < unparseable
{ "A1", "1.0" }, // unparseable < parseable
};
}
/**
* Basic test for unparseable module versions
*/
@Test(dataProvider = "unparseableVersions")
public void testUnparseableModuleVersion(String vs1, String vs2) {
ModuleDescriptor descriptor1 = newModule("m", vs1);
ModuleDescriptor descriptor2 = newModule("m", vs2);
if (vs1 != null && !isParsableVersion(vs1)) {
assertFalse(descriptor1.version().isPresent());
assertTrue(descriptor1.rawVersion().isPresent());
assertEquals(descriptor1.rawVersion().get(), vs1);
}
if (vs2 != null && !isParsableVersion(vs2)) {
assertFalse(descriptor2.version().isPresent());
assertTrue(descriptor2.rawVersion().isPresent());
assertEquals(descriptor2.rawVersion().get(), vs2);
}
assertFalse(descriptor1.equals(descriptor2));
assertFalse(descriptor2.equals(descriptor1));
assertTrue(descriptor1.compareTo(descriptor2) == -1);
assertTrue(descriptor2.compareTo(descriptor1) == 1);
}
/**
* Basic test for requiring a module with an unparseable version recorded
* at compile version.
*/
@Test(dataProvider = "unparseableVersions")
public void testUnparseableCompiledVersion(String vs1, String vs2) {
Requires r1 = newRequires("m", vs1);
Requires r2 = newRequires("m", vs2);
if (vs1 != null && !isParsableVersion(vs1)) {
assertFalse(r1.compiledVersion().isPresent());
assertTrue(r1.rawCompiledVersion().isPresent());
assertEquals(r1.rawCompiledVersion().get(), vs1);
}
if (vs2 != null && !isParsableVersion(vs2)) {
assertFalse(r2.compiledVersion().isPresent());
assertTrue(r2.rawCompiledVersion().isPresent());
assertEquals(r2.rawCompiledVersion().get(), vs2);
}
assertFalse(r1.equals(r2));
assertFalse(r2.equals(r1));
assertTrue(r1.compareTo(r2) == -1);
assertTrue(r2.compareTo(r1) == 1);
}
private ModuleDescriptor newModule(String name, String vs) {
JavaLangModuleAccess JLMA = SharedSecrets.getJavaLangModuleAccess();
Builder builder = JLMA.newModuleBuilder(name, false, Set.of());
if (vs != null)
builder.version(vs);
builder.requires("java.base");
ByteBuffer bb = ModuleInfoWriter.toByteBuffer(builder.build());
return ModuleDescriptor.read(bb);
}
private Requires newRequires(String name, String vs) {
JavaLangModuleAccess JLMA = SharedSecrets.getJavaLangModuleAccess();
Builder builder = JLMA.newModuleBuilder("foo", false, Set.of());
if (vs == null) {
builder.requires(name);
} else {
JLMA.requires(builder, Set.of(), name, vs);
}
Set<ModuleDescriptor.Requires> requires = builder.build().requires();
Iterator<ModuleDescriptor.Requires> iterator = requires.iterator();
ModuleDescriptor.Requires r = iterator.next();
if (r.name().equals("java.base")) {
r = iterator.next();
}
return r;
}
private boolean isParsableVersion(String vs) {
try {
Version.parse(vs);
return true;
} catch (IllegalArgumentException e) {
return false;
}
}
// toNameAndVersion // toNameAndVersion
public void testToNameAndVersion() { public void testToNameAndVersion() {
@ -1170,59 +1273,6 @@ public class ModuleDescriptorTest {
} }
// osName
public void testOsName() {
String osName = ModuleDescriptor.newModule("foo").osName("Linux").build().osName().get();
assertEquals(osName, "Linux");
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testNullOsName() {
ModuleDescriptor.newModule("foo").osName(null);
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testEmptyOsName() {
ModuleDescriptor.newModule("foo").osName("");
}
// osArch
public void testOsArch() {
String osArch = ModuleDescriptor.newModule("foo").osName("arm").build().osName().get();
assertEquals(osArch, "arm");
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testNullOsArch() {
ModuleDescriptor.newModule("foo").osArch(null);
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testEmptyOsArch() {
ModuleDescriptor.newModule("foo").osArch("");
}
// osVersion
public void testOsVersion() {
String osVersion = ModuleDescriptor.newModule("foo").osName("11.2").build().osName().get();
assertEquals(osVersion, "11.2");
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testNullOsVersion() {
ModuleDescriptor.newModule("foo").osVersion(null);
}
@Test(expectedExceptions = IllegalArgumentException.class)
public void testEmptyOsVersion() {
ModuleDescriptor.newModule("foo").osVersion("");
}
// reads // reads
private static InputStream EMPTY_INPUT_STREAM = new InputStream() { private static InputStream EMPTY_INPUT_STREAM = new InputStream() {
@ -1239,7 +1289,9 @@ public class ModuleDescriptorTest {
} }
}; };
// basic test reading module-info.class /**
* Basic test reading module-info.class
*/
public void testRead() throws Exception { public void testRead() throws Exception {
Module base = Object.class.getModule(); Module base = Object.class.getModule();
@ -1256,6 +1308,7 @@ public class ModuleDescriptorTest {
assertEquals(descriptor.name(), "java.base"); assertEquals(descriptor.name(), "java.base");
} }
} }
/** /**
* Test ModuleDescriptor with a packager finder * Test ModuleDescriptor with a packager finder
*/ */

View File

@ -373,7 +373,7 @@ public class ModuleFinderTest {
/** /**
* Test ModuleModule with a JAR file containing a .class file in the top * Test ModuleFinder with a JAR file containing a .class file in the top
* level directory. * level directory.
*/ */
public void testOfOneJarFileWithTopLevelClass() throws Exception { public void testOfOneJarFileWithTopLevelClass() throws Exception {
@ -386,6 +386,7 @@ public class ModuleFinderTest {
assertTrue(false); assertTrue(false);
} catch (FindException e) { } catch (FindException e) {
assertTrue(e.getCause() instanceof InvalidModuleDescriptorException); assertTrue(e.getCause() instanceof InvalidModuleDescriptorException);
assertTrue(e.getCause().getMessage().contains("Mojo.class"));
} }
finder = ModuleFinder.of(jar); finder = ModuleFinder.of(jar);
@ -394,11 +395,12 @@ public class ModuleFinderTest {
assertTrue(false); assertTrue(false);
} catch (FindException e) { } catch (FindException e) {
assertTrue(e.getCause() instanceof InvalidModuleDescriptorException); assertTrue(e.getCause() instanceof InvalidModuleDescriptorException);
assertTrue(e.getCause().getMessage().contains("Mojo.class"));
} }
} }
/** /**
* Test ModuleModule with a JAR file containing a .class file in the top * Test ModuleFinder with a JAR file containing a .class file in the top
* level directory. * level directory.
*/ */
public void testOfOneExplodedModuleWithTopLevelClass() throws Exception { public void testOfOneExplodedModuleWithTopLevelClass() throws Exception {
@ -411,6 +413,7 @@ public class ModuleFinderTest {
assertTrue(false); assertTrue(false);
} catch (FindException e) { } catch (FindException e) {
assertTrue(e.getCause() instanceof InvalidModuleDescriptorException); assertTrue(e.getCause() instanceof InvalidModuleDescriptorException);
assertTrue(e.getCause().getMessage().contains("Mojo.class"));
} }
finder = ModuleFinder.of(m_dir); finder = ModuleFinder.of(m_dir);
@ -419,6 +422,7 @@ public class ModuleFinderTest {
assertTrue(false); assertTrue(false);
} catch (FindException e) { } catch (FindException e) {
assertTrue(e.getCause() instanceof InvalidModuleDescriptorException); assertTrue(e.getCause() instanceof InvalidModuleDescriptorException);
assertTrue(e.getCause().getMessage().contains("Mojo.class"));
} }
} }

View File

@ -0,0 +1,70 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @run main/othervm ModuleFinderWithSecurityManager allow
* @run main/othervm ModuleFinderWithSecurityManager deny
* @summary Basic test for ModuleFinder.ofSystem() with security manager
*/
import java.lang.module.ModuleFinder;
import java.lang.module.ModuleReference;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Set;
public class ModuleFinderWithSecurityManager {
public static void main(String[] args) throws Exception {
boolean allow = args[0].equals("allow");
// set security policy to allow access
if (allow) {
String testSrc = System.getProperty("test.src");
if (testSrc == null)
testSrc = ".";
Path policyFile = Paths.get(testSrc, "java.policy");
System.setProperty("java.security.policy", policyFile.toString());
}
System.setSecurityManager(new SecurityManager());
ModuleFinder finder = null;
try {
finder = ModuleFinder.ofSystem();
if (!allow) throw new RuntimeException("SecurityException expected");
} catch (SecurityException e) {
if (allow) throw new RuntimeException("SecurityException not expected");
}
// no additional permissions should be required to locate modules
if (finder != null) {
ModuleReference base = finder.find("java.base").orElse(null);
if (base == null)
throw new RuntimeException("java.base not found");
Set<ModuleReference> allModules = finder.findAll();
if (!allModules.contains(base))
throw new RuntimeException("java.base not in all modules");
}
}
}

View File

@ -79,17 +79,21 @@ public class ModuleReaderTest {
"java/lang/Object.class" "java/lang/Object.class"
}; };
// resource names that should not be found in the base module // (directory) resources that may be in the base module
private static final String[] BAD_BASE_RESOURCES = { private static final String[] MAYBE_BASE_RESOURCES = {
"NotFound",
"java", "java",
"/java",
"//java",
"java/", "java/",
"java/lang", "java/lang",
"java/lang/",
};
// resource names that should not be found in the base module
private static final String[] NOT_BASE_RESOURCES = {
"NotFound",
"/java",
"//java",
"/java/lang", "/java/lang",
"//java/lang", "//java/lang",
"java/lang/",
"java//lang", "java//lang",
"/java/lang/Object.class", "/java/lang/Object.class",
"//java/lang/Object.class", "//java/lang/Object.class",
@ -109,13 +113,17 @@ public class ModuleReaderTest {
"p/Main.class" "p/Main.class"
}; };
// resource names that should not be found in the test module // (directory) resources that may be in the test module
private static final String[] BAD_TEST_RESOURCES = { private static final String[] MAYBE_TEST_RESOURCES = {
"NotFound",
"p", "p",
"p/"
};
// resource names that should not be found in the test module
private static final String[] NOT_TEST_RESOURCES = {
"NotFound",
"/p", "/p",
"//p", "//p",
"p/",
"/p/Main.class", "/p/Main.class",
"//p/Main.class", "//p/Main.class",
"p/Main.class/", "p/Main.class/",
@ -160,11 +168,19 @@ public class ModuleReaderTest {
testOpen(reader, name, expectedBytes); testOpen(reader, name, expectedBytes);
testRead(reader, name, expectedBytes); testRead(reader, name, expectedBytes);
testList(reader, name); testList(reader, name);
} }
// test "not found" // test resources that may be in the base module
for (String name : BAD_BASE_RESOURCES) { for (String name : MAYBE_BASE_RESOURCES) {
Optional<URI> ouri = reader.find(name);
ouri.ifPresent(uri -> {
if (name.endsWith("/"))
assertTrue(uri.toString().endsWith("/"));
});
}
// test "not found" in java.base module
for (String name : NOT_BASE_RESOURCES) {
assertFalse(reader.find(name).isPresent()); assertFalse(reader.find(name).isPresent());
assertFalse(reader.open(name).isPresent()); assertFalse(reader.open(name).isPresent());
assertFalse(reader.read(name).isPresent()); assertFalse(reader.read(name).isPresent());
@ -261,7 +277,7 @@ public class ModuleReaderTest {
try (reader) { try (reader) {
// test each of the known resources in the module // test resources in test module
for (String name : TEST_RESOURCES) { for (String name : TEST_RESOURCES) {
byte[] expectedBytes byte[] expectedBytes
= Files.readAllBytes(MODS_DIR = Files.readAllBytes(MODS_DIR
@ -274,8 +290,18 @@ public class ModuleReaderTest {
testList(reader, name); testList(reader, name);
} }
// test "not found" // test resources that may be in the test module
for (String name : BAD_TEST_RESOURCES) { for (String name : MAYBE_TEST_RESOURCES) {
System.out.println(name);
Optional<URI> ouri = reader.find(name);
ouri.ifPresent(uri -> {
if (name.endsWith("/"))
assertTrue(uri.toString().endsWith("/"));
});
}
// test "not found" in test module
for (String name : NOT_TEST_RESOURCES) {
assertFalse(reader.find(name).isPresent()); assertFalse(reader.find(name).isPresent());
assertFalse(reader.open(name).isPresent()); assertFalse(reader.open(name).isPresent());
assertFalse(reader.read(name).isPresent()); assertFalse(reader.read(name).isPresent());
@ -394,9 +420,6 @@ public class ModuleReaderTest {
for (String e : names) { for (String e : names) {
assertTrue(reader.find(e).isPresent()); assertTrue(reader.find(e).isPresent());
} }
// should not contain directories
names.forEach(e -> assertFalse(e.endsWith("/")));
} }
} }

View File

@ -0,0 +1,4 @@
grant {
// ModuleFinder.ofSystem() needs this
permission java.lang.RuntimePermission "accessSystemModules";
};

View File

@ -45,6 +45,8 @@ import java.util.Set;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import jdk.internal.misc.SharedSecrets; import jdk.internal.misc.SharedSecrets;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test; import org.testng.annotations.Test;
import static org.testng.Assert.*; import static org.testng.Assert.*;
@ -669,6 +671,258 @@ public class BasicLayerTest {
} }
/**
* Test layers with a qualified export. The module exporting the package
* does not read the target module.
*
* m1 { exports p to m2 }
* m2 { }
*/
public void testQualifiedExports1() {
ModuleDescriptor descriptor1 = newBuilder("m1").
exports("p", Set.of("m2"))
.build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf = resolve(finder1, "m1", "m2");
ClassLoader cl = new ClassLoader() { };
Layer layer = Layer.empty().defineModules(cf, mn -> cl);
assertTrue(layer.modules().size() == 2);
Module m1 = layer.findModule("m1").get();
Module m2 = layer.findModule("m2").get();
// check m1 exports p to m2
assertFalse(m1.isExported("p"));
assertTrue(m1.isExported("p", m2));
assertFalse(m1.isOpen("p", m2));
}
/**
* Test layers with a qualified export. The module exporting the package
* reads the target module.
*
* m1 { exports p to m2; }
* m2 { requires m1; }
*/
public void testQualifiedExports2() {
ModuleDescriptor descriptor1 = newBuilder("m1")
.exports("p", Set.of("m2"))
.build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires("m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf = resolve(finder1, "m2");
ClassLoader cl = new ClassLoader() { };
Layer layer = Layer.empty().defineModules(cf, mn -> cl);
assertTrue(layer.modules().size() == 2);
Module m1 = layer.findModule("m1").get();
Module m2 = layer.findModule("m2").get();
// check m1 exports p to m2
assertFalse(m1.isExported("p"));
assertTrue(m1.isExported("p", m2));
assertFalse(m1.isOpen("p", m2));
}
/**
* Test layers with a qualified export. The module exporting the package
* does not read the target module in the parent layer.
*
* - Configuration/layer1: m1 { }
* - Configuration/layer2: m2 { exports p to m1; }
*/
public void testQualifiedExports3() {
// create layer1 with m1
ModuleDescriptor descriptor1 = newBuilder("m1").build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolve(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
assertTrue(layer1.modules().size() == 1);
// create layer2 with m2
ModuleDescriptor descriptor2 = newBuilder("m2")
.exports("p", Set.of("m1"))
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2);
Configuration cf2 = resolve(cf1, finder2, "m2");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer2.modules().size() == 1);
Module m1 = layer1.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
// check m2 exports p to layer1/m1
assertFalse(m2.isExported("p"));
assertTrue(m2.isExported("p", m1));
assertFalse(m2.isOpen("p", m1));
}
/**
* Test layers with a qualified export. The module exporting the package
* reads the target module in the parent layer.
*
* - Configuration/layer1: m1 { }
* - Configuration/layer2: m2 { requires m1; exports p to m1; }
*/
public void testQualifiedExports4() {
// create layer1 with m1
ModuleDescriptor descriptor1 = newBuilder("m1").build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolve(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
assertTrue(layer1.modules().size() == 1);
// create layer2 with m2
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires("m1")
.exports("p", Set.of("m1"))
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2);
Configuration cf2 = resolve(cf1, finder2, "m2");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer2.modules().size() == 1);
Module m1 = layer1.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
// check m2 exports p to layer1/m1
assertFalse(m2.isExported("p"));
assertTrue(m2.isExported("p", m1));
assertFalse(m2.isOpen("p", m1));
}
/**
* Test layers with a qualified export. The module exporting the package
* does not read the target module.
*
* - Configuration/layer1: m1
* - Configuration/layer2: m1, m2 { exports p to m1; }
*/
public void testQualifiedExports5() {
// create layer1 with m1
ModuleDescriptor descriptor1 = newBuilder("m1").build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolve(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
assertTrue(layer1.modules().size() == 1);
// create layer2 with m1 and m2
ModuleDescriptor descriptor2 = newBuilder("m2").exports("p", Set.of("m1")).build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf2 = resolve(cf1, finder2, "m1", "m2");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer2.modules().size() == 2);
Module m1_v1 = layer1.findModule("m1").get();
Module m1_v2 = layer2.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
// check m2 exports p to layer2/m2
assertFalse(m2.isExported("p"));
assertTrue(m2.isExported("p", m1_v2));
assertFalse(m2.isExported("p", m1_v1));
}
/**
* Test layers with a qualified export. The module exporting the package
* reads the target module in the parent layer (due to requires transitive).
*
* - Configuration/layer1: m1, m2 { requires transitive m1; }
* - Configuration/layer2: m1, m3 { requires m2; exports p to m1; }
*/
public void testQualifiedExports6() {
// create layer1 with m1 and m2
ModuleDescriptor descriptor1 = newBuilder("m1").build();
ModuleDescriptor descriptor2 = newBuilder("m2")
.requires(Set.of(Requires.Modifier.TRANSITIVE), "m1")
.build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1, descriptor2);
Configuration cf1 = resolve(finder1, "m2");
ClassLoader loader1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> loader1);
assertTrue(layer1.modules().size() == 2);
// create layer2 with m1 and m3
ModuleDescriptor descriptor3 = newBuilder("m3")
.requires("m2")
.exports("p", Set.of("m1"))
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor1, descriptor3);
Configuration cf2 = resolve(cf1, finder2, "m1", "m3");
ClassLoader loader2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> loader2);
assertTrue(layer2.modules().size() == 2);
Module m1_v1 = layer1.findModule("m1").get();
Module m2 = layer1.findModule("m2").get();
Module m1_v2 = layer2.findModule("m1").get();
Module m3 = layer2.findModule("m3").get();
assertTrue(m3.canRead(m1_v1));
assertFalse(m3.canRead(m1_v2));
assertFalse(m3.isExported("p"));
assertTrue(m3.isExported("p", m1_v1));
assertFalse(m3.isExported("p", m1_v2));
assertFalse(m3.isExported("p", m2));
}
/**
* Test layers with a qualified export. The target module is not in any layer.
*
* - Configuration/layer1: m1 { }
* - Configuration/layer2: m2 { exports p to m3; }
*/
public void testQualifiedExports7() {
// create layer1 with m1
ModuleDescriptor descriptor1 = newBuilder("m1").build();
ModuleFinder finder1 = ModuleUtils.finderOf(descriptor1);
Configuration cf1 = resolve(finder1, "m1");
ClassLoader cl1 = new ClassLoader() { };
Layer layer1 = Layer.empty().defineModules(cf1, mn -> cl1);
assertTrue(layer1.modules().size() == 1);
// create layer2 with m2
ModuleDescriptor descriptor2 = newBuilder("m2")
.exports("p", Set.of("m3"))
.build();
ModuleFinder finder2 = ModuleUtils.finderOf(descriptor2);
Configuration cf2 = resolve(cf1, finder2, "m2");
ClassLoader cl2 = new ClassLoader() { };
Layer layer2 = layer1.defineModules(cf2, mn -> cl2);
assertTrue(layer2.modules().size() == 1);
Module m1 = layer1.findModule("m1").get();
Module m2 = layer2.findModule("m2").get();
// check m2 does not export p to anyone
assertFalse(m2.isExported("p"));
assertFalse(m2.isExported("p", m1));
}
/** /**
* Attempt to use Layer defineModules to create a layer with a module * Attempt to use Layer defineModules to create a layer with a module
* defined to a class loader that already has a module of the same name * defined to a class loader that already has a module of the same name
@ -796,29 +1050,31 @@ public class BasicLayerTest {
} }
@DataProvider(name = "javaPackages")
public Object[][] javaPackages() {
return new Object[][] { { "m1", "java" }, { "m2", "java.x" } };
}
/** /**
* Attempt to create a Layer with a module containing a "java." package. * Attempt to create a Layer with a module containing a "java" package.
* This should only be allowed when the module is defined to the platform * This should only be allowed when the module is defined to the platform
* class loader. * class loader.
*/ */
@Test(enabled = false) @Test(dataProvider = "javaPackages")
public void testLayerWithJavaPackage() { public void testLayerWithJavaPackage(String mn, String pn) {
ModuleDescriptor descriptor = newBuilder("foo") ModuleDescriptor descriptor = newBuilder(mn).packages(Set.of(pn)).build();
.packages(Set.of("java.foo"))
.build();
ModuleFinder finder = ModuleUtils.finderOf(descriptor); ModuleFinder finder = ModuleUtils.finderOf(descriptor);
Configuration cf = Layer.boot() Configuration cf = Layer.boot()
.configuration() .configuration()
.resolve(finder, ModuleFinder.of(), Set.of("foo")); .resolve(finder, ModuleFinder.of(), Set.of(mn));
assertTrue(cf.modules().size() == 1); assertTrue(cf.modules().size() == 1);
ClassLoader pcl = ClassLoader.getPlatformClassLoader(); ClassLoader pcl = ClassLoader.getPlatformClassLoader();
ClassLoader scl = ClassLoader.getSystemClassLoader(); ClassLoader scl = ClassLoader.getSystemClassLoader();
try { try {
Layer.boot().defineModules(cf, mn -> new ClassLoader() { }); Layer.boot().defineModules(cf, _mn -> new ClassLoader() { });
assertTrue(false); assertTrue(false);
} catch (LayerInstantiationException e) { } } catch (LayerInstantiationException e) { }
@ -833,13 +1089,13 @@ public class BasicLayerTest {
} catch (LayerInstantiationException e) { } } catch (LayerInstantiationException e) { }
// create layer with module defined to platform class loader // create layer with module defined to platform class loader
Layer layer = Layer.boot().defineModules(cf, mn -> pcl); Layer layer = Layer.boot().defineModules(cf, _mn -> pcl);
Optional<Module> om = layer.findModule("foo"); Optional<Module> om = layer.findModule(mn);
assertTrue(om.isPresent()); assertTrue(om.isPresent());
Module foo = om.get(); Module foo = om.get();
assertTrue(foo.getClassLoader() == pcl); assertTrue(foo.getClassLoader() == pcl);
assertTrue(foo.getPackages().length == 1); assertTrue(foo.getPackages().length == 1);
assertTrue(foo.getPackages()[0].equals("java.foo")); assertTrue(foo.getPackages()[0].equals(pn));
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,5 +23,6 @@
grant { grant {
permission java.lang.RuntimePermission "getClassLoader"; permission java.lang.RuntimePermission "getClassLoader";
permission java.io.FilePermission "${java.home}/-", "read"; permission java.lang.RuntimePermission "accessSystemModules";
permission java.io.FilePermission "${java.home}/modules/-", "read"; // exploded build
}; };

View File

@ -76,6 +76,7 @@ if [ \! -d $EXTD ]; then
(cd $JARD; "$JAR" ${TESTTOOLVMOPTS} -cf ../p$n.jar *) (cd $JARD; "$JAR" ${TESTTOOLVMOPTS} -cf ../p$n.jar *)
done done
cp p2.jar p2dup.jar
mv p3.jar $EXTD mv p3.jar $EXTD
cp $TESTCLASSES/Load.class $TESTD cp $TESTCLASSES/Load.class $TESTD
@ -117,6 +118,8 @@ go "$TESTD" "" FooProvider1
go ".${SEP}p2.jar" "" FooProvider2 go ".${SEP}p2.jar" "" FooProvider2
go ".${SEP}p2.jar${SEP}p2dup.jar" "" FooProvider2
go "${P3JAR}${SEP}p2.jar" "" FooProvider3 FooProvider2 go "${P3JAR}${SEP}p2.jar" "" FooProvider3 FooProvider2
go "$TESTD${SEP}p2.jar" "" FooProvider1 FooProvider2 go "$TESTD${SEP}p2.jar" "" FooProvider1 FooProvider2

View File

@ -1,3 +1,3 @@
grant { grant {
permission java.io.FilePermission "${java.home}/-", "read"; permission java.lang.RuntimePermission "accessSystemModules";
}; };

View File

@ -0,0 +1,101 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import java.io.InputStream;
import java.io.IOException;
import java.lang.module.ModuleReader;
import java.lang.module.ModuleReference;
import java.net.URI;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import jdk.internal.module.ClassFileConstants;
import jdk.internal.module.ClassFileAttributes;
import jdk.internal.module.ClassFileAttributes.ModuleTargetAttribute;
import jdk.internal.org.objectweb.asm.Attribute;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.org.objectweb.asm.ClassVisitor;
import jdk.internal.org.objectweb.asm.Opcodes;
public class ModuleTargetHelper {
private ModuleTargetHelper() {}
public static final class ModuleTarget {
private String osName, osArch;
public ModuleTarget(String osName, String osArch) {
this.osName = osName;
this.osArch = osArch;
}
public String osName() {
return osName;
}
public String osArch() {
return osArch;
}
}
public static ModuleTarget getJavaBaseTarget() throws IOException {
Path p = Paths.get(URI.create("jrt:/modules/java.base/module-info.class"));
try (InputStream in = Files.newInputStream(p)) {
return read(in);
}
}
public static ModuleTarget read(InputStream in) throws IOException {
ModuleTargetAttribute[] modTargets = new ModuleTargetAttribute[1];
ClassVisitor cv = new ClassVisitor(Opcodes.ASM5) {
@Override
public void visitAttribute(Attribute attr) {
if (attr instanceof ModuleTargetAttribute) {
modTargets[0] = (ModuleTargetAttribute)attr;
}
}
};
// prototype of attributes that should be parsed
Attribute[] attrs = new Attribute[] {
new ModuleTargetAttribute()
};
// parse module-info.class
ClassReader cr = new ClassReader(in);
cr.accept(cv, attrs, 0);
if (modTargets[0] != null) {
return new ModuleTarget(modTargets[0].osName(), modTargets[0].osArch());
}
return null;
}
public static ModuleTarget read(ModuleReference modRef) throws IOException {
ModuleReader reader = modRef.open();
try (InputStream in = reader.open("module-info.class").get()) {
return read(in);
} finally {
reader.close();
}
}
}

View File

@ -1,3 +1,3 @@
grant { grant {
permission java.io.FilePermission "${java.home}/-", "read"; permission java.lang.RuntimePermission "accessSystemModules";
}; };

View File

@ -28,6 +28,7 @@
* java.security.jgss/sun.security.krb5.internal:+open * java.security.jgss/sun.security.krb5.internal:+open
* java.security.jgss/sun.security.jgss * java.security.jgss/sun.security.jgss
* java.security.jgss/sun.security.krb5:+open * java.security.jgss/sun.security.krb5:+open
* java.security.jgss/sun.security.krb5.internal.ccache
* java.security.jgss/sun.security.krb5.internal.crypto * java.security.jgss/sun.security.krb5.internal.crypto
* java.security.jgss/sun.security.krb5.internal.ktab * java.security.jgss/sun.security.krb5.internal.ktab
* jdk.security.auth * jdk.security.auth

View File

@ -214,7 +214,8 @@ public class IntegrationTest {
checkReleaseProperty(props, "JAVA_FULL_VERSION"); checkReleaseProperty(props, "JAVA_FULL_VERSION");
checkReleaseProperty(props, "OS_NAME"); checkReleaseProperty(props, "OS_NAME");
checkReleaseProperty(props, "OS_ARCH"); checkReleaseProperty(props, "OS_ARCH");
checkReleaseProperty(props, "OS_VERSION"); // OS_VERSION is added from makefile. We're testing API-way to create image here!
// checkReleaseProperty(props, "OS_VERSION");
if (!Files.exists(output.resolve("toto.txt"))) { if (!Files.exists(output.resolve("toto.txt"))) {
throw new AssertionError("Post processing not called"); throw new AssertionError("Post processing not called");

View File

@ -193,7 +193,7 @@ public class JLinkNegativeTest {
.output(imageFile) .output(imageFile)
.addMods("not_zip") .addMods("not_zip")
.modulePath(helper.defaultModulePath()) .modulePath(helper.defaultModulePath())
.call().assertFailure("Error: java.io.IOException: Invalid jmod file"); .call().assertFailure("Error: java.io.IOException: Invalid JMOD file");
} finally { } finally {
deleteDirectory(jmod); deleteDirectory(jmod);
} }

View File

@ -81,9 +81,13 @@ public class CompiledVersionTest {
Path msrc = SRC_DIR.resolve(mn); Path msrc = SRC_DIR.resolve(mn);
if (version.equals("0")) { if (version.equals("0")) {
assertTrue(CompilerUtils.compile(msrc, MODS_DIR, assertTrue(CompilerUtils.compile(msrc, MODS_DIR,
"--add-exports", "java.base/jdk.internal.module=m1",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1",
"--module-source-path", SRC_DIR.toString())); "--module-source-path", SRC_DIR.toString()));
} else { } else {
assertTrue(CompilerUtils.compile(msrc, MODS_DIR, assertTrue(CompilerUtils.compile(msrc, MODS_DIR,
"--add-exports", "java.base/jdk.internal.module=m1",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1",
"--module-source-path", SRC_DIR.toString(), "--module-source-path", SRC_DIR.toString(),
"--module-version", version)); "--module-version", version));
} }

View File

@ -25,6 +25,8 @@ import java.lang.module.ModuleDescriptor;
import java.lang.module.ModuleDescriptor.*; import java.lang.module.ModuleDescriptor.*;
import java.lang.module.ModuleFinder; import java.lang.module.ModuleFinder;
import java.lang.module.ModuleReference; import java.lang.module.ModuleReference;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.Paths; import java.nio.file.Paths;
@ -40,7 +42,11 @@ import static org.testng.Assert.*;
/** /**
* @test * @test
* @bug 8142968 8173381 * @bug 8142968 8173381
* @library /lib/testlibrary
* @modules java.base/jdk.internal.misc * @modules java.base/jdk.internal.misc
* @modules java.base/jdk.internal.module
* @modules java.base/jdk.internal.org.objectweb.asm
* @build ModuleTargetHelper
* @run testng SystemModulesTest * @run testng SystemModulesTest
* @summary Verify the properties of ModuleDescriptor created * @summary Verify the properties of ModuleDescriptor created
* by SystemModules * by SystemModules
@ -62,7 +68,6 @@ public class SystemModulesTest {
return; return;
ModuleFinder.ofSystem().findAll().stream() ModuleFinder.ofSystem().findAll().stream()
.map(ModuleReference::descriptor)
.forEach(this::checkAttributes); .forEach(this::checkAttributes);
} }
@ -72,9 +77,7 @@ public class SystemModulesTest {
if (name.equals(OS_NAME)) if (name.equals(OS_NAME))
return true; return true;
if (OS_NAME.equals("Mac OS X")) { if (OS_NAME.startsWith("Windows")) {
return name.equals("Darwin");
} else if (OS_NAME.startsWith("Windows")) {
return name.startsWith("Windows"); return name.startsWith("Windows");
} else { } else {
System.err.println("ERROR: " + name + " but expected: " + OS_NAME); System.err.println("ERROR: " + name + " but expected: " + OS_NAME);
@ -89,28 +92,28 @@ public class SystemModulesTest {
switch (OS_ARCH) { switch (OS_ARCH) {
case "i386": case "i386":
case "x86": case "x86":
return name.equals("i586"); return name.equals("x86");
case "amd64":
return name.equals("x86_64");
default: default:
System.err.println("ERROR: " + name + " but expected: " + OS_ARCH); System.err.println("ERROR: " + name + " but expected: " + OS_ARCH);
return false; return false;
} }
} }
private void checkAttributes(ModuleDescriptor md) { private void checkAttributes(ModuleReference modRef) {
System.out.format("%s %s %s %s%n", md.name(), try {
md.osName(), md.osArch(), md.osVersion()); if (modRef.descriptor().name().equals("java.base")) {
ModuleTargetHelper.ModuleTarget mt = ModuleTargetHelper.read(modRef);
if (md.name().equals("java.base")) { assertTrue(checkOSName(mt.osName()));
assertTrue(checkOSName(md.osName().get())); assertTrue(checkOSArch(mt.osArch()));
assertTrue(checkOSArch(md.osArch().get()));
assertTrue(md.osVersion().isPresent());
} else { } else {
// target platform attribute is dropped by jlink plugin // target platform attribute is dropped by jlink plugin for other modules
assertFalse(md.osName().isPresent()); ModuleTargetHelper.ModuleTarget mt = ModuleTargetHelper.read(modRef);
assertFalse(md.osArch().isPresent()); assertTrue(mt == null || (mt.osName() == null && mt.osArch() == null));
assertFalse(md.osVersion().isPresent()); }
assertTrue(md.packages().size() > 0 } catch (IOException exp) {
|| EMPTY_MODULES.contains(md.name()), md.name()); throw new UncheckedIOException(exp);
} }
} }

View File

@ -38,7 +38,6 @@ import jdk.testlibrary.FileUtils;
import static jdk.testlibrary.ProcessTools.*; import static jdk.testlibrary.ProcessTools.*;
import org.testng.annotations.BeforeTest; import org.testng.annotations.BeforeTest;
import org.testng.annotations.Test; import org.testng.annotations.Test;
import static org.testng.Assert.*; import static org.testng.Assert.*;
@ -48,7 +47,9 @@ import static org.testng.Assert.*;
* @bug 8142968 8173381 8174740 * @bug 8142968 8173381 8174740
* @library /lib/testlibrary * @library /lib/testlibrary
* @modules jdk.compiler jdk.jlink * @modules jdk.compiler jdk.jlink
* @build UserModuleTest CompilerUtils jdk.testlibrary.FileUtils jdk.testlibrary.ProcessTools * @modules java.base/jdk.internal.module
* @modules java.base/jdk.internal.org.objectweb.asm
* @build ModuleTargetHelper UserModuleTest CompilerUtils jdk.testlibrary.FileUtils jdk.testlibrary.ProcessTools
* @run testng UserModuleTest * @run testng UserModuleTest
*/ */
@ -85,7 +86,9 @@ public class UserModuleTest {
for (String mn : modules) { for (String mn : modules) {
Path msrc = SRC_DIR.resolve(mn); Path msrc = SRC_DIR.resolve(mn);
assertTrue(CompilerUtils.compile(msrc, MODS_DIR, assertTrue(CompilerUtils.compile(msrc, MODS_DIR,
"--module-source-path", SRC_DIR.toString())); "--module-source-path", SRC_DIR.toString(),
"--add-exports", "java.base/jdk.internal.module=" + mn,
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=" + mn));
} }
if (Files.exists(IMAGE)) { if (Files.exists(IMAGE)) {
@ -106,7 +109,10 @@ public class UserModuleTest {
if (!hasJmods()) return; if (!hasJmods()) return;
Path java = IMAGE.resolve("bin").resolve("java"); Path java = IMAGE.resolve("bin").resolve("java");
assertTrue(executeProcess(java.toString(), "-m", MAIN_MID) assertTrue(executeProcess(java.toString(),
"--add-exports", "java.base/jdk.internal.module=m1,m4",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1,m4",
"-m", MAIN_MID)
.outputTo(System.out) .outputTo(System.out)
.errorTo(System.out) .errorTo(System.out)
.getExitValue() == 0); .getExitValue() == 0);
@ -136,6 +142,8 @@ public class UserModuleTest {
Path java = IMAGE.resolve("bin").resolve("java"); Path java = IMAGE.resolve("bin").resolve("java");
assertTrue(executeProcess(java.toString(), assertTrue(executeProcess(java.toString(),
"--add-exports", "java.base/jdk.internal.module=m1,m4",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1,m4",
"-Djdk.system.module.finder.disabledFastPath", "-Djdk.system.module.finder.disabledFastPath",
"-m", MAIN_MID) "-m", MAIN_MID)
.outputTo(System.out) .outputTo(System.out)
@ -154,7 +162,10 @@ public class UserModuleTest {
Path dir = Paths.get("dedupSetTest"); Path dir = Paths.get("dedupSetTest");
createImage(dir, "m1", "m2", "m3", "m4"); createImage(dir, "m1", "m2", "m3", "m4");
Path java = dir.resolve("bin").resolve("java"); Path java = dir.resolve("bin").resolve("java");
assertTrue(executeProcess(java.toString(), "-m", MAIN_MID) assertTrue(executeProcess(java.toString(),
"--add-exports", "java.base/jdk.internal.module=m1,m4",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1,m4",
"-m", MAIN_MID)
.outputTo(System.out) .outputTo(System.out)
.errorTo(System.out) .errorTo(System.out)
.getExitValue() == 0); .getExitValue() == 0);
@ -205,11 +216,13 @@ public class UserModuleTest {
} }
private void createJmods(String... modules) throws IOException { private void createJmods(String... modules) throws IOException {
// use the same target platform as in java.base ModuleTargetHelper.ModuleTarget mt = ModuleTargetHelper.getJavaBaseTarget();
ModuleDescriptor md = Layer.boot().findModule("java.base").get() if (mt == null) {
.getDescriptor(); throw new RuntimeException("ModuleTarget is missing for java.base");
String osName = md.osName().get(); }
String osArch = md.osArch().get();
String osName = mt.osName();
String osArch = mt.osArch();
// create JMOD files // create JMOD files
Files.createDirectories(JMODS_DIR); Files.createDirectories(JMODS_DIR);
@ -246,6 +259,8 @@ public class UserModuleTest {
// verify ModuleDescriptor // verify ModuleDescriptor
Path java = dir.resolve("bin").resolve("java"); Path java = dir.resolve("bin").resolve("java");
assertTrue(executeProcess(java.toString(), assertTrue(executeProcess(java.toString(),
"--add-exports", "java.base/jdk.internal.module=m1,m4",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1,m4",
"--add-modules=m1", "-m", "m4") "--add-modules=m1", "-m", "m4")
.outputTo(System.out) .outputTo(System.out)
.errorTo(System.out) .errorTo(System.out)
@ -275,6 +290,8 @@ public class UserModuleTest {
// verify ModuleDescriptor // verify ModuleDescriptor
Path java = dir.resolve("bin").resolve("java"); Path java = dir.resolve("bin").resolve("java");
assertTrue(executeProcess(java.toString(), assertTrue(executeProcess(java.toString(),
"--add-exports", "java.base/jdk.internal.module=m1,m4",
"--add-exports", "java.base/jdk.internal.org.objectweb.asm=m1,m4",
"--add-modules=m1", "-m", "m4", "retainModuleTarget") "--add-modules=m1", "-m", "m4", "retainModuleTarget")
.outputTo(System.out) .outputTo(System.out)
.errorTo(System.out) .errorTo(System.out)

View File

@ -23,6 +23,7 @@
package p1; package p1;
import java.io.InputStream;
import java.io.IOException; import java.io.IOException;
import java.lang.module.ModuleDescriptor; import java.lang.module.ModuleDescriptor;
import java.lang.reflect.Layer; import java.lang.reflect.Layer;
@ -35,7 +36,38 @@ import java.nio.file.Path;
import java.util.Collections; import java.util.Collections;
import java.util.Set; import java.util.Set;
import jdk.internal.module.ClassFileAttributes;
import jdk.internal.module.ClassFileAttributes.ModuleTargetAttribute;
import jdk.internal.module.ClassFileConstants;
import jdk.internal.org.objectweb.asm.Attribute;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.org.objectweb.asm.ClassVisitor;
import jdk.internal.org.objectweb.asm.Opcodes;
public class Main { public class Main {
private static boolean hasModuleTarget(InputStream in) throws IOException {
ModuleTargetAttribute[] modTargets = new ModuleTargetAttribute[1];
ClassVisitor cv = new ClassVisitor(Opcodes.ASM5) {
@Override
public void visitAttribute(Attribute attr) {
if (attr instanceof ModuleTargetAttribute) {
modTargets[0] = (ModuleTargetAttribute)attr;
}
}
};
// prototype of attributes that should be parsed
Attribute[] attrs = new Attribute[] {
new ModuleTargetAttribute()
};
// parse module-info.class
ClassReader cr = new ClassReader(in);
cr.accept(cv, attrs, 0);
return modTargets[0] != null &&
(modTargets[0].osName() != null || modTargets[0].osArch() != null);
}
public static void main(String... args) throws Exception { public static void main(String... args) throws Exception {
// load another package // load another package
p2.T.test(); p2.T.test();
@ -44,12 +76,13 @@ public class Main {
validate(Main.class.getModule()); validate(Main.class.getModule());
// validate the Moduletarget attribute for java.base // validate the Moduletarget attribute for java.base
ModuleDescriptor md = Layer.boot().findModule("java.base").get() FileSystem fs = FileSystems.newFileSystem(URI.create("jrt:/"),
.getDescriptor(); Collections.emptyMap());
if (!md.osName().isPresent() || !md.osArch().isPresent() || Path path = fs.getPath("/", "modules", "java.base", "module-info.class");
!md.osVersion().isPresent()) { try (InputStream in = Files.newInputStream(path)) {
throw new RuntimeException("java.base: " + md.osName() + " " + if (! hasModuleTarget(in)) {
md.osArch() + " " + md.osVersion()); throw new RuntimeException("Missing ModuleTarget for java.base");
}
} }
} }
@ -67,9 +100,9 @@ public class Main {
checkPackages(md.packages(), "p1", "p2"); checkPackages(md.packages(), "p1", "p2");
checkPackages(md1.packages(), "p1", "p2"); checkPackages(md1.packages(), "p1", "p2");
// check ModuleTarget attribute try (InputStream in = Files.newInputStream(path)) {
checkModuleTargetAttribute(md); checkModuleTargetAttribute(in, "p1");
checkModuleTargetAttribute(md1); }
} }
static void checkPackages(Set<String> pkgs, String... expected) { static void checkPackages(Set<String> pkgs, String... expected) {
@ -78,10 +111,9 @@ public class Main {
} }
} }
static void checkModuleTargetAttribute(ModuleDescriptor md) { static void checkModuleTargetAttribute(InputStream in, String modName) throws IOException {
if (md.osName().isPresent() || md.osArch().isPresent() || if (hasModuleTarget(in)) {
md.osVersion().isPresent()) { throw new RuntimeException("ModuleTarget present for " + modName);
throw new RuntimeException(md.osName() + " " + md.osArch() + " " + md.osVersion());
} }
} }
} }

View File

@ -36,7 +36,47 @@ import java.nio.file.Path;
import java.util.Collections; import java.util.Collections;
import java.util.Set; import java.util.Set;
import jdk.internal.module.ClassFileAttributes;
import jdk.internal.module.ClassFileAttributes.ModuleTargetAttribute;
import jdk.internal.module.ClassFileConstants;
import jdk.internal.org.objectweb.asm.Attribute;
import jdk.internal.org.objectweb.asm.ClassReader;
import jdk.internal.org.objectweb.asm.ClassVisitor;
import jdk.internal.org.objectweb.asm.Opcodes;
public class Main { public class Main {
private static boolean hasModuleTarget(InputStream in) throws IOException {
ModuleTargetAttribute[] modTargets = new ModuleTargetAttribute[1];
ClassVisitor cv = new ClassVisitor(Opcodes.ASM5) {
@Override
public void visitAttribute(Attribute attr) {
if (attr instanceof ModuleTargetAttribute) {
modTargets[0] = (ModuleTargetAttribute)attr;
}
}
};
// prototype of attributes that should be parsed
Attribute[] attrs = new Attribute[] {
new ModuleTargetAttribute()
};
// parse module-info.class
ClassReader cr = new ClassReader(in);
cr.accept(cv, attrs, 0);
return modTargets[0] != null &&
(modTargets[0].osName() != null || modTargets[0].osArch() != null);
}
private static boolean hasModuleTarget(String modName) throws IOException {
FileSystem fs = FileSystems.newFileSystem(URI.create("jrt:/"),
Collections.emptyMap());
Path path = fs.getPath("/", "modules", modName, "module-info.class");
try (InputStream in = Files.newInputStream(path)) {
return hasModuleTarget(in);
}
}
// the system module plugin by default drops ModuleTarget attribute // the system module plugin by default drops ModuleTarget attribute
private static boolean expectModuleTarget = false; private static boolean expectModuleTarget = false;
public static void main(String... args) throws IOException { public static void main(String... args) throws IOException {
@ -49,13 +89,8 @@ public class Main {
} }
// java.base is packaged with osName/osArch/osVersion // java.base is packaged with osName/osArch/osVersion
ModuleDescriptor md = Layer.boot().findModule("java.base").get() if (! hasModuleTarget("java.base")) {
.getDescriptor(); throw new RuntimeException("ModuleTarget absent for java.base");
if (!md.osName().isPresent() ||
!md.osArch().isPresent() ||
!md.osVersion().isPresent()) {
throw new RuntimeException("osName/osArch/osVersion is missing: " +
md.osName() + " " + md.osArch() + " " + md.osVersion());
} }
// verify module-info.class for m1 and m4 // verify module-info.class for m1 and m4
@ -82,7 +117,7 @@ public class Main {
checkModuleDescriptor(ModuleDescriptor.read(Files.newInputStream(path)), packages); checkModuleDescriptor(ModuleDescriptor.read(Files.newInputStream(path)), packages);
} }
static void checkModuleDescriptor(ModuleDescriptor md, String... packages) { static void checkModuleDescriptor(ModuleDescriptor md, String... packages) throws IOException {
String mainClass = md.name().replace('m', 'p') + ".Main"; String mainClass = md.name().replace('m', 'p') + ".Main";
if (!md.mainClass().get().equals(mainClass)) { if (!md.mainClass().get().equals(mainClass)) {
throw new RuntimeException(md.mainClass().toString()); throw new RuntimeException(md.mainClass().toString());
@ -90,22 +125,16 @@ public class Main {
if (expectModuleTarget) { if (expectModuleTarget) {
// ModuleTarget attribute is retained // ModuleTarget attribute is retained
if (!md.osName().isPresent() || !md.osArch().isPresent()) { if (! hasModuleTarget(md.name())) {
throw new RuntimeException("osName or osArch is missing: " + throw new RuntimeException("ModuleTarget missing for " + md.name());
md.osName() + " " + md.osArch());
} }
} else { } else {
// by default ModuleTarget attribute is dropped // by default ModuleTarget attribute is dropped
if (md.osName().isPresent() || md.osArch().isPresent()) { if (hasModuleTarget(md.name())) {
throw new RuntimeException("osName and osArch should not be set: " + throw new RuntimeException("ModuleTarget present for " + md.name());
md.osName() + " " + md.osArch());
} }
} }
if (md.osVersion().isPresent()) {
throw new RuntimeException("Expected no osVersion set: " + md.osVersion());
}
Set<String> pkgs = md.packages(); Set<String> pkgs = md.packages();
if (!pkgs.equals(Set.of(packages))) { if (!pkgs.equals(Set.of(packages))) {
throw new RuntimeException(pkgs + " expected: " + Set.of(packages)); throw new RuntimeException(pkgs + " expected: " + Set.of(packages));

View File

@ -134,16 +134,16 @@ public class AddExportsTestWarningError {
return new Object[][]{ return new Object[][]{
// source not found // source not found
{"DoesNotExist/p=m1", "WARNING: Unknown module: DoesNotExist specified in --add-exports"}, {"DoesNotExist/p=m1", "WARNING: Unknown module: DoesNotExist specified to --add-exports"},
{"m1/DoesNotExist=m2", "WARNING: package DoesNotExist not in m1"}, {"m1/DoesNotExist=m2", "WARNING: package DoesNotExist not in m1"},
// target not found // target not found
{"m1/p1=DoesNotExist", "WARNING: Unknown module: DoesNotExist specified in --add-exports"}, {"m1/p1=DoesNotExist", "WARNING: Unknown module: DoesNotExist specified to --add-exports"},
// bad names // bad names
{"m*/p1=m2", "WARNING: Unknown module: m* specified in --add-exports"}, {"m*/p1=m2", "WARNING: Unknown module: m* specified to --add-exports"},
{"m1/p!=m2", "WARNING: package p! not in m1"}, {"m1/p!=m2", "WARNING: package p! not in m1"},
{"m1/p1=m!", "WARNING: Unknown module: m! specified in --add-exports"}, {"m1/p1=m!", "WARNING: Unknown module: m! specified to --add-exports"},
}; };
} }

View File

@ -168,14 +168,14 @@ public class AddReadsTestWarningError {
return new Object[][]{ return new Object[][]{
// source not found // source not found
{"DoesNotExist=m2", "WARNING: Unknown module: DoesNotExist specified in --add-reads"}, {"DoesNotExist=m2", "WARNING: Unknown module: DoesNotExist specified to --add-reads"},
// target not found // target not found
{"m2=DoesNotExist", "WARNING: Unknown module: DoesNotExist specified in --add-reads"}, {"m2=DoesNotExist", "WARNING: Unknown module: DoesNotExist specified to --add-reads"},
// bad names // bad names
{"m*=m2", "WARNING: Unknown module: m* specified in --add-reads"}, {"m*=m2", "WARNING: Unknown module: m* specified to --add-reads"},
{"m2=m!", "WARNING: Unknown module: m! specified in --add-reads"}, {"m2=m!", "WARNING: Unknown module: m! specified to --add-reads"},
}; };
} }

View File

@ -0,0 +1,116 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @library /lib/testlibrary
* @build InitErrors jdk.testlibrary.*
* @run testng InitErrors
* @summary Basic test to ensure that module system initialization errors
* go the right stream and with the right level of verbosity
*/
import java.util.Arrays;
import jdk.testlibrary.ProcessTools;
import jdk.testlibrary.OutputAnalyzer;
import org.testng.annotations.Test;
import static org.testng.Assert.*;
public class InitErrors {
// the option to cause module initialization to fail
private static final String ADD_UNKNOWN_MODULE = "--add-modules=XXX";
// the expected error message
private static final String UNKNOWN_MODULE_NOT_FOUND= "Module XXX not found";
// output expected in the stack trace when using -Xlog:init=debug
private static final String STACK_FRAME = "java.base/java.lang.System.initPhase2";
/**
* Default behavior, send error message to stdout
*/
@Test
public void testDefaultOutput() throws Exception {
expectFail(showVersion(ADD_UNKNOWN_MODULE)
.stdoutShouldContain(UNKNOWN_MODULE_NOT_FOUND)
.stdoutShouldNotContain(STACK_FRAME)
.stderrShouldNotContain(UNKNOWN_MODULE_NOT_FOUND)
.stderrShouldNotContain(STACK_FRAME));
}
/**
* -XX:+DisplayVMOutputToStderr should send error message to stderr
*/
@Test
public void testOutputToStderr() throws Exception {
expectFail(showVersion(ADD_UNKNOWN_MODULE, "-XX:+DisplayVMOutputToStderr")
.stdoutShouldNotContain(UNKNOWN_MODULE_NOT_FOUND)
.stdoutShouldNotContain(STACK_FRAME)
.stderrShouldContain(UNKNOWN_MODULE_NOT_FOUND)
.stderrShouldNotContain(STACK_FRAME));
}
/**
* -Xlog:init=debug should print stack trace to stdout
*/
@Test
public void testStackTrace() throws Exception {
expectFail(showVersion(ADD_UNKNOWN_MODULE, "-Xlog:init=debug")
.stdoutShouldContain(UNKNOWN_MODULE_NOT_FOUND)
.stdoutShouldContain(STACK_FRAME)
.stderrShouldNotContain(UNKNOWN_MODULE_NOT_FOUND)
.stderrShouldNotContain(STACK_FRAME));
}
/**
* -Xlog:init=debug -XX:+DisplayVMOutputToStderr should print stack trace
* to stderr
*/
@Test
public void testStackTraceToStderr() throws Exception {
expectFail(showVersion(ADD_UNKNOWN_MODULE,
"-Xlog:init=debug",
"-XX:+DisplayVMOutputToStderr")
.stdoutShouldNotContain(UNKNOWN_MODULE_NOT_FOUND)
.stdoutShouldNotContain(STACK_FRAME)
.stderrShouldContain(UNKNOWN_MODULE_NOT_FOUND)
.stderrShouldContain(STACK_FRAME));
}
private OutputAnalyzer showVersion(String... args) throws Exception {
int len = args.length;
args = Arrays.copyOf(args, len+1);
args[len] = "-version";
return ProcessTools.executeTestJava(args)
.outputTo(System.out)
.errorTo(System.out);
}
private void expectFail(OutputAnalyzer output) {
assertFalse(output.getExitValue() == 0);
}
}

View File

@ -179,7 +179,7 @@ public class PatchTestWarningError {
"-m", "test/jdk.test.Main", arg) "-m", "test/jdk.test.Main", arg)
.outputTo(System.out) .outputTo(System.out)
.errorTo(System.out) .errorTo(System.out)
.shouldContain("WARNING: Unknown module: DoesNotExist specified in --patch-module") .shouldContain("WARNING: Unknown module: DoesNotExist specified to --patch-module")
.getExitValue(); .getExitValue();
assertTrue(exitValue == 0); assertTrue(exitValue == 0);

View File

@ -0,0 +1,68 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
/**
* Launched by PermitIllegalAccess to attempt illegal access.
*/
public class AttemptAccess {
public static void main(String[] args) throws Exception {
String action = args[0];
int count = Integer.parseInt(args[1]);
for (int i=0; i<count; i++) {
switch (action) {
case "access":
tryAccess();
break;
case "setAccessible":
trySetAccessible();
break;
case "trySetAccessible":
tryTrySetAccessible();
break;
}
}
}
static void tryAccess() throws Exception {
Class<?> clazz = Class.forName("sun.security.x509.X500Name");
Constructor<?> ctor = clazz.getConstructor(String.class);
Object name = ctor.newInstance("CN=user");
}
static void trySetAccessible() throws Exception {
Method find = ClassLoader.class.getDeclaredMethod("findClass", String.class);
find.setAccessible(true);
}
static void tryTrySetAccessible() throws Exception {
Method find = ClassLoader.class.getDeclaredMethod("findClass", String.class);
find.trySetAccessible();
}
}

View File

@ -0,0 +1,208 @@
/*
* Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @library /lib/testlibrary
* @build PermitIllegalAccess AttemptAccess jdk.testlibrary.*
* @run testng PermitIllegalAccess
* @summary Basic test for java --permit-illegal-access
*/
import java.util.List;
import jdk.testlibrary.ProcessTools;
import jdk.testlibrary.OutputAnalyzer;
import org.testng.annotations.Test;
import static org.testng.Assert.*;
/**
* Basic test of --permit-illegal-access to ensure that it permits access
* via core reflection and setAccessible/trySetAccessible.
*/
@Test
public class PermitIllegalAccess {
static final String TEST_CLASSES = System.getProperty("test.classes");
static final String TEST_MAIN = "AttemptAccess";
static final String WARNING = "WARNING";
static final String STARTUP_WARNING =
"WARNING: --permit-illegal-access will be removed in the next major release";
static final String ILLEGAL_ACCESS_WARNING =
"WARNING: Illegal access by " + TEST_MAIN;
/**
* Launches AttemptAccess to execute an action, returning the OutputAnalyzer
* to analyze the output/exitCode.
*/
private OutputAnalyzer tryAction(String action, int count) throws Exception {
String arg = "" + count;
return ProcessTools
.executeTestJava("-cp", TEST_CLASSES, TEST_MAIN, action, arg)
.outputTo(System.out)
.errorTo(System.out);
}
/**
* Launches AttemptAccess with --permit-illegal-access to execute an action,
* returning the OutputAnalyzer to analyze the output/exitCode.
*/
private OutputAnalyzer tryActionPermittingIllegalAccess(String action,
int count)
throws Exception
{
String arg = "" + count;
return ProcessTools
.executeTestJava("-cp", TEST_CLASSES, "--permit-illegal-access",
TEST_MAIN, action, arg)
.outputTo(System.out)
.errorTo(System.out);
}
/**
* Sanity check to ensure that IllegalAccessException is thrown.
*/
public void testAccessFail() throws Exception {
int exitValue = tryAction("access", 1)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("IllegalAccessException")
.stderrShouldNotContain(WARNING)
.stderrShouldContain("IllegalAccessException")
.getExitValue();
assertTrue(exitValue != 0);
}
/**
* Sanity check to ensure that InaccessibleObjectException is thrown.
*/
public void testSetAccessibleFail() throws Exception {
int exitValue = tryAction("setAccessible", 1)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("InaccessibleObjectException")
.stderrShouldNotContain(WARNING)
.stderrShouldContain("InaccessibleObjectException")
.getExitValue();
assertTrue(exitValue != 0);
}
/**
* Permit illegal access to succeed
*/
public void testAccessPermitted() throws Exception {
tryActionPermittingIllegalAccess("access", 1)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("IllegalAccessException")
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldNotContain("IllegalAccessException")
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);
}
/**
* Permit repeated illegal access to succeed
*/
public void testRepeatedAccessPermitted() throws Exception {
OutputAnalyzer outputAnalyzer = tryActionPermittingIllegalAccess("access", 10)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("IllegalAccessException")
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldNotContain("IllegalAccessException")
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);;
// should only have one illegal access warning
assertTrue(containsCount(outputAnalyzer.asLines(), ILLEGAL_ACCESS_WARNING) == 1);
}
/**
* Permit setAccessible to succeed
*/
public void testSetAccessiblePermitted() throws Exception {
tryActionPermittingIllegalAccess("setAccessible", 1)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);
}
/**
* Permit repeated calls to setAccessible to succeed
*/
public void testRepeatedSetAccessiblePermitted() throws Exception {
OutputAnalyzer outputAnalyzer = tryActionPermittingIllegalAccess("setAccessible", 10)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);
// should only have one illegal access warning
assertTrue(containsCount(outputAnalyzer.asLines(), ILLEGAL_ACCESS_WARNING) == 1);
}
/**
* Permit trySetAccessible to succeed
*/
public void testTrySetAccessiblePermitted() throws Exception {
tryActionPermittingIllegalAccess("trySetAccessible", 1)
.stdoutShouldNotContain(WARNING)
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);
}
/**
* Permit repeated calls to trySetAccessible to succeed
*/
public void testRepeatedTrySetAccessiblePermitted() throws Exception {
OutputAnalyzer outputAnalyzer = tryActionPermittingIllegalAccess("trySetAccessible", 10)
.stdoutShouldNotContain(WARNING)
.stdoutShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(STARTUP_WARNING)
.stderrShouldNotContain("InaccessibleObjectException")
.stderrShouldContain(ILLEGAL_ACCESS_WARNING)
.shouldHaveExitValue(0);
// should only have one illegal access warning
assertTrue(containsCount(outputAnalyzer.asLines(), ILLEGAL_ACCESS_WARNING) == 1);
}
/**
* Returns the number of lines in the given input that contain the
* given char sequence.
*/
private int containsCount(List<String> lines, CharSequence cs) {
int count = 0;
for (String line : lines) {
if (line.contains(cs)) count++;
}
return count;
}
}

Some files were not shown because too many files have changed in this diff Show More