| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.apache.bcel.util; |
| |
| import java.io.ByteArrayInputStream; |
| import java.io.IOException; |
| import java.util.Hashtable; |
| |
| import org.apache.bcel.Const; |
| import org.apache.bcel.classfile.ClassParser; |
| import org.apache.bcel.classfile.ConstantClass; |
| import org.apache.bcel.classfile.ConstantPool; |
| import org.apache.bcel.classfile.ConstantUtf8; |
| import org.apache.bcel.classfile.JavaClass; |
| import org.apache.bcel.classfile.Utility; |
| |
| /** |
| * <p> |
| * Drop in replacement for the standard class loader of the JVM. You can use it in conjunction with the JavaWrapper to |
| * dynamically modify/create classes as they're requested. |
| * </p> |
| * |
| * <p> |
| * This class loader recognizes special requests in a distinct format, i.e., when the name of the requested class |
| * contains with "$$BCEL$$" it calls the createClass() method with that name (everything bevor the $$BCEL$$ is |
| * considered to be the package name. You can subclass the class loader and override that method. "Normal" classes class |
| * can be modified by overriding the modifyClass() method which is called just before defineClass(). |
| * </p> |
| * |
| * <p> |
| * There may be a number of packages where you have to use the default class loader (which may also be faster). You can |
| * define the set of packages where to use the system class loader in the constructor. The default value contains |
| * "java.", "sun.", "javax." |
| * </p> |
| * |
| * @see JavaWrapper |
| * @see ClassPath |
| * @deprecated 6.0 Do not use - does not work |
| */ |
| @Deprecated |
| public class ClassLoader extends java.lang.ClassLoader { |
| |
| private static final String BCEL_TOKEN = "$$BCEL$$"; |
| |
| public static final String[] DEFAULT_IGNORED_PACKAGES = {"java.", "javax.", "sun."}; |
| |
| private final Hashtable<String, Class<?>> classes = new Hashtable<>(); |
| // Hashtable is synchronized thus thread-safe |
| private final String[] ignoredPackages; |
| private Repository repository = SyntheticRepository.getInstance(); |
| |
| /** |
| * Ignored packages are by default ( "java.", "sun.", "javax."), i.e. loaded by system class loader |
| */ |
| public ClassLoader() { |
| this(DEFAULT_IGNORED_PACKAGES); |
| } |
| |
| /** |
| * @param deferTo delegate class loader to use for ignored packages |
| */ |
| public ClassLoader(final java.lang.ClassLoader deferTo) { |
| super(deferTo); |
| this.ignoredPackages = DEFAULT_IGNORED_PACKAGES; |
| this.repository = new ClassLoaderRepository(deferTo); |
| } |
| |
| /** |
| * @param ignoredPackages classes contained in these packages will be loaded with the system class loader |
| * @param deferTo delegate class loader to use for ignored packages |
| */ |
| public ClassLoader(final java.lang.ClassLoader deferTo, final String[] ignoredPackages) { |
| this(ignoredPackages); |
| this.repository = new ClassLoaderRepository(deferTo); |
| } |
| |
| /** |
| * @param ignoredPackages classes contained in these packages will be loaded with the system class loader |
| */ |
| public ClassLoader(final String[] ignoredPackages) { |
| this.ignoredPackages = ignoredPackages; |
| } |
| |
| /** |
| * Override this method to create you own classes on the fly. The name contains the special token $$BCEL$$. Everything |
| * before that token is considered to be a package name. You can encode your own arguments into the subsequent string. |
| * You must ensure however not to use any "illegal" characters, i.e., characters that may not appear in a Java class |
| * name too |
| * <p> |
| * The default implementation interprets the string as a encoded compressed Java class, unpacks and decodes it with the |
| * Utility.decode() method, and parses the resulting byte array and returns the resulting JavaClass object. |
| * </p> |
| * |
| * @param className compressed byte code with "$$BCEL$$" in it |
| */ |
| protected JavaClass createClass(final String className) { |
| final int index = className.indexOf(BCEL_TOKEN); |
| final String realName = className.substring(index + BCEL_TOKEN.length()); |
| JavaClass clazz = null; |
| try { |
| final byte[] bytes = Utility.decode(realName, true); |
| final ClassParser parser = new ClassParser(new ByteArrayInputStream(bytes), "foo"); |
| clazz = parser.parse(); |
| } catch (final IOException e) { |
| e.printStackTrace(); |
| return null; |
| } |
| // Adapt the class name to the passed value |
| final ConstantPool cp = clazz.getConstantPool(); |
| final ConstantClass cl = cp.getConstant(clazz.getClassNameIndex(), Const.CONSTANT_Class, ConstantClass.class); |
| final ConstantUtf8 name = cp.getConstantUtf8(cl.getNameIndex()); |
| name.setBytes(Utility.packageToPath(className)); |
| return clazz; |
| } |
| |
| @Override |
| protected Class<?> loadClass(final String className, final boolean resolve) throws ClassNotFoundException { |
| Class<?> cl = null; |
| /* |
| * First try: lookup hash table. |
| */ |
| if ((cl = classes.get(className)) == null) { |
| /* |
| * Second try: Load system class using system class loader. You better don't mess around with them. |
| */ |
| for (final String ignoredPackage : ignoredPackages) { |
| if (className.startsWith(ignoredPackage)) { |
| cl = getParent().loadClass(className); |
| break; |
| } |
| } |
| if (cl == null) { |
| JavaClass clazz = null; |
| /* |
| * Third try: Special request? |
| */ |
| if (className.contains(BCEL_TOKEN)) { |
| clazz = createClass(className); |
| } else { // Fourth try: Load classes via repository |
| if ((clazz = repository.loadClass(className)) == null) { |
| throw new ClassNotFoundException(className); |
| } |
| clazz = modifyClass(clazz); |
| } |
| if (clazz != null) { |
| final byte[] bytes = clazz.getBytes(); |
| cl = defineClass(className, bytes, 0, bytes.length); |
| } else { |
| cl = Class.forName(className); |
| } |
| } |
| if (resolve) { |
| resolveClass(cl); |
| } |
| } |
| classes.put(className, cl); |
| return cl; |
| } |
| |
| /** |
| * Override this method if you want to alter a class before it gets actually loaded. Does nothing by default. |
| */ |
| protected JavaClass modifyClass(final JavaClass clazz) { |
| return clazz; |
| } |
| } |