blob: 0f0b02b92c24713e65474de63db942de6f404081 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2005, 2017 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.osgi.internal.hookregistry;
import java.io.FileNotFoundException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import org.eclipse.osgi.internal.framework.EquinoxConfiguration;
import org.eclipse.osgi.internal.loader.BundleLoader;
import org.eclipse.osgi.internal.loader.ModuleClassLoader;
import org.eclipse.osgi.internal.loader.classpath.ClasspathEntry;
import org.eclipse.osgi.internal.loader.classpath.ClasspathManager;
import org.eclipse.osgi.storage.BundleInfo.Generation;
import org.eclipse.osgi.storage.bundlefile.BundleEntry;
/**
* A class loading hook that hooks into a module class loader
*/
public abstract class ClassLoaderHook {
/**
* Gets called by a classpath manager before defining a class. This method allows a class loading hook
* to process the bytes of a class that is about to be defined and return a transformed byte array.
* @param name the name of the class being defined
* @param classbytes the bytes of the class being defined
* @param classpathEntry the ClasspathEntry where the class bytes have been read from.
* @param entry the BundleEntry source of the class bytes
* @param manager the class path manager used to define the requested class
* @return a transformed array of classbytes or null if the original bytes should be used.
*/
public byte[] processClass(String name, byte[] classbytes, ClasspathEntry classpathEntry, BundleEntry entry, ClasspathManager manager) {
return null;
}
/**
* Gets called by a classpath manager before defining a class. This method allows a class loading hook
* to reject a transformation to the class bytes by a
* {@link #processClass(String, byte[], ClasspathEntry, BundleEntry, ClasspathManager) processClass} method.
* @param name the name of the class being defined
* @param transformedBytes the transformed bytes of the class being defined
* @param classpathEntry the ClasspathEntry where the class bytes have been read from
* @param entry the BundleEntry source of the class bytes
* @param manager the class path manager used to define the requested class
* @return returns true if the modified bytes should be rejected; otherwise false is returned
*/
public boolean rejectTransformation(String name, byte[] transformedBytes, ClasspathEntry classpathEntry, BundleEntry entry, ClasspathManager manager) {
return false;
}
/**
* Gets called by a classpath manager when looking for ClasspathEntry objects. This method allows
* a classloading hook to add additional ClasspathEntry objects
* @param cpEntries the list of ClasspathEntry objects currently available for the requested classpath
* @param cp the name of the requested classpath
* @param hostmanager the classpath manager the requested ClasspathEntry is for
* @param sourceGeneration the source generation of the requested ClasspathEntry
* @return true if a ClasspathEntry has been added to cpEntries
*/
public boolean addClassPathEntry(ArrayList<ClasspathEntry> cpEntries, String cp, ClasspathManager hostmanager, Generation sourceGeneration) {
return false;
}
/**
* Gets called by a base data during {@link ModuleClassLoader#findLibrary(String)}.
* A this method is called for each configured class loading hook until one
* class loading hook returns a non-null value. If no class loading hook returns
* a non-null value then the default behavior will be used.
* @param generation the bundle generation to find a native library for.
* @param libName the name of the native library.
* @return The absolute path name of the native library or null.
*/
public String findLocalLibrary(Generation generation, String libName) {
return null;
}
/**
* Gets called by a bundle loader when {@link BundleLoader#getClassLoader()}
* is called the first time in order to allow a hook to create the class loader.
* This should rarely, if ever be overridden. The default implementation
* returns null indicating the built-in implementation should be used.
* Only one hook is able to provide the implementation of the module class loader
* and the first one to return non-null wins.
*
* @param parent the parent classloader
* @param configuration the equinox configuration
* @param delegate the delegate for this classloader
* @param generation the generation for this class loader
* @return returns an implementation of a module class loader or <code>null</code>
* if the built-in implemention is to be used.
*/
public ModuleClassLoader createClassLoader(ClassLoader parent, EquinoxConfiguration configuration, BundleLoader delegate, Generation generation) {
// do nothing
return null;
}
/**
* Gets called by a classpath manager at the end of
* {@link BundleLoader#getClassLoader()} is called the first time and a class loader is created.
* @param classLoader the newly created bundle classloader
*/
public void classLoaderCreated(ModuleClassLoader classLoader) {
// do nothing
}
/**
* Called by a {@link BundleLoader#findClass(String)} method before delegating to the resolved constraints and
* local bundle for a class load. If this method returns null then normal delegation is done. If this method
* returns a non-null value then the rest of the delegation process is skipped and the returned value is used.
* If this method throws a <code>ClassNotFoundException</code> then the calling
* {@link BundleLoader#findClass(String)} method re-throws the exception.
* @param name the name of the class to find
* @param classLoader the module class loader
* @return the class found by this hook or null if normal delegation should continue
* @throws ClassNotFoundException to terminate the delegation and throw an exception
*/
public Class<?> preFindClass(String name, ModuleClassLoader classLoader) throws ClassNotFoundException {
return null;
}
/**
* Called by a {@link BundleLoader#findClass(String)} method after delegating to the resolved constraints and
* local bundle for a class load. This method will only be called if no class was found
* from the normal delegation.
* @param name the name of the class to find
* @param classLoader the bundle class loader
* @return the class found by this hook or null if normal delegation should continue
* @throws ClassNotFoundException to terminate the delegation and throw an exception
*/
public Class<?> postFindClass(String name, ModuleClassLoader classLoader) throws ClassNotFoundException {
return null;
}
/**
* Called by a {@link BundleLoader#findResource(String)} before delegating to the resolved constraints and
* local bundle for a resource load. If this method returns null then normal delegation is done.
* If this method returns a non-null value then the rest of the delegation process is skipped and the returned value is used.
* If this method throws an <code>FileNotFoundException</code> then the delegation is terminated.
* @param name the name of the resource to find
* @param classLoader the bundle class loader
* @return the resource found by this hook or null if normal delegation should continue
* @throws FileNotFoundException to terminate the delegation
*/
public URL preFindResource(String name, ModuleClassLoader classLoader) throws FileNotFoundException {
return null;
}
/**
* Called by a {@link BundleLoader#findResource(String)} after delegating to the resolved constraints and
* local bundle for a resource load. This method will only be called if no resource was found
* from the normal delegation.
* @param name the name of the resource to find
* @param classLoader the bundle class loader
* @return the resource found by this hook or null if normal delegation should continue
* @throws FileNotFoundException to terminate the delegation
*/
public URL postFindResource(String name, ModuleClassLoader classLoader) throws FileNotFoundException {
return null;
}
/**
* Called by a {@link BundleLoader#findResources(String)} before delegating to the resolved constraints and
* local bundle for a resource load. If this method returns null then normal delegation is done.
* If this method returns a non-null value then the rest of the delegation process is skipped and the returned value is used.
* If this method throws an <code>FileNotFoundException</code> then the delegation is terminated
* @param name the name of the resource to find
* @param classLoader the bundle class loader
* @return the resources found by this hook or null if normal delegation should continue
* @throws FileNotFoundException to terminate the delegation
*/
public Enumeration<URL> preFindResources(String name, ModuleClassLoader classLoader) throws FileNotFoundException {
return null;
}
/**
* Called by a {@link BundleLoader#findResources(String)} after delegating to the resolved constraints and
* local bundle for a resource load. This method will only be called if no resources were found
* from the normal delegation.
* @param name the name of the resource to find
* @param classLoader the bundle class loader
* @return the resources found by this hook or null if normal delegation should continue
* @throws FileNotFoundException to terminate the delegation
*/
public Enumeration<URL> postFindResources(String name, ModuleClassLoader classLoader) throws FileNotFoundException {
return null;
}
/**
* Called by a {@link ClasspathManager} before normal delegation. If this method returns
* a non-null value then the rest of the delegation process is skipped and the returned value
* is used.
* @param name the name of the library to find
* @param classLoader the bundle class loader
* @return the library found by this hook or null if normal delegation should continue
* @throws FileNotFoundException to terminate the delegation
*/
public String preFindLibrary(String name, ModuleClassLoader classLoader) throws FileNotFoundException {
return null;
}
/**
* Called by a {@link ClasspathManager} after normal delegation. This method will only be called
* if no library was found from the normal delegation.
* @param name the name of the library to find
* @param classLoader the bundle class loader
* @return the library found by this hook or null if normal delegation should continue
*/
public String postFindLibrary(String name, ModuleClassLoader classLoader) {
return null;
}
/**
* Gets called by a classpath manager during {@link ClasspathManager#findLocalClass(String)} before
* searching the local classloader for a class. A classpath manager will call this method for
* each configured class loading stat hook.
* @param name the name of the requested class
* @param manager the classpath manager used to find and load the requested class
* @throws ClassNotFoundException to prevent the requested class from loading
*/
public void preFindLocalClass(String name, ClasspathManager manager) throws ClassNotFoundException {
// do nothing
}
/**
* Gets called by a classpath manager during {@link ClasspathManager#findLocalClass(String)} after
* searching the local classloader for a class. A classpath manager will call this method for
* each configured class loading stat hook.
* @param name the name of the requested class
* @param clazz the loaded class or null if not found
* @param manager the classpath manager used to find and load the requested class
* @throws ClassNotFoundException
*/
public void postFindLocalClass(String name, Class<?> clazz, ClasspathManager manager) throws ClassNotFoundException {
// do nothing
}
/**
* Gets called by a classpath manager during {@link ClasspathManager#findLocalResource(String)} before
* searching the local classloader for a resource. A classpath manager will call this method for
* each configured class loading stat hook.
* @param name the name of the requested resource
* @param manager the classpath manager used to find the requested resource
*/
public void preFindLocalResource(String name, ClasspathManager manager) {
// do nothing
}
/**
* Gets called by a classpath manager during {@link ClasspathManager#findLocalResource(String)} after
* searching the local classloader for a resource. A classpath manager will call this method for
* each configured class loading stat hook.
* @param name the name of the requested resource
* @param resource the URL to the requested resource or null if not found
* @param manager the classpath manager used to find the requested resource
*/
public void postFindLocalResource(String name, URL resource, ClasspathManager manager) {
// do nothing
}
/**
* Gets called by a classpath manager after an attempt is made to define a class. This method allows
* a class loading stat hook to record data about a class definition.
* @param name the name of the class that got defined
* @param clazz the class object that got defined or null if an error occurred while defining a class
* @param classbytes the class bytes used to define the class
* @param classpathEntry the ClasspathEntry where the class bytes got read from
* @param entry the BundleEntyr source of the class bytes
* @param manager the classpath manager used to define the class
*/
public void recordClassDefine(String name, Class<?> clazz, byte[] classbytes, ClasspathEntry classpathEntry, BundleEntry entry, ClasspathManager manager) {
// do nothing
}
/**
* Returns the parent class loader to be used by all ModuleClassLoaders.
* A {@code null} value may be returned if this hook does not supply the parent.
* Only one hook is able to provide the implementation of the parent class loader
* and the first one to return non-null wins.
* @param configuration the equinox configuration
* @return the parent class loader to be used by all ModuleClassLoaders
*/
public ClassLoader getModuleClassLoaderParent(EquinoxConfiguration configuration) {
// do nothing by default
return null;
}
/**
* Returns true if this hook can support invoking
* {@link ClassLoaderHook#processClass(String, byte[], ClasspathEntry, BundleEntry, ClasspathManager) processClass}
* recursively for the same class name. If false is returned then a class
* loading error will occur if recursive class processing is detected.
* <p>
* This method must return a constant boolean value.
* @return true if recursing class processing is supported
*/
public boolean isProcessClassRecursionSupported() {
return false;
}
}