8164742: ServiceConfigurationError on invoke of getServiceLoader method of StandardJavaFileManager

Reviewed-by: jlahoda
This commit is contained in:
Srikanth Adayapalam 2016-09-19 05:31:53 +05:30
parent f4abd474cc
commit ac462ed848
7 changed files with 157 additions and 84 deletions

View File

@ -105,12 +105,12 @@ public class MutableFieldsAnalyzer extends AbstractCodingRulesAnalyzer {
"configurationClass", "resolveRequiresAndUsesMethod"); "configurationClass", "resolveRequiresAndUsesMethod");
ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$Layer", ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$Layer",
"layerClass", "bootMethod", "defineModulesWithOneLoaderMethod", "configurationMethod"); "layerClass", "bootMethod", "defineModulesWithOneLoaderMethod", "configurationMethod");
ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$Module",
"addExportsMethod", "addUsesMethod", "getModuleMethod", "getUnnamedModuleMethod");
ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$ServiceLoaderHelper", ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$ServiceLoaderHelper",
"loadMethod"); "loadMethod");
ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$VMHelper", ignoreFields("com.sun.tools.javac.util.JDK9Wrappers$VMHelper",
"vmClass", "getRuntimeArgumentsMethod"); "vmClass", "getRuntimeArgumentsMethod");
ignoreFields("com.sun.tools.javac.util.ModuleHelper",
"addExportsMethod", "getUnnamedModuleMethod", "getModuleMethod");
} }
} }

View File

@ -78,6 +78,7 @@ import com.sun.tools.javac.util.ListBuffer;
import com.sun.tools.javac.util.JDK9Wrappers.Configuration; import com.sun.tools.javac.util.JDK9Wrappers.Configuration;
import com.sun.tools.javac.util.JDK9Wrappers.Layer; import com.sun.tools.javac.util.JDK9Wrappers.Layer;
import com.sun.tools.javac.util.JDK9Wrappers.ModuleFinder; import com.sun.tools.javac.util.JDK9Wrappers.ModuleFinder;
import com.sun.tools.javac.util.JDK9Wrappers.Module;
import com.sun.tools.javac.util.JDK9Wrappers.ServiceLoaderHelper; import com.sun.tools.javac.util.JDK9Wrappers.ServiceLoaderHelper;
import static java.nio.file.FileVisitOption.FOLLOW_LINKS; import static java.nio.file.FileVisitOption.FOLLOW_LINKS;
@ -957,6 +958,7 @@ public class JavacFileManager extends BaseFileManager implements StandardJavaFil
public <S> ServiceLoader<S> getServiceLoader(Location location, Class<S> service) throws IOException { public <S> ServiceLoader<S> getServiceLoader(Location location, Class<S> service) throws IOException {
nullCheck(location); nullCheck(location);
nullCheck(service); nullCheck(service);
Module.getModule(getClass()).addUses(service);
if (location.isModuleLocation()) { if (location.isModuleLocation()) {
Collection<Path> paths = locations.getLocation(location); Collection<Path> paths = locations.getLocation(location);
ModuleFinder finder = ModuleFinder.of(paths.toArray(new Path[paths.size()])); ModuleFinder finder = ModuleFinder.of(paths.toArray(new Path[paths.size()]));

View File

@ -79,6 +79,7 @@ import com.sun.tools.javac.util.DefinedBy;
import com.sun.tools.javac.util.DefinedBy.Api; import com.sun.tools.javac.util.DefinedBy.Api;
import com.sun.tools.javac.util.Iterators; import com.sun.tools.javac.util.Iterators;
import com.sun.tools.javac.util.JCDiagnostic; import com.sun.tools.javac.util.JCDiagnostic;
import com.sun.tools.javac.util.JDK9Wrappers.Module;
import com.sun.tools.javac.util.JavacMessages; import com.sun.tools.javac.util.JavacMessages;
import com.sun.tools.javac.util.List; import com.sun.tools.javac.util.List;
import com.sun.tools.javac.util.Log; import com.sun.tools.javac.util.Log;
@ -119,7 +120,6 @@ public class JavacProcessingEnvironment implements ProcessingEnvironment, Closea
private final JavacTypes typeUtils; private final JavacTypes typeUtils;
private final JavaCompiler compiler; private final JavaCompiler compiler;
private final Modules modules; private final Modules modules;
private final ModuleHelper moduleHelper;
private final Types types; private final Types types;
/** /**
@ -227,7 +227,6 @@ public class JavacProcessingEnvironment implements ProcessingEnvironment, Closea
enter = Enter.instance(context); enter = Enter.instance(context);
initialCompleter = ClassFinder.instance(context).getCompleter(); initialCompleter = ClassFinder.instance(context).getCompleter();
chk = Check.instance(context); chk = Check.instance(context);
moduleHelper = ModuleHelper.instance(context);
initProcessorLoader(); initProcessorLoader();
defaultModule = source.allowModules() && options.isUnset("noModules") defaultModule = source.allowModules() && options.isUnset("noModules")
@ -265,7 +264,8 @@ public class JavacProcessingEnvironment implements ProcessingEnvironment, Closea
? fileManager.getClassLoader(ANNOTATION_PROCESSOR_PATH) ? fileManager.getClassLoader(ANNOTATION_PROCESSOR_PATH)
: fileManager.getClassLoader(CLASS_PATH); : fileManager.getClassLoader(CLASS_PATH);
moduleHelper.addExports(processorClassLoader); if (options.isSet("accessInternalAPI"))
ModuleHelper.addExports(Module.getModule(getClass()), Module.getUnnamedModule(processorClassLoader));
if (processorClassLoader != null && processorClassLoader instanceof Closeable) { if (processorClassLoader != null && processorClassLoader instanceof Closeable) {
compiler.closeables = compiler.closeables.prepend((Closeable) processorClassLoader); compiler.closeables = compiler.closeables.prepend((Closeable) processorClassLoader);

View File

@ -129,6 +129,85 @@ public class JDK9Wrappers {
} }
} }
/**
* Wrapper class for java.lang.reflect.Module. To materialize a handle use the static factory
* methods Module#getModule(Class<?>) or Module#getUnnamedModule(ClassLoader).
*/
public static class Module {
private final Object theRealModule;
private Module(Object module) {
this.theRealModule = module;
init();
}
public static Module getModule(Class<?> clazz) {
try {
init();
Object result = getModuleMethod.invoke(clazz, new Object[0]);
return new Module(result);
} catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException
| SecurityException ex) {
throw new Abort(ex);
}
}
public static Module getUnnamedModule(ClassLoader classLoader) {
try {
init();
Object result = getUnnamedModuleMethod.invoke(classLoader, new Object[0]);
return new Module(result);
} catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException
| SecurityException ex) {
throw new Abort(ex);
}
}
public Module addExports(String pn, Module other) {
try {
addExportsMethod.invoke(theRealModule, new Object[] { pn, other.theRealModule});
} catch (IllegalAccessException | InvocationTargetException ex) {
throw new Abort(ex);
}
return this;
}
public Module addUses(Class<?> st) {
try {
addUsesMethod.invoke(theRealModule, new Object[] { st });
} catch (IllegalAccessException | InvocationTargetException ex) {
throw new Abort(ex);
}
return this;
}
// -----------------------------------------------------------------------------------------
// on java.lang.reflect.Module
private static Method addExportsMethod = null;
// on java.lang.reflect.Module
private static Method addUsesMethod = null;
// on java.lang.Class
private static Method getModuleMethod;
// on java.lang.ClassLoader
private static Method getUnnamedModuleMethod;
private static void init() {
if (addExportsMethod == null) {
try {
Class<?> moduleClass = Class.forName("java.lang.reflect.Module", false, null);
addUsesMethod = moduleClass.getDeclaredMethod("addUses", new Class<?>[] { Class.class });
addExportsMethod = moduleClass.getDeclaredMethod("addExports",
new Class<?>[] { String.class, moduleClass });
getModuleMethod = Class.class.getDeclaredMethod("getModule", new Class<?>[0]);
getUnnamedModuleMethod = ClassLoader.class.getDeclaredMethod("getUnnamedModule", new Class<?>[0]);
} catch (ClassNotFoundException | NoSuchMethodException | SecurityException ex) {
throw new Abort(ex);
}
}
}
}
/** /**
* Wrapper class for java.lang.module.Configuration. * Wrapper class for java.lang.module.Configuration.
*/ */

View File

@ -25,88 +25,30 @@
package com.sun.tools.javac.util; package com.sun.tools.javac.util;
import java.lang.reflect.InvocationTargetException; import com.sun.tools.javac.util.JDK9Wrappers.Module;
import java.lang.reflect.Method;
public class ModuleHelper { public class ModuleHelper {
/** The context key for the module helper. */
protected static final Context.Key<ModuleHelper> moduleHelperKey = new Context.Key<>();
/** Get the JavaCompiler instance for this context. */ private static final String[] javacInternalPackages = new String[] {
public static ModuleHelper instance(Context context) { "com.sun.tools.javac.api",
ModuleHelper instance = context.get(moduleHelperKey); "com.sun.tools.javac.code",
if (instance == null) "com.sun.tools.javac.comp",
instance = new ModuleHelper(context); "com.sun.tools.javac.file",
return instance; "com.sun.tools.javac.jvm",
} "com.sun.tools.javac.main",
"com.sun.tools.javac.model",
"com.sun.tools.javac.parser",
"com.sun.tools.javac.platform",
"com.sun.tools.javac.processing",
"com.sun.tools.javac.tree",
"com.sun.tools.javac.util",
public ModuleHelper(Context context) { "com.sun.tools.doclint",
context.put(moduleHelperKey, this);
Options options = Options.instance(context);
allowAccessToInternalAPI = options.isSet("accessInternalAPI");
}
final boolean allowAccessToInternalAPI;
private void exportPackageToModule(String packageName, Object target)
throws ClassNotFoundException, NoSuchMethodException, IllegalArgumentException,
InvocationTargetException, IllegalAccessException {
if (addExportsMethod == null) {
Class<?> moduleClass = Class.forName("java.lang.reflect.Module");
addExportsMethod = moduleClass.getDeclaredMethod("addExports",
new Class<?>[] { String.class, moduleClass });
}
addExportsMethod.invoke(from, new Object[] { packageName, target });
}
static final String[] javacInternalPackages = new String[] {
"com.sun.tools.javac.api",
"com.sun.tools.javac.code",
"com.sun.tools.javac.comp",
"com.sun.tools.javac.file",
"com.sun.tools.javac.jvm",
"com.sun.tools.javac.main",
"com.sun.tools.javac.model",
"com.sun.tools.javac.parser",
"com.sun.tools.javac.platform",
"com.sun.tools.javac.processing",
"com.sun.tools.javac.tree",
"com.sun.tools.javac.util",
"com.sun.tools.doclint",
}; };
public void addExports(ClassLoader classLoader) { public static void addExports(Module from, Module to) {
try { for (String pack: javacInternalPackages) {
if (allowAccessToInternalAPI) { from.addExports(pack, to);
if (from == null) {
if (getModuleMethod == null) {
getModuleMethod = Class.class.getDeclaredMethod("getModule", new Class<?>[0]);
}
from = getModuleMethod.invoke(getClass(), new Object[0]);
}
if (getUnnamedModuleMethod == null) {
getUnnamedModuleMethod = ClassLoader.class.getDeclaredMethod("getUnnamedModule", new Class<?>[0]);
}
Object target = getUnnamedModuleMethod.invoke(classLoader, new Object[0]);
for (String pack: javacInternalPackages) {
exportPackageToModule(pack, target);
}
}
} catch (Exception e) {
// do nothing
} }
} }
}
// a module instance
private Object from = null;
// on java.lang.reflect.Module
private static Method addExportsMethod = null;
// on java.lang.ClassLoader
private static Method getUnnamedModuleMethod = null;
// on java.lang.Class
private static Method getModuleMethod = null;
}

View File

@ -106,12 +106,12 @@ public class DetectMutableStaticFields {
// The following static fields are used for caches of information obtained // The following static fields are used for caches of information obtained
// by reflective lookup, to avoid explicit references that are not available // by reflective lookup, to avoid explicit references that are not available
// when running javac on JDK 8. // when running javac on JDK 8.
ignore("com/sun/tools/javac/util/ModuleHelper",
"addExportsMethod", "getModuleMethod", "getUnnamedModuleMethod");
ignore("com/sun/tools/javac/util/JDK9Wrappers$Configuration", ignore("com/sun/tools/javac/util/JDK9Wrappers$Configuration",
"resolveRequiresAndUsesMethod", "configurationClass"); "resolveRequiresAndUsesMethod", "configurationClass");
ignore("com/sun/tools/javac/util/JDK9Wrappers$Layer", ignore("com/sun/tools/javac/util/JDK9Wrappers$Layer",
"bootMethod", "defineModulesWithOneLoaderMethod", "configurationMethod", "layerClass"); "bootMethod", "defineModulesWithOneLoaderMethod", "configurationMethod", "layerClass");
ignore("com/sun/tools/javac/util/JDK9Wrappers$Module",
"addExportsMethod", "addUsesMethod", "getModuleMethod", "getUnnamedModuleMethod");
ignore("com/sun/tools/javac/util/JDK9Wrappers$ModuleFinder", ignore("com/sun/tools/javac/util/JDK9Wrappers$ModuleFinder",
"moduleFinderClass", "ofMethod"); "moduleFinderClass", "ofMethod");
ignore("com/sun/tools/javac/util/JDK9Wrappers$ServiceLoaderHelper", ignore("com/sun/tools/javac/util/JDK9Wrappers$ServiceLoaderHelper",

View File

@ -0,0 +1,50 @@
/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 8164742
* @summary Test that jdk.compiler can materialize a service loader for arbitrary services
* @run main FileManagerGetServiceLoaderTest
*/
import javax.tools.*;
public class FileManagerGetServiceLoaderTest {
public static void main(String... args) throws Exception {
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
StandardJavaFileManager fm = compiler.getStandardFileManager(null, null, null);
/* FileManagerGetServiceLoaderTest.class is not really a service, but that is
immaterial to the test which just verifies addUses would have been called
so module boundary is not an issue for a class outside of jdk.compiler
*/
java.util.ServiceLoader<?> loader = fm.getServiceLoader(StandardLocation.CLASS_PATH,
FileManagerGetServiceLoaderTest.class);
if (loader == null) {
throw new AssertionError("Could not obtain service loader");
}
}
}