com.hazelcast.org.codehaus.janino.JavaSourceClassLoader Maven / Gradle / Ivy
/*
* Janino - An embedded Java[TM] compiler
*
* Copyright (c) 2001-2010 Arno Unkrig. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
* following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the
* following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
* following disclaimer in the documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.hazelcast.org.codehaus.janino;
import java.io.File;
import java.io.Reader;
import java.nio.charset.Charset;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import com.hazelcast.org.codehaus.commons.compiler.AbstractJavaSourceClassLoader;
import com.hazelcast.org.codehaus.commons.compiler.CompileException;
import com.hazelcast.org.codehaus.commons.compiler.ErrorHandler;
import com.hazelcast.org.codehaus.commons.compiler.InternalCompilerException;
import com.hazelcast.org.codehaus.commons.compiler.WarningHandler;
import com.hazelcast.org.codehaus.commons.compiler.lang.ClassLoaders;
import com.hazelcast.org.codehaus.commons.compiler.util.Disassembler;
import com.hazelcast.org.codehaus.commons.compiler.util.resource.DirectoryResourceFinder;
import com.hazelcast.org.codehaus.commons.compiler.util.resource.PathResourceFinder;
import com.hazelcast.org.codehaus.commons.compiler.util.resource.ResourceFinder;
import com.hazelcast.org.codehaus.commons.nullanalysis.Nullable;
import com.hazelcast.org.codehaus.janino.util.ClassFile;
/**
* A {@link ClassLoader} that, unlike usual {@link ClassLoader}s, does not load byte code, but reads Java source code
* and then scans, parses, compiles and loads it into the virtual machine.
*
* As with any {@link ClassLoader}, it is not possible to "update" classes after they've been loaded. The way to
* achieve this is to give up on the {@link JavaSourceClassLoader} and create a new one.
*
*
* Notice that this class loader does not support resoures in the sense of {@link ClassLoader#getResource(String)},
* {@link ClassLoader#getResourceAsStream(String)} nd {@link ClassLoader#getResources(String)}.
*
*
* @see ClassLoaders
*/
public
class JavaSourceClassLoader extends AbstractJavaSourceClassLoader {
public
JavaSourceClassLoader() { this(ClassLoader.getSystemClassLoader()); }
public
JavaSourceClassLoader(ClassLoader parentClassLoader) {
this(
parentClassLoader,
(File[]) null, // sourcePath
null // characterEncoding
);
}
/**
* Sets up a {@link JavaSourceClassLoader} that finds Java source code in a file that resides in either of
* the directories specified by the given source path.
*
* @param parentClassLoader See {@link ClassLoader}
* @param sourcePath A collection of directories that are searched for Java source files in
* the given order
* @param characterEncoding The encoding of the Java source files ({@code null} for platform
* default encoding)
*/
public
JavaSourceClassLoader(
ClassLoader parentClassLoader,
@Nullable File[] sourcePath,
@Nullable String characterEncoding
) {
this(
parentClassLoader, // parentClassLoader
( // sourceFinder
sourcePath == null
? new DirectoryResourceFinder(new File("."))
: new PathResourceFinder(sourcePath)
),
characterEncoding // characterEncoding
);
}
/**
* Constructs a {@link JavaSourceClassLoader} that finds Java source code through a given {@link
* ResourceFinder}.
*
* You can specify to include certain debugging information in the generated class files, which is useful if you
* want to debug through the generated classes (see {@link Scanner#Scanner(String, Reader)}).
*
*
* @param parentClassLoader See {@link ClassLoader}
* @param sourceFinder Used to locate additional source files
* @param characterEncoding The encoding of the Java source files ({@code null} for platform
* default encoding)
*/
public
JavaSourceClassLoader(
ClassLoader parentClassLoader,
ResourceFinder sourceFinder,
@Nullable String characterEncoding
) {
this(parentClassLoader, new JavaSourceIClassLoader(
sourceFinder, // sourceFinder
characterEncoding, // characterEncoding
new ClassLoaderIClassLoader(parentClassLoader) // parentIClassLoader
));
}
/**
* Constructs a {@link JavaSourceClassLoader} that finds classes through an {@link JavaSourceIClassLoader}.
*/
public
JavaSourceClassLoader(ClassLoader parentClassLoader, JavaSourceIClassLoader iClassLoader) {
super(parentClassLoader);
this.iClassLoader = iClassLoader;
}
@Override public void
setSourcePath(File[] sourcePath) { this.setSourceFinder(new PathResourceFinder(sourcePath)); }
@Override public void
setSourceFinder(ResourceFinder sourceFinder) { this.iClassLoader.setSourceFinder(sourceFinder); }
@Override public void
setSourceCharset(Charset charset) { this.iClassLoader.setSourceCharset(charset); }
@Override public void
setDebuggingInfo(boolean debugSource, boolean debugLines, boolean debugVars) {
this.debugSource = debugSource;
this.debugLines = debugLines;
this.debugVars = debugVars;
}
/**
* @see UnitCompiler#setCompileErrorHandler
*/
public void
setCompileErrorHandler(@Nullable ErrorHandler compileErrorHandler) {
this.iClassLoader.setCompileErrorHandler(compileErrorHandler);
}
/**
* @see Parser#setWarningHandler(WarningHandler)
* @see UnitCompiler#setCompileErrorHandler
*/
public void
setWarningHandler(@Nullable WarningHandler warningHandler) {
this.iClassLoader.setWarningHandler(warningHandler);
}
/**
* Implementation of {@link ClassLoader#findClass(String)}.
*
* @throws ClassNotFoundException
*/
@Override protected /*synchronized <- No need to synchronize, because 'loadClass()' is synchronized */ Class>
findClass(@Nullable String name) throws ClassNotFoundException {
assert name != null;
// Check if the bytecode for that class was generated already.
byte[] bytecode = (byte[]) this.precompiledClasses.remove(name);
if (bytecode == null) {
// Read, scan, parse and compile the right compilation unit.
{
Map bytecodes = this.generateBytecodes(name);
if (bytecodes == null) throw new ClassNotFoundException(name);
this.precompiledClasses.putAll(bytecodes);
}
// Now the bytecode for our class should be available.
bytecode = (byte[]) this.precompiledClasses.remove(name);
if (bytecode == null) {
throw new InternalCompilerException(
"SNO: Scanning, parsing and compiling class \""
+ name
+ "\" did not create a class file!?"
);
}
}
if (Boolean.getBoolean("disasm")) Disassembler.disassembleToStdout(bytecode);
return this.defineBytecode(name, bytecode);
}
private final Set compiledUnitCompilers = new HashSet();
/**
* This {@link Map} keeps those classes which were already compiled, but not yet defined i.e. which were not yet
* passed to {@link ClassLoader#defineClass(java.lang.String, byte[], int, int)}.
*/
private final Map precompiledClasses = new HashMap();
/**
* Finds, scans, parses the right compilation unit. Compile the parsed compilation unit to bytecode. This may cause
* more compilation units being scanned and parsed. Continue until all compilation units are compiled.
*
* @return String name => byte[] bytecode, or {@code null} if no source code could be found
* @throws ClassNotFoundException on compilation problems
*/
@Nullable protected Map
generateBytecodes(String name) throws ClassNotFoundException {
if (this.iClassLoader.loadIClass(Descriptor.fromClassName(name)) == null) return null;
Map bytecodes = new HashMap();
COMPILE_UNITS:
for (;;) {
for (UnitCompiler uc : this.iClassLoader.getUnitCompilers()) {
if (!this.compiledUnitCompilers.contains(uc)) {
ClassFile[] cfs;
try {
cfs = uc.compileUnit(this.debugSource, this.debugLines, this.debugVars);
} catch (CompileException ex) {
throw new ClassNotFoundException(ex.getMessage(), ex);
}
for (ClassFile cf : cfs) bytecodes.put(cf.getThisClassName(), cf.toByteArray());
this.compiledUnitCompilers.add(uc);
continue COMPILE_UNITS;
}
}
return bytecodes;
}
}
/**
* @throws ClassFormatError
* @see #setProtectionDomainFactory
*/
private Class>
defineBytecode(String className, byte[] ba) {
return this.defineClass(className, ba, 0, ba.length, (
this.protectionDomainFactory != null
? this.protectionDomainFactory.getProtectionDomain(ClassFile.getSourceResourceName(className))
: null
));
}
private final JavaSourceIClassLoader iClassLoader;
private boolean debugSource = Boolean.getBoolean(Scanner.SYSTEM_PROPERTY_SOURCE_DEBUGGING_ENABLE);
private boolean debugLines = this.debugSource;
private boolean debugVars = this.debugSource;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy